nielsm-metric_fu 1.1.1 → 1.3.1

Sign up to get free protection for your applications and to get access to all the features.
Files changed (65) hide show
  1. data/HISTORY +46 -0
  2. data/Rakefile +4 -7
  3. data/TODO +0 -4
  4. data/lib/base/base_template.rb +1 -2
  5. data/lib/base/configuration.rb +9 -35
  6. data/lib/base/generator.rb +9 -1
  7. data/lib/base/graph.rb +5 -3
  8. data/lib/base/report.rb +3 -3
  9. data/lib/generators/churn.rb +15 -72
  10. data/lib/generators/flay.rb +3 -2
  11. data/lib/generators/flog.rb +33 -6
  12. data/lib/generators/rcov.rb +32 -37
  13. data/lib/generators/reek.rb +28 -1
  14. data/lib/generators/roodi.rb +2 -1
  15. data/lib/generators/saikuro.rb +53 -40
  16. data/lib/generators/stats.rb +29 -13
  17. data/lib/graphs/engines/bluff.rb +86 -0
  18. data/lib/graphs/engines/gchart.rb +118 -0
  19. data/lib/graphs/flay_grapher.rb +6 -20
  20. data/lib/graphs/flog_grapher.rb +25 -22
  21. data/lib/graphs/grapher.rb +11 -0
  22. data/lib/graphs/rcov_grapher.rb +2 -16
  23. data/lib/graphs/reek_grapher.rb +12 -24
  24. data/lib/graphs/roodi_grapher.rb +6 -20
  25. data/lib/metric_fu.rb +13 -6
  26. data/lib/templates/awesome/awesome_template.rb +7 -0
  27. data/lib/templates/awesome/churn.html.erb +39 -0
  28. data/lib/templates/awesome/css/buttons.css +82 -0
  29. data/lib/templates/awesome/{default.css → css/default.css} +16 -0
  30. data/lib/templates/awesome/css/integrity.css +335 -0
  31. data/lib/templates/awesome/css/reset.css +7 -0
  32. data/lib/templates/awesome/flay.html.erb +8 -1
  33. data/lib/templates/awesome/flog.html.erb +8 -1
  34. data/lib/templates/awesome/layout.html.erb +7 -4
  35. data/lib/templates/awesome/rcov.html.erb +7 -1
  36. data/lib/templates/awesome/reek.html.erb +7 -1
  37. data/lib/templates/awesome/roodi.html.erb +7 -1
  38. data/lib/templates/javascripts/bluff-min.js +1 -0
  39. data/lib/templates/javascripts/excanvas.js +35 -0
  40. data/lib/templates/javascripts/js-class.js +1 -0
  41. data/spec/base/configuration_spec.rb +21 -54
  42. data/spec/base/generator_spec.rb +63 -0
  43. data/spec/base/graph_spec.rb +24 -0
  44. data/spec/generators/churn_spec.rb +16 -125
  45. data/spec/generators/flay_spec.rb +11 -5
  46. data/spec/generators/flog_spec.rb +19 -0
  47. data/spec/generators/reek_spec.rb +65 -0
  48. data/spec/graphs/engines/bluff_spec.rb +16 -0
  49. data/spec/graphs/engines/gchart_spec.rb +108 -0
  50. data/spec/graphs/flay_grapher_spec.rb +37 -0
  51. data/spec/graphs/flog_grapher_spec.rb +40 -10
  52. data/spec/graphs/rcov_grapher_spec.rb +37 -0
  53. data/spec/graphs/reek_grapher_spec.rb +46 -0
  54. data/spec/graphs/roodi_grapher_spec.rb +37 -0
  55. data/spec/resources/saikuro/app/controllers/sessions_controller.rb_cyclo.html +0 -0
  56. data/spec/resources/saikuro/app/controllers/users_controller.rb_cyclo.html +0 -0
  57. data/spec/resources/saikuro/index_cyclo.html +0 -0
  58. data/spec/resources/saikuro_sfiles/thing.rb_cyclo.html +1 -0
  59. data/spec/resources/yml/20090630.yml +7844 -0
  60. data/spec/spec.opts +2 -4
  61. data/spec/spec_helper.rb +2 -23
  62. data/tasks/metric_fu.rake +1 -1
  63. metadata +72 -14
  64. data/vendor/_fonts/monaco.ttf +0 -0
  65. data/vendor/saikuro/saikuro.rb +0 -1214
@@ -1,35 +1,38 @@
1
- require 'gruff'
2
1
  module MetricFu
3
2
 
4
- class FlogGrapher
3
+ class FlogGrapher < Grapher
5
4
 
6
- attr_accessor :flog_total, :flog_average, :labels
5
+ attr_accessor :flog_average, :labels, :top_five_percent_average
7
6
 
8
7
  def initialize
9
- self.flog_total = []
10
- self.flog_average = []
11
- self.labels = {}
8
+ super
9
+ @flog_average = []
10
+ @labels = {}
11
+ @top_five_percent_average =[]
12
12
  end
13
13
 
14
14
  def get_metrics(metrics, date)
15
- self.flog_total.push(metrics[:flog][:total])
16
- self.flog_average.push(metrics[:flog][:average])
17
- self.labels.update( { self.labels.size => date })
15
+ @top_five_percent_average.push(calc_top_five_percent_average(metrics))
16
+ @flog_average.push(metrics[:flog][:average])
17
+ @labels.update( { @labels.size => date })
18
18
  end
19
19
 
20
- def graph!
21
- g = Gruff::Line.new(MetricFu.graph_size)
22
- g.title = "Flog: code complexity"
23
- g.theme = MetricFu.graph_theme
24
- g.font = MetricFu.graph_font
25
- g.data('flog total', self.flog_total)
26
- g.data('flog average', self.flog_average)
27
- g.labels = self.labels
28
- g.title_font_size = MetricFu.graph_title_font_size
29
- g.legend_box_size = MetricFu.graph_legend_box_size
30
- g.legend_font_size = MetricFu.graph_legend_font_size
31
- g.marker_font_size = MetricFu.graph_marker_font_size
32
- g.write(File.join(MetricFu.output_directory, 'flog.png'))
20
+ private
21
+
22
+ def calc_top_five_percent_average(metrics)
23
+ methods = metrics[:flog][:pages].inject([]) {|methods, page| methods << page[:scanned_methods]}
24
+ methods.flatten!
25
+ methods = methods.sort_by {|method| method[:score]}.reverse
26
+
27
+ number_of_methods_that_is_five_percent = (methods.size * 0.05).ceil
28
+
29
+ total_for_five_percent =
30
+ methods[0...number_of_methods_that_is_five_percent].inject(0) {|total, method| total += method[:score] }
31
+ if number_of_methods_that_is_five_percent == 0
32
+ 0.0
33
+ else
34
+ total_for_five_percent / number_of_methods_that_is_five_percent.to_f
35
+ end
33
36
  end
34
37
 
35
38
  end
@@ -0,0 +1,11 @@
1
+ module MetricFu
2
+ class Grapher
3
+ def initialize
4
+ self.class.require_graphing_gem
5
+ end
6
+
7
+ def self.require_graphing_gem
8
+ # to be overridden by charting engines
9
+ end
10
+ end
11
+ end
@@ -1,11 +1,11 @@
1
- require 'gruff'
2
1
  module MetricFu
3
2
 
4
- class RcovGrapher
3
+ class RcovGrapher < Grapher
5
4
 
6
5
  attr_accessor :rcov_percent, :labels
7
6
 
8
7
  def initialize
8
+ super
9
9
  self.rcov_percent = []
10
10
  self.labels = {}
11
11
  end
@@ -15,20 +15,6 @@ module MetricFu
15
15
  self.labels.update( { self.labels.size => date })
16
16
  end
17
17
 
18
- def graph!
19
- g = Gruff::Line.new(MetricFu.graph_size)
20
- g.title = "Rcov: code coverage"
21
- g.theme = MetricFu.graph_theme
22
- g.font = MetricFu.graph_font
23
- g.data('rcov', self.rcov_percent)
24
- g.labels = self.labels
25
- g.title_font_size = MetricFu.graph_title_font_size
26
- g.legend_box_size = MetricFu.graph_legend_box_size
27
- g.legend_font_size = MetricFu.graph_legend_font_size
28
- g.marker_font_size = MetricFu.graph_marker_font_size
29
- g.write(File.join(MetricFu.output_directory, 'rcov.png'))
30
- end
31
-
32
18
  end
33
19
 
34
20
  end
@@ -1,44 +1,32 @@
1
- require 'gruff'
2
1
  module MetricFu
3
2
 
4
- class ReekGrapher
3
+ class ReekGrapher < Grapher
5
4
 
6
5
  attr_accessor :reek_count, :labels
7
6
 
8
7
  def initialize
9
- self.reek_count = {}
10
- self.labels= {}
8
+ super
9
+ @reek_count = {}
10
+ @labels = {}
11
11
  end
12
12
 
13
13
  def get_metrics(metrics, date)
14
- counter = self.labels.size
15
- self.labels.update( { counter => date })
14
+ counter = @labels.size
15
+ @labels.update( { @labels.size => date })
16
16
 
17
17
  metrics[:reek][:matches].each do |reek_chunk|
18
18
  reek_chunk[:code_smells].each do |code_smell|
19
19
  # speaking of code smell...
20
- self.reek_count[code_smell[:type]] = [] if self.reek_count[code_smell[:type]].nil?
21
- self.reek_count[code_smell[:type]][counter].nil? ? self.reek_count[code_smell[:type]][counter] = 1 : self.reek_count[code_smell[:type]][counter] += 1
20
+ @reek_count[code_smell[:type]] = [] if @reek_count[code_smell[:type]].nil?
21
+ if @reek_count[code_smell[:type]][counter].nil?
22
+ @reek_count[code_smell[:type]][counter] = 1
23
+ else
24
+ @reek_count[code_smell[:type]][counter] += 1
25
+ end
22
26
  end
23
27
  end
24
28
  end
25
29
 
26
- def graph!
27
- g = Gruff::Line.new(MetricFu.graph_size)
28
- g.title = "Reek: code smells"
29
- g.theme = MetricFu.graph_theme
30
- g.font = MetricFu.graph_font
31
- self.reek_count.each_pair do |type, count|
32
- g.data(type, count)
33
- end
34
- g.labels = self.labels
35
- g.title_font_size = MetricFu.graph_title_font_size
36
- g.legend_box_size = MetricFu.graph_legend_box_size
37
- g.legend_font_size = MetricFu.graph_legend_font_size
38
- g.marker_font_size = MetricFu.graph_marker_font_size
39
- g.write(File.join(MetricFu.output_directory, 'reek.png'))
40
- end
41
-
42
30
  end
43
31
 
44
32
  end
@@ -1,32 +1,18 @@
1
- require 'gruff'
2
1
  module MetricFu
3
2
 
4
- class RoodiGrapher
3
+ class RoodiGrapher < Grapher
5
4
 
6
5
  attr_accessor :roodi_count, :labels
7
6
 
8
7
  def initialize
9
- self.roodi_count = []
10
- self.labels = {}
8
+ super
9
+ @roodi_count = []
10
+ @labels = {}
11
11
  end
12
12
 
13
13
  def get_metrics(metrics, date)
14
- self.roodi_count.push(metrics[:roodi][:problems].size)
15
- self.labels.update( { self.labels.size => date })
16
- end
17
-
18
- def graph!
19
- g = Gruff::Line.new(MetricFu.graph_size)
20
- g.title = "Roodi: design problems"
21
- g.theme = MetricFu.graph_theme
22
- g.font = MetricFu.graph_font
23
- g.data('roodi', self.roodi_count)
24
- g.labels = self.labels
25
- g.title_font_size = MetricFu.graph_title_font_size
26
- g.legend_box_size = MetricFu.graph_legend_box_size
27
- g.legend_font_size = MetricFu.graph_legend_font_size
28
- g.marker_font_size = MetricFu.graph_marker_font_size
29
- g.write(File.join(MetricFu.output_directory, 'roodi.png'))
14
+ @roodi_count.push(metrics[:roodi][:problems].size)
15
+ @labels.update( { @labels.size => date })
30
16
  end
31
17
 
32
18
  end
@@ -1,11 +1,13 @@
1
+ require 'rake'
2
+ require 'yaml'
1
3
  # Load a few things to make our lives easier elsewhere.
2
4
  module MetricFu
3
5
  LIB_ROOT = File.dirname(__FILE__)
4
6
  end
5
7
  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
- graph_dir = File.join(MetricFu::LIB_ROOT, 'graphs')
8
+ generator_dir = File.join(MetricFu::LIB_ROOT, 'generators')
9
+ template_dir = File.join(MetricFu::LIB_ROOT, 'templates')
10
+ graph_dir = File.join(MetricFu::LIB_ROOT, 'graphs')
9
11
 
10
12
  # We need to require these two things first because our other classes
11
13
  # depend on them.
@@ -13,12 +15,17 @@ require File.join(base_dir, 'report')
13
15
  require File.join(base_dir, 'generator')
14
16
  require File.join(base_dir, 'graph')
15
17
 
16
- # Load the rakefile so users of the gem get the default metric_fu task
17
- load File.join(MetricFu::LIB_ROOT, '..', 'tasks', 'metric_fu.rake')
18
+ # prevent the task from being run multiple times.
19
+ unless Rake::Task.task_defined? "metrics:all"
20
+ # Load the rakefile so users of the gem get the default metric_fu task
21
+ load File.join(MetricFu::LIB_ROOT, '..', 'tasks', 'metric_fu.rake')
22
+ end
18
23
 
19
24
  # Now load everything else that's in the directory
20
25
  Dir[File.join(base_dir, '*.rb')].each{|l| require l }
21
26
  Dir[File.join(generator_dir, '*.rb')].each {|l| require l }
22
27
  Dir[File.join(template_dir, 'standard/*.rb')].each {|l| require l}
23
28
  Dir[File.join(template_dir, 'awesome/*.rb')].each {|l| require l}
24
- Dir[File.join(graph_dir, '*.rb')].each {|l| require l}
29
+ require graph_dir + "/grapher"
30
+ Dir[File.join(graph_dir, '*.rb')].each {|l| require l}
31
+ Dir[File.join(graph_dir, 'engines', '*.rb')].each {|l| require l}
@@ -1,9 +1,16 @@
1
+ require 'fileutils'
2
+
1
3
  class AwesomeTemplate < MetricFu::Template
2
4
 
3
5
  def write
4
6
  # Getting rid of the crap before and after the project name from integrity
5
7
  @name = File.basename(Dir.pwd).gsub(/^\w+-|-\w+$/, "")
6
8
 
9
+ # Copy Bluff javascripts to output directory
10
+ Dir[File.join(this_directory, '..', 'javascripts', '*')].each do |f|
11
+ FileUtils.copy(f, File.join(MetricFu.output_directory, File.basename(f)))
12
+ end
13
+
7
14
  report.each_pair do |section, contents|
8
15
  if template_exists?(section)
9
16
  create_instance_var(section, contents)
@@ -16,4 +16,43 @@
16
16
  <% count += 1 %>
17
17
  <% end %>
18
18
  </table>
19
+
20
+ <% if @churn[:class_churn] %>
21
+ <p>Classes that change a lot in your project may be bad a sign.</p>
22
+ <table>
23
+ <tr>
24
+ <th>File Path</th>
25
+ <th>Times Changed</th>
26
+ </tr>
27
+ <% count = 0 %>
28
+ <% @churn[:class_churn].each do |change| %>
29
+ <tr>
30
+ <td><%= link_to_filename(change['klass']['file']) %> <%= change['klass']['klass'] %></td>
31
+ <td><%= change['times_changed'] %></td>
32
+ </tr>
33
+ <% count += 1 %>
34
+ <% end %>
35
+ </table>
36
+ <% end %>
37
+
38
+ <% if @churn[:method_churn] %>
39
+ <p>Methods that change a lot in your project may be bad a sign.</p>
40
+ <table>
41
+ <tr>
42
+ <th>File Path</th>
43
+ <th>Times Changed</th>
44
+ </tr>
45
+ <% count = 0 %>
46
+ <% @churn[:method_churn].each do |change| %>
47
+ <tr>
48
+ <td><%= link_to_filename(change['method']['file']) %> <%= change['method']['method'] %></td>
49
+ <td><%= change['times_changed'] %></td>
50
+ </tr>
51
+ <% count += 1 %>
52
+ <% end %>
53
+ </table>
54
+ <% end %>
55
+
56
+
57
+
19
58
  <p>Generated on <%= Time.now.localtime %></p>
@@ -0,0 +1,82 @@
1
+ /* --------------------------------------------------------------
2
+
3
+ buttons.css
4
+ * Gives you some great CSS-only buttons.
5
+
6
+ Created by Kevin Hale [particletree.com]
7
+ * particletree.com/features/rediscovering-the-button-element
8
+
9
+ See Readme.txt in this folder for instructions.
10
+
11
+ -------------------------------------------------------------- */
12
+
13
+ button {
14
+ display:block;
15
+ float:left;
16
+ margin:0 0.583em 0.667em 0;
17
+ padding:5px 10px 5px 7px; /* Links */
18
+
19
+ border:1px solid #dedede;
20
+ border-top:1px solid #eee;
21
+ border-left:1px solid #eee;
22
+
23
+ background-color:#f5f5f5;
24
+ font-family:"Lucida Grande", Tahoma, Arial, Verdana, sans-serif;
25
+ font-size:100%;
26
+ line-height:130%;
27
+ text-decoration:none;
28
+ font-weight:bold;
29
+ color:#565656;
30
+ cursor:pointer;
31
+ }
32
+ button {
33
+ width:auto;
34
+ overflow:visible;
35
+ padding:4px 10px 3px 7px; /* IE6 */
36
+ }
37
+ button[type] {
38
+ padding:4px 10px 4px 7px; /* Firefox */
39
+ line-height:17px; /* Safari */
40
+ }
41
+ *:first-child+html button[type] {
42
+ padding:4px 10px 3px 7px; /* IE7 */
43
+ }
44
+ button img {
45
+ margin:0 3px -3px 0 !important;
46
+ padding:0;
47
+ border:none;
48
+ width:16px;
49
+ height:16px;
50
+ float:none;
51
+ }
52
+
53
+
54
+ /* Button colors
55
+ -------------------------------------------------------------- */
56
+
57
+ /* Standard */
58
+ button:hover {
59
+ background-color:#dff4ff;
60
+ border:1px solid #c2e1ef;
61
+ color:#336699;
62
+ }
63
+
64
+ /* Positive */
65
+ body .positive {
66
+ color:#529214;
67
+ }
68
+ button.positive:hover {
69
+ background-color:#E6EFC2;
70
+ border:1px solid #C6D880;
71
+ color:#529214;
72
+ }
73
+
74
+ /* Negative */
75
+ body .negative {
76
+ color:#d12f19;
77
+ }
78
+ button.negative:hover {
79
+ background:#fbe3e4;
80
+ border:1px solid #fbc2c4;
81
+ color:#d12f19;
82
+ }
@@ -73,3 +73,19 @@ table td.score {
73
73
  overflow: auto;
74
74
  font-size: 50%;
75
75
  }
76
+
77
+ .bluff-tooltip {
78
+ background: #fff;
79
+ border: 1px solid #d1edf5;
80
+ padding: 8px 8px 6px;
81
+ }
82
+ .bluff-tooltip .color {
83
+ display: block;
84
+ height: 4px;
85
+ width: 30px;
86
+ margin: 0 0 4px;
87
+ overflow: hidden;
88
+ }
89
+ .bluff-tooltip .data {
90
+ font-weight: bold;
91
+ }
@@ -0,0 +1,335 @@
1
+ html {
2
+ background-color: #e0e0e0; }
3
+
4
+ body {
5
+ font-size: 100%;
6
+ font-family: Helvetica Neue, Helvetica, Arial, sans-serif;
7
+ color: #333333; }
8
+
9
+ a {
10
+ color: #ed1556;
11
+ text-decoration: none; }
12
+ a:hover {
13
+ color: #ffffff;
14
+ background-color: #ed1556; }
15
+
16
+ #header, #content, #footer {
17
+ margin: 0 auto;
18
+ background: #eeeeee;
19
+ padding: 0 2em;
20
+ z-index: 0;
21
+ position: relative;
22
+ font-size: 1em; }
23
+
24
+ #header {
25
+ background: #ffffff; }
26
+ #header h1 {
27
+ font-weight: bold;
28
+ font-size: 1.5em; }
29
+ #header address.watermark {
30
+ position: absolute;
31
+ font-weight: bold;
32
+ right: 3em;
33
+ top: 0;
34
+ font-size: .75em;
35
+ color: #cccccc; }
36
+ #header address.watermark a {
37
+ color: #cccccc;
38
+ font-weight: bold;
39
+ font-size: 2em; }
40
+ #header address.watermark a:hover {
41
+ background: transparent;
42
+ color: #aaaaaa; }
43
+
44
+ #content {
45
+ padding-top: 1em;
46
+ padding-bottom: 2em; }
47
+ #content strong {
48
+ font-weight: bold; }
49
+ #content em {
50
+ font-style: italic; }
51
+ #content h1, #content h2, #content h3, #content h4, #content h5, #content h6 {
52
+ color: #4e4e4e; }
53
+ #content h1 {
54
+ font-size: 2em;
55
+ font-weight: bold;
56
+ margin-bottom: .75em;
57
+ padding: .25em 0;
58
+ line-height: 1.2;
59
+ border-bottom: 1px solid #c0c0c0; }
60
+ #content h2 {
61
+ font-weight: bold;
62
+ font-size: 1.5em;
63
+ margin: 1em 0 .2em; }
64
+ #content h3 {
65
+ font-weight: bold;
66
+ font-size: 1.25em;
67
+ margin: .25em 0; }
68
+ #content h4, #content h5, #content h6 {
69
+ font-weight: bold;
70
+ margin-top: .5em; }
71
+ #content code, #content pre, #content textarea, #content input {
72
+ font-family: Monaco, Deja Vu Sans Mono, Inconsolata, Consolas, monospace; }
73
+ #content form p {
74
+ margin-top: 1em;
75
+ position: relative; }
76
+ #content form p.checkbox label {
77
+ margin-top: 0 !important; }
78
+ #content form input.text, #content form textarea {
79
+ width: 30em;
80
+ padding: .2em .4em;
81
+ color: #4e4e4e; }
82
+ #content form input.text {
83
+ height: 1.4em; }
84
+ #content form label {
85
+ float: left;
86
+ display: block;
87
+ margin-top: .5em;
88
+ width: 8em;
89
+ margin-right: .75em; }
90
+ #content form .with_errors label {
91
+ background: red;
92
+ color: white;
93
+ position: relative;
94
+ top: -.7em; }
95
+ #content form .with_errors.required label {
96
+ position: static;
97
+ margin-right: .25em;
98
+ padding: 0 .2em; }
99
+ #content form .with_errors input, #content form .with_errors textarea {
100
+ border: 2px solid #f22;
101
+ background: #fee;
102
+ color: #222222; }
103
+ #content form .required label {
104
+ float: none;
105
+ display: block;
106
+ width: auto;
107
+ position: relative;
108
+ font-weight: bold;
109
+ margin-top: 1em;
110
+ text-indent: -.65em; }
111
+ #content form .required label:before {
112
+ content: "* ";
113
+ color: #ed1556; }
114
+ #content form .required input.text {
115
+ width: 25.6em;
116
+ font-size: 24px;
117
+ font-weight: bold; }
118
+ #content form .normal {
119
+ margin-top: 2em; }
120
+ #content form h2.notifier label {
121
+ float: none;
122
+ width: auto;
123
+ margin-right: 0; }
124
+ #content form h2.notifier label .warning {
125
+ font-size: .5em;
126
+ font-weight: normal;
127
+ color: #999999; }
128
+ #content form fieldset {
129
+ padding-bottom: 1em;
130
+ margin-left: 1.35em;
131
+ border-bottom: 1px solid #c0c0c0;
132
+ margin-bottom: 1em; }
133
+ #content form fieldset h3 {
134
+ margin-top: 1em;
135
+ margin-bottom: 0; }
136
+ #content form fieldset p.normal {
137
+ margin-top: 1em; }
138
+ #content form fieldset p label {
139
+ width: 6.7em; }
140
+ #content form p.submit {
141
+ margin-top: 2em; }
142
+ #content form p.submit:after {
143
+ display: block;
144
+ clear: both;
145
+ float: none;
146
+ content: ".";
147
+ text-indent: -9999em;
148
+ text-align: left; }
149
+ #content form p.submit.destroy button, #content form p.submit.manual-build button {
150
+ float: none;
151
+ display: inline; }
152
+ #content form p.submit.manual-build button {
153
+ margin-right: 0; }
154
+ #content #build form, #content #last_build form {
155
+ font-size: .75em; }
156
+ #content #build form p.submit, #content #last_build form p.submit {
157
+ margin: 0;
158
+ padding: 0;
159
+ position: absolute;
160
+ right: .5em;
161
+ top: 1.25em; }
162
+ #content .blank_slate p, #content .error p {
163
+ position: relative;
164
+ top: .3em; }
165
+ #content .blank_slate h1, #content .error h1 {
166
+ border-width: 0;
167
+ margin: 0;
168
+ padding: 0; }
169
+ #content .blank_slate h1 button, #content .error h1 button {
170
+ float: none;
171
+ border: 0 none;
172
+ background: transparent;
173
+ display: inline;
174
+ color: #ed1556;
175
+ padding: 0.25em 0;
176
+ margin: 0; }
177
+ #content .blank_slate h1 button:hover, #content .error h1 button:hover {
178
+ background: #ed1556;
179
+ color: #ffffff; }
180
+ #content .error dt {
181
+ margin-top: 1.4em;
182
+ margin-bottom: .3em;
183
+ font-size: 1.75em;
184
+ font-family: Georgia, Times New Roman, serif; }
185
+ #content .error dd {
186
+ line-height: 1.4; }
187
+ #content .error .backtrace {
188
+ margin: 1em 0;
189
+ overflow: scroll;
190
+ height: 30em;
191
+ border: 1px solid #c0c0c0;
192
+ line-height: 1.6; }
193
+ #content #projects {
194
+ margin: 1em 0 2em;
195
+ border-top: 1px solid #c0c0c0; }
196
+ #content #projects li {
197
+ position: relative;
198
+ border-bottom: 1px solid #c0c0c0; }
199
+ #content #projects li.odd {
200
+ background: #e6e6e6; }
201
+ #content #projects li.building {
202
+ background: transparent url(/spinner.gif) no-repeat scroll right; }
203
+ #content #projects li a {
204
+ font-size: 2em;
205
+ padding: .25em;
206
+ line-height: 1.2;
207
+ font-weight: bold;
208
+ display: block; }
209
+ #content #projects li a.success {
210
+ color: #337022; }
211
+ #content #projects li a.failed {
212
+ color: #ff1100; }
213
+ #content #projects li .meta {
214
+ position: absolute;
215
+ right: .6em;
216
+ top: 1.5em;
217
+ font-size: 0.8em;
218
+ color: #999999;
219
+ text-align: right; }
220
+ #content #projects li.building .meta {
221
+ right: 1.6em; }
222
+ #content #projects li.success .meta {
223
+ color: #337022; }
224
+ #content #projects li.failed .meta {
225
+ color: #ff1100; }
226
+ #content #previous_builds li a {
227
+ display: block;
228
+ padding: .25em;
229
+ margin-bottom: .25em;
230
+ border-width: 1px;
231
+ border-style: solid; }
232
+ #content #previous_builds li a strong {
233
+ font-size: 1.3em; }
234
+ #content #previous_builds li a .attribution {
235
+ font-size: .9em; }
236
+ #content #projects li.success a, #content #previous_builds li.success a {
237
+ background-color: #bbf8aa;
238
+ border-color: #99d688;
239
+ color: #337022; }
240
+ #content #projects li.success a .attribution, #content #previous_builds li.success a .attribution {
241
+ color: #77b466; }
242
+ #content #projects li.success a:hover, #content #previous_builds li.success a:hover {
243
+ background-color: #ddffcc; }
244
+ #content #projects li.failed a, #content #previous_builds li.failed a {
245
+ background-color: #ffbbaa;
246
+ border-color: #dd9988;
247
+ color: #ff1100; }
248
+ #content #projects li.failed a .attribution, #content #previous_builds li.failed a .attribution {
249
+ color: #bb7766; }
250
+ #content #projects li.failed a:hover, #content #previous_builds li.failed a:hover {
251
+ background-color: #ffddcc; }
252
+ #content #build, #content #last_build {
253
+ position: relative; }
254
+ #content #build h1, #content #build blockquote, #content #last_build h1, #content #last_build blockquote {
255
+ border-width: 0 1px;
256
+ border-style: solid; }
257
+ #content #build h1, #content #last_build h1 {
258
+ border-top-width: 1px; }
259
+ #content #build blockquote, #content #last_build blockquote {
260
+ bottom-bottom-width: 1px;
261
+ line-height: 1.4; }
262
+ #content #build.success h1, #content #build.success blockquote, #content #last_build.success h1, #content #last_build.success blockquote {
263
+ background-color: #bbf8aa;
264
+ border-color: #99d688 #ccffbb #ccffbb #99d688; }
265
+ #content #build.success h1, #content #last_build.success h1 {
266
+ color: #337022; }
267
+ #content #build.success .meta, #content #last_build.success .meta {
268
+ color: #77b466; }
269
+ #content #build.failed h1, #content #build.failed blockquote, #content #last_build.failed h1, #content #last_build.failed blockquote {
270
+ background-color: #ffbbaa;
271
+ border-color: #dd9988 #ffccbb #ffccbb #dd9988; }
272
+ #content #build.failed h1, #content #last_build.failed h1 {
273
+ color: #ff1100; }
274
+ #content #build.failed .meta, #content #last_build.failed .meta {
275
+ color: #bb7766; }
276
+ #content #build h1, #content #last_build h1 {
277
+ margin-top: .5em;
278
+ margin-bottom: 0;
279
+ padding: .25em;
280
+ color: #337022; }
281
+ #content #build blockquote, #content #last_build blockquote {
282
+ padding: .75em;
283
+ margin-bottom: 2em; }
284
+ #content #build blockquote .meta, #content #last_build blockquote .meta {
285
+ margin-top: 1em;
286
+ display: block;
287
+ font-size: .9em; }
288
+ #content #build pre.output, #content #last_build pre.output {
289
+ background: #111;
290
+ color: #fff;
291
+ padding: .5em;
292
+ overflow: auto;
293
+ max-height: 50em;
294
+ font-size: .825em; }
295
+ #content #build pre.output .color30, #content #last_build pre.output .color30 {
296
+ color: #333; }
297
+ #content #build pre.output .color31, #content #last_build pre.output .color31 {
298
+ color: #e33; }
299
+ #content #build pre.output .color32, #content #last_build pre.output .color32 {
300
+ color: #3e3; }
301
+ #content #build pre.output .color33, #content #last_build pre.output .color33 {
302
+ color: #ee3; }
303
+ #content #build pre.output .color34, #content #last_build pre.output .color34 {
304
+ color: #33e; }
305
+ #content #build pre.output .color35, #content #last_build pre.output .color35 {
306
+ color: #e3e; }
307
+ #content #build pre.output .color36, #content #last_build pre.output .color36 {
308
+ color: #3ee; }
309
+ #content #build pre.output .color37, #content #last_build pre.output .color37 {
310
+ color: #fff; }
311
+ #content #push_path {
312
+ display: block;
313
+ margin-top: 1em;
314
+ margin-left: 2em; }
315
+
316
+ a.success {
317
+ color: #bbf8aa; }
318
+ a.success:hover {
319
+ background-color: #bbf8aa;
320
+ color: white; }
321
+ a.failed {
322
+ color: #ffbbaa; }
323
+ a.failed:hover {
324
+ background-color: #ffbbaa;
325
+ color: white; }
326
+
327
+ #footer {
328
+ padding: 1.5em 2.5em;
329
+ border-top: 1px solid #ccc;
330
+ font-size: .8em;
331
+ color: #666;
332
+ text-align: right; }
333
+ #footer strong {
334
+ font-weight: bold; }
335
+