1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75
|
require 'rack/body_proxy'
module Flipper
module Middleware
class Memoizer
# Public: Initializes an instance of the Memoizer middleware. Flipper must
# be configured with a default instance or the flipper instance must be
# setup in the env of the request. You can do this by using the
# Flipper::Middleware::SetupEnv middleware.
#
# app - The app this middleware is included in.
# opts - The Hash of options.
# :preload_all - Boolean of whether or not to preload all features.
# :preload - Array of Symbol feature names to preload.
#
# Examples
#
# use Flipper::Middleware::Memoizer
#
# # using with preload_all features
# use Flipper::Middleware::Memoizer, preload_all: true
#
# # using with preload specific features
# use Flipper::Middleware::Memoizer, preload: [:stats, :search, :some_feature]
#
def initialize(app, opts = {})
if opts.is_a?(Flipper::DSL) || opts.is_a?(Proc)
raise 'Flipper::Middleware::Memoizer no longer initializes with a flipper instance or block. Read more at: https://git.io/vSo31.' # rubocop:disable LineLength
end
@app = app
@opts = opts
@env_key = opts.fetch(:env_key, 'flipper')
end
def call(env)
request = Rack::Request.new(env)
if skip_memoize?(request)
@app.call(env)
else
memoized_call(env)
end
end
private
def skip_memoize?(request)
@opts[:unless] && @opts[:unless].call(request)
end
def memoized_call(env)
reset_on_body_close = false
flipper = env.fetch(@env_key) { Flipper }
original = flipper.memoizing?
flipper.memoize = true
flipper.preload_all if @opts[:preload_all]
if (preload = @opts[:preload])
flipper.preload(preload)
end
response = @app.call(env)
response[2] = Rack::BodyProxy.new(response[2]) do
flipper.memoize = original
end
reset_on_body_close = true
response
ensure
flipper.memoize = original if flipper && !reset_on_body_close
end
end
end
end
|