Object Mentor Blog: Testing GUIs Part I: RoR. http://blog.objectmentor.com/articles/2007/01/13/testing-guis-part-i-ror en-us 40 Testing GUIs Part I: RoR. <p>Testing GUIs is one of the the holy grails of Test Driven Develoment (TDD). Many teams who have adopted <span class="caps">TDD</span> for other parts of their projects have, for one reason or another, been unable to adequately test the <span class="caps">GUI</span> portion of their code.</p> <p>In this series of article I will show that <span class="caps">GUI</span> testing is a solved problem. Over the years the <span class="caps">TDD</span> community has produced and accumulated tools, frameworks, libraries, and techniques that allow any team to test their <span class="caps">GUI</span> code as fully as any other part of their code.</p> <h1>Testing GUIs Part I: <span style="color:red;">Ruby on Rails</span></h1> <p>In the world of web development, no community has solved the problem of <span class="caps">GUI</span> testing better than the Ruby on Rails community. When you work on a rails project, testing the <span class="caps">GUI</span> is simply <em>de-rigeur</em>. The rails framework provides all the necessary tools and access points for testing all aspects of the application, including the generation of <span class="caps">HTML</span> and the structure of the resulting web pages.</p> <p>Web pages in rails are specified by <code>.rhtml</code> files that contain a mixture of <span class="caps">HTML</span> and ruby code similar to the way Java and <span class="caps">HTML</span> are mixed in <code>.jsp</code> files. The difference is that <code>.rhtml</code> files are translated at runtime rather than being compiled into servlets the way <code>.jsp</code> pages are. This makes it very easy for the rails environment to generate the <span class="caps">HTML</span> for a web page <em>outside</em> of the web container. Indeed, the web server does not need to be running.</p> <p>This ease and portability of generating <span class="caps">HTML</span> means that the rails test framework merely needs to set up the variables needed by the ruby scriptlets within the <code>.rhtml</code> files, generate the <span class="caps">HTML</span>, and then parse that <span class="caps">HTML</span> into a form that the tests can query.</p> <h3>A typical example.</h3> The tests query the <span class="caps">HTML</span> using an xpath-like syntax coupled with a suite of very powerful assertion functions. The best way to understand this is to see it. So here is a simple file named: <code>autocomplete_teacher.rhtml</code>. <pre><code>&lt;ul class="autocomplete_list"&gt; &lt;% @autocompleted_teachers.each do |t| %&gt; &lt;li class="autocomplete_item"&gt;&lt;%= "#{create_name_adornment(t)} #{t.last_name}, #{t.first_name}"%&gt;&lt;/li&gt; &lt;% end %&gt; &lt;/ul&gt; </code></pre> You don&#8217;t have to be a ruby programmer to understand this. All it is doing is building an <span class="caps">HTML</span> list. The Ruby scriptlet between <code>&lt;% and %&gt;</code> tokens simple loops for each teacher creating an <code>&lt;li&gt;</code> tag from an &#8220;adornment&#8221;, and the first and last name. (The adornment happens to be the database id of the teacher in parentheses.) A simple test for this <code>.rhtml</code> file is: <pre><code> def test_autocomplete_teacher_finds_one_in_first_name post :autocomplete_teacher, :request=&gt;{:teacher=&gt;"B"} assert_template "autocomplete_teacher" assert_response :success assert_select "ul.autocomplete_list" do assert_select "li.autocomplete_item", :count =&gt; 1 assert_select "li", "(1) Martin, Bob" end end</code></pre> <ul> <li>The <code>post</code> statement simply invokes the controller that would normally be invoked by a <span class="caps">POST</span> url of the form: <code>POST /teachers/autocomplete_teacher</code> with the <code>teacher</code> parameter set to <code>"B"</code>.</li> <li>The first assertion makes sure that the controller rendered the <code>autocomplete_teacher.rhtml</code> template.</li> <li>The next makes sure that the controller returned success.</li> <li>the third is a compound assertion that starts by finding the <code>&lt;ul&gt;</code> tag with a <code>class="autocomplete_list"</code> attribute. (Notice the use of <code>css</code> syntax.) <ul> <li>Within this tag there should be an <code>&lt;li&gt;</code> tag with a <code>class="autocomplete_item"</code> attribute, </li> <li>and containing the text <code>(1) Martin, Bob</code>.</li> </ul></li> </ul> <p><br> It should not come as any surprise that this test runs in a test environment in which the database has been pre-loaded with very specific data. For example, this test database always has &#8220;Bob Martin&#8221; being the first row (<code>id=1</code>) in the <code>Teacher</code> table.</p> <p>The <code>assert_select</code> function is <em>very</em> powerful, and allows you to query large and complex <span class="caps">HTML</span> documents with surgical precision. Although this example give you just a glimpse of that power, you should be able to see that the rails testing scheme allows you to test that all the scriptlets in an <code>.rhtml</code> file are behaving correctly, and are correctly extracting data from the variables set by the controller.</p> <h3>An example using <em>RSpec</em> and <em>Behavior Driven Design</em>.</h3> <p>What follows is a more significant rails example that uses an alternate testing syntax known as Behavior Driven Design (BDD). The tool that accepts this syntax is called <em>RSpec</em>.</p> Imagine that we have a page that records telephone messages taken from teachers at different schools. Part of that page might have an .rhtml syntax that looks like this: <pre><code>&lt;h1&gt;Message List&lt;/h1&gt; &lt;table id="list"&gt; &lt;tr class="list_header_row"&gt; &lt;th class="list_header"&gt;Time&lt;/th&gt; &lt;th class="list_header"&gt;Caller&lt;/th&gt; &lt;th class="list_header"&gt;School&lt;/th&gt; &lt;th class="list_header"&gt;IEP&lt;/th&gt; &lt;/tr&gt; &lt;%time_chooser = TimeChooser.new%&gt; &lt;% for message in @messages %&gt; &lt;%cell_class = cycle("list_content_even", "list_content_odd")%&gt; &lt;tr id="list_content_row"&gt; &lt;td id="time" class="&lt;%=cell_class%&gt;"&gt;&lt;%=h(time_chooser.format_time(message.time)) %&gt;&lt;/td&gt; &lt;td id="caller" class="&lt;%=cell_class%&gt;"&gt;&lt;%=h person_name(message.caller) %&gt;&lt;/td&gt; &lt;td id="school" class="&lt;%=cell_class%&gt;"&gt;&lt;%=h message.school.name %&gt;&lt;/td&gt; &lt;td id="iep" class="&lt;%=cell_class%&gt;"&gt;&lt;%=h (message.iep ? "X" : "") %&gt;&lt;/td&gt; &lt;/tr&gt; &lt;% end %&gt; &lt;/table&gt; </code></pre> Clearly each message has a time, caller, school, and some kind of boolean field named &#8220;IEP&#8221;. We can test this <code>.rhtml</code> file with the following <em>RSpec</em> specification: <pre><code>context "Given a request to render message/list with one message the page" do setup do m = mock "message" caller = mock "person",:null_object=&gt;true school = mock "school" m.should_receive(:school).and_return(school) m.should_receive(:time).and_return(Time.parse("1/1/06")) m.should_receive(:caller).any_number_of_times.and_return(caller) m.should_receive(:iep).and_return(true) caller.should_receive(:first_name).and_return("Bob") caller.should_receive(:last_name).and_return("Martin") school.should_receive(:name).and_return("Jefferson") assigns[:messages]=[m] assigns[:message_pages] = mock "message_pages", :null_object=&gt;true render 'message/list' end specify "should show the time" do response.should_have_tag :td, :content=&gt;"12:00 AM 1/1", :attributes=&gt;{:id=&gt;"time"} end specify "should show caller first and last name" do response.should_have_tag :td, :content=&gt;"Bob Martin", :attributes=&gt;{:id=&gt;"caller"} end specify "should show school name" do response.should_have_tag :td, :content=&gt;"Jefferson", :attributes=&gt;{:id=&gt;"school"} end specify "should show the IEP field" do response.should_have_tag :td, :content=&gt;"X",:attributes=&gt;{:id=&gt;"iep"} end end </code></pre> I&#8217;m not going to explain the <code>setup</code> function containing all that <em>mock</em> stuff you see at the start. Let me just say that the mocking facilities of <em>RSpec</em> are both powerful and convenient. Actually you shouldn&#8217;t have too much trouble understanding the <code>setup</code> if you try; but understanding it is not essential for this example. The interesting testing is in the <code>specify</code> blocks. <p>You shouldn&#8217;t have too much trouble reading the <code>specify</code> blocks. You can understand all of them if you understand the first. Here is what it does:</p> <ul> <li>The first spec ensures that <code>&lt;td id="time"&gt;12:00 AM 1/1&lt;/td&gt;</code> exists in the <span class="caps">HTML</span> document. This is not a string compare. Rather it is a semantic equivalence. Whitespace, and other attributes and complications are ignored. This spec will pass as long as there is a <code>td</code> tag with the appropriate id and contents.</li> </ul> <h3><span class="caps">HTML</span> Testing Discipline and Strategy</h3> <p>One of the reasons that <span class="caps">GUI</span> testing has been so problematic in the <code>.jsp</code> world is that the java scriptlets in those files often reach out into the overall application domain and touch code that ties them to the web container and the application server. For example, if you make a call from a <code>.jsp</code> page to a database gateway, or an entity bean, or some other structure that is tied to the database; then in order to test the <code>.jsp</code> you have to have the full enabling context running. Rails gets away with this because the enabling context is lightweight, portable, and disconnected from the web container, and the live database. Even so, rails applications are not always as decoupled as they should be.</p> <p>In Rails, Java, or any other web context, the discipline should be to make sure that none of the scriptlets in the <code>.jsp</code>, <code>.rhtml</code>, etc. files know anything at all about the rest of the application. Rather, the controller code should load up data into simple objects and pass them to the scriptlets (typically in the <code>attributes</code> field of the <code>HttpServletRequest</code> object or its equivalent). The scriptlets can fiddle with the format of this data (e.g. data formats, money formats, etc.) but <em>should not do any calculation, querying, or other business rule or database processing.</em> Nor should the scriptlets navigate through the model objects or entities. Rather the controller should do all the navigating, gathering, and calculating and present the data to the scriptlets in a nice little package.</p> <p>If you follow this simple design discipline, then your web pages can be generated completely outside of the web environment, and your tests can parse and inspect the html in a simple and friendly environment.</p> <h3>Conclusion</h3> <p>I&#8217;ll have more to say about <em>RSpec</em> in a future blog. <span class="caps">BDD</span> is an exciting twist on the syntax of testing, that has an effect far greater than the simple syntax shift would imply.</p> <p>I hope this article has convinced you that the rails community has solved the problem of testing <span class="caps">HTML</span> generation. This solution can be easily extrapolated back to Java and .NET as future blogs in this series will show.</p> <p>Clearly the problem of testing Javascript, and the ever more complex issues of Web2.0 and <span class="caps">GTK</span> are not addressed by this scheme. However, there are solutions for Javascript that we will investigate in future blogs in this series.</p> <p>Finally, this technique does not test the integration and workflow of a whole application. Again, those are topics for later blogs.</p> <p>I hope this kickoff blog has been informative. If you have a comment, question, or even a rant, please don&#8217;t hesitate to add a comment to this blog.</p> Sat, 13 Jan 2007 17:24:00 -0600 urn:uuid:6fcd0b8d-2d9f-4d06-b44d-926bb905a097 Uncle Bob http://blog.objectmentor.com/articles/2007/01/13/testing-guis-part-i-ror Uncle Bob's Blatherings Testing GUIs "Testing GUIs Part I: RoR." by ping irons <p>Just know teachers collective strike, to<a href="http://www.sportsoutsale.com" rel="nofollow">ping g20 driver</a> blocking defaults salary ready to &#8216;run road&#8217; boss.&#8221; The teachers that night to haikou meilan airport &#8220;intercept&#8221; and clashes happened.Hundreds of students and parents know the truth, worry about the tuition &#8220;skip stone&#8221;, also collective action<a href="http://www.sportsoutsale.com" rel="nofollow">ping g20 fairway woods</a> find li wei money back. Teachers and parents in the police station found that li wei said the company account is only 600 yuan, can&#8217;t pay the teacher&#8217;s salary. But li wei denied &#8220;run road&#8221;, said that the company did not collapse, although there are a wage but have been<a href="http://www.sportsoutsale.com" rel="nofollow">ping g20 irons</a> trying to make up for. According to preliminary statistics, the company defaults teacher pay plus education service fees have not yet provide a total of more than 100 ten thousand yuan.</p> Wed, 16 May 2012 02:23:05 -0500 urn:uuid:d91ed7ac-22fa-4b7a-ad72-42a0d78e2f56 http://blog.objectmentor.com/articles/2007/01/13/testing-guis-part-i-ror#comment-224864 "Testing GUIs Part I: RoR." by ??????? <p>Strongly recommended for shopping at ??? <a href="http://www.cafe-hbal.com/vb/" rel="nofollow">&#1605;&#1606;&#1578;&#1583;&#1610;&#1575;&#1578;</a></p> Fri, 11 May 2012 07:23:04 -0500 urn:uuid:d1b20914-47f1-473a-a044-9a9332d30777 http://blog.objectmentor.com/articles/2007/01/13/testing-guis-part-i-ror#comment-223791 "Testing GUIs Part I: RoR." by Link Building Company India <p>Promote your websites with Linkbuildingtraffic.com: a search engine optimization and internet marketing company specializing in Professional Link Building + SEO and SEM services. please contact at Mail Id: <a href="mailto:linkbuildingtraffic@gmail.com" rel="nofollow">linkbuildingtraffic@gmail.com</a></p> Tue, 01 May 2012 14:35:25 -0500 urn:uuid:997410de-08b9-4428-a594-edfa8cb2c46b http://blog.objectmentor.com/articles/2007/01/13/testing-guis-part-i-ror#comment-221083 "Testing GUIs Part I: RoR." by Craig Chilvers <p>Your article is extremely impressive. I never considered that it was feasible to accomplish something like that until after I looked over your post . Please visit our website for more information .</p> Tue, 01 May 2012 12:30:24 -0500 urn:uuid:534d0d79-81c5-458b-8a73-83336abbf651 http://blog.objectmentor.com/articles/2007/01/13/testing-guis-part-i-ror#comment-221070 "Testing GUIs Part I: RoR." by louboutin sales <p>Testing GUIs Part I: RoR. 137 hoo,good article!!I like the post!4</p> Fri, 20 Apr 2012 18:17:01 -0500 urn:uuid:dcb91a4e-c66e-4328-988b-0f5086923070 http://blog.objectmentor.com/articles/2007/01/13/testing-guis-part-i-ror#comment-218962 "Testing GUIs Part I: RoR." by you <p>I have bookmarked it. <a href="http://mytorridcoupons.blogspot.com" rel="nofollow"> torrid coupons </a></p> Fri, 20 Apr 2012 10:36:48 -0500 urn:uuid:30841825-dd89-4c63-9dd8-7d4163aafb81 http://blog.objectmentor.com/articles/2007/01/13/testing-guis-part-i-ror#comment-218921 "Testing GUIs Part I: RoR." by re <p>This is fantastic article. <a href="http://medicalbillingandcodingjobs.biz" rel="nofollow"> Medical billing and coding jobs </a></p> Fri, 20 Apr 2012 10:34:37 -0500 urn:uuid:3527abca-2728-4d4b-8638-544b92e933db http://blog.objectmentor.com/articles/2007/01/13/testing-guis-part-i-ror#comment-218919 "Testing GUIs Part I: RoR." by youngbrown <p>Thanks for the information, I&#8217;ll visit the site again to get update information <a href="http://toys.shop.ebay.in/" rel="nofollow">Toys</a></p> Tue, 27 Mar 2012 03:46:26 -0500 urn:uuid:ae41ff16-5ca0-4e25-8ae4-e6a7e6c6aea6 http://blog.objectmentor.com/articles/2007/01/13/testing-guis-part-i-ror#comment-213304 "Testing GUIs Part I: RoR." by taylormade r11 driver <p>to the hegemonictitleist 910D3 control of her life, love Ping G20 is not desire Continuous stretch of the war is about people&#8217;s nerves TaylorMade R11s Driverscut off powerful,</p> Mon, 26 Mar 2012 20:12:20 -0500 urn:uuid:a41f3d0a-5e2d-4063-a129-1476a9f6b1d1 http://blog.objectmentor.com/articles/2007/01/13/testing-guis-part-i-ror#comment-213117 "Testing GUIs Part I: RoR." by china crafts <p>I wanted to thank you for this great read!! I definitely enjoying every little bit of it. I have bookmarked it to check out new stuff you post.</p> Wed, 07 Mar 2012 10:42:41 -0600 urn:uuid:5d95fcd9-d6a6-4d39-b189-2ca0f655ce3b http://blog.objectmentor.com/articles/2007/01/13/testing-guis-part-i-ror#comment-208776 "Testing GUIs Part I: RoR." by Massagem Tantrica <p>They</p> <p>’ll normally inform you whether the boots growing to be bought is</p> <p>genuine or an imitation.</p> Mon, 05 Mar 2012 16:11:40 -0600 urn:uuid:3f52c61e-6e28-4c7c-801d-c433b4ad62a8 http://blog.objectmentor.com/articles/2007/01/13/testing-guis-part-i-ror#comment-208060 "Testing GUIs Part I: RoR." by Payday Loans <p>Great post! Thank you so much for sharing this article about Testing GUI&#8217;s.</p> Thu, 16 Feb 2012 14:27:17 -0600 urn:uuid:00e9ea3c-6389-45e2-8f64-af03838d8cf3 http://blog.objectmentor.com/articles/2007/01/13/testing-guis-part-i-ror#comment-203996 "Testing GUIs Part I: RoR." by evdo wireless internet <p>That is one fine ride. The Shimano&#8217;s Dura-Ace Di2 and MSC Koncept frame are sublime. <a href="http://www.evdodepotusa.com" rel="nofollow">evdo wireless internet</a>.</p> Thu, 16 Feb 2012 06:01:56 -0600 urn:uuid:2b03a365-903d-4035-978e-77357153c450 http://blog.objectmentor.com/articles/2007/01/13/testing-guis-part-i-ror#comment-203953 "Testing GUIs Part I: RoR." by iphone sms to mac backup <p>Most of us will delete the SMS file if the iPhone inbox is full. For some of the very important text file, you would like to save it to Mac hard drive and read it later or you need print them. So, why not export the text message to HDD and save it now?</p> Wed, 15 Feb 2012 20:21:25 -0600 urn:uuid:7829ca05-0f6c-4e41-8d51-bd9f00b6976d http://blog.objectmentor.com/articles/2007/01/13/testing-guis-part-i-ror#comment-203777 "Testing GUIs Part I: RoR." by PDF ? PPT <p><a href="http://www.pdftoppt.ru" rel="nofollow">PDF ? PPT</a> PDF to PowerPoint Converter lets you convert PDF to PowerPoint (PPT, PPTX, PPS, PPTM, PPSX, PPSM, POT, etc.), fully supports MS 2010, 2007, 2003.</p> Sun, 29 Jan 2012 19:05:30 -0600 urn:uuid:13916883-83f7-4ea7-a857-e76c200b17f8 http://blog.objectmentor.com/articles/2007/01/13/testing-guis-part-i-ror#comment-200126 "Testing GUIs Part I: RoR." by shakopee townhomes <p>Thanks for the post. It was informative and interesting.</p> Fri, 20 Jan 2012 02:42:15 -0600 urn:uuid:b765d040-0fbd-4c5d-b778-6619f5eeb80d http://blog.objectmentor.com/articles/2007/01/13/testing-guis-part-i-ror#comment-198669 "Testing GUIs Part I: RoR." by Wholesale clothing manufacturers <p>Nice article, I&#8217;ve bookmarked this one.</p> Thu, 19 Jan 2012 22:04:01 -0600 urn:uuid:0904542f-1900-4014-9e95-51d352b9432b http://blog.objectmentor.com/articles/2007/01/13/testing-guis-part-i-ror#comment-198646 "Testing GUIs Part I: RoR." by led displays <p>I have to say this is huge work.I have read the whole article and it is really superbly written article.Testing guis part 1 is really an informative article and wanna say we the readers wanna see some more like this.</p> Fri, 13 Jan 2012 00:24:08 -0600 urn:uuid:395565dc-6e2c-4679-a091-1cbc95b9d885 http://blog.objectmentor.com/articles/2007/01/13/testing-guis-part-i-ror#comment-197646 "Testing GUIs Part I: RoR." by Urdu poetry <p>Thanks for sharing these wonderful comic videos with us. They are really funny. Will look after for some more updates.</p> Thu, 29 Dec 2011 01:06:11 -0600 urn:uuid:cc560cb8-15e0-44ec-9859-b6c574c8ac97 http://blog.objectmentor.com/articles/2007/01/13/testing-guis-part-i-ror#comment-193114 "Testing GUIs Part I: RoR." by dentist in houston <p>First I must tell you we appreciate the great and informative entry.Need to admit that I never learned about this review, I noticed many new facts, in my opinion. God bless you for sharing this information useful and interesting and I also will probably anticipate other interesting posts you closer to the inside future.keep.</p> Tue, 27 Dec 2011 09:16:02 -0600 urn:uuid:8be42780-82d3-48a3-99ff-e22f4386eae6 http://blog.objectmentor.com/articles/2007/01/13/testing-guis-part-i-ror#comment-192606 "Testing GUIs Part I: RoR." by Software Testing Services <p>The article very surprised to me! Your writing is good. In this I learned a lot! Thank you! Share with us the good times. <b>&lt;<a href="http://www.kualitatem.com/" rel="nofollow">Software Quality Assurance</a></b></p> Mon, 28 Nov 2011 00:17:44 -0600 urn:uuid:321e36a1-298a-481c-86d7-3515c80ce34a http://blog.objectmentor.com/articles/2007/01/13/testing-guis-part-i-ror#comment-179021 "Testing GUIs Part I: RoR." by banyo dekorasyon <p>Yes, I plan on discussing the testing of thick client GUIs, <a href="http://www.akinyapi.net/BanyoDekorasyon.html" rel="nofollow">Banyo Dekorasyon</a> and GUIs with lots of JavaScript.</p> Thu, 24 Nov 2011 06:48:34 -0600 urn:uuid:5bafb167-36a4-4113-bf88-2c7835bee858 http://blog.objectmentor.com/articles/2007/01/13/testing-guis-part-i-ror#comment-177517 "Testing GUIs Part I: RoR." by car rental service delhi <p>If only I had found this blog before. The advice in this post are very helpful and I will certainly read the other posts in this series too. Thank you for posting this.</p> Thu, 24 Nov 2011 03:59:56 -0600 urn:uuid:5422e7df-1f60-41ff-ba86-9707839547dd http://blog.objectmentor.com/articles/2007/01/13/testing-guis-part-i-ror#comment-177442 "Testing GUIs Part I: RoR." by Cheap Beats By Dre <p>bus today, make them <a href="http://www.cheap-monsterbeatsbydrdre.co.uk" rel="nofollow">Cheap Beats By Dre</a>miserable. One said: &#8220;I ??am really unlucky it! I was packed <a href="http://www.cheap-monsterbeatsbydrdre.co.uk/beats-by-drdre-studio-c-89.html" rel="nofollow">Beats By Dre Studio</a>in the car to flow production.&#8221; One said: &#8220;I ??called <a href="http://www.cheap-monsterbeatsbydrdre.co.uk/beats-by-drdre-solo-c-91.html" rel="nofollow">Beats By Dre Solo</a>it bad luck! In I was packed car are pregnant.<a href="http://www.cheap-monsterbeatsbydrdre.co.uk/beats-by-drdre-pro-c-90.html" rel="nofollow">Beats By Dre Pro</a> Classic joke: I TVU A university student<a href="http://www.drdrebeatsheadphones-australia.com" rel="nofollow">beats by dr dre</a> caught by the enemy, the enemy tied him at the poles,<a href="http://www.drdrebeatsheadphones-australia.com/justbeats-solo-purple-onear-headphones-with-controltalk-p-234.html" rel="nofollow">just beats solo headphones purple</a> and then asked him: say, where are you? You do not say it electrocuted! S<a href="http://www.drdrebeatsheadphones-australia.com/cheap-drdre-beats-studio-limited-edition-headphones-blackyellow-p-185.html" rel="nofollow">cheap dr.dre beats studio headphones balck/yellow</a>tudents back to the enemy a word, the result was electrocuted, he said: I am TVU.<a href="http://www.drdrebeatsheadphones-australia.com/cheap-beats-by-drdre-pro-performance-professional-headphones-white-p-192.html" rel="nofollow">Hot sale beats by dr dre pro headphones</a></p> Tue, 15 Nov 2011 03:52:02 -0600 urn:uuid:d2d34063-c466-4d7a-a76a-c016cb70ad31 http://blog.objectmentor.com/articles/2007/01/13/testing-guis-part-i-ror#comment-173174 "Testing GUIs Part I: RoR." by Gate Valve <p>Thanks for sharing! please allow me to twitter it.</p> Mon, 07 Nov 2011 21:14:06 -0600 urn:uuid:cea8aee0-300f-44b8-89ce-0798292ca204 http://blog.objectmentor.com/articles/2007/01/13/testing-guis-part-i-ror#comment-169544