Skip to content

Activity (a.k.a) "News" Feed implementation focused on simplicity, scalability and read-time performance. It is powered by a Redis-based backend, and is compatible with sharding data using Twemproxy.

License

Notifications You must be signed in to change notification settings

anfinil/activity-feed

Repository files navigation

ActivityFeed

This is a ruby implementation of a fast activity feed commonly used in a typical social network-like applications. The implementation is optimized for read-time performance and high concurrency (lots of users). A default Redis-based backend implementation is provided, with the API supporting new backends very easily.

This project is sponsored by Simbi, Inc.

WARNING: this project is under active development, and is not yet finished

Gem Version MIT licensed

Build Status Code Climate Test Coverage Issue Count

Overview, Usage and Installation

Design

Serialization and De-Serialization

Key Features

What's an Activity Feed?

Here is an example of a text-based activity feed that is very common today on social networking sites.

Example

The stories in the feed depend entirely on the application using this library, therefore to integrate with ActivityFeed requires a few additional glue points in your code, mostly around serializing your objects to and from the feed.

Overview

This project has been developed from scratch.

The feed library aims to address the following goals:

  • To define a minimalistic API for a typical event-based activity feed, without tying it to any concrete backend
  • To make it easy to implement and plug in a new type of backend, eg. using Couchbase or MongoDB
  • To provide a scalable default backend implementation using Redis, which can support millions of users via sharding
  • To support multiple activity feeds within the same application, but used for different purposes, eg. activity feed of my followers, versus activity feed of my own actions.

Usage

First you need to configure the Feed with a valid backend implementation.

Configuration

    require 'activity-feed'
    require 'activity-feed/backend/redis_backend'
      
    ActivityFeed.feed(:friends_news) do |config|
      config.backend      = ActivityFeed::Backend::RedisBackend.new(
        config: config,
        redis: -> { ::Redis.new(host: '127.0.0.1') },
      )
      config.max_size     = 1000 # how many items can be in the feed
      config.per_page     = 20   # default page size
    end

Above we've configured the Redis client, passed the proc that creates new Redis clients into the Redis Backend for ActivityFeed. We've also limited the max size of the feed to a 1000 items – which are typically 1000 most recent events.

Multiple Independent Activity Feeds

But sometimes a single feed is not enough. What if we wanted to maintain two separate personalized feeds for each user: one would be news articles the user subscribes to, and the other would be a more typical activity feed.

We can create an additional activity feed, say for followers, and call it :followers at the same time, and configure it with a slightly different backend. Because we expect this activity feed to be more taxing – as events might have large audiences — we'll wrap it in the ConnectionPool that will create several connections that can be used concurrently:

  require 'activity-feed'
  require 'activity-feed/backend/redis_backend'
  require 'activity-feed/backend/hash_backend'
  
  # This is the feed of news articles based on user
  # subscription preferences, use a local hash implementation
  ActivityFeed.feed(:friends_news) do |config|
    config.backend = ActivityFeed::Backend::HashBackend.new(
      config: config
    )
    config.per_page = 20
  end

  # This is the feed of events associated with the followers.
  # We use ConnectionPool because we anticipate higher load.
  ActivityFeed.feed(:followers) do |config|
    config.backend = ActivityFeed::Backend::RedisBackend.new(
      config: config,
      redis: ::ConnectionPool.new(size: 5, timeout: 5) do
                ::Redis.new(host: '192.168.10.10', port: 9000)
              end
    )
    config.per_page = 50
  end
Referencing Multiple Feeds

So how do you access the feed from your code? Please check the UML diagram above to see how objects are returned.

When we called ActivityFeed.feed(:friends_news) for the very first time, the library created a hash key :friends_news that from now on will point to this instance of the feed configuration within the application.

In addition, the gem also created a constant and a method under the ActivityFeed namespace. For example, given a name such as :friends_news the following are all valid ways of accessing the feed:

  • ActivityFeed::FriendsNews
  • ActivityFeed.friends_news
  • ActivityFeed.feed(:friends_news)

You can also get a full list of currently defined feeds with ActivityFeed.feed_names method.

Publishing Data to the Feed

When we publish events to the feeds, we typically (although not always) do it for many feeds at the same time. This is why the write operations expect a list of users, or an enumeration, or a block yielding batches of the users:

    require 'activity-feed'
    
    # First we define list of users (or "owners") of the activity feed to be
    # populated with the given event 
    users = [User.find(1), User.find(2), User.find(3) ]
    
    # Next, we instantiate the feed by passing the list of users,
    # and then we publish the event across all of the corresponding feeds.
    @feed = ActivityFeed.friends_news.for(users)
    # And then we publish the event to each feed:
    @feed.publish(sort: Time.now, event: event)

Instead of passing the list of user IDs, you can pass an ActiveRecord::Relation, or a block — which should yield the next element in the array when called, or nil when exhausted.

For any object types besides Integer, ActivityFeed will call a method #to_af on the object, in order to receive a string representation of that object.

    # This is just an example of how you could return AREL statement
    # which can then be fetched in groups (pages) of users and split into
    # several parallel jobs by ActivityFeed.
    
    @follower = User.where(follower: @event.actor)
    @feed = ActivityFeed.feed(:followers_feed).for(@follower)
    @feed.publish(event: @event, sort: Time.now) # publish the event sorted by time.
Writing Efficiently, and/or Concurrently

For large data sets it is generally required to use batch operations, instead of looping for each user. If you are using Rails, then the corresponding method of interest is #find_in_batches, which can apply to any ActiveRecord::Relation instance. This method retrieves a batch of records and then yields the entire batch to the block as an array of models.

If you are not using Rails, you can still use any custom method that yields batches, one by one, to the block, where each batch can be as an array of integers or models.

    @feed = ActivityFeed.feed(:news_feed).for do
      User.where(followee: @event.actor)
          .find_in_batches(batch_size: 1000) { |users| yield(users) }
    end
    
    # Now the #publish method can batch pushing the event to the users, 
    # possibly in parallel as a possible optimization.
    @feed.publish(event: @event)

Reading the Feed Using #paginate and #find_in_batches

  require 'activity-feed'

  # You can also use just #reader method, instead of #create_reader
  @feed = ActivityFeed.feed(:news_feed).for(User.where(username: 'kig').first)
  @feed.paginate(page: 1, per_page: 20)  
  # => [ <Events::FavoriteCommentEvent#0x2134afa user: ..., comment: ...>, <Events::StoryPostedEvent...>]

OR You can also use another method #find_in_batches, which is meant to emulate similar method available in Rails framework. The method can be configured with different batch size, and yields a up to max events to the block defined.

  @feed.find_in_batches(batch_size: 100) do |events|
    # do something with the list of events for this batch.
  end
end

Rendering a Single Page

To actually render/display the feed to the user, we can render each element (or event) returned by the #paginate call:

  json = @feed.paginate(page: 1, per_page: 20).map do |event|
    event.render(:json)                                    
    # => { "name": "FavoriteComment", "user": { "username": "kig" }, .... }"
  end.join(', ')

Installation

Add this line to your application's Gemfile:

    gem 'activity-feed'

And then execute:

$ bundle

Or install it yourself as:

$ gem install activity-feed

Development

After checking out the repo, run bin/setup to install dependencies. Then, run rake spec to run the tests. You can also run bin/console for an interactive prompt that will allow you to experiment.

To install this gem onto your local machine, run bundle exec rake install. To release a new version, update the version number in version.rb, and then run bundle exec rake release, which will create a git tag for the version, push git commits and tags, and push the .gem file to rubygems.org.

Contributing

Bug reports and pull requests are welcome on GitHub at https://github.com/kigster/activity-feed

License

The gem is available as open source under the terms of the MIT License.

Acknowledgements

  • This project is conceived and sponsored by Simbi, Inc..
  • Author's personal experience at Wanelo, Inc. has served as an inspiration.

About

Activity (a.k.a) "News" Feed implementation focused on simplicity, scalability and read-time performance. It is powered by a Redis-based backend, and is compatible with sharding data using Twemproxy.

Resources

License

Stars

Watchers

Forks

Packages

No packages published