/flexcon

"Spread" scope variables and pass them to lambdas

Primary LanguageRuby

Flexcon

Flexible consignment of arguments to lambdas using scopes.

Installation

Add this line to your application's Gemfile:

gem 'flexcon'

And then execute:

$ bundle

Or install it yourself as:

$ gem install flexcon

Usage

And then execute:

$ bundle

Or install it yourself as:

$ gem install Flexcon

Basic Usage

Given a scope which could be an array, a hash, an object, or a proc - dispatch attributes of the scope into lambdas:

array_scope = ['John', 'Doe', 'Acme Corporation']
hash_scope = { last_name: 'Doe', first_name: 'John', company: 'Acme Corporation' }
object_scope = OpenStruct.new(hash_scope)

full_name = lambda do |first_name, last_name|
  "#{first_name} #{last_name}"
end

employee = lambda do |first_name, last_name, company|
  "#{first_name} #{last_name} of #{company}"
end

# The following returns 'John Doe'
Flexcon.dispatch(array_scope, full_name)
Flexcon.dispatch(hash_scope, full_name)
Flexcon.dispatch(object_scope, full_name)

# The following returns 'John Doe of Acme Corporation' 
Flexcon.dispatch(array_scope, employee)
Flexcon.dispatch(hash_scope, employee)
Flexcon.dispatch(object_scope, employee)

# The following returns "['user', 'student', 'university']"
proc_scope = -> () do
  {
    models: [:user, :student, :university],
    api: -> (params) do
      params.to_json
    end
  }
end

api_wrapper = lambda do |api, models| 
  api.call(models)
end

Flexcon.dispatch(proc_scope, api_wrapper)

Advanced Usage

If the method signature needed for certain functions are difficult to predict, instead of enforcing a single method signature, allow methods to ask for what they need from a scope.

class Operation
  def initialize
    @steps = []
    @named = {}
  end

  def step(name, &block)
    @steps << block
    @named[name] = block
  end

  def call(scope)
    @steps.each do |step|
      Flexcon.dispatch(scope, step)
    end
  end
end

class Scope
  attr_accessor :params, :models

  def initialize(params={})
    self.params = params
    self.models = {}
  end
end

op = Operation.new

op.step :find_or_init_user do |params, models|
  user = User.find_by(email: params[:user][:email])
  user ||= User.new(params[:user])

  models[:user] = user
end

op.step :find_or_init_student do |params, models|
  student = Student.find_or_initialize_by(user: models[:user], university_id: params[:university_id])
  student.assign_attributes(params[:student])

  models[:student] = student
end

op.step :save_user do |models|
  models[:user].save!
end

op.step :save_student do |models|
  models[:student].save!
end

op.call(Scope.new({}))

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/rcpedro/flexcon.