webvalve 0.9.0
Sign up to get free protection for your applications and to get access to all the features.
- checksums.yaml +7 -0
- data/LICENSE +21 -0
- data/README.md +195 -0
- data/Rakefile +22 -0
- data/lib/generators/webvalve/fake_service_generator.rb +70 -0
- data/lib/generators/webvalve/install_generator.rb +33 -0
- data/lib/tasks/webvalve_tasks.rake +0 -0
- data/lib/webvalve.rb +40 -0
- data/lib/webvalve/engine.rb +26 -0
- data/lib/webvalve/fake_service.rb +22 -0
- data/lib/webvalve/fake_service_config.rb +53 -0
- data/lib/webvalve/fake_service_wrapper.rb +18 -0
- data/lib/webvalve/instrumentation.rb +4 -0
- data/lib/webvalve/instrumentation/log_subscriber.rb +16 -0
- data/lib/webvalve/instrumentation/middleware.rb +24 -0
- data/lib/webvalve/manager.rb +76 -0
- data/lib/webvalve/rspec.rb +8 -0
- data/lib/webvalve/version.rb +3 -0
- data/spec/dummy/README.rdoc +28 -0
- data/spec/dummy/Rakefile +6 -0
- data/spec/dummy/app/assets/javascripts/application.js +13 -0
- data/spec/dummy/app/assets/stylesheets/application.css +15 -0
- data/spec/dummy/app/controllers/application_controller.rb +5 -0
- data/spec/dummy/app/helpers/application_helper.rb +2 -0
- data/spec/dummy/app/views/layouts/application.html.erb +13 -0
- data/spec/dummy/bin/bundle +3 -0
- data/spec/dummy/bin/rails +4 -0
- data/spec/dummy/bin/rake +4 -0
- data/spec/dummy/bin/setup +29 -0
- data/spec/dummy/config.ru +4 -0
- data/spec/dummy/config/application.rb +31 -0
- data/spec/dummy/config/boot.rb +5 -0
- data/spec/dummy/config/database.yml +25 -0
- data/spec/dummy/config/environment.rb +5 -0
- data/spec/dummy/config/environments/development.rb +41 -0
- data/spec/dummy/config/environments/production.rb +79 -0
- data/spec/dummy/config/environments/test.rb +42 -0
- data/spec/dummy/config/initializers/assets.rb +11 -0
- data/spec/dummy/config/initializers/backtrace_silencers.rb +7 -0
- data/spec/dummy/config/initializers/cookies_serializer.rb +3 -0
- data/spec/dummy/config/initializers/filter_parameter_logging.rb +4 -0
- data/spec/dummy/config/initializers/inflections.rb +16 -0
- data/spec/dummy/config/initializers/mime_types.rb +4 -0
- data/spec/dummy/config/initializers/session_store.rb +3 -0
- data/spec/dummy/config/initializers/wrap_parameters.rb +14 -0
- data/spec/dummy/config/locales/en.yml +23 -0
- data/spec/dummy/config/routes.rb +3 -0
- data/spec/dummy/config/secrets.yml +22 -0
- data/spec/dummy/db/test.sqlite3 +0 -0
- data/spec/dummy/log/development.log +0 -0
- data/spec/dummy/log/test.log +3938 -0
- data/spec/dummy/public/404.html +67 -0
- data/spec/dummy/public/422.html +67 -0
- data/spec/dummy/public/500.html +66 -0
- data/spec/dummy/public/favicon.ico +0 -0
- data/spec/examples.txt +30 -0
- data/spec/rails_helper.rb +14 -0
- data/spec/spec_helper.rb +22 -0
- data/spec/support/helpers.rb +24 -0
- data/spec/webvalve/fake_service_config_spec.rb +133 -0
- data/spec/webvalve/fake_service_spec.rb +47 -0
- data/spec/webvalve/manager_spec.rb +132 -0
- data/spec/webvalve_spec.rb +19 -0
- metadata +268 -0
@@ -0,0 +1,53 @@
|
|
1
|
+
module WebValve
|
2
|
+
class FakeServiceConfig
|
3
|
+
ENABLED_VALUES = %w(1 t true)
|
4
|
+
|
5
|
+
attr_reader :service
|
6
|
+
|
7
|
+
def initialize(service:, url: nil)
|
8
|
+
@service = service
|
9
|
+
@custom_service_url = url
|
10
|
+
end
|
11
|
+
|
12
|
+
def should_intercept?
|
13
|
+
Rails.env.test? ||
|
14
|
+
(WebValve.enabled? && !service_enabled_in_env?)
|
15
|
+
end
|
16
|
+
|
17
|
+
def service_url
|
18
|
+
@service_url ||= begin
|
19
|
+
url = custom_service_url || default_service_url
|
20
|
+
raise missing_url_message if url.blank?
|
21
|
+
strip_basic_auth url
|
22
|
+
end
|
23
|
+
end
|
24
|
+
|
25
|
+
private
|
26
|
+
|
27
|
+
attr_reader :custom_service_url
|
28
|
+
|
29
|
+
def missing_url_message
|
30
|
+
<<~MESSAGE
|
31
|
+
There is no URL defined for #{service.name}.
|
32
|
+
Configure one by setting the ENV variable "#{service_name.to_s.upcase}_API_URL"
|
33
|
+
or by using WebValve.register #{service.name}, url: "http://something.dev"
|
34
|
+
MESSAGE
|
35
|
+
end
|
36
|
+
|
37
|
+
def strip_basic_auth(url)
|
38
|
+
url.to_s.sub(%r(\Ahttp(s)?://[^@/]+@), 'http\1://')
|
39
|
+
end
|
40
|
+
|
41
|
+
def service_enabled_in_env?
|
42
|
+
ENABLED_VALUES.include?(ENV["#{service_name.to_s.upcase}_ENABLED"])
|
43
|
+
end
|
44
|
+
|
45
|
+
def default_service_url
|
46
|
+
ENV["#{service_name.to_s.upcase}_API_URL"]
|
47
|
+
end
|
48
|
+
|
49
|
+
def service_name
|
50
|
+
@service_name ||= service.name.demodulize.underscore.sub 'fake_', ''
|
51
|
+
end
|
52
|
+
end
|
53
|
+
end
|
@@ -0,0 +1,18 @@
|
|
1
|
+
module WebValve
|
2
|
+
class FakeServiceWrapper
|
3
|
+
# lazily resolve the app constant to leverage rails class reloading
|
4
|
+
def initialize(app)
|
5
|
+
@app_klass_name = app.name
|
6
|
+
end
|
7
|
+
|
8
|
+
def call(env)
|
9
|
+
app.call(env)
|
10
|
+
end
|
11
|
+
|
12
|
+
private
|
13
|
+
|
14
|
+
def app
|
15
|
+
@app_klass_name.constantize
|
16
|
+
end
|
17
|
+
end
|
18
|
+
end
|
@@ -0,0 +1,16 @@
|
|
1
|
+
module WebValve
|
2
|
+
module Instrumentation
|
3
|
+
class LogSubscriber < ActiveSupport::LogSubscriber
|
4
|
+
def request(event)
|
5
|
+
return unless logger.debug?
|
6
|
+
status = event.payload[:status]
|
7
|
+
method = event.payload[:method].to_s.upcase
|
8
|
+
url = event.payload[:url]
|
9
|
+
host = event.payload[:host]
|
10
|
+
name = '%s %s (%.1fms)' % ["WebValve", "Request Captured", event.duration]
|
11
|
+
details = "#{host} #{method} #{url} [#{status}]"
|
12
|
+
debug " #{color(name, YELLOW, true)} #{color(details, BOLD, true)}"
|
13
|
+
end
|
14
|
+
end
|
15
|
+
end
|
16
|
+
end
|
@@ -0,0 +1,24 @@
|
|
1
|
+
module WebValve
|
2
|
+
module Instrumentation
|
3
|
+
class Middleware
|
4
|
+
METHOD = 'REQUEST_METHOD'.freeze
|
5
|
+
PATH = 'PATH_INFO'.freeze
|
6
|
+
HOST = 'SERVER_NAME'.freeze
|
7
|
+
|
8
|
+
def initialize(app)
|
9
|
+
@app = app
|
10
|
+
end
|
11
|
+
|
12
|
+
def call(env)
|
13
|
+
ActiveSupport::Notifications.instrument('request.webvalve') do |payload|
|
14
|
+
payload[:method] = env[METHOD]
|
15
|
+
payload[:url] = env[PATH]
|
16
|
+
payload[:host] = env[HOST]
|
17
|
+
@app.call(env).tap do |status, _header, _body|
|
18
|
+
payload[:status] = status
|
19
|
+
end
|
20
|
+
end
|
21
|
+
end
|
22
|
+
end
|
23
|
+
end
|
24
|
+
end
|
@@ -0,0 +1,76 @@
|
|
1
|
+
require 'webmock'
|
2
|
+
require 'singleton'
|
3
|
+
require 'set'
|
4
|
+
|
5
|
+
module WebValve
|
6
|
+
# @api private
|
7
|
+
class Manager
|
8
|
+
include Singleton
|
9
|
+
|
10
|
+
def register(fake_service, **args)
|
11
|
+
raise "#{fake_service.inspect} already registered" if fake_service_configs.any? { |c| c.service == fake_service }
|
12
|
+
fake_service_configs << FakeServiceConfig.new(service: fake_service, **args)
|
13
|
+
end
|
14
|
+
|
15
|
+
def whitelist_url(url)
|
16
|
+
raise "#{url} already registered" if whitelisted_urls.include?(url)
|
17
|
+
whitelisted_urls << url
|
18
|
+
end
|
19
|
+
|
20
|
+
def setup
|
21
|
+
fake_service_configs.each do |config|
|
22
|
+
if config.should_intercept?
|
23
|
+
webmock_service config
|
24
|
+
else
|
25
|
+
whitelist_service config
|
26
|
+
end
|
27
|
+
end
|
28
|
+
|
29
|
+
WebMock.enable!
|
30
|
+
WebMock.disable_net_connect! webmock_disable_options
|
31
|
+
end
|
32
|
+
|
33
|
+
# @api private
|
34
|
+
def reset
|
35
|
+
whitelisted_urls.clear
|
36
|
+
fake_service_configs.clear
|
37
|
+
end
|
38
|
+
|
39
|
+
# @api private
|
40
|
+
def fake_service_configs
|
41
|
+
@fake_service_configs ||= []
|
42
|
+
end
|
43
|
+
|
44
|
+
# @api private
|
45
|
+
def whitelisted_urls
|
46
|
+
@whitelisted_urls ||= Set.new
|
47
|
+
end
|
48
|
+
|
49
|
+
private
|
50
|
+
|
51
|
+
def webmock_disable_options
|
52
|
+
{ allow_localhost: true }.tap do |opts|
|
53
|
+
opts[:allow] = whitelisted_url_regexps unless Rails.env.test?
|
54
|
+
end
|
55
|
+
end
|
56
|
+
|
57
|
+
def whitelisted_url_regexps
|
58
|
+
whitelisted_urls.map { |url| url_to_regexp url }
|
59
|
+
end
|
60
|
+
|
61
|
+
def webmock_service(config)
|
62
|
+
WebMock.stub_request(
|
63
|
+
:any,
|
64
|
+
url_to_regexp(config.service_url)
|
65
|
+
).to_rack(FakeServiceWrapper.new(config.service))
|
66
|
+
end
|
67
|
+
|
68
|
+
def whitelist_service(config)
|
69
|
+
whitelisted_urls << config.service_url
|
70
|
+
end
|
71
|
+
|
72
|
+
def url_to_regexp(url)
|
73
|
+
%r(\A#{Regexp.escape url})
|
74
|
+
end
|
75
|
+
end
|
76
|
+
end
|
@@ -0,0 +1,28 @@
|
|
1
|
+
== README
|
2
|
+
|
3
|
+
This README would normally document whatever steps are necessary to get the
|
4
|
+
application up and running.
|
5
|
+
|
6
|
+
Things you may want to cover:
|
7
|
+
|
8
|
+
* Ruby version
|
9
|
+
|
10
|
+
* System dependencies
|
11
|
+
|
12
|
+
* Configuration
|
13
|
+
|
14
|
+
* Database creation
|
15
|
+
|
16
|
+
* Database initialization
|
17
|
+
|
18
|
+
* How to run the test suite
|
19
|
+
|
20
|
+
* Services (job queues, cache servers, search engines, etc.)
|
21
|
+
|
22
|
+
* Deployment instructions
|
23
|
+
|
24
|
+
* ...
|
25
|
+
|
26
|
+
|
27
|
+
Please feel free to use a different markup language if you do not plan to run
|
28
|
+
<tt>rake doc:app</tt>.
|
data/spec/dummy/Rakefile
ADDED
@@ -0,0 +1,13 @@
|
|
1
|
+
// This is a manifest file that'll be compiled into application.js, which will include all the files
|
2
|
+
// listed below.
|
3
|
+
//
|
4
|
+
// Any JavaScript/Coffee file within this directory, lib/assets/javascripts, vendor/assets/javascripts,
|
5
|
+
// or any plugin's vendor/assets/javascripts directory can be referenced here using a relative path.
|
6
|
+
//
|
7
|
+
// It's not advisable to add code directly here, but if you do, it'll appear at the bottom of the
|
8
|
+
// compiled file.
|
9
|
+
//
|
10
|
+
// Read Sprockets README (https://github.com/rails/sprockets#sprockets-directives) for details
|
11
|
+
// about supported directives.
|
12
|
+
//
|
13
|
+
//= require_tree .
|
@@ -0,0 +1,15 @@
|
|
1
|
+
/*
|
2
|
+
* This is a manifest file that'll be compiled into application.css, which will include all the files
|
3
|
+
* listed below.
|
4
|
+
*
|
5
|
+
* Any CSS and SCSS file within this directory, lib/assets/stylesheets, vendor/assets/stylesheets,
|
6
|
+
* or any plugin's vendor/assets/stylesheets directory can be referenced here using a relative path.
|
7
|
+
*
|
8
|
+
* You're free to add application-wide styles to this file and they'll appear at the bottom of the
|
9
|
+
* compiled file so the styles you add here take precedence over styles defined in any styles
|
10
|
+
* defined in the other CSS/SCSS files in this directory. It is generally better to create a new
|
11
|
+
* file per style scope.
|
12
|
+
*
|
13
|
+
*= require_tree .
|
14
|
+
*= require_self
|
15
|
+
*/
|
data/spec/dummy/bin/rake
ADDED
@@ -0,0 +1,29 @@
|
|
1
|
+
#!/usr/bin/env ruby
|
2
|
+
require 'pathname'
|
3
|
+
|
4
|
+
# path to your application root.
|
5
|
+
APP_ROOT = Pathname.new File.expand_path('../../', __FILE__)
|
6
|
+
|
7
|
+
Dir.chdir APP_ROOT do
|
8
|
+
# This script is a starting point to setup your application.
|
9
|
+
# Add necessary setup steps to this file:
|
10
|
+
|
11
|
+
puts "== Installing dependencies =="
|
12
|
+
system "gem install bundler --conservative"
|
13
|
+
system "bundle check || bundle install"
|
14
|
+
|
15
|
+
# puts "\n== Copying sample files =="
|
16
|
+
# unless File.exist?("config/database.yml")
|
17
|
+
# system "cp config/database.yml.sample config/database.yml"
|
18
|
+
# end
|
19
|
+
|
20
|
+
puts "\n== Preparing database =="
|
21
|
+
system "bin/rake db:setup"
|
22
|
+
|
23
|
+
puts "\n== Removing old logs and tempfiles =="
|
24
|
+
system "rm -f log/*"
|
25
|
+
system "rm -rf tmp/cache"
|
26
|
+
|
27
|
+
puts "\n== Restarting application server =="
|
28
|
+
system "touch tmp/restart.txt"
|
29
|
+
end
|
@@ -0,0 +1,31 @@
|
|
1
|
+
require File.expand_path('../boot', __FILE__)
|
2
|
+
|
3
|
+
# Pick the frameworks you want:
|
4
|
+
require "active_record/railtie"
|
5
|
+
require "action_controller/railtie"
|
6
|
+
require "action_mailer/railtie"
|
7
|
+
require "action_view/railtie"
|
8
|
+
require "sprockets/railtie"
|
9
|
+
|
10
|
+
Bundler.require(*Rails.groups)
|
11
|
+
require "webvalve"
|
12
|
+
|
13
|
+
module Dummy
|
14
|
+
class Application < Rails::Application
|
15
|
+
# Settings in config/environments/* take precedence over those specified here.
|
16
|
+
# Application configuration should go into files in config/initializers
|
17
|
+
# -- all .rb files in that directory are automatically loaded.
|
18
|
+
|
19
|
+
# Set Time.zone default to the specified zone and make Active Record auto-convert to this zone.
|
20
|
+
# Run "rake -D time" for a list of tasks for finding time zone names. Default is UTC.
|
21
|
+
# config.time_zone = 'Central Time (US & Canada)'
|
22
|
+
|
23
|
+
# The default locale is :en and all translations from config/locales/*.rb,yml are auto loaded.
|
24
|
+
# config.i18n.load_path += Dir[Rails.root.join('my', 'locales', '*.{rb,yml}').to_s]
|
25
|
+
# config.i18n.default_locale = :de
|
26
|
+
|
27
|
+
# Do not swallow errors in after_commit/after_rollback callbacks.
|
28
|
+
config.active_record.raise_in_transactional_callbacks = true
|
29
|
+
end
|
30
|
+
end
|
31
|
+
|
@@ -0,0 +1,25 @@
|
|
1
|
+
# SQLite version 3.x
|
2
|
+
# gem install sqlite3
|
3
|
+
#
|
4
|
+
# Ensure the SQLite 3 gem is defined in your Gemfile
|
5
|
+
# gem 'sqlite3'
|
6
|
+
#
|
7
|
+
default: &default
|
8
|
+
adapter: sqlite3
|
9
|
+
pool: 5
|
10
|
+
timeout: 5000
|
11
|
+
|
12
|
+
development:
|
13
|
+
<<: *default
|
14
|
+
database: db/development.sqlite3
|
15
|
+
|
16
|
+
# Warning: The database defined as "test" will be erased and
|
17
|
+
# re-generated from your development database when you run "rake".
|
18
|
+
# Do not set this db to the same as development or production.
|
19
|
+
test:
|
20
|
+
<<: *default
|
21
|
+
database: db/test.sqlite3
|
22
|
+
|
23
|
+
production:
|
24
|
+
<<: *default
|
25
|
+
database: db/production.sqlite3
|
@@ -0,0 +1,41 @@
|
|
1
|
+
Rails.application.configure do
|
2
|
+
# Settings specified here will take precedence over those in config/application.rb.
|
3
|
+
|
4
|
+
# In the development environment your application's code is reloaded on
|
5
|
+
# every request. This slows down response time but is perfect for development
|
6
|
+
# since you don't have to restart the web server when you make code changes.
|
7
|
+
config.cache_classes = false
|
8
|
+
|
9
|
+
# Do not eager load code on boot.
|
10
|
+
config.eager_load = false
|
11
|
+
|
12
|
+
# Show full error reports and disable caching.
|
13
|
+
config.consider_all_requests_local = true
|
14
|
+
config.action_controller.perform_caching = false
|
15
|
+
|
16
|
+
# Don't care if the mailer can't send.
|
17
|
+
config.action_mailer.raise_delivery_errors = false
|
18
|
+
|
19
|
+
# Print deprecation notices to the Rails logger.
|
20
|
+
config.active_support.deprecation = :log
|
21
|
+
|
22
|
+
# Raise an error on page load if there are pending migrations.
|
23
|
+
config.active_record.migration_error = :page_load
|
24
|
+
|
25
|
+
# Debug mode disables concatenation and preprocessing of assets.
|
26
|
+
# This option may cause significant delays in view rendering with a large
|
27
|
+
# number of complex assets.
|
28
|
+
config.assets.debug = true
|
29
|
+
|
30
|
+
# Asset digests allow you to set far-future HTTP expiration dates on all assets,
|
31
|
+
# yet still be able to expire them through the digest params.
|
32
|
+
config.assets.digest = true
|
33
|
+
|
34
|
+
# Adds additional error checking when serving assets at runtime.
|
35
|
+
# Checks for improperly declared sprockets dependencies.
|
36
|
+
# Raises helpful error messages.
|
37
|
+
config.assets.raise_runtime_errors = true
|
38
|
+
|
39
|
+
# Raises error for missing translations
|
40
|
+
# config.action_view.raise_on_missing_translations = true
|
41
|
+
end
|