Skip to content

Latest commit

 

History

History
163 lines (110 loc) · 6.29 KB

README.md

File metadata and controls

163 lines (110 loc) · 6.29 KB

Agis

Gem Version Build Status

A Redis-based stateless Actor library designed with ActiveRecord in mind. Built on deferred retrying, message boxes are only locked and executed when a method is called, and are executed sequentially in the same thread. We use Agis to implement per-request behavior.

Both parameters and return values are stored as JSON entities. Actor calls are retried until they return without failure.

Features

  • Limited class mixin - methods for actors are explicitly selected with agis_defm
  • Actors are procedures, not processes - they run in the same thread
  • Redis-based data structures (redis lock; message and return boxes)
  • Deferred retrying of failed calls
  • Method calls are only removed from a message box when they return a value
  • Message box renaming with agis_id for more specific data models
  • Method-specific custom timeouts for lock expiry
  • Not many other features - very simple!

Installation

Requires mlanett-redis-lock, required as "redis-lock"

gem 'agis'

Example

require 'agis'
require 'redis'

$redis = Redis.new

class PerCounter < ActiveRecord::Base
  attr_accessor :value
  
  include Agis
  
  def agis_id; "any"; end # Agis id is "any", the name of the message box of any instance of PerCounter
                          # will be PerCounter : any
  
  def incif(arg1) # only increment the value if it's eq to arg1 when the actor calls this method
    @value = $redis.get("counter:" + self.id) or 0
    if(@value == arg1)
      @value += 1
      $redis.set("counter:" + self.id, @value)
    end
  end
  
  def initialize
    agis_defm1 :incif, :retry # retry is default
  end
end

pc = PerCounter.new
pc.agis_call($redis, :incif, pc.value)
puts pc.value

Bank accounts

Let's assume we have a transaction class like this one:

class Transaction
  attr_accessor :id, :amount, :sender, :receiver, :lastref, :senderbalance
  
  def agis_id
    sender.id
  end
end

For simplicity, let's assume that a receiving account can accumulate money indefinitely, but a sending account cannot go below 0. We also need to keep track of the sending balance for fast access, and keep the transactions in the correct order via lastref to prevent double spending.

We can imagine making a transaction like so:

Transaction.create(50, sender.id, recv.id) # agis_id refers to sender_id, this is the number that will be put on the message box, rather than the Transaction's own id

# underneath this calls the agis call like so:
agis_call($redis, :create, [50, sender.id, recv.id])

We defined agis_id as being the sender's id, since the sender is the sensitive part of the transaction and the one that senderbalance refers to.

The agis:create method will probably access the database to get lastref. Meanwhile, no one else is allowed to deal with the Transactions with this sender id, because the message box for it is locked.

Even better, if there's any reason for the transaction to fall out of order in-between, we can retry the transaction by calling the agis:create method again, in the agis method itself - and only give up if the sender's balance isn't enough (or some other reason like frozen accounts), possibly raising an error.

Retrying

Agis allows retrying with agis_recall(), which accepts the same parameters as agis_call(). It doesn't tackle the message box, since it's already locked when called in an Agis method. This does nothing but call the given method among the agis_methods.

Agis doesn't remove any call from the message box that crashes or raises an exception. Instead it will retry it each time. A single agis_call will retry the failed call once, after that it raises an AgisRetryAttemptsExceeded error. As a result, you must write methods which:

  • Assume they will be retried several times
  • Will be retried if they raise an exception that isn't handled in the method itself

However, these restrictions are balanced by the following guarantees :

  • Methods will be retried as many times as needed until they succeed
  • Methods called through the same message box (classname + Object#agis_id) are guaranteed to run in a single thread, in sequence

The message box will effectively not move forward until the method call returns. Provided your method doesn't raise an exception or crash, it's guaranteed to run exactly as many times as it is called.

once vs retry

agis_defm has a second parameter that accepts either :retry or :once. :retry is default:

  • :retry removes the method call from the message box AFTER the call returns without an error
  • :once removes the method call from the message box BEFORE executing it

Example:

agis_defm1 :add_chat_message, :once

:once isn't always necessary and there may be a way that involves unique identifiers. Retrying may be better whenever multiple actors are involved and are locked into some procedural step they otherwise can't get out of.

Instance variables

If you write code that you assume will be retried, the only thing you can be sure of is the classname and agis_id on the message box, everything else is variable. If you write code like this:

class User
...
  def bind # agis method
    v = self.bind_commit_id
    exceptional_procedure(v)
  end
...
end

usr = User.find(711)
usr.bind_commit_id = Bind.create.id
usr.agis_call($redis, :bind)

This code reads bind_commit_id from the current instance, but Agis almost assumes that the instance isn't the same as it was when the actor call was made. A working version of this code would accept instance variables as parameters:

class User
...
  def bind(bid) # agis method
    exceptional_procedure(bid)
  end
...
end

usr = User.find(711)
bid = Bind.create.id
usr.agis_call($redis, :bind, bid)

No-op call

Version 0.2.9 added a no-op call:

usr = User.find(id)
usr.agis_call($redis)

This executes the message box whenever needed, such as when transactions have to be resolved for a total balance.