Class: ConvenientService::Dependencies::Extractions::RubyMiddleware::Middleware::Builder

Inherits:
Object
  • Object
show all
Defined in:
lib/convenient_service/dependencies/extractions/ruby_middleware/middleware/builder.rb

Overview

This provides a DSL for building up a stack of middlewares.

This code is based heavily off of Rack::Builder and ActionDispatch::MiddlewareStack in Rack and Rails, respectively.

Usage

Building a middleware stack is very easy:

app = Middleware::Builder.new do |b|
  b.use A
  b.use B
end

# Call the middleware
app.call(7)

Instance Method Summary collapse

Constructor Details

#initialize(opts = nil) { ... } ⇒ Builder

Initializes the builder. An optional block can be passed which will either yield the builder or be evaluated in the context of the instance.

Example:

Builder.new do |b|
  b.use A
  b.use B
end

Builder.new do
  use A
  use B
end

Parameters:

  • opts (Hash) (defaults to: nil)

    Options hash

Options Hash (opts):

  • :runner_class (Class)

    The class to wrap the middleware stack in which knows how to run them.

Yields:

  • [] Evaluated in this instance which allows you to use methods like #use and such.



63
64
65
66
67
68
69
70
71
72
73
74
75
# File 'lib/convenient_service/dependencies/extractions/ruby_middleware/middleware/builder.rb', line 63

def initialize(opts = nil, &block)
  opts ||= {}
  @runner_class = opts.fetch(:runner_class, Runner)
  @middleware_name = opts.fetch(:name, 'Middleware')

  if block_given?
    if block.arity == 1
      yield self
    else
      instance_eval(&block)
    end
  end
end

Instance Method Details

#call(env = nil) ⇒ Object

Runs the builder stack with the given environment.



152
153
154
# File 'lib/convenient_service/dependencies/extractions/ruby_middleware/middleware/builder.rb', line 152

def call(env = nil)
  to_app.call(env)
end

#delete(index) ⇒ Object

Deletes the given middleware object or index



146
147
148
149
# File 'lib/convenient_service/dependencies/extractions/ruby_middleware/middleware/builder.rb', line 146

def delete(index)
  index = self.index(index) unless index.is_a?(Integer)
  stack.delete_at(index)
end

#flattenObject

Returns a mergeable version of the builder. If use is called with the return value of this method, then the stack will merge, instead of being treated as a separate single middleware.



85
86
87
88
89
# File 'lib/convenient_service/dependencies/extractions/ruby_middleware/middleware/builder.rb', line 85

def flatten
  lambda do |env|
    call(env)
  end
end

#inject_logger(logger) ⇒ Object



163
164
165
166
167
# File 'lib/convenient_service/dependencies/extractions/ruby_middleware/middleware/builder.rb', line 163

def inject_logger logger
  insert_before_each Middleware::Logger, logger, name

  self
end

#insert(index, middleware, *args, &block) ⇒ Object Also known as: insert_before

Inserts a middleware at the given index or directly before the given middleware object.



108
109
110
111
112
# File 'lib/convenient_service/dependencies/extractions/ruby_middleware/middleware/builder.rb', line 108

def insert(index, middleware, *args, &block)
  index = self.index(index) unless index.is_a?(Integer)
  fail "no such middleware to insert before: #{index.inspect}" unless index
  stack.insert(index, [middleware, args, block])
end

#insert_after(index, middleware, *args, &block) ⇒ Object

Inserts a middleware after the given index or middleware object.



117
118
119
120
121
# File 'lib/convenient_service/dependencies/extractions/ruby_middleware/middleware/builder.rb', line 117

def insert_after(index, middleware, *args, &block)
  index = self.index(index) unless index.is_a?(Integer)
  fail "no such middleware to insert after: #{index.inspect}" unless index
  insert(index + 1, middleware, *args, &block)
end

#insert_after_each(middleware, *args, &block) ⇒ Object

Inserts a middleware after each middleware object



131
132
133
134
135
# File 'lib/convenient_service/dependencies/extractions/ruby_middleware/middleware/builder.rb', line 131

def insert_after_each(middleware, *args, &block)
  self.stack = stack.reduce([]) do |carry, item|
    carry.push(item, [middleware, args, block])
  end
end

#insert_before_each(middleware, *args, &block) ⇒ Object

Inserts a middleware before each middleware object



124
125
126
127
128
# File 'lib/convenient_service/dependencies/extractions/ruby_middleware/middleware/builder.rb', line 124

def insert_before_each(middleware, *args, &block)
  self.stack = stack.reduce([]) do |carry, item|
    carry.push([middleware, args, block], item)
  end
end

#inspectObject



156
157
158
159
160
161
# File 'lib/convenient_service/dependencies/extractions/ruby_middleware/middleware/builder.rb', line 156

def inspect
  name+'[' + stack.map do |middleware|
    name = middleware[0].is_a?(Proc) ? 'Proc' : middleware[0].name
    "#{name}(#{middleware[1].join(', ')})"
  end.join(', ') + ']'
end

#nameObject

Returns the name of the current middleware



78
79
80
# File 'lib/convenient_service/dependencies/extractions/ruby_middleware/middleware/builder.rb', line 78

def name
  @middleware_name
end

#replace(index, middleware, *args, &block) ⇒ Object

Replaces the given middleware object or index with the new middleware.



138
139
140
141
142
143
# File 'lib/convenient_service/dependencies/extractions/ruby_middleware/middleware/builder.rb', line 138

def replace(index, middleware, *args, &block)
  index = self.index index unless index.is_a? Integer

  delete(index)
  insert(index, middleware, *args, &block)
end

#use(middleware, *args, &block) ⇒ Object

Adds a middleware class to the middleware stack. Any additional args and a block, if given, are saved and passed to the initializer of the middleware.

Parameters:

  • middleware (Class)

    The middleware class



96
97
98
99
100
101
102
103
104
105
# File 'lib/convenient_service/dependencies/extractions/ruby_middleware/middleware/builder.rb', line 96

def use(middleware, *args, &block)
  if middleware.is_a?(Builder)
    # Merge in the other builder's stack into our own
    stack.concat(middleware.stack)
  else
    stack << [middleware, args, block]
  end

  self
end