diff options
| -rw-r--r-- | Gemfile | 4 | ||||
| -rw-r--r-- | Gemfile.lock | 20 | ||||
| -rw-r--r-- | app/controllers/v1/configs_controller.rb | 11 | ||||
| -rw-r--r-- | config/cucumber.yml | 8 | ||||
| -rw-r--r-- | config/routes.rb | 1 | ||||
| -rw-r--r-- | features/config.feature | 39 | ||||
| -rw-r--r-- | features/step_definitions/.gitkeep | 0 | ||||
| -rw-r--r-- | features/step_definitions/api_steps.rb | 132 | ||||
| -rw-r--r-- | features/step_definitions/config_steps.rb | 6 | ||||
| -rw-r--r-- | features/support/env.rb | 58 | ||||
| -rw-r--r-- | features/support/hooks.rb | 6 | ||||
| -rw-r--r-- | lib/tasks/cucumber.rake | 65 | ||||
| -rwxr-xr-x | script/cucumber | 10 | 
13 files changed, 360 insertions, 0 deletions
| @@ -55,6 +55,10 @@ group :test do    # billing tests    gem 'fake_braintree', require: false + +  # we use cucumber to document and test the api +  gem 'cucumber-rails', require: false +  gem 'jsonpath', require: false  end  group :test, :development do diff --git a/Gemfile.lock b/Gemfile.lock index 1060d70..a286d6f 100644 --- a/Gemfile.lock +++ b/Gemfile.lock @@ -88,6 +88,18 @@ GEM        actionpack        couchrest        couchrest_model +    cucumber (1.3.15) +      builder (>= 2.1.2) +      diff-lcs (>= 1.1.3) +      gherkin (~> 2.12) +      multi_json (>= 1.7.5, < 2.0) +      multi_test (>= 0.1.1) +    cucumber-rails (1.4.1) +      capybara (>= 1.1.2, < 3) +      cucumber (>= 1.3.8, < 2) +      mime-types (~> 1.16) +      nokogiri (~> 1.5) +      rails (>= 3, < 5)      daemons (1.1.9)      debugger (1.6.6)        columnize (>= 0.3.1) @@ -95,6 +107,7 @@ GEM        debugger-ruby_core_source (~> 1.3.2)      debugger-linecache (1.2.0)      debugger-ruby_core_source (1.3.2) +    diff-lcs (1.2.5)      erubis (2.7.0)      eventmachine (1.0.3)      execjs (2.0.2) @@ -113,6 +126,8 @@ GEM      faker (1.2.0)        i18n (~> 0.5)      ffi (1.9.3) +    gherkin (2.12.2) +      multi_json (~> 1.3)      haml (3.1.8)      haml-rails (0.3.5)        actionpack (>= 3.1, < 4.1) @@ -129,6 +144,8 @@ GEM        railties (>= 3.0, < 5.0)        thor (>= 0.14, < 2.0)      json (1.8.1) +    jsonpath (0.5.6) +      multi_json      kaminari (0.13.0)        actionpack (>= 3.0.0)        activesupport (>= 3.0.0) @@ -146,6 +163,7 @@ GEM      mocha (0.13.3)        metaclass (~> 0.0.1)      multi_json (1.10.0) +    multi_test (0.1.1)      nokogiri (1.6.1)        mini_portile (~> 0.5.0)      phantomjs-binaries (1.9.2.3) @@ -249,6 +267,7 @@ DEPENDENCIES    couchrest (~> 1.1.3)    couchrest_model (~> 2.0.0)    couchrest_session_store (~> 0.2.4) +  cucumber-rails    debugger    factory_girl_rails    fake_braintree @@ -259,6 +278,7 @@ DEPENDENCIES    i18n-missing_translations    jquery-rails    json +  jsonpath    kaminari (= 0.13.0)    launchy    leap_web_billing! diff --git a/app/controllers/v1/configs_controller.rb b/app/controllers/v1/configs_controller.rb new file mode 100644 index 0000000..a43861b --- /dev/null +++ b/app/controllers/v1/configs_controller.rb @@ -0,0 +1,11 @@ +class V1::ConfigsController < ApplicationController + +  before_filter :require_login + +  def index +  end + +  def show +  end + +end diff --git a/config/cucumber.yml b/config/cucumber.yml new file mode 100644 index 0000000..19b288d --- /dev/null +++ b/config/cucumber.yml @@ -0,0 +1,8 @@ +<% +rerun = File.file?('rerun.txt') ? IO.read('rerun.txt') : "" +rerun_opts = rerun.to_s.strip.empty? ? "--format #{ENV['CUCUMBER_FORMAT'] || 'progress'} features" : "--format #{ENV['CUCUMBER_FORMAT'] || 'pretty'} #{rerun}" +std_opts = "--format #{ENV['CUCUMBER_FORMAT'] || 'pretty'} --strict --tags ~@wip" +%> +default: <%= std_opts %> features +wip: --tags @wip:3 --wip features +rerun: <%= rerun_opts %> --format rerun --out rerun.txt --strict --tags ~@wip diff --git a/config/routes.rb b/config/routes.rb index 468e14e..3936824 100644 --- a/config/routes.rb +++ b/config/routes.rb @@ -35,6 +35,7 @@ LeapWeb::Application.routes.draw do      resource :cert, :only => [:show, :create]      resource :smtp_cert, :only => [:create]      resource :service, :only => [:show] +    resources :configs, :only => [:index, :show]    end    scope "(:locale)", :locale => MATCH_LOCALE do diff --git a/features/config.feature b/features/config.feature new file mode 100644 index 0000000..2d237f2 --- /dev/null +++ b/features/config.feature @@ -0,0 +1,39 @@ +Feature: Download Provider Configuration + +  The LEAP Provider exposes parts of its configuration through the API. + +  This can be used to find out about services offered. The big picture can be retrieved from `/provider.json`. More detailed settings of the services are available after authentication. You can get a list of the available settings from `/1/configs.json`. + +  Background: +    Given I set headers: +      | Accept       | application/json | +      | Content-Type | application/json | + +  @tempfile +  Scenario: Fetch provider config +    Given the provider config is: +      """ +      {"config": "me"} +      """ +    When I send a GET request to "/provider.json" +    Then the response status should be "200" +    And the response should be: +      """ +      {"config": "me"} +      """ + +  Scenario: Missing provider config +    When I send a GET request to "/provider.json" +    Then the response status should be "404" +    And the response should be: +      """ +      {"error": "not found"} +      """ + +  Scenario: Authentication required for list of configs +    When I send a GET request to "/1/configs" +    Then the response status should be "401" +    And the response should be: +      """ +      {"error": "Please log in to perform that action."} +      """ diff --git a/features/step_definitions/.gitkeep b/features/step_definitions/.gitkeep new file mode 100644 index 0000000..e69de29 --- /dev/null +++ b/features/step_definitions/.gitkeep diff --git a/features/step_definitions/api_steps.rb b/features/step_definitions/api_steps.rb new file mode 100644 index 0000000..0e52f7a --- /dev/null +++ b/features/step_definitions/api_steps.rb @@ -0,0 +1,132 @@ +require 'jsonpath' + +if defined?(Rack) + +  # Monkey patch Rack::MockResponse to work properly with response debugging +  class Rack::MockResponse +    def to_str +      body +    end +  end + +  World(Rack::Test::Methods) + +end + +Given /^I set headers:$/ do |headers| +  headers.rows_hash.each {|k,v| header k, v } +end + +Given /^I send and accept (XML|JSON)$/ do |type| +  header 'Accept', "application/#{type.downcase}" +  header 'Content-Type', "application/#{type.downcase}" +end + +Given /^I send and accept HTML$/ do +  header 'Accept', "text/html" +  header 'Content-Type', "application/x-www-form-urlencoded" +end + +When /^I authenticate as the user "([^"]*)" with the password "([^"]*)"$/ do |user, pass| +  authorize user, pass +end + +When /^I digest\-authenticate as the user "(.*?)" with the password "(.*?)"$/ do |user, pass| +  digest_authorize user, pass +end + +When /^I send a (GET|POST|PUT|DELETE) request (?:for|to) "([^"]*)"(?: with the following:)?$/ do |*args| +  request_type = args.shift +  path = args.shift +  input = args.shift + +  request_opts = {method: request_type.downcase.to_sym} + +  unless input.nil? +    if input.class == Cucumber::Ast::Table +      request_opts[:params] = input.rows_hash +    else +      request_opts[:input] = input +    end +  end + +  request path, request_opts +end + +Then /^show me the (unparsed)?\s?response$/ do |unparsed| +  if unparsed == 'unparsed' +    puts last_response.body +  elsif last_response.headers['Content-Type'] =~ /json/ +    json_response = JSON.parse(last_response.body) +    puts JSON.pretty_generate(json_response) +  else +    puts last_response.headers +    puts last_response.body +  end +end + +Then /^the response status should be "([^"]*)"$/ do |status| +  if self.respond_to? :should +    last_response.status.should == status.to_i +  else +    assert_equal status.to_i, last_response.status +  end +end + +Then /^the response should (not)?\s?have "([^"]*)"$/ do |negative, json_path| +  json    = JSON.parse(last_response.body) +  results = JsonPath.new(json_path).on(json).to_a.map(&:to_s) +  if self.respond_to?(:should) +    if negative.present? +      results.should be_empty +    else +      results.should_not be_empty +    end +  else +    if negative.present? +      assert results.empty? +    else +      assert !results.empty? +    end +  end +end + + +Then /^the response should (not)?\s?have "([^"]*)" with the text "([^"]*)"$/ do |negative, json_path, text| +  json    = JSON.parse(last_response.body) +  results = JsonPath.new(json_path).on(json).to_a.map(&:to_s) +  if self.respond_to?(:should) +    if negative.present? +      results.should_not include(text) +    else +      results.should include(text) +    end +  else +    if negative.present? +      assert !results.include?(text) +    else +      assert results.include?(text) +    end +  end +end + +Then /^the response should be:$/ do |json| +  expected = JSON.parse(json) +  actual = JSON.parse(last_response.body) + +  if self.respond_to?(:should) +    actual.should == expected +  else +    assert_equal expected, actual +  end +end + +Then /^the response should have "([^"]*)" with a length of (\d+)$/ do |json_path, length| +  json = JSON.parse(last_response.body) +  results = JsonPath.new(json_path).on(json) +  if self.respond_to?(:should) +    results.length.should == length.to_i +  else +    assert_equal length.to_i, results.length +  end +end diff --git a/features/step_definitions/config_steps.rb b/features/step_definitions/config_steps.rb new file mode 100644 index 0000000..50ae829 --- /dev/null +++ b/features/step_definitions/config_steps.rb @@ -0,0 +1,6 @@ +Given /the provider config is:$/ do |config| +  @tempfile = Tempfile.new('provider.json') +  @tempfile.write config +  @tempfile.close +  StaticConfigController::PROVIDER_JSON = @tempfile.path +end diff --git a/features/support/env.rb b/features/support/env.rb new file mode 100644 index 0000000..d3067db --- /dev/null +++ b/features/support/env.rb @@ -0,0 +1,58 @@ +# IMPORTANT: This file is generated by cucumber-rails - edit at your own peril. +# It is recommended to regenerate this file in the future when you upgrade to a +# newer version of cucumber-rails. Consider adding your own code to a new file +# instead of editing this one. Cucumber will automatically load all features/**/*.rb +# files. + +require 'cucumber/rails' + +# Capybara defaults to CSS3 selectors rather than XPath. +# If you'd prefer to use XPath, just uncomment this line and adjust any +# selectors in your step definitions to use the XPath syntax. +# Capybara.default_selector = :xpath + +# By default, any exception happening in your Rails application will bubble up +# to Cucumber so that your scenario will fail. This is a different from how +# your application behaves in the production environment, where an error page will +# be rendered instead. +# +# Sometimes we want to override this default behaviour and allow Rails to rescue +# exceptions and display an error page (just like when the app is running in production). +# Typical scenarios where you want to do this is when you test your error pages. +# There are two ways to allow Rails to rescue exceptions: +# +# 1) Tag your scenario (or feature) with @allow-rescue +# +# 2) Set the value below to true. Beware that doing this globally is not +# recommended as it will mask a lot of errors for you! +# +ActionController::Base.allow_rescue = false + +# Remove/comment out the lines below if your app doesn't have a database. +# For some databases (like MongoDB and CouchDB) you may need to use :truncation instead. +begin +  #DatabaseCleaner.strategy = :truncation +rescue NameError +  raise "You need to add database_cleaner to your Gemfile (in the :test group) if you wish to use it." +end + +# You may also want to configure DatabaseCleaner to use different strategies for certain features and scenarios. +# See the DatabaseCleaner documentation for details. Example: +# +#   Before('@no-txn,@selenium,@culerity,@celerity,@javascript') do +#     # { :except => [:widgets] } may not do what you expect here +#     # as Cucumber::Rails::Database.javascript_strategy overrides +#     # this setting. +#     DatabaseCleaner.strategy = :truncation +#   end +# +#   Before('~@no-txn', '~@selenium', '~@culerity', '~@celerity', '~@javascript') do +#     DatabaseCleaner.strategy = :transaction +#   end +# + +# Possible values are :truncation and :transaction +# The :transaction strategy is faster, but might give you threading problems. +# See https://github.com/cucumber/cucumber-rails/blob/master/features/choose_javascript_database_strategy.feature +Cucumber::Rails::Database.javascript_strategy = :truncation + diff --git a/features/support/hooks.rb b/features/support/hooks.rb new file mode 100644 index 0000000..360f231 --- /dev/null +++ b/features/support/hooks.rb @@ -0,0 +1,6 @@ +After('@tempfile') do +  if @tempfile +    @tempfile.close +    @tempfile.unlink +  end +end diff --git a/lib/tasks/cucumber.rake b/lib/tasks/cucumber.rake new file mode 100644 index 0000000..9f53ce4 --- /dev/null +++ b/lib/tasks/cucumber.rake @@ -0,0 +1,65 @@ +# IMPORTANT: This file is generated by cucumber-rails - edit at your own peril. +# It is recommended to regenerate this file in the future when you upgrade to a +# newer version of cucumber-rails. Consider adding your own code to a new file +# instead of editing this one. Cucumber will automatically load all features/**/*.rb +# files. + + +unless ARGV.any? {|a| a =~ /^gems/} # Don't load anything when running the gems:* tasks + +vendored_cucumber_bin = Dir["#{Rails.root}/vendor/{gems,plugins}/cucumber*/bin/cucumber"].first +$LOAD_PATH.unshift(File.dirname(vendored_cucumber_bin) + '/../lib') unless vendored_cucumber_bin.nil? + +begin +  require 'cucumber/rake/task' + +  namespace :cucumber do +    Cucumber::Rake::Task.new({:ok => 'test:prepare'}, 'Run features that should pass') do |t| +      t.binary = vendored_cucumber_bin # If nil, the gem's binary is used. +      t.fork = true # You may get faster startup if you set this to false +      t.profile = 'default' +    end + +    Cucumber::Rake::Task.new({:wip => 'test:prepare'}, 'Run features that are being worked on') do |t| +      t.binary = vendored_cucumber_bin +      t.fork = true # You may get faster startup if you set this to false +      t.profile = 'wip' +    end + +    Cucumber::Rake::Task.new({:rerun => 'test:prepare'}, 'Record failing features and run only them if any exist') do |t| +      t.binary = vendored_cucumber_bin +      t.fork = true # You may get faster startup if you set this to false +      t.profile = 'rerun' +    end + +    desc 'Run all features' +    task :all => [:ok, :wip] + +    task :statsetup do +      require 'rails/code_statistics' +      ::STATS_DIRECTORIES << %w(Cucumber\ features features) if File.exist?('features') +      ::CodeStatistics::TEST_TYPES << "Cucumber features" if File.exist?('features') +    end +  end +  desc 'Alias for cucumber:ok' +  task :cucumber => 'cucumber:ok' + +  task :default => :cucumber + +  task :features => :cucumber do +    STDERR.puts "*** The 'features' task is deprecated. See rake -T cucumber ***" +  end + +  # In case we don't have the generic Rails test:prepare hook, append a no-op task that we can depend upon. +  task 'test:prepare' do +  end + +  task :stats => 'cucumber:statsetup' +rescue LoadError +  desc 'cucumber rake task not available (cucumber not installed)' +  task :cucumber do +    abort 'Cucumber rake task is not available. Be sure to install cucumber as a gem or plugin' +  end +end + +end diff --git a/script/cucumber b/script/cucumber new file mode 100755 index 0000000..7fa5c92 --- /dev/null +++ b/script/cucumber @@ -0,0 +1,10 @@ +#!/usr/bin/env ruby + +vendored_cucumber_bin = Dir["#{File.dirname(__FILE__)}/../vendor/{gems,plugins}/cucumber*/bin/cucumber"].first +if vendored_cucumber_bin +  load File.expand_path(vendored_cucumber_bin) +else +  require 'rubygems' unless ENV['NO_RUBYGEMS'] +  require 'cucumber' +  load Cucumber::BINARY +end | 
