{"id":1196,"date":"2012-01-07T12:45:36","date_gmt":"2012-01-07T10:45:36","guid":{"rendered":"http:\/\/blog.railsware.com\/?p=1196"},"modified":"2021-08-13T11:39:37","modified_gmt":"2021-08-13T08:39:37","slug":"testing-gem-integration-with-multiple-ruby-frameworks","status":"publish","type":"post","link":"https:\/\/railsware.com\/blog\/testing-gem-integration-with-multiple-ruby-frameworks\/","title":{"rendered":"Testing gem integration with multiple ruby frameworks"},"content":{"rendered":"\n<h2 class=\"wp-block-heading\">Preface<\/h2>\n\n\n\n<p>When you develop some cool gem that should work in different <a href=\"https:\/\/railsware.com\/blog\/ruby-on-rails-guide\/\">ruby frameworks<\/a><br>you definitely should write acceptance tests.<\/p>\n\n\n\n<p>Nowadays it&#8217;s pretty easy to do with RSpec+Capybara.<\/p>\n\n\n\n<h2 class=\"wp-block-heading\">Your goal<\/h2>\n\n\n\n<p>Assume you develop MyRackMiddleware gem that should work in Sinatra and Rails application.<\/p>\n\n\n\n<h2 class=\"wp-block-heading\">Your actions<\/h2>\n\n\n\n<h3 class=\"wp-block-heading\">Create gem layout<\/h3>\n\n\n\n<p>To create layout use well-known <em>bundle gem<\/em> command:<\/p>\n\n\n\n<pre class=\"wp-block-preformatted\">$ bundle gem my_rack_middleware\n<\/pre>\n\n\n\n<p><br>It&#8217;s generate something like:<\/p>\n\n\n\n<!--more-->\n\n\n\n<pre class=\"wp-block-preformatted\">\u251c\u2500\u2500 lib\n\u2502&nbsp;&nbsp; \u251c\u2500\u2500 my_rack_middleware\n\u2502&nbsp;&nbsp; \u2502&nbsp;&nbsp; \u2514\u2500\u2500 version.rb\n\u2502&nbsp;&nbsp; \u2514\u2500\u2500 my_rack_middleware.rb\n\u251c\u2500\u2500 Gemfile\n\u251c\u2500\u2500 my_rack_middleware.gemspec\n\u2514\u2500\u2500 Rakefile\n<\/pre>\n\n\n\n<p>Add to <em>Gemfile<\/em> :<\/p>\n\n\n\n<pre class=\"wp-block-preformatted\">source \"http:\/\/rubygems.org\"\n\ngemspec\n\ngem 'rspec'\ngem 'capybara'\n\ngem 'sinatra'\ngem 'rails'\n<\/pre>\n\n\n\n<p>Now start use BDD :) Before start implementation we need two sample applications.<br>Let&#8217;s put they into <em>apps<\/em> directory.<\/p>\n\n\n\n<h3 class=\"wp-block-heading\">Creating Sinatra test application<\/h3>\n\n\n\n<p>Thus Sinatra is simple framework it&#8217;s trivial to create test application:<\/p>\n\n\n\n<pre class=\"wp-block-preformatted\">$ cat apps\/test_sinatra_app.rb \n<\/pre>\n\n\n\n<pre class=\"wp-block-preformatted\">require 'sinatra'\nclass TestSinatraApp &lt; Sinatra::Base\n  enable :sessions\n\n  get '\/login' do\n    body \"Please log in\"\n  end\n\n  post '\/login' do\n    session[:user_email] = params[:user_email]\n    redirect to('\/profile')\n  end\n\n  get '\/profile' do\n    if user_email = session[:user_email]\n      body \"Welcome, #{user_email}!\"\n    else\n      redirect to('\/login')\n    end\n  end\nend\n<\/pre>\n\n\n\n<h3 class=\"wp-block-heading\">Creating Rails3 test application<\/h3>\n\n\n\n<p>When you think that with rails it will be hard then your are wrong :)<br>Thanks to rails3 design its also rather simple! Let&#8217;s see:<\/p>\n\n\n\n<pre class=\"wp-block-preformatted\">$ cat apps\/test_rails_app.rb \n<\/pre>\n\n\n\n<pre class=\"wp-block-preformatted\">require 'rails'\nrequire 'action_controller\/railtie'\n\nmodule TestRailsApp\n  class Application &lt; Rails::Application config.secret_token = '572c86f5ede338bd8aba8dae0fd3a326aabababc98d1e6ce34b9f5' routes.draw do get '\/login' =&gt; 'test_rails_app\/sessions#new'\n      post '\/login'   =&gt; 'test_rails_app\/sessions#create'\n      get  '\/profile' =&gt; 'test_rails_app\/profiles#show'\n    end\n  end\n\n  class SessionsController &lt; ActionController::Base def new render :text =&gt; \"Please log in\"\n    end\n\n    def create\n      session[:user_email] = params[:user_email]\n      redirect_to '\/profile'\n    end\n  end\n\n  class ProfilesController &lt; ActionController::Base def show if user_email = session[:user_email] render :text =&gt; \"Welcome, #{user_email}!\"\n      else\n        redirect_to '\/login' \n      end\n    end\n  end\nend\n<\/pre>\n\n\n\n<p>So you have <em>whole rails application<\/em> in one file. Nice! :)<\/p>\n\n\n\n<h3 class=\"wp-block-heading\">Write integration tests<\/h3>\n\n\n\n<p>Now is most interesting part of this article.<br>Capybara gem itself has rspec helpers that allows you to write rspec examples<br>in way of acceptance testing (aka <em>feature<\/em> and <em>scenario<\/em>).<\/p>\n\n\n\n<p>Actually internally <em>feature<\/em> is just alias to ExampleGroup (describe) and <em>scenario<\/em> is alias to Example (it).<br>But this syntax follows you to think different.<\/p>\n\n\n\n<p>Put in <em>spec\/spec_helper.rb<\/em><\/p>\n\n\n\n<pre class=\"wp-block-preformatted\">require 'capybara\/rspec'\n\nrequire 'rack_session_access'\nrequire 'rack_session_access\/capybara'\n\n# load test applications\nDir[File.expand_path('..\/..\/apps\/*.rb', __FILE__)].each do |f|\n  require f\nend\n\n# configure sinatra application to use MyRackMiddleware\nTestSinatraApp.configure do |app|\n  app.use MyRackMiddleware\nend\n\n# configure rails application to use MyRackMiddleware\nTestRailsApp::Application.configure do |app|\n  app.middleware.use MyRackMiddleware\nend\n<\/pre>\n\n\n\n<p>Write acceptance spec for example in <em>spec\/middleware_spec.rb<\/em> :<br>Follow DRY principle and use <em>shared_examples<\/em>!<\/p>\n\n\n\n<pre class=\"wp-block-preformatted\">require 'spec_helper'\n\nshared_examples \"common scenarios\" do\n  scenario \"test application itself\" do\n    page.visit \"\/login\"\n    page.should have_content(\"Please log in\")\n\n    page.visit \"\/profile\"\n    page.should have_content(\"Please log in\")\n  end\n\n  scenario \"MyRackMiddleware related scenario ...\" do\n    ...\n  end\nend\n\nfeature \"MyRackMiddleware\", %q(\n  As ROLE\n  I want FEATURE\n  So I have BENEFIT\n) do\n\n  context \"with Sinatra application\" do\n    background do\n      Capybara.app = TestSinatraApp\n    end\n\n    include_examples \"common scenarios\"\n  end\n\n  context \"with Rails application\" do\n    background do\n      Capybara.app = TestRailsApp::Application\n    end\n\n    include_examples \"common scenarios\"\n  end\nend\n\n<\/pre>\n\n\n\n<h3 class=\"wp-block-heading\">Run tests<\/h3>\n\n\n\n<p>Before implementation run specs<\/p>\n\n\n\n<pre class=\"wp-block-preformatted\">$ bundle exec rspec -fs -c spec\/middleware_spec.rb\n<\/pre>\n\n\n\n<p>Now you may add scenarios that cover your MyRackMiddleware influence on applications and start your middleware implementation!<\/p>\n\n\n\n<h3 class=\"wp-block-heading\">Conclusion<\/h3>\n\n\n\n<p>We hope this article will help you to write better tests for your gems.<\/p>\n\n\n\n<h2 class=\"wp-block-heading\">References<\/h2>\n\n\n\n<ul class=\"wp-block-list\"><li><a href=\"https:\/\/github.com\/jnicklas\/capybara\">capybara<\/a><\/li><li><a href=\"https:\/\/www.relishapp.com\/rspec\">rspec<\/a><\/li><\/ul>\n","protected":false},"excerpt":{"rendered":"<p>Preface When you develop some cool gem that should work in different ruby frameworksyou definitely should write acceptance tests. Nowadays it&#8217;s pretty easy to do with RSpec+Capybara. Your goal Assume you develop MyRackMiddleware gem that should work in Sinatra and Rails application. Your actions Create gem layout To create layout use well-known bundle gem command:&#8230;<\/p>\n","protected":false},"author":19,"featured_media":1745,"comment_status":"open","ping_status":"open","sticky":false,"template":"","format":"standard","meta":{"_acf_changed":false,"footnotes":""},"categories":[3],"tags":[],"coauthors":["Andriy Yanko"],"class_list":["post-1196","post","type-post","status-publish","format-standard","has-post-thumbnail","hentry","category-development"],"acf":[],"aioseo_notices":[],"categories_data":[{"name":"Engineering","link":"https:\/\/railsware.com\/blog?category=development"}],"post_thumbnails":"https:\/\/railsware.com\/blog\/wp-content\/themes\/railsware\/vendors\/images\/article-thumbnail-default.jpg","amp_enabled":true,"_links":{"self":[{"href":"https:\/\/railsware.com\/blog\/wp-json\/wp\/v2\/posts\/1196","targetHints":{"allow":["GET"]}}],"collection":[{"href":"https:\/\/railsware.com\/blog\/wp-json\/wp\/v2\/posts"}],"about":[{"href":"https:\/\/railsware.com\/blog\/wp-json\/wp\/v2\/types\/post"}],"author":[{"embeddable":true,"href":"https:\/\/railsware.com\/blog\/wp-json\/wp\/v2\/users\/19"}],"replies":[{"embeddable":true,"href":"https:\/\/railsware.com\/blog\/wp-json\/wp\/v2\/comments?post=1196"}],"version-history":[{"count":8,"href":"https:\/\/railsware.com\/blog\/wp-json\/wp\/v2\/posts\/1196\/revisions"}],"predecessor-version":[{"id":14029,"href":"https:\/\/railsware.com\/blog\/wp-json\/wp\/v2\/posts\/1196\/revisions\/14029"}],"wp:featuredmedia":[{"embeddable":true,"href":"https:\/\/railsware.com\/blog\/wp-json\/wp\/v2\/media\/1745"}],"wp:attachment":[{"href":"https:\/\/railsware.com\/blog\/wp-json\/wp\/v2\/media?parent=1196"}],"wp:term":[{"taxonomy":"category","embeddable":true,"href":"https:\/\/railsware.com\/blog\/wp-json\/wp\/v2\/categories?post=1196"},{"taxonomy":"post_tag","embeddable":true,"href":"https:\/\/railsware.com\/blog\/wp-json\/wp\/v2\/tags?post=1196"},{"taxonomy":"author","embeddable":true,"href":"https:\/\/railsware.com\/blog\/wp-json\/wp\/v2\/coauthors?post=1196"}],"curies":[{"name":"wp","href":"https:\/\/api.w.org\/{rel}","templated":true}]}}