first stab at docs

This commit is contained in:
John Bintz 2012-04-10 16:40:53 -04:00
parent 39bf60c9b1
commit c3ec22b326
11 changed files with 522 additions and 10 deletions

View File

@ -2,10 +2,10 @@
# More info at https://github.com/guard/guard#readme # More info at https://github.com/guard/guard#readme
guard 'compass' do guard 'compass', :configuration_file => 'config/compass.rb' do
watch('^sass/(.*)\.s[ac]ss') watch(%r{^sass/(.*)\.s[ac]ss$})
end end
guard 'haml' do guard 'haml', :input => '_layouts/haml', :output => '_layouts' do
watch(/^.+(\.html\.haml)/) watch(/^.+(\.html\.haml)/)
end end

View File

@ -1,2 +1,4 @@
markdown: redcarpet markdown: redcarpet
auto: true
pygments: true

14
_layouts/default.html Normal file
View File

@ -0,0 +1,14 @@
<html>
<head>
<title>Flowerbox</title>
<link href='http://fonts.googleapis.com/css?family=Pontano+Sans' rel='stylesheet' type='text/css' />
<link href='http://fonts.googleapis.com/css?family=Gorditas' rel='stylesheet' type='text/css' />
<link href='stylesheets/screen.css' rel='stylesheet' type='text/css' />
<link href='stylesheets/syntax.css' rel='stylesheet' type='text/css' />
</head>
<body>
<div id='container'>
{{ content }}
</div>
</body>
</html>

View File

@ -0,0 +1,11 @@
%html
%head
%title Flowerbox
%link{:href => 'http://fonts.googleapis.com/css?family=Pontano+Sans', :rel => :stylesheet, :type => 'text/css'}/
%link{:href => 'http://fonts.googleapis.com/css?family=Gorditas', :rel => :stylesheet, :type => 'text/css'}/
%link{:rel => :stylesheet, :href => 'stylesheets/screen.css', :type => 'text/css'}/
%link{:rel => :stylesheet, :href => 'stylesheets/syntax.css', :type => 'text/css'}/
%body
#container
{{ content }}

3
config/compass.rb Normal file
View File

@ -0,0 +1,3 @@
css_dir = 'stylesheets'
sass_dir = 'sass'

209
index.md
View File

@ -1,11 +1,12 @@
--- ---
title: Flowerbox -- a multi-environment, multi-runner JavaScript testing framework framework. title: Flowerbox -- a multi-environment, multi-runner, super-easy JavaScript testing framework framework.
layout: default layout: default
--- ---
# Flowerbox # Flowerbox
## A multi-environment, multi-runner JavaScript testing framework framework. ## A multi-environment, multi-runner, super-easy JavaScript testing framework framework.
---
## Super-fast unit testing getting started ## Super-fast unit testing getting started super guide!
### (you need Firefox at the very least)
{% highlight sh %} {% highlight sh %}
gem install flowerbox gem install flowerbox
@ -14,13 +15,207 @@ flowerbox spec/javascripts
{% endhighlight %} {% endhighlight %}
## ...and what about integration testing? ## ...and what about integration testing?
### (whoa nelly, pure JS Cucumber?)
{% highlight sh %} {% highlight sh %}
flowerbox plant cucumber flowerbox plant cucumber
flowerbox js-features flowerbox js-features
{% end %} {% endhighlight %}
## Flowerbox! ...wha? ## Hey, I'm already using `jasmine-gem` or `jasmine-headless-webkit`
### (you're smart!)
{% highlight sh %}
flowerbox transplant spec/javascripts
flowerbox spec/javascripts
{% endhighlight %}
## I use Sprockets and need to configure asset paths and templates and stuff!
### (...may the force be with you...)
{% highlight ruby %}
# in spec/javascripts/spec_helper.rb
require 'haml-sprockets'
Flowerbox.configure do |c|
c.asset_paths << "app/assets/javascripts"
c.asset_paths << "vendor/assets/javascripts"
c.additional_files << "applications.js.coffee"
# ... other stuff ...
end
{% endhighlight %}
## Firefox is so 2008. Can I run my tests on Chrome?
### (we like Selenium here!)
{% highlight ruby %}
Flowerbox.configure do |c|
c.run_with :chrome
end
{% endhighlight %}
## Chrome is so 2010. Can I run my tests on node.js?
### (evented hipness! jsdom will give you a fake DOM to work with.)
{% highlight ruby %}
# make sure node and npm are in your path
# jsdom and ws get installed in your project under node_modules
Flowerbox.configure do |c|
c.run_with :node
end
{% endhighlight %}
{% highlight coffeescript %}
# make sure you bind your global things to this
class @MyCoolClass extends @MyBaseClass
cool: -> "yeah man"
{% endhighlight %}
## Where'd headless support go?!
No, I knew you'd ask. Read the long bit of text below. _tl;dr_: Selenium with Chrome is
pretty much the same setup, just as fast, and gives you stack traces and other debugging assistance.
---
# Plant some Jasmine
---
# Plant a Cucumber or two
## Make a feature like in big boy Cucumber
### (that's 'cause it *is* big boy Cucumber, just in JS form)
{% highlight gherkin %}
# js-features/features/my_cool_thing.feature
Feature: My Cool Thing
Scenario: Show a cool thing
Given I have a cool thing
And I have a cool thing viewer
When I render the viewer
Then I should see the cool thing in the viewer
@wip # <= tags work, too!
Scenario: Maybe do something else
Given something
When yadda
Then BAM
{% endhighlight %}
## Write some steps
### (Flowerbox gives you a donut-full of syntactic sugar that wraps around Cucumber.js. Be sure to drink lots of water!)
{% highlight coffeescript %}
# js-features/steps/given/i_have_a_cool_thing.js.coffee
Flowerbox.Given /^I have a cool thing$/, ->
@data =
one: 'two'
three: 'four'
@coolThing = new CoolThing(@data)
{% endhighlight %}
{% highlight coffeescript %}
# js-features/steps/given/i_have_a_cool_thing_viewer.js.coffee
Flowerbox.Given /^I have a cool thing viewer$/, ->
@coolThingViewer = new CoolThingViewer(@coolThing)
{% endhighlight %}
{% highlight coffeescript %}
# js-features/steps/when/i_render_the_viewer.js.coffee
Flowerbox.When /^I render the viewer$/, ->
@coolThingViewer.render()
{% endhighlight %}
{% highlight coffeescript %}
# js-features/steps/then/i_should_see_the_cool_thing.js.coffee
Flowerbox.Then /^I should see the cool thing in the viewer$/, ->
@expect(
@coolThingViewer.$('input[name="one"]').val()
).toEqual(@data.one)
@expect(
@coolThingViewer.$('input[name="three"]').val()
).toEqual(@data.three)
{% endhighlight %}
## Get some hooks up in here!
### (and other fun environment stuff)
{% highlight coffeescript %}
# js-features/support/env.js.coffee
#
#= require ../steps
Flowerbox.World ->
@Before (callback) ->
@something = 'is available everywhere'
callback()
@After (callback) ->
Put = 'some stuff back'
callback()
{% endhighlight %}
---
# Flowerbox! Yea...wha?
So I (John) have been in this JavaScript testing game for a while, and, after about So I (John) have been in this JavaScript testing game for a while, and, after about
three years of the stuff, decided to finally take all the knowledge I had o three years of doing JS testing for Web apps, decided to finally take all the knowledge I had
and make something that was dead-simple to use.
## You already did once...
`jasmine-headless-webkit` was my first attempt, but it's a C++/Qt mess. Downloading all of
Qt is a total pain. Also, Qt4's JavaScript engine, JavaScriptCore, doesn't have stack trace
support, so it got pretty useless for me for bigger, messy projects (looking at you, Backbone).
Qt5 has V8, but whatever, I just decided to ditch the whole
compiling software thing and replace it with running tests in Selenium and/or node.js.
## Selenium faster than headless? But CI servers--
Sure, spinning up that instance of Firefox or Chrome is slower than starting up a small
QtWebKit widget, but there's a lot less that can go wrong when using a full-blown
browser. This also means your CI server config stays leaner. You may already have Qt installed
for capybara-webkit, but one less thing depending on Qt just makes your life (and mine) easier.
Also, when running Flowerbox with Guard, your browser stays open between test runs, so you only
pay the startup penalty once.
And, if you write everything correctly, you can just run your tests on node.js and not even have
to worry about a browser.
## But you said fake browsers are teh bad--
I changed my mind. As long as I'm not testing for browser-only things, like CSS interaction
or position elements, running tests on node.js with a browser-like DOM provided by jsdom
really is good enough. I use full-blown Ruby Cucumber if I need to test more complex browser
things.
## Cucumber.js? Full-stack testing is the only way!
Yeah, and it's slow. I'll carve out what I can in Cucumber.js while I work on Ruby Cucumber
features. Makes the process of getting those complex integration tests written a lot faster.
Also, I've changed my views on unit testing as a process. Unit testing for me is now about
two things:
* Code design through a liberal use of stubs and mocks.
* Really quickly running through blg blobs of inputs/outputs (parsers and such).
Treating unit testing this way taught me how to better organize and design my code, and I
only use it for really testing code when I need to test those big blobs of things.
This is my opinion, YMMV, Flowerbox lets you do what you want. Never use Cucumber.js support
if you don't want to. Write your integration bits in Jasmine. I don't mind.

View File

@ -4,3 +4,79 @@
* <link href="/stylesheets/screen.css" media="screen, projection" rel="stylesheet" type="text/css" /> */ * <link href="/stylesheets/screen.css" media="screen, projection" rel="stylesheet" type="text/css" /> */
@import "compass/reset"; @import "compass/reset";
@import "compass/css3";
$base_color: #d9b741;
$header_color: #000;
$header_blend_color: #3a60e5;
$code_border_color: $header_blend_color;
body {
background-color: $base_color;
}
#container {
width: 550px;
margin: 0 auto;
}
h1, h2 {
font-weight: bold;
}
h1 {
font-size: 1.75em;
margin: 0.35em 0;
color: mix($header_color, $header_blend_color, 90);
}
h2 {
font-size: 1.45em;
margin: 0.25em 0;
color: mix($header_color, $header_blend_color, 70);
}
h3 {
font-size: 1.1em;
margin: 0.2em 0;
color: mix($header_color, $header_blend_color, 50);
}
h1, h2, h3 {
text-align: center;
font-family: 'Gorditas', 'Helvetica Neue', 'Helvetica';
}
p, li {
font-family: 'Pontano Sans', 'Helvetica Neue', 'Helvetica';
line-height: 120%;
}
code, pre {
font-family: 'Courier New';
}
.highlight {
font-size: 0.9em;
padding: 1em;
border: solid $code_border_color 1px;
margin: 1em 0;
}
em {
font-style: italic;
}
p {
margin: 0.7em 0;
}
li {
list-style-type: disc;
}
ul {
padding-left: 2em;
}

View File

@ -66,3 +66,85 @@ a img {
article, aside, details, figcaption, figure, footer, header, hgroup, menu, nav, section, summary { article, aside, details, figcaption, figure, footer, header, hgroup, menu, nav, section, summary {
display: block; display: block;
} }
/* line 15, ../sass/screen.scss */
body {
background-color: #d9b741;
}
/* line 19, ../sass/screen.scss */
#container {
width: 550px;
margin: 0 auto;
}
/* line 24, ../sass/screen.scss */
h1, h2 {
font-weight: bold;
}
/* line 28, ../sass/screen.scss */
h1 {
font-size: 1.75em;
margin: 0.35em 0;
color: #050916;
}
/* line 34, ../sass/screen.scss */
h2 {
font-size: 1.45em;
margin: 0.25em 0;
color: #111c44;
}
/* line 40, ../sass/screen.scss */
h3 {
font-size: 1.1em;
margin: 0.2em 0;
color: #1d3072;
}
/* line 46, ../sass/screen.scss */
h1, h2, h3 {
text-align: center;
font-family: 'Gorditas', 'Helvetica Neue', 'Helvetica';
}
/* line 51, ../sass/screen.scss */
p, li {
font-family: 'Pontano Sans', 'Helvetica Neue', 'Helvetica';
line-height: 120%;
}
/* line 56, ../sass/screen.scss */
code, pre {
font-family: 'Courier New';
}
/* line 60, ../sass/screen.scss */
.highlight {
font-size: 0.9em;
padding: 1em;
border: solid #3a60e5 1px;
margin: 1em 0;
}
/* line 68, ../sass/screen.scss */
em {
font-style: italic;
}
/* line 72, ../sass/screen.scss */
p {
margin: 0.7em 0;
}
/* line 76, ../sass/screen.scss */
li {
list-style-type: disc;
}
/* line 80, ../sass/screen.scss */
ul {
padding-left: 2em;
}

60
stylesheets/syntax.css Normal file
View File

@ -0,0 +1,60 @@
.highlight { background: #ffffff; }
.highlight .c { color: #999988; font-style: italic } /* Comment */
.highlight .err { color: #a61717; background-color: #e3d2d2 } /* Error */
.highlight .k { font-weight: bold } /* Keyword */
.highlight .o { font-weight: bold } /* Operator */
.highlight .cm { color: #999988; font-style: italic } /* Comment.Multiline */
.highlight .cp { color: #999999; font-weight: bold } /* Comment.Preproc */
.highlight .c1 { color: #999988; font-style: italic } /* Comment.Single */
.highlight .cs { color: #999999; font-weight: bold; font-style: italic } /* Comment.Special */
.highlight .gd { color: #000000; background-color: #ffdddd } /* Generic.Deleted */
.highlight .gd .x { color: #000000; background-color: #ffaaaa } /* Generic.Deleted.Specific */
.highlight .ge { font-style: italic } /* Generic.Emph */
.highlight .gr { color: #aa0000 } /* Generic.Error */
.highlight .gh { color: #999999 } /* Generic.Heading */
.highlight .gi { color: #000000; background-color: #ddffdd } /* Generic.Inserted */
.highlight .gi .x { color: #000000; background-color: #aaffaa } /* Generic.Inserted.Specific */
.highlight .go { color: #888888 } /* Generic.Output */
.highlight .gp { color: #555555 } /* Generic.Prompt */
.highlight .gs { font-weight: bold } /* Generic.Strong */
.highlight .gu { color: #aaaaaa } /* Generic.Subheading */
.highlight .gt { color: #aa0000 } /* Generic.Traceback */
.highlight .kc { font-weight: bold } /* Keyword.Constant */
.highlight .kd { font-weight: bold } /* Keyword.Declaration */
.highlight .kp { font-weight: bold } /* Keyword.Pseudo */
.highlight .kr { font-weight: bold } /* Keyword.Reserved */
.highlight .kt { color: #445588; font-weight: bold } /* Keyword.Type */
.highlight .m { color: #009999 } /* Literal.Number */
.highlight .s { color: #d14 } /* Literal.String */
.highlight .na { color: #008080 } /* Name.Attribute */
.highlight .nb { color: #0086B3 } /* Name.Builtin */
.highlight .nc { color: #445588; font-weight: bold } /* Name.Class */
.highlight .no { color: #008080 } /* Name.Constant */
.highlight .ni { color: #800080 } /* Name.Entity */
.highlight .ne { color: #990000; font-weight: bold } /* Name.Exception */
.highlight .nf { color: #990000; font-weight: bold } /* Name.Function */
.highlight .nn { color: #555555 } /* Name.Namespace */
.highlight .nt { color: #000080 } /* Name.Tag */
.highlight .nv { color: #008080 } /* Name.Variable */
.highlight .ow { font-weight: bold } /* Operator.Word */
.highlight .w { color: #bbbbbb } /* Text.Whitespace */
.highlight .mf { color: #009999 } /* Literal.Number.Float */
.highlight .mh { color: #009999 } /* Literal.Number.Hex */
.highlight .mi { color: #009999 } /* Literal.Number.Integer */
.highlight .mo { color: #009999 } /* Literal.Number.Oct */
.highlight .sb { color: #d14 } /* Literal.String.Backtick */
.highlight .sc { color: #d14 } /* Literal.String.Char */
.highlight .sd { color: #d14 } /* Literal.String.Doc */
.highlight .s2 { color: #d14 } /* Literal.String.Double */
.highlight .se { color: #d14 } /* Literal.String.Escape */
.highlight .sh { color: #d14 } /* Literal.String.Heredoc */
.highlight .si { color: #d14 } /* Literal.String.Interpol */
.highlight .sx { color: #d14 } /* Literal.String.Other */
.highlight .sr { color: #009926 } /* Literal.String.Regex */
.highlight .s1 { color: #d14 } /* Literal.String.Single */
.highlight .ss { color: #990073 } /* Literal.String.Symbol */
.highlight .bp { color: #999999 } /* Name.Builtin.Pseudo */
.highlight .vc { color: #008080 } /* Name.Variable.Class */
.highlight .vg { color: #008080 } /* Name.Variable.Global */
.highlight .vi { color: #008080 } /* Name.Variable.Instance */
.highlight .il { color: #009999 } /* Literal.Number.Integer.Long */

69
stylesheets/vim.css Normal file
View File

@ -0,0 +1,69 @@
.codehilite .hll { background-color: #222222 }
.codehilite .c { color: #000080 } /* Comment */
.codehilite .err { color: #cccccc; border: 1px solid #FF0000 } /* Error */
.codehilite .g { color: #cccccc } /* Generic */
.codehilite .k { color: #cdcd00 } /* Keyword */
.codehilite .l { color: #cccccc } /* Literal */
.codehilite .n { color: #cccccc } /* Name */
.codehilite .o { color: #3399cc } /* Operator */
.codehilite .x { color: #cccccc } /* Other */
.codehilite .p { color: #cccccc } /* Punctuation */
.codehilite .cm { color: #000080 } /* Comment.Multiline */
.codehilite .cp { color: #000080 } /* Comment.Preproc */
.codehilite .c1 { color: #000080 } /* Comment.Single */
.codehilite .cs { color: #cd0000; font-weight: bold } /* Comment.Special */
.codehilite .gd { color: #cd0000 } /* Generic.Deleted */
.codehilite .ge { color: #cccccc; font-style: italic } /* Generic.Emph */
.codehilite .gr { color: #FF0000 } /* Generic.Error */
.codehilite .gh { color: #000080; font-weight: bold } /* Generic.Heading */
.codehilite .gi { color: #00cd00 } /* Generic.Inserted */
.codehilite .go { color: #808080 } /* Generic.Output */
.codehilite .gp { color: #000080; font-weight: bold } /* Generic.Prompt */
.codehilite .gs { color: #cccccc; font-weight: bold } /* Generic.Strong */
.codehilite .gu { color: #800080; font-weight: bold } /* Generic.Subheading */
.codehilite .gt { color: #0040D0 } /* Generic.Traceback */
.codehilite .kc { color: #cdcd00 } /* Keyword.Constant */
.codehilite .kd { color: #00cd00 } /* Keyword.Declaration */
.codehilite .kn { color: #cd00cd } /* Keyword.Namespace */
.codehilite .kp { color: #cdcd00 } /* Keyword.Pseudo */
.codehilite .kr { color: #cdcd00 } /* Keyword.Reserved */
.codehilite .kt { color: #00cd00 } /* Keyword.Type */
.codehilite .ld { color: #cccccc } /* Literal.Date */
.codehilite .m { color: #cd00cd } /* Literal.Number */
.codehilite .s { color: #cd0000 } /* Literal.String */
.codehilite .na { color: #cccccc } /* Name.Attribute */
.codehilite .nb { color: #cd00cd } /* Name.Builtin */
.codehilite .nc { color: #00cdcd } /* Name.Class */
.codehilite .no { color: #cccccc } /* Name.Constant */
.codehilite .nd { color: #cccccc } /* Name.Decorator */
.codehilite .ni { color: #cccccc } /* Name.Entity */
.codehilite .ne { color: #666699; font-weight: bold } /* Name.Exception */
.codehilite .nf { color: #cccccc } /* Name.Function */
.codehilite .nl { color: #cccccc } /* Name.Label */
.codehilite .nn { color: #cccccc } /* Name.Namespace */
.codehilite .nx { color: #cccccc } /* Name.Other */
.codehilite .py { color: #cccccc } /* Name.Property */
.codehilite .nt { color: #cccccc } /* Name.Tag */
.codehilite .nv { color: #00cdcd } /* Name.Variable */
.codehilite .ow { color: #cdcd00 } /* Operator.Word */
.codehilite .w { color: #cccccc } /* Text.Whitespace */
.codehilite .mf { color: #cd00cd } /* Literal.Number.Float */
.codehilite .mh { color: #cd00cd } /* Literal.Number.Hex */
.codehilite .mi { color: #cd00cd } /* Literal.Number.Integer */
.codehilite .mo { color: #cd00cd } /* Literal.Number.Oct */
.codehilite .sb { color: #cd0000 } /* Literal.String.Backtick */
.codehilite .sc { color: #cd0000 } /* Literal.String.Char */
.codehilite .sd { color: #cd0000 } /* Literal.String.Doc */
.codehilite .s2 { color: #cd0000 } /* Literal.String.Double */
.codehilite .se { color: #cd0000 } /* Literal.String.Escape */
.codehilite .sh { color: #cd0000 } /* Literal.String.Heredoc */
.codehilite .si { color: #cd0000 } /* Literal.String.Interpol */
.codehilite .sx { color: #cd0000 } /* Literal.String.Other */
.codehilite .sr { color: #cd0000 } /* Literal.String.Regex */
.codehilite .s1 { color: #cd0000 } /* Literal.String.Single */
.codehilite .ss { color: #cd0000 } /* Literal.String.Symbol */
.codehilite .bp { color: #cd00cd } /* Name.Builtin.Pseudo */
.codehilite .vc { color: #00cdcd } /* Name.Variable.Class */
.codehilite .vg { color: #00cdcd } /* Name.Variable.Global */
.codehilite .vi { color: #00cdcd } /* Name.Variable.Instance */
.codehilite .il { color: #cd00cd } /* Literal.Number.Integer.Long */