require 'cucumber/formatter/ordered_xml_markup'
require 'cucumber/formatter/duration'
require 'cucumber/formatter/summary'
module Cucumber
module Formatter
# The formatter used for --format html
class Html
include ERB::Util # for the #h method
include Duration
include Summary
def initialize(step_mother, io, options)
@io = io
@options = options
@buffer = {}
@step_mother = step_mother
@current_builder = create_builder(@io)
end
def before_features(features)
start_buffering :features
end
def after_features(features)
stop_buffering :features
#
builder.declare!(
:DOCTYPE,
:html,
:PUBLIC,
'-//W3C//DTD XHTML 1.0 Strict//EN',
'http://www.w3.org/TR/xhtml1/DTD/xhtml1-strict.dtd'
)
builder.html(:xmlns => 'http://www.w3.org/1999/xhtml') do
builder.head do
builder.meta(:content => 'text/html;charset=utf-8')
builder.title 'Cucumber'
inline_css
end
builder.body do
builder.div(:class => 'cucumber') do
builder << buffer(:features)
builder.div(scenario_summary(@step_mother) {|status_count, _| status_count}, :class => 'summary')
builder.div(step_summary(@step_mother) {|status_count, _| status_count}, :class => 'summary')
builder.div(format_duration(features.duration), :class => 'duration')
end
end
end
end
def before_feature(feature)
start_buffering :feature
@exceptions = []
end
def after_feature(feature)
stop_buffering :feature
builder.div(:class => 'feature') do
builder << buffer(:feature)
end
end
def before_comment(comment)
start_buffering :comment
end
def after_comment(comment)
stop_buffering :comment
builder.pre(:class => 'comment') do
builder << buffer(:comment)
end
end
def comment_line(comment_line)
builder.text!(comment_line)
builder.br
end
def after_tags(tags)
@tag_spacer = nil
end
def tag_name(tag_name)
builder.text!(@tag_spacer) if @tag_spacer
@tag_spacer = ' '
builder.span(tag_name, :class => 'tag')
end
def feature_name(name)
lines = name.split(/\r?\n/)
return if lines.empty?
builder.h2 do |h2|
builder.span(lines[0], :class => 'val')
end
builder.p(:class => 'narrative') do
lines[1..-1].each do |line|
builder.text!(line.strip)
builder.br
end
end
end
def before_background(background)
@in_background = true
start_buffering :background
end
def after_background(background)
stop_buffering :background
@in_background = nil
builder.div(:class => 'background') do
builder << buffer(:background)
end
end
def background_name(keyword, name, file_colon_line, source_indent)
@listing_background = true
builder.h3 do |h3|
builder.span(keyword, :class => 'keyword')
builder.text!(' ')
builder.span(name, :class => 'val')
end
end
def before_feature_element(feature_element)
start_buffering :feature_element
end
def after_feature_element(feature_element)
stop_buffering :feature_element
css_class = {
Ast::Scenario => 'scenario',
Ast::ScenarioOutline => 'scenario outline'
}[feature_element.class]
builder.div(:class => css_class) do
builder << buffer(:feature_element)
end
@open_step_list = true
end
def scenario_name(keyword, name, file_colon_line, source_indent)
@listing_background = false
builder.h3 do
builder.span(keyword, :class => 'keyword')
builder.text!(' ')
builder.span(name, :class => 'val')
end
end
def before_outline_table(outline_table)
@outline_row = 0
start_buffering :outline_table
end
def after_outline_table(outline_table)
stop_buffering :outline_table
builder.table do
builder << buffer(:outline_table)
end
@outline_row = nil
end
def before_examples(examples)
start_buffering :examples
end
def after_examples(examples)
stop_buffering :examples
builder.div(:class => 'examples') do
builder << buffer(:examples)
end
end
def examples_name(keyword, name)
builder.h4 do
builder.span(keyword, :class => 'keyword')
builder.text!(' ')
builder.span(name, :class => 'val')
end
end
def before_steps(steps)
start_buffering :steps
end
def after_steps(steps)
stop_buffering :steps
builder.ol do
builder << buffer(:steps)
end
end
def before_step(step)
@step_id = step.dom_id
end
def before_step_result(keyword, step_match, multiline_arg, status, exception, source_indent, background)
start_buffering :step_result
@hide_this_step = false
if exception
if @exceptions.include?(exception)
@hide_this_step = true
return
end
@exceptions << exception
end
if status != :failed && @in_background ^ background
@hide_this_step = true
return
end
@status = status
end
def after_step_result(keyword, step_match, multiline_arg, status, exception, source_indent, background)
stop_buffering :step_result
return if @hide_this_step
builder.li(:id => @step_id, :class => "step #{status}") do
builder << buffer(:step_result)
end
end
def step_name(keyword, step_match, status, source_indent, background)
@step_matches ||= []
background_in_scenario = background && !@listing_background
@skip_step = @step_matches.index(step_match) || background_in_scenario
@step_matches << step_match
unless @skip_step
build_step(keyword, step_match, status)
end
end
def exception(exception, status)
return if @hide_this_step
builder.pre(format_exception(exception), :class => status)
end
def before_multiline_arg(multiline_arg)
start_buffering :multiline_arg
end
def after_multiline_arg(multiline_arg)
stop_buffering :multiline_arg
return if @hide_this_step || @skip_step
if Ast::Table === multiline_arg
builder.table do
builder << buffer(:multiline_arg)
end
else
builder << buffer(:multiline_arg)
end
end
def py_string(string)
return if @hide_this_step
builder.pre(:class => 'val') do |pre|
builder << string.gsub("\n", '
')
end
end
def before_table_row(table_row)
@row_id = table_row.dom_id
@col_index = 0
start_buffering :table_row
end
def after_table_row(table_row)
stop_buffering :table_row
return if @hide_this_step
builder.tr(:id => @row_id) do
builder << buffer(:table_row)
end
if table_row.exception
builder.tr do
builder.td(:colspan => @col_index.to_s, :class => 'failed') do
builder.pre do |pre|
pre << format_exception(table_row.exception)
end
end
end
end
@outline_row += 1 if @outline_row
end
def table_cell_value(value, status)
return if @hide_this_step
cell_type = @outline_row == 0 ? :th : :td
attributes = {:id => "#{@row_id}_#{@col_index}", :class => 'val'}
attributes[:class] += " #{status}" if status
build_cell(cell_type, value, attributes)
@col_index += 1
end
def announce(announcement)
builder.pre(announcement, :class => 'announcement')
end
def embed(file, mime_type)
case(mime_type)
when /^image\/(png|gif|jpg)/
embed_image(file)
end
end
private
def embed_image(file)
id = file.hash
builder.pre(:class => 'embed') do |pre|
pre << %{Screenshot
}
end
end
def build_step(keyword, step_match, status)
step_name = step_match.format_args(lambda{|param| %{#{param}}})
builder.div do |div|
builder.span(keyword, :class => 'keyword')
builder.text!(' ')
builder.span(:class => 'step val') do |name|
name << h(step_name).gsub(/<span class="(.*?)">/, '').gsub(/<\/span>/, '')
end
end
end
def build_cell(cell_type, value, attributes)
builder.__send__(cell_type, value, attributes)
end
def inline_css
builder.style(:type => 'text/css') do
builder.text!(File.read(File.dirname(__FILE__) + '/cucumber.css'))
end
end
def format_exception(exception)
h((["#{exception.message} (#{exception.class})"] + exception.backtrace).join("\n"))
end
def builder
@current_builder
end
def buffer(label)
result = @buffer[label]
@buffer[label] = ''
result
end
def start_buffering(label)
@buffer[label] ||= ''
@parent_builder ||= {}
@parent_builder[label] = @current_builder
@current_builder = create_builder(@buffer[label])
end
def stop_buffering(label)
@current_builder = @parent_builder[label]
end
def create_builder(io)
OrderedXmlMarkup.new(:target => io, :indent => 0)
end
end
end
end