require 'rails' require 'flipper/engine' RSpec.describe Flipper::Engine do let(:application) do Class.new(Rails::Application) do config.eager_load = false config.logger = ActiveSupport::Logger.new($stdout) end.instance end before do Rails.application = nil ActiveSupport::Dependencies.autoload_paths = ActiveSupport::Dependencies.autoload_paths.dup ActiveSupport::Dependencies.autoload_once_paths = ActiveSupport::Dependencies.autoload_once_paths.dup end # Reset Rails.env around each example around do |example| begin env = Rails.env.to_s example.run ensure Rails.env = env end end let(:config) { application.config.flipper } subject { application.initialize! } shared_examples 'config.strict' do let(:adapter) { Flipper.adapter.adapter } it 'can set strict=true from ENV' do with_env 'FLIPPER_STRICT' => 'true' do subject expect(config.strict).to eq(:raise) expect(adapter).to be_instance_of(Flipper::Adapters::Strict) end end it 'can set strict=warn from ENV' do with_env 'FLIPPER_STRICT' => 'warn' do subject expect(config.strict).to eq(:warn) expect(adapter).to be_instance_of(Flipper::Adapters::Strict) expect(adapter.handler).to be(Flipper::Adapters::Strict::HANDLERS.fetch(:warn)) end end it 'can set strict=false from ENV' do with_env 'FLIPPER_STRICT' => 'false' do subject expect(config.strict).to eq(false) expect(adapter).to be_instance_of(Flipper::Adapters::Memory) end end it "defaults to strict=false in RAILS_ENV=production" do Rails.env = "production" subject expect(config.strict).to eq(false) expect(adapter).to be_instance_of(Flipper::Adapters::Memory) end %w(development test).each do |env| it "defaults to strict=warn in RAILS_ENV=#{env}" do Rails.env = env expect(Rails.env).to eq(env) subject expect(config.strict).to eq(:warn) expect(adapter).to be_instance_of(Flipper::Adapters::Strict) expect(adapter.handler).to be(Flipper::Adapters::Strict::HANDLERS.fetch(:warn)) end end end context 'cloudless' do it_behaves_like 'config.strict' it 'can set env_key from ENV' do with_env 'FLIPPER_ENV_KEY' => 'flopper' do subject expect(config.env_key).to eq('flopper') end end it 'can set memoize from ENV' do with_env 'FLIPPER_MEMOIZE' => 'false' do subject expect(config.memoize).to eq(false) end end it 'can set preload from ENV' do with_env 'FLIPPER_PRELOAD' => 'false' do subject expect(config.preload).to eq(false) end end it 'can set instrumenter from ENV' do stub_const('My::Cool::Instrumenter', Class.new) with_env 'FLIPPER_INSTRUMENTER' => 'My::Cool::Instrumenter' do subject expect(config.instrumenter).to eq(My::Cool::Instrumenter) end end it 'can set log from ENV' do with_env 'FLIPPER_LOG' => 'false' do subject expect(config.log).to eq(false) end end it 'sets defaults' do subject # initialize expect(config.env_key).to eq("flipper") expect(config.memoize).to be(true) expect(config.preload).to be(true) end it "configures instrumentor on default instance" do subject # initialize expect(Flipper.instance.instrumenter).to eq(ActiveSupport::Notifications) end it 'uses Memoizer middleware if config.memoize = true' do initializer { config.memoize = true } expect(subject.middleware).to include(Flipper::Middleware::Memoizer) end it 'does not use Memoizer middleware if config.memoize = false' do initializer { config.memoize = false } expect(subject.middleware).not_to include(Flipper::Middleware::Memoizer) end it 'passes config to memoizer' do initializer do config.update( env_key: 'my_flipper', preload: [:stats, :search] ) end expect(subject.middleware).to include(Flipper::Middleware::Memoizer) middleware = subject.middleware.detect { |m| m.klass == Flipper::Middleware::Memoizer } expect(middleware.args[0]).to eq({ env_key: config.env_key, preload: config.preload, if: nil }) end end context 'with cloud' do around do |example| with_env "FLIPPER_CLOUD_TOKEN" => "test-token" do example.run end end # App for Rack::Test let(:app) { application.routes } it_behaves_like 'config.strict' do let(:adapter) do dual_write = Flipper.adapter.adapter poll = dual_write.local poll.adapter end end it "initializes cloud configuration" do stub_request(:get, /flippercloud\.io/).to_return(status: 200, body: "{}") application.initialize! expect(Flipper.instance).to be_a(Flipper::Cloud::DSL) expect(Flipper.instance.instrumenter).to be(ActiveSupport::Notifications) end context "with CLOUD_SYNC_SECRET" do around do |example| with_env "FLIPPER_CLOUD_SYNC_SECRET" => "test-secret" do example.run end end let(:request_body) do JSON.generate({ "environment_id" => 1, "webhook_id" => 1, "delivery_id" => SecureRandom.uuid, "action" => "sync", }) end let(:timestamp) { Time.now } let(:signature) { Flipper::Cloud::MessageVerifier.new(secret: ENV["FLIPPER_CLOUD_SYNC_SECRET"]).generate(request_body, timestamp) } let(:signature_header_value) { Flipper::Cloud::MessageVerifier.new(secret: "").header(signature, timestamp) } it "configures webhook app" do application.initialize! stub = stub_request(:get, "https://www.flippercloud.io/adapter/features?exclude_gate_names=true").with({ headers: { "Flipper-Cloud-Token" => ENV["FLIPPER_CLOUD_TOKEN"] }, }).to_return(status: 200, body: JSON.generate({ features: {} }), headers: {}) post "/_flipper", request_body, { "HTTP_FLIPPER_CLOUD_SIGNATURE" => signature_header_value } expect(last_response.status).to eq(200) expect(stub).to have_been_requested end end context "without CLOUD_SYNC_SECRET" do it "does not configure webhook app" do application.initialize! post "/_flipper" expect(last_response.status).to eq(404) end end context "without FLIPPER_CLOUD_TOKEN" do it "gracefully skips configuring webhook app" do with_env "FLIPPER_CLOUD_TOKEN" => nil do application.initialize! expect(Flipper.instance).to be_a(Flipper::DSL) end post "/_flipper" expect(last_response.status).to eq(404) end end end context 'with cloud secrets in Rails.credentials' do around do |example| # Create temporary directory for Rails.root to write credentials to # Once Rails 5.2 support is dropped, this can all be replaced with # `config.credentials.content_path = Tempfile.new.path` Dir.mktmpdir do |dir| Dir.chdir(dir) do Dir.mkdir("#{dir}/config") example.run end end end before do # Set master key which is needed to write credentials ENV["RAILS_MASTER_KEY"] = "a" * 32 application.credentials.write(YAML.dump({ flipper: { cloud_token: "credentials-token", cloud_sync_secret: "credentials-secret", } })) end it "enables cloud" do application.initialize! expect(ENV["FLIPPER_CLOUD_TOKEN"]).to eq("credentials-token") expect(ENV["FLIPPER_CLOUD_SYNC_SECRET"]).to eq("credentials-secret") expect(Flipper.instance).to be_a(Flipper::Cloud::DSL) end end it "includes model methods" do subject require 'active_record' expect(ActiveRecord::Base.ancestors).to include(Flipper::Model::ActiveRecord) end # Add app initializer in the same order as config/initializers/* def initializer(&block) application.initializer 'spec', before: :load_config_initializers do block.call end end end