title: AWDwR subtitle: RailsConf date: System Testing Rails itself script: - pager.js stylesheet: - presentation.css watermark: | #agenda: Agenda * *Intro* ** "Introduction":intro, "System Testing":systest * Live example ** "rjs":rjs, "render":render, "test-unit":test-unit, "dashboard":dashboard * Context ** "motivation":motivation, "evolution":evolution * Components ** "gorp":gorp, "testrails":testrails, "dashboard":dashboard, "cgi-spa":cgi-spa * Conclusion ** "observation":observation, "conclusion":conclusion, "sources":sources #intro: Introduction * My name is Sam Ruby #intro: Introduction * My name is Sam Ruby * IBM, Emerging Technology * Co-author, "Agile Web Development with Rails":http://www.pragprog.com/titles/rails3/agile-web-development-with-rails-third-edition #intro: Introduction * My name is Sam Ruby * IBM, Emerging Technology * Co-author, "Agile Web Development with Rails":http://www.pragprog.com/titles/rails3/agile-web-development-with-rails-third-edition * "Weblog":http://intertwingly.net/blog/ * "More info on Wikipedia":http://en.wikipedia.org/wiki/Sam_Ruby #systest: System Testing System testing of software or hardware is testing conducted on a complete, integrated system to evaluate the system's compliance with its specified requirements. System testing falls within the scope of black box testing, and as such, should require no knowledge of the inner design of the code or logic. -- "wikipedia":http://en.wikipedia.org/wiki/System_testing #agenda: Agenda * Intro ** "Introduction":intro, "System Testing":systest * *Live example* ** "rjs":rjs, "render":render, "test-unit":test-unit, "dashboard":dashboard * Context ** "motivation":motivation, "evolution":evolution * Components ** "gorp":gorp, "testrails":testrails, "dashboard":dashboard, "cgi-spa":cgi-spa * Conclusion ** "observation":observation, "conclusion":conclusion, "sources":sources #rjs: rjs test output ~06/rjs_functional_test-187.html # rjs test on 1.9.2 ~06/rjs_functional_test-192.html # rjs test on 1.9.2 ~06/rjs_functional_test.rb # rjs test on 1.9.2 ~06/bisect.log #render: rjs render test output ~07/rjs_render_test.html # rjs render test on 1.9.2 ~07/rjs_render_test.rb # rjs render test on 1.9.2 ~07/bisect.out #testunit: uninitialized constant on 1.9.2 ~08/makedepot.log # uninitialized constant on 1.9.2 * "reported":http://blade.nagaokaut.ac.jp/cgi-bin/scat.rb/ruby/ruby-core/30688 on ruby-core * isolated to "revision 28200":http://blade.nagaokaut.ac.jp/cgi-bin/scat.rb/ruby/ruby-core/30688 #dashboard: dashboard ~dashboard.html #agenda: Agenda * Intro ** "Introduction":intro, "System Testing":systest * Live example ** "rjs":rjs, "render":render, "test-unit":test-unit, "dashboard":dashboard * *Context* ** "motivation":motivation, "evolution":evolution * Components ** "gorp":gorp, "testrails":testrails, "dashboard":dashboard, "cgi-spa":cgi-spa * Conclusion ** "observation":observation, "conclusion":conclusion, "sources":sources #motivation: Motivation How did I get here? # Motivation In my case, I did this entirely out of self defense. # Motivation In my case, I did this entirely out of self defense. System testing rails turned out to be a completely unintentional side effect. # Motivation _Really, I'm not out to destroy Microsoft. That will just be a completely unintentional side effect._ - "Linus Torvalds":http://en.wikiquote.org/wiki/Linus_Torvalds # Motivation I refuse to be a victim. # Motivation I refuse to be a victim. The all too common cycle I see: # Motivation I refuse to be a victim. The all too common cycle I see: * Don't test the betas # Motivation I refuse to be a victim. The all too common cycle I see: * Don't test the betas * Wait for a release # Motivation I refuse to be a victim. The all too common cycle I see: * Don't test the betas * Wait for a release * Complain on blogs and twitter # Motivation Don't be a victim. #evolution: Evolution I started out with a simple script, in one file, with a micro DSL. #Evolution I started out with a simple script, in one file, with a micro DSL. It ran a few commands, a few HTTP requests, captured the output, and formatted the results as HTML for visual inspection. #Evolution I started out with a simple script, in one file, with a micro DSL. It ran a few commands, a few HTTP requests, captured the output, and formatted the results as HTML for visual inspection. Manually verifying this turned out to be a bit of a drag. #Evolution Next step was a separate script to verify the output of the first script. #Evolution Next step was a separate script to verify the output of the first script. It did that by using assert_select. #Evolution Once I started work on edition 4, I had need for a second script. #Evolution Once I started work on edition 4, I had need for a second script. I factored out the common code into a library. #Evolution Once I started work on edition 4, I had need for a second script. I factored out the common code into a library. Best frameworks are factored out of a running application. #Evolution Once I started work on edition 4, I had need for a second script. I factored out the common code into a library. Best frameworks are factored out of a running application. This one may turn out to be a good counter-example. #Evolution Running two scripts against two versions of Rails and five versions of Ruby inspired additional automation. #Evolution Running two scripts against two versions of Rails and five versions of Ruby inspired additional automation. One to set up the directories, use rvm to manage Ruby VMs (even building them when necessary), use bundler to install dependencies, and to run the script. #Evolution Running two scripts against two versions of Rails and five versions of Ruby inspired additional automation. One to set up the directories, use rvm to manage Ruby VMs (even building them when necessary), use bundler to install dependencies, and to run the script. One to present the overall status, and launch new tests. #Evolution Running two scripts against two versions of Rails and five versions of Ruby inspired additional automation. One to set up the directories, use rvm to manage Ruby VMs (even building them when necessary), use bundler to install dependencies, and to run the script. One to present the overall status, and launch new tests. The second script used another library that builds on CGI. #agenda: Agenda * Intro ** "Introduction":intro, "System Testing":systest * Live example ** "rjs":rjs, "render":render, "test-unit":test-unit, "dashboard":dashboard * Context ** "motivation":motivation, "evolution":evolution * *Components* ** "gorp":gorp, "testrails":testrails, "dashboard":dashboard, "cgi-spa":cgi-spa * Conclusion ** "observation":observation, "conclusion":conclusion, "sources":sources #recap: List of components * testrails: select vm, install bundle, run test * makedepot: run a single scenario (edition 3 or edition 4) * gorp: common library * dashboard: display test results, and submit new tests * cgi-spa: makes cgi more tolerable #gorp: List of components * testrails: select vm, install bundle, run test * makedepot: run a single scenario (edition 3 or edition 4) * *gorp: common library* * dashboard: display test results, and submit new tests * cgi-spa: makes cgi more tolerable # Gorp Gorp is the common library * Name has no special meaning # Gorp.xhtmlparse * originally to handle html entities * monkey patch for 1.8.6 (obsolete) * nokogiri support * backports (via monkeypatch) nokogiri API to REXml # Gorp.Commands (executable) * db: sqlite3 command * ruby: run a script * rake: run a rake * console: run rails console * generate: rails generate * cmd: arbitrary OS command * irb: interactive ruby * rails: set up a new application # Gorp.Commands (admin) * omit: indicate what steps should be omitted * section: a block of commands * overview: description for a section * note: description for a step * log: line to output to the screen * issue: track bug reports # Gorp.test * suite: a test suite (enables assert_select) * input: parses the _output_ of a run * output: declares where to place the output * section: matches a section in the input * collect_stdout: extract the stdout lines (common idiom) * sort_hash: simplifies comparison of hashes # Gorp.test (other) * cmd, get, post, rake, ruby, console: augmented with assert_select * html_fault: captures errors (and matches issues) * html_summary: hook on test completion (stops server, produces output) * at_exit: additional support for standalone tests * test: individual step # Gorp.net * snap: capture body of a response * get: does a HTTP get * post: does a HTTP get (manages cookies, fills in forms) # Gorp.env * dump_env * log * path (work path) # Gorp.edit * unindent * highlight * clear_highlights * clear_all_marks * mark * edit * msub * all= * read * dcl # Other: * restart_server, stop_server: both do what you would expect * verify that $PORT (defaults to 3000) is available * parse command for _n.n_ syntax * --rails-debug (ENV['BACKTRACE'] = '1') * output: produce test output, using builder. Runs sections, captures errors. * which_rails: rails command to use (git, gem) * Commands know about OS and Rails version changes # Obsolete: * alternatives to bundler (rails.env) # Gorp.Globals * $x: builder * $toc: table of contents * $todos: list of todos * $issue: current issue number * $style: CSS to be output #testrails: List of components * *testrails*: select vm, install bundle, run test * makedepot: run a single scenario (edition 3 or edition 4) * gorp: common library * dashboard: display test results, and submit new tests * cgi-spa: makes cgi more tolerable # Testrails * update Rails from git * update tests from git * build a new rvm, if necessary * update gems using bundler * run the script * paginate the output #dashboard: List of components * testrails: select vm, install bundle, run test * makedepot: run a single scenario (edition 3 or edition 4) * gorp: common library * *dashboard*: display test results, and submit new tests * cgi-spa: makes cgi more tolerable # Dashboard * displays colorized status in tabular form * submits new tests in background * displays live status of running test (AJAX) * configured via YAML #cgi-spa: List of components * testrails: select vm, install bundle, run test * makedepot: run a single scenario (edition 3 or edition 4) * gorp: common library * dashboard: display test results, and submit new tests * *cgi-spa*: makes cgi more tolerable # cgi-spa * methods: json, json!, html, html!, post, post! * html: style!, script! * environment: $HTTP_GET, $HTTP_POST, $XHR_JSON, $XHTML, $param, $HOME, $SERVER * builder: indented_text, indented_data, traceback * submit * installation #agenda: Agenda * Intro ** "Introduction":intro, "System Testing":systest * Live example ** "rjs":rjs, "render":render, "test-unit":test-unit, "dashboard":dashboard * Context ** "motivation":motivation, "evolution":evolution * Components ** "gorp":gorp, "testrails":testrails, "dashboard":dashboard, "cgi-spa":cgi-spa * *Conclusion* ** "observation":observation, "conclusion":conclusion, "sources":sources #observation: Observation Edge of a major new release is atypical # Observation Edge of a major new release is atypical More common errors to be found: * rails server won't start * rake test won't test * permutations not covered by unit tests #conclusion: Conclusion Don't be a victim # Conclusion Don't be a victim * incorporate edge Rails into your CI # Conclusion Don't be a victim * incorporate edge Rails into your CI * report what you find early #sources: Sources * "testrails":http://github.com/rubys/awdwr/blob/master/testrails.rb: select vm, install bundle, run test * "makedepot":http://github.com/rubys/awdwr/blob/master/makedepot.rb: run a single scenario (edition 3) * "checkdepot":http://github.com/rubys/awdwr/blob/master/checkdepot.rb: verify * output of a single scenario (edition 3) * "gorp":http://github.com/rubys/gorp: common library * "dashboard":http://github.com/rubys/awdwr/blob/master/dashboard.rb: display test results, and submit new tests * "cgi-spa":http://github.com/rubys/cgi-spa: makes cgi more tolerable #Thanks: Thanks! Any questions?