jscruggs-metric_fu 0.9.0 → 1.0.0

Sign up to get free protection for your applications and to get access to all the features.
Files changed (69) hide show
  1. data/HISTORY +10 -0
  2. data/README +1 -154
  3. data/Rakefile +37 -2
  4. data/TODO +2 -1
  5. data/lib/base/base_template.rb +134 -0
  6. data/lib/base/configuration.rb +187 -0
  7. data/lib/base/generator.rb +144 -0
  8. data/lib/{metric_fu → base}/md5_tracker.rb +0 -0
  9. data/lib/base/report.rb +100 -0
  10. data/lib/{metric_fu → generators}/churn.rb +20 -22
  11. data/lib/generators/flay.rb +29 -0
  12. data/lib/generators/flog.rb +130 -0
  13. data/lib/generators/rcov.rb +74 -0
  14. data/lib/generators/reek.rb +31 -0
  15. data/lib/generators/roodi.rb +28 -0
  16. data/lib/generators/saikuro.rb +201 -0
  17. data/lib/generators/stats.rb +43 -0
  18. data/lib/metric_fu.rb +20 -3
  19. data/lib/templates/{churn.html.erb → standard/churn.html.erb} +12 -4
  20. data/lib/templates/{default.css → standard/default.css} +20 -1
  21. data/lib/templates/{flay.html.erb → standard/flay.html.erb} +12 -9
  22. data/lib/templates/standard/flog.html.erb +52 -0
  23. data/lib/templates/standard/index.html.erb +38 -0
  24. data/lib/templates/standard/rcov.html.erb +42 -0
  25. data/lib/templates/standard/reek.html.erb +41 -0
  26. data/lib/templates/{roodi.html.erb → standard/roodi.html.erb} +10 -8
  27. data/lib/templates/standard/saikuro.html.erb +83 -0
  28. data/lib/templates/standard/standard_template.rb +26 -0
  29. data/lib/templates/standard/stats.html.erb +54 -0
  30. data/spec/base/base_template_spec.rb +140 -0
  31. data/spec/base/configuration_spec.rb +303 -0
  32. data/spec/base/generator_spec.rb +159 -0
  33. data/spec/{md5_tracker_spec.rb → base/md5_tracker_spec.rb} +1 -1
  34. data/spec/base/report_spec.rb +139 -0
  35. data/spec/generators/churn_spec.rb +152 -0
  36. data/spec/generators/flay_spec.rb +101 -0
  37. data/spec/generators/flog_spec.rb +189 -0
  38. data/spec/generators/reek_spec.rb +47 -0
  39. data/spec/generators/saikuro_spec.rb +35 -0
  40. data/spec/generators/stats_spec.rb +74 -0
  41. data/spec/spec_helper.rb +24 -7
  42. data/tasks/metric_fu.rake +14 -0
  43. data/{lib/tasks → tasks}/railroad.rake +0 -0
  44. data/{lib/metric_fu → vendor}/saikuro/saikuro.rb +0 -0
  45. metadata +58 -47
  46. data/lib/metric_fu/base.rb +0 -160
  47. data/lib/metric_fu/flay.rb +0 -17
  48. data/lib/metric_fu/flog.rb +0 -129
  49. data/lib/metric_fu/reek.rb +0 -17
  50. data/lib/metric_fu/roodi.rb +0 -17
  51. data/lib/tasks/churn.rake +0 -9
  52. data/lib/tasks/coverage.rake +0 -54
  53. data/lib/tasks/flay.rake +0 -6
  54. data/lib/tasks/flog.rake +0 -69
  55. data/lib/tasks/metric_fu.rake +0 -24
  56. data/lib/tasks/metric_fu.rb +0 -6
  57. data/lib/tasks/reek.rake +0 -6
  58. data/lib/tasks/roodi.rake +0 -7
  59. data/lib/tasks/saikuro.rake +0 -35
  60. data/lib/tasks/stats.rake +0 -14
  61. data/lib/templates/flog.html.erb +0 -38
  62. data/lib/templates/flog_page.html.erb +0 -25
  63. data/lib/templates/reek.html.erb +0 -30
  64. data/spec/base_spec.rb +0 -57
  65. data/spec/churn_spec.rb +0 -117
  66. data/spec/config_spec.rb +0 -110
  67. data/spec/flay_spec.rb +0 -19
  68. data/spec/flog_spec.rb +0 -208
  69. data/spec/reek_spec.rb +0 -26
@@ -0,0 +1,43 @@
1
+ module MetricFu
2
+
3
+ class Stats < Generator
4
+
5
+ def emit
6
+ `rake stats > #{metric_directory + '/stats.txt'}`
7
+ end
8
+
9
+ def analyze
10
+ output = File.open(metric_directory + '/stats.txt').read
11
+ output = output.split("\n")
12
+ output = output.find_all {|line| line[0].chr != "+" }
13
+ output = output.find_all {|line| line[0].chr != "(" }
14
+ output.shift
15
+ totals = output.pop
16
+ totals = totals.split(" ").find_all {|el| ! el.empty? }
17
+ @stats = {}
18
+ @stats[:codeLOC] = totals[0].match(/\d.*/)[0].to_i
19
+ @stats[:testLOC] = totals[1].match(/\d.*/)[0].to_i
20
+ @stats[:code_to_test_ratio] = totals[2].match(/1\:(\d.*)/)[1].to_f
21
+
22
+ @stats[:lines] = output.map do |line|
23
+ elements = line.split("|")
24
+ elements.map! {|el| el.strip }
25
+ elements = elements.find_all {|el| ! el.empty? }
26
+ info_line = {}
27
+ info_line[:name] = elements.shift
28
+ elements.map! {|el| el.to_i }
29
+ [:lines, :loc, :classes, :methods,
30
+ :methods_per_class, :loc_per_method].each do |sym|
31
+ info_line[sym] = elements.shift
32
+ end
33
+ info_line
34
+ end
35
+ @stats
36
+ end
37
+
38
+ def to_h
39
+ {:stats => @stats}
40
+ end
41
+
42
+ end
43
+ end
data/lib/metric_fu.rb CHANGED
@@ -1,3 +1,20 @@
1
- require File.join(File.dirname(__FILE__), 'metric_fu', 'base') #require first because of dependecies
2
- require File.join(File.dirname(__FILE__), 'tasks', 'metric_fu')
3
- Dir[File.join(File.dirname(__FILE__), 'metric_fu/*.rb')].each{|l| require l }
1
+ # Load a few things to make our lives easier elsewhere.
2
+ module MetricFu
3
+ LIB_ROOT = File.dirname(__FILE__)
4
+ end
5
+ base_dir = File.join(MetricFu::LIB_ROOT, 'base')
6
+ generator_dir = File.join(MetricFu::LIB_ROOT, 'generators')
7
+ template_dir = File.join(MetricFu::LIB_ROOT, 'templates')
8
+
9
+ # We need to require these two things first because our other classes
10
+ # depend on them.
11
+ require File.join(base_dir, 'report')
12
+ require File.join(base_dir, 'generator')
13
+
14
+ # Load the rakefile so users of the gem get the default metric_fu task
15
+ load File.join(MetricFu::LIB_ROOT, '..', 'tasks', 'metric_fu.rake')
16
+
17
+ # Now load everything else that's in the directory
18
+ Dir[File.join(base_dir, '*.rb')].each{|l| require l }
19
+ Dir[File.join(generator_dir, '*.rb')].each {|l| require l }
20
+ Dir[File.join(template_dir, 'standard/*.rb')].each {|l| require l}
@@ -12,11 +12,19 @@
12
12
  This task uses your source control log to identify those files.
13
13
  </p>
14
14
  <table>
15
- <tr><th>File Path</th><th>Times Changed</th></tr>
16
- <% @changes.to_a.sort {|x,y| y[1] <=> x[1]}.each do |change| %>
17
- <tr><td><%= change[0] %></td><td class='warning'><%= change[1] %></td></tr>
15
+ <tr>
16
+ <th>File Path</th>
17
+ <th>Times Changed</th>
18
+ </tr>
19
+ <% count = 0 %>
20
+ <% @churn[:changes].each do |change| %>
21
+ <tr>
22
+ <td><%= change[:file_path] %></td>
23
+ <td><%= change[:times_changed] %></td>
24
+ </tr>
25
+ <% count += 1 %>
18
26
  <% end %>
19
27
  </table>
20
28
  <p>Generated on <%= Time.now.localtime %></p>
21
29
  </body>
22
- </html>
30
+ </html>
@@ -42,4 +42,23 @@ table {
42
42
 
43
43
  .warning {
44
44
  background: yellow;
45
- }
45
+ }
46
+ .rcov_code td {
47
+ border-bottom: 1px solid #ddd ;
48
+ padding: 0;
49
+ margin: 0;
50
+ }
51
+ .rcov_code tr {
52
+ border: 0px;
53
+ padding:0px;
54
+ margin: 0px;
55
+ }
56
+ .rcov_code pre {
57
+ border: 0px;
58
+ padding: 0px;
59
+ margin: 0px;
60
+ }
61
+ .rcov_run {}
62
+ .rcov_not_run {
63
+ background-color: #d88;
64
+ }
@@ -8,21 +8,24 @@
8
8
 
9
9
  <body>
10
10
  <h1>Flay Results</h1>
11
+ <h2>Total Score (lower is better): <%= @flay[:total_score] %></h2>
11
12
  <p><a href='http://ruby.sadi.st/Flay.html'>Flay</a> analyzes ruby code for structural similarities.</p>
12
13
  <table>
13
- <tr>
14
+ <tr>
14
15
  <th>Files</th>
15
16
  <th>Matches</th>
16
- </tr>
17
- <% @matches.each_with_index do |match, count| %>
17
+ </tr>
18
+ <% count = 0 %>
19
+ <% @flay[:matches].each do |match| %>
18
20
  <tr class='<%= cycle("light", "dark", count) %>'>
19
- <td>
20
- <% match[1..-1].each do |filename| %>
21
- <%= link_to_filename(*filename.split(":")) %><br>
22
- <% end %>
21
+ <td>
22
+ <% match[:matches].each do |file| %>
23
+ <%= file[:name] %>:<%= file[:line] %><br />
24
+ <% end %>
23
25
  </td>
24
- <td><%= match.first %></td>
25
- </tr>
26
+ <td><%= match[:reason] %></td>
27
+ </tr>
28
+ <% count += 1 %>
26
29
  <% end %>
27
30
  </table>
28
31
  <p>Generated on <%= Time.now.localtime %></p>
@@ -0,0 +1,52 @@
1
+ <html>
2
+ <head>
3
+ <title>Flog Reporter</title>
4
+ <style>
5
+ <%= inline_css("default.css") %>
6
+ </style>
7
+ </head>
8
+ <body>
9
+ <h1>Flog Results</h1>
10
+ <p><a href='http://ruby.sadi.st/Flog.html'>Flog</a> measures code complexity.</p>
11
+ <h2>Total Flog score for all methods: <%= @flog[:total]%></h2>
12
+ <h2>Average Flog score for all methods: <%= @flog[:average]%></h2>
13
+ <table>
14
+ <tr>
15
+ <th>File</th>
16
+ <th>Total score</th>
17
+ <th>Methods</th>
18
+ <th>Average score</th>
19
+ <th>Highest score</th>
20
+ </tr>
21
+ <% @flog[:pages].each do |page| %>
22
+ <tr>
23
+ <td><%= page[:path] %></td>
24
+ <td><%= page[:score].round %></td>
25
+ <td><%= page[:scanned_methods].length %></td>
26
+ <td><%= page[:average_score].round %></td>
27
+ <td><%= page[:highest_score].round %></td>
28
+ </tr>
29
+ <% end %>
30
+ </table>
31
+
32
+ <% @flog[:pages].each do |page| %>
33
+ <h2><%= page[:path] %></h2>
34
+ <% page[:scanned_methods].each do |sm| %>
35
+ <p><%= sm[:name] %></p>
36
+ <table>
37
+ <tr>
38
+ <th>Score</th>
39
+ <th>Operator</th>
40
+ </tr>
41
+ <% sm[:operators].each do |operator| %>
42
+ <tr>
43
+ <td><%= operator[:score] %></td>
44
+ <td><%= operator[:operator] %></td>
45
+ </tr>
46
+ <% end %>
47
+ </table>
48
+ <% end %>
49
+ <% end %>
50
+ <p>Generated on <%= Time.now.localtime %></p>
51
+ </body>
52
+ </html>
@@ -0,0 +1,38 @@
1
+ <html>
2
+ <head>
3
+ <title>Metric Fu Results</title>
4
+ <style>
5
+ <%= inline_css("default.css") %>
6
+ </style>
7
+ </head>
8
+
9
+ <body>
10
+ <h1>Metric Fu Results</h1>
11
+ <% if @churn %>
12
+ <p><a href="churn.html">Churn report</a></p>
13
+ <% end %>
14
+ <% if @flay %>
15
+ <p><a href="flay.html">Flay report</a></p>
16
+ <% end %>
17
+ <% if @flog %>
18
+ <p><a href="flog.html">Flog report</a></p>
19
+ <% end %>
20
+ <% if @rcov %>
21
+ <p><a href="rcov.html">Rcov report</a></p>
22
+ <% end %>
23
+ <% if @reek %>
24
+ <p><a href="reek.html">Reek report</a></p>
25
+ <% end %>
26
+ <% if @roodi %>
27
+ <p><a href="roodi.html">Roodi report</a></p>
28
+ <% end %>
29
+ <% if @saikuro %>
30
+ <p><a href="saikuro.html">Saikuro report</a></p>
31
+ <% end %>
32
+ <% if @stats %>
33
+ <p><a href="stats.html">Stats report</a></p>
34
+ <% end %>
35
+ <p>Generated on <%= Time.now.localtime %></p>
36
+ </body>
37
+ </html>
38
+
@@ -0,0 +1,42 @@
1
+ <html>
2
+ <head>
3
+ <title>Rcov Code Coverage Results</title>
4
+ <style>
5
+ <%= inline_css("default.css") %>
6
+ </style>
7
+ </head>
8
+
9
+ <body>
10
+ <h1>Rcov Code Coverage Results</h1>
11
+ <p>C0 code coverage information.</p>
12
+
13
+ <table>
14
+ <tr>
15
+ <th>File Path</th>
16
+ <th>Percent run</th>
17
+ </tr>
18
+ <% count = 0 %>
19
+ <% @rcov.each_pair do |fname, file| %>
20
+ <tr>
21
+ <td><%= fname %></td>
22
+ <td><%= file[:percent_run] %></td>
23
+ </tr>
24
+ <% count += 1 %>
25
+ <% end %>
26
+ </table>
27
+
28
+ <% @rcov.each_pair do |fname, file| %>
29
+ <h2> <%= fname %></h2>
30
+ <table class="rcov_code">
31
+ <% file[:lines].each do |line| %>
32
+ <tr>
33
+ <% css_class = line[:was_run] ? "rcov_run" : "rcov_not_run" %>
34
+ <td class="<%= css_class %>"><pre><%= line[:content] %></pre></td>
35
+ </tr>
36
+ <% end %>
37
+ </table>
38
+ <% end %>
39
+ <p>Generated on <%= Time.now.localtime %></p>
40
+ </body>
41
+ </html>
42
+
@@ -0,0 +1,41 @@
1
+ <html>
2
+ <head>
3
+ <title>Reek Results</title>
4
+ <style>
5
+ <%= inline_css("default.css") %>
6
+ </style>
7
+ </head>
8
+
9
+ <body>
10
+ <h1>Reek Results</h1>
11
+ <p><a href="http://reek.rubyforge.org/">Reek</a> detects common code smells in ruby code.</p>
12
+ <table>
13
+ <tr>
14
+ <th>File Path</th>
15
+ <th>Method</th>
16
+ <th>Description</th>
17
+ <th>Type</th>
18
+ </tr>
19
+ <% count = 0 %>
20
+ <% @reek[:matches].each do |match| %>
21
+ <% match[:code_smells].each do |smell| %>
22
+ <tr class='<%= cycle("light", "dark", count) %>'>
23
+ <td><%= match[:file_path] %></td>
24
+ <td>
25
+ <%= smell[:method] %>
26
+ </td>
27
+ <td>
28
+ <%= smell[:message] %>
29
+ </td>
30
+ <td>
31
+ <%= smell[:type] %>
32
+ </td>
33
+ </tr>
34
+ <% count += 1 %>
35
+ <% end %>
36
+ <% end %>
37
+
38
+ </table>
39
+ <p>Generated on <%= Time.now.localtime %></p>
40
+ </body>
41
+ </html>
@@ -13,14 +13,16 @@
13
13
  <tr>
14
14
  <th>File Path</th>
15
15
  <th>Warning</th>
16
- </tr>
17
- <% @matches.each_with_index do |match, count| %>
18
- <tr class='<%= cycle("light", "dark", count) %>'>
19
- <td><%= link_to_filename(match.first.split(':').first) if match.first %></td>
20
- <td><%= match[1] %></td>
21
- </tr>
22
- <% end %>
16
+ </tr>
17
+ <% count = 0 %>
18
+ <% @roodi[:problems].each do |problem| %>
19
+ <tr class='<%= cycle("light", "dark", count) %>'>
20
+ <td><%= problem[:file] %>:<%= problem[:line] %></td>
21
+ <td><%= problem[:problem] %></td>
22
+ </tr>
23
+ <% count += 1 %>
24
+ <% end %>
23
25
  </table>
24
26
  <p>Generated on <%= Time.now.localtime %></p>
25
27
  </body>
26
- </html>
28
+ </html>
@@ -0,0 +1,83 @@
1
+ <html>
2
+ <head>
3
+ <title>Saikuro Results</title>
4
+ <style>
5
+ <%= inline_css("default.css") %>
6
+ </style>
7
+ </head>
8
+
9
+ <body>
10
+ <h1>Saikuro Results</h1>
11
+ <p><a href='http://saikuro.rubyforge.org/'>Saikuro</a> analyzes ruby code for cyclomatic complexity.</p>
12
+
13
+ <h2>Analyzed Methods</h2>
14
+ <table>
15
+ <tr>
16
+ <th>Method Name</th>
17
+ <th>Complexity</th>
18
+ <th># Lines</th>
19
+ </tr>
20
+ <% @saikuro[:methods].each do |method| %>
21
+ <tr>
22
+ <td><%= method[:name] %></td>
23
+ <td><%= method[:complexity] %></td>
24
+ <td><%= method[:lines] %></td>
25
+ </tr>
26
+ <% end %>
27
+ </table>
28
+
29
+
30
+
31
+ <h2>Analyzed Classes</h2>
32
+ <table>
33
+ <tr>
34
+ <th>Class Name</th>
35
+ <th>Complexity</th>
36
+ <th># Lines</th>
37
+ </tr>
38
+ <% @saikuro[:classes].each do |klass| %>
39
+ <tr>
40
+ <td><%= klass[:name] %></td>
41
+ <td><%= klass[:complexity] %></td>
42
+ <td><%= klass[:lines] %></td>
43
+ </tr>
44
+ <% end %>
45
+ </table>
46
+
47
+
48
+ <h2>Analyzed Files</h2>
49
+ <% @saikuro[:files].each do |file| %>
50
+ <% file[:classes].each do |klass| %>
51
+ <% if !klass[:methods].empty? %>
52
+ <h3><%= file[:filename] %></h3>
53
+ <h4>Class : <%= klass[:class_name] %></h4>
54
+ <h5>Total complexity : <%= klass[:complexity] %></h5>
55
+ <h5>Total lines : <%= klass[:lines] %></h5>
56
+ <table>
57
+ <tr>
58
+ <th>Method</th>
59
+ <th>Complexity</th>
60
+ <th># Lines</th>
61
+ </tr>
62
+ <% klass[:methods].each do |method| %>
63
+ <tr>
64
+ <td>
65
+ <%= method[:name] %>
66
+ </td>
67
+ <td>
68
+ <%= method[:complexity] %>
69
+ </td>
70
+ <td>
71
+ <%= method[:lines] %>
72
+ </td>
73
+ </tr>
74
+ <% end %>
75
+ </table>
76
+ <% end %>
77
+ <% end %>
78
+ <% end %>
79
+
80
+ <p>Generated on <%= Time.now.localtime %></p>
81
+ </body>
82
+ </html>
83
+
@@ -0,0 +1,26 @@
1
+ class StandardTemplate < MetricFu::Template
2
+
3
+
4
+ def write
5
+ report.each_pair do |section, contents|
6
+ if template_exists?(section)
7
+ create_instance_var(section, contents)
8
+ html = erbify(section)
9
+ fn = output_filename(section)
10
+ MetricFu.report.save_output(html, MetricFu.output_directory, fn)
11
+ end
12
+ end
13
+
14
+ # Instance variables we need should already be created from above
15
+ if template_exists?('index')
16
+ html = erbify('index')
17
+ fn = output_filename('index')
18
+ MetricFu.report.save_output(html, MetricFu.output_directory, fn)
19
+ end
20
+ end
21
+
22
+ def this_directory
23
+ File.dirname(__FILE__)
24
+ end
25
+ end
26
+
@@ -0,0 +1,54 @@
1
+ <html>
2
+ <head>
3
+ <title>Rake Stats Results</title>
4
+ <style>
5
+ <%= inline_css("default.css") %>
6
+ </style>
7
+ </head>
8
+
9
+ <body>
10
+ <h1>Rake Stats Results</h1>
11
+ <p>Rails rake stats results.</p>
12
+ <table>
13
+ <tr>
14
+ <th>Lines of Code</th>
15
+ <th>Lines of Test</th>
16
+ <th>Code to test ratio</th>
17
+ </tr>
18
+ <tr>
19
+ <td><%= @stats[:codeLOC] %></td>
20
+ <td><%= @stats[:testLOC] %></td>
21
+ <td>1:<%= @stats[:code_to_test_ratio] %></td>
22
+ </tr>
23
+ </table>
24
+
25
+ <table>
26
+ <tr>
27
+ <th>Name</th>
28
+ <th>Lines</th>
29
+ <th>LOC</th>
30
+ <th>Classes</th>
31
+ <th>Methods</th>
32
+ <th>Methods per class</th>
33
+ <th>LOC per method</th>
34
+ </tr>
35
+ <% count = 0 %>
36
+ <% @stats[:lines].each do |line| %>
37
+ <tr>
38
+ <td><%= line[:name] %></td>
39
+ <td><%= line[:lines] %></td>
40
+ <td><%= line[:loc] %></td>
41
+ <td><%= line[:classes] %></td>
42
+ <td><%= line[:methods] %></td>
43
+ <td><%= line[:methods_per_class] %></td>
44
+ <td><%= line[:loc_per_method] %></td>
45
+ </tr>
46
+ <% count += 1 %>
47
+ <% end %>
48
+ </table>
49
+
50
+ <p>Generated on <%= Time.now.localtime %></p>
51
+ </body>
52
+ </html>
53
+
54
+
@@ -0,0 +1,140 @@
1
+ require File.dirname(__FILE__) + "/../spec_helper.rb"
2
+ require 'erb'
3
+
4
+ describe MetricFu::Template do
5
+
6
+ before(:each) do
7
+ @template = Template.new
8
+ end
9
+
10
+ describe "#erbify" do
11
+ it 'should evaluate a erb doc' do
12
+ section = 'section'
13
+ File.stub!(:read).and_return('foo')
14
+ erb = mock('erb')
15
+ erb.should_receive(:result)
16
+ ERB.should_receive(:new).with('foo').and_return(erb)
17
+ @template.should_receive(:template).and_return('foo')
18
+ @template.send(:erbify, section)
19
+ end
20
+ end
21
+
22
+ describe "#template_exists? " do
23
+
24
+ before(:each) do
25
+ @section = mock('section')
26
+ @template.should_receive(:template).
27
+ with(@section).and_return(@section)
28
+ end
29
+
30
+ describe 'if the template exists' do
31
+ it 'should return true' do
32
+ File.should_receive(:exist?).with(@section).and_return(true)
33
+ result = @template.send(:template_exists?, @section)
34
+ result.should be_true
35
+ end
36
+ end
37
+
38
+ describe 'if the template does not exist' do
39
+ it 'should return false' do
40
+ File.should_receive(:exist?).with(@section).and_return(false)
41
+ result = @template.send(:template_exists?, @section)
42
+ result.should be_false
43
+ end
44
+ end
45
+ end
46
+
47
+ describe "#create_instance_var" do
48
+ it 'should set an instance variable with the passed contents' do
49
+ section = 'section'
50
+ contents = 'contents'
51
+ @template.send(:create_instance_var, section, contents)
52
+ @template.instance_variable_get(:@section).should == contents
53
+ end
54
+ end
55
+
56
+ describe "#template" do
57
+ it 'should generate the filename of the template file' do
58
+ section = mock('section')
59
+ section.should_receive(:to_s).and_return('section')
60
+ @template.should_receive(:this_directory).and_return('dir')
61
+ result = @template.send(:template, section)
62
+ result.should == "dir/section.html.erb"
63
+ end
64
+ end
65
+
66
+ describe "#output_filename" do
67
+ it 'should generate the filename of the output file' do
68
+ section = mock('section')
69
+ section.should_receive(:to_s).and_return('section')
70
+ result = @template.send(:output_filename, section)
71
+ result.should == "section.html"
72
+ end
73
+ end
74
+
75
+ describe "#inline_css" do
76
+ it 'should return the contents of a css file' do
77
+ css = 'mycss.css'
78
+ @template.should_receive(:this_directory).and_return('dir')
79
+ io = mock('io', :read => "css contents")
80
+ @template.should_receive(:open).and_yield(io)
81
+ result = @template.send(:inline_css, css)
82
+ result.should == 'css contents'
83
+ end
84
+ end
85
+
86
+ describe "#link_to_filename " do
87
+ describe "when on OS X" do
88
+ before(:each) do
89
+ config = mock("configuration")
90
+ config.should_receive(:platform).and_return('universal-darwin-9.0')
91
+ MetricFu.stub!(:configuration).and_return(config)
92
+ File.should_receive(:expand_path).and_return('filename')
93
+ end
94
+
95
+ it 'should return a textmate protocol link' do
96
+ name = "filename"
97
+ result = @template.send(:link_to_filename, name)
98
+ result.should eql("<a href='txmt://open/?url=file://" \
99
+ + "filename&line='>filename:</a>")
100
+ end
101
+
102
+ end
103
+
104
+ describe "when on other platforms" do
105
+ before(:each) do
106
+ config = mock("configuration")
107
+ config.should_receive(:platform).and_return('other')
108
+ MetricFu.stub!(:configuration).and_return(config)
109
+ File.should_receive(:expand_path).and_return('filename')
110
+ end
111
+
112
+ it 'should return a file protocol link' do
113
+ name = "filename"
114
+ result = @template.send(:link_to_filename, name)
115
+ result.should == "<a href='file://filename'>filename:</a>"
116
+ end
117
+ end
118
+ end
119
+
120
+ describe "#cycle" do
121
+ it 'should return the first_value passed if iteration passed is even' do
122
+ first_val = "first"
123
+ second_val = "second"
124
+ iter = 2
125
+ result = @template.send(:cycle, first_val, second_val, iter)
126
+ result.should == first_val
127
+ end
128
+
129
+ it 'should return the second_value passed if iteration passed is odd' do
130
+ first_val = "first"
131
+ second_val = "second"
132
+ iter = 1
133
+ result = @template.send(:cycle, first_val, second_val, iter)
134
+ result.should == second_val
135
+ end
136
+ end
137
+
138
+ end
139
+
140
+