rack-webprofiler 0.1.0.pre.beta2 → 0.1.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (38) hide show
  1. checksums.yaml +5 -13
  2. data/CHANGELOG.md +36 -12
  3. data/README.md +16 -12
  4. data/docs/DSL.md +152 -0
  5. data/docs/GettingStarted.md +51 -0
  6. data/docs/README.md +6 -0
  7. data/lib/rack/templates/assets/css/profiler.css +1 -1
  8. data/lib/rack/templates/assets/css/rwpt.css +1 -1
  9. data/lib/rack/templates/assets/sass/_variables.scss +21 -2
  10. data/lib/rack/templates/assets/sass/profiler.scss +73 -61
  11. data/lib/rack/templates/assets/sass/rwpt.scss +2 -2
  12. data/lib/rack/templates/panel/show.erb +4 -4
  13. data/lib/rack/templates/profiler.erb +4 -0
  14. data/lib/rack/web_profiler/collector.rb +197 -125
  15. data/lib/rack/web_profiler/collectors/rack_collector.rb +74 -0
  16. data/lib/rack/web_profiler/{collector/rack → collectors}/request_collector.rb +43 -60
  17. data/lib/rack/web_profiler/{collector → collectors}/ruby_collector.rb +4 -3
  18. data/lib/rack/web_profiler/{collector → collectors}/time_collector.rb +12 -6
  19. data/lib/rack/web_profiler/collectors.rb +21 -27
  20. data/lib/rack/web_profiler/config.rb +4 -9
  21. data/lib/rack/web_profiler/controller.rb +131 -126
  22. data/lib/rack/web_profiler/engine.rb +10 -14
  23. data/lib/rack/web_profiler/model.rb +74 -23
  24. data/lib/rack/web_profiler/request.rb +22 -7
  25. data/lib/rack/web_profiler/response.rb +27 -0
  26. data/lib/rack/web_profiler/rouge/html_formatter.rb +25 -0
  27. data/lib/rack/web_profiler/router.rb +11 -6
  28. data/lib/rack/web_profiler/version.rb +1 -1
  29. data/lib/rack/web_profiler/view.rb +255 -139
  30. data/lib/rack/web_profiler.rb +47 -4
  31. data/rack-webprofiler.gemspec +1 -3
  32. metadata +32 -32
  33. data/lib/rack/web_profiler/collector/debug_collector.rb +0 -31
  34. data/lib/rack/web_profiler/collector/erb_collector.rb +0 -0
  35. data/lib/rack/web_profiler/collector/performance_collector.rb +0 -1
  36. data/lib/rack/web_profiler/collector/rack/rack_collector.rb +0 -23
  37. data/lib/rack/web_profiler/collector/view.rb +0 -44
  38. data/lib/rack/web_profiler/model/collection_record.rb +0 -46
@@ -217,7 +217,8 @@ body {
217
217
  -webkit-user-select: none;
218
218
  user-select: none;
219
219
 
220
- &:hover {
220
+ &:hover,
221
+ &.active {
221
222
  background-color: $profiler_panel__sidebar_item_hover;
222
223
  }
223
224
  &.sidebar__item--blank {
@@ -263,7 +264,6 @@ body {
263
264
  width: 100%;
264
265
  min-height: calc(100% - 80px);
265
266
  padding: 20px;
266
- margin-bottom: 50px;
267
267
  }
268
268
  .wrapper__container {
269
269
  box-sizing: border-box;
@@ -7,13 +7,13 @@
7
7
  <% content_for :sidebar do %>
8
8
  <% @collectors.each do |_k, collector| %>
9
9
  <% if collector_has_panel?(collector, @collection) %>
10
- <div class="sidebar__item">
10
+ <div class="sidebar__item<%= collector == @collector ? " active" : nil %>">
11
11
  <div class="sidebar__item-inner">
12
- <a href="<%=h WebProfiler::Router.url_for_profiler(@collection.token, collector.name) %>">
12
+ <a href="<%=h WebProfiler::Router.url_for_profiler(@collection.token, collector.identifier) %>">
13
13
  <i class="icn">
14
14
  <% unless collector.icon.nil? %><img alt="" src="<%=h collector.icon %>" /><% end %>
15
15
  </i>
16
- <span><%=h collector.name.capitalize.gsub(/_/, ' ') %></span>
16
+ <span><%=h collector.label %></span>
17
17
  </a>
18
18
  </div>
19
19
  </div>
@@ -21,6 +21,6 @@
21
21
  <% end %>
22
22
  <% end %>
23
23
 
24
- <div class="rack-webprofiler_collectors_<%=h @collector.name.to_s %>">
24
+ <div class="rack-webprofiler_collectors_<%=h @collector.identifier.to_s %>">
25
25
  <%= collector_panel(@collector, @collection) %>
26
26
  </div>
@@ -9,6 +9,8 @@
9
9
  <div class="rack-webprofiler__item rack-webprofiler_collectors_<%=h name.to_s %>">
10
10
  <% if collector_has_panel?(collector, @collection) %>
11
11
  <a href="<%=h WebProfiler::Router.url_for_profiler(@collection.token, name) %>">
12
+ <% else %>
13
+ <span class="rack-webprofiler__item-content">
12
14
  <% end %>
13
15
  <% unless collector.icon.nil? %>
14
16
  <i class="icn"><img alt="" src="<%=h collector.icon %>" /></i>
@@ -19,6 +21,8 @@
19
21
  <span class="rack-webprofiler__item__status-color <%=h status.to_s %>"></span>
20
22
  <% if collector_has_panel?(collector, @collection) %>
21
23
  </a>
24
+ <% else %>
25
+ </span>
22
26
  <% end %>
23
27
  </div>
24
28
  <% end %>
@@ -1,155 +1,227 @@
1
1
  require "docile"
2
2
 
3
3
  module Rack
4
+ class WebProfiler
5
+ # Collector
6
+ class Collector
7
+ # DSL
8
+ module DSL
9
+ def self.included(base)
10
+ base.extend(ClassMethods)
11
+ base.class_eval do
12
+ @definition = Definition.new
13
+ @definition.position = 1
14
+ @definition.is_enabled = true
15
+ @definition.klass = self
16
+ end
17
+ end
4
18
 
5
- # Collector
6
- class WebProfiler::Collector
7
- autoload :RubyCollector, "rack/web_profiler/collector/ruby_collector"
8
- autoload :TimeCollector, "rack/web_profiler/collector/time_collector"
9
- autoload :View, "rack/web_profiler/collector/view"
19
+ module ClassMethods
20
+ attr_reader :definition
21
+
22
+ # Set the icon of the {Rack::WebProfiler::Collector}.
23
+ #
24
+ # @param icon [String, nil]
25
+ def icon(icon = nil); definition.icon = icon; end
26
+
27
+ # Set the identifier of the {Rack::WebProfiler::Collector}.
28
+ #
29
+ # @param identifier [String, nil]
30
+ def identifier(identifier = nil); definition.identifier = identifier; end
31
+
32
+ # Set the label of the {Rack::WebProfiler::Collector}.
33
+ #
34
+ # @param label [String, nil]
35
+ def label(label = nil); definition.label = label; end
36
+
37
+ #
38
+ def position(position = nil); definition.position = position.to_i; end
39
+
40
+ #
41
+ def collect(&block); definition.collect = block; end
42
+
43
+ # Set the template of the {Rack::WebProfiler::Collector}.
44
+ #
45
+ # @param template [String, nil]
46
+ # @option type [Symbol] :file or :DATA
47
+ def template(template = nil, type: :file)
48
+ template = get_data_contents(template) if type == :DATA
49
+ definition.template = template
50
+ end
10
51
 
11
- module Rack
12
- autoload :RackCollector, "rack/web_profiler/collector/rack/rack_collector"
13
- autoload :RequestCollector, "rack/web_profiler/collector/rack/request_collector"
14
- end
52
+ # Tell if the {Rack::WebProfiler::Collector} is enabled.
53
+ #
54
+ # @param is_enabled [Boolean, Block]
55
+ def is_enabled?(is_enabled = true)
56
+ definition.is_enabled = Proc.new if block_given?
57
+ definition.is_enabled = is_enabled
58
+ end
15
59
 
16
- # DSL
17
- module DSL
18
- def self.included(base)
19
- base.extend(ClassMethods)
20
- base.class_eval do
21
- @definition = Definition.new
22
- @definition.position = 1
23
- @definition.is_enabled = true
24
- @definition.klass = self
60
+ private
61
+
62
+ def get_data_contents(path)
63
+ data = ""
64
+ ::File.open(path, "rb") do |f|
65
+ begin
66
+ line = f.gets
67
+ end until line.nil? || /^__END__$/ === line
68
+ data << line while line = f.gets
69
+ end
70
+ data
71
+ end
25
72
  end
26
73
  end
27
74
 
28
- module ClassMethods
29
- attr_reader :definition
75
+ # Definition
76
+ #
77
+ # Collector definition.
78
+ class Definition
79
+ attr_accessor :icon, :identifier, :label, :position, :collect, :template, :is_enabled, :klass
80
+ attr_reader :data_storage
81
+
82
+ # Collect the data who the Collector need.
83
+ #
84
+ # @param request [Rack::WebProfiler::Request]
85
+ # @param response [Rack::Response]
86
+ #
87
+ # @return [Rack::WebProfiler::Collector::DSL::DataStorage]
88
+ def collect!(request, response)
89
+ @data_storage = Docile.dsl_eval(DataStorage.new, request, response, &collect)
90
+ end
30
91
 
31
- def icon(icon = nil); definition.icon = icon; end
32
- def collector_name(name = nil); definition.name = name; end
33
- def position(position = nil); definition.position = position.to_i; end
34
- def collect(&block); definition.collect = block; end
92
+ # Is the collector enabled.
93
+ #
94
+ # @return [Boolean]
95
+ def is_enabled?
96
+ return !!@is_enabled.call if @is_enabled.is_a?(Proc)
97
+ !!@is_enabled
98
+ end
99
+ end
35
100
 
36
- def template(template = nil, type: :file)
37
- template = get_data_contents(template) if type == :DATA
38
- definition.template = template
101
+ # DataStorage
102
+ #
103
+ # Used to store datas who Collector needs.
104
+ #
105
+ # @todo do DataStorage compatible with Marshal
106
+ class DataStorage
107
+ attr_reader :datas
108
+
109
+ def initialize
110
+ @datas = Hash.new
111
+ @status = nil
112
+ @show_tab = true
113
+ @show_panel = true
39
114
  end
40
115
 
41
- def is_enabled?(is_enabled = true)
42
- definition.is_enabled = Proc.new if block_given?
43
- definition.is_enabled = is_enabled
116
+ # Store a value.
117
+ #
118
+ # @param k [String, Symbol]
119
+ # @param v
120
+ def store(k, v)
121
+ # @todo check data format (must be compatible with Marshal)
122
+ @datas[k.to_sym] = v
44
123
  end
45
124
 
46
- private
125
+ # Status.
126
+ #
127
+ # @param v [Symbol, nil]
128
+ #
129
+ # @return [Symbol, nil]
130
+ def status(v = nil)
131
+ # @todo check status?
132
+ # raise Exception, "" unless [:success, :warning, :danger].include?(v)
133
+ @status = v.to_sym unless v.nil?
134
+ @status
135
+ end
47
136
 
48
- def get_data_contents(path)
49
- data = ""
50
- ::File.open(path, "rb") do |f|
51
- begin
52
- line = f.gets
53
- end until line.nil? || /^__END__$/ === line
54
- data << line while line = f.gets
55
- end
56
- data
137
+ #
138
+ #
139
+ # @param b [Boolean, nil]
140
+ #
141
+ # @return [Boolean]
142
+ def show_panel(b = nil)
143
+ @show_panel = !!b unless b.nil?
144
+ @show_panel
57
145
  end
58
- end
59
- end
60
146
 
61
- # Definition
62
- #
63
- # Collector definition.
64
- class Definition
65
- attr_accessor :icon, :name, :position, :collect, :template, :is_enabled, :klass
66
- attr_reader :data_storage
147
+ #
148
+ #
149
+ # @param b [Boolean, nil]
150
+ #
151
+ # @return [Boolean]
152
+ def show_tab(b = nil)
153
+ @show_tab = !!b unless b.nil?
154
+ @show_tab
155
+ end
67
156
 
68
- # Collect the data who the Collector need.
69
- #
70
- # @param request [Rack::WebProfiler::Request]
71
- # @param response [Rack::Response]
72
- #
73
- # @return [Rack::WebProfiler::Collector::DSL::DataStorage]
74
- def collect!(request, response)
75
- @data_storage = Docile.dsl_eval(DataStorage.new, request, response, &collect)
157
+ # Transform DataStorage to an Hash
158
+ #
159
+ # @return [Hash<Symbol, Object>]
160
+ def to_h
161
+ {
162
+ datas: @datas,
163
+ status: @status,
164
+ show_panel: @show_panel,
165
+ show_tab: @show_tab,
166
+ }
167
+ end
76
168
  end
77
169
 
78
- # Is the collector enabled.
79
- #
80
- # @return [Boolean]
81
- def is_enabled?
82
- return !!@is_enabled.call if @is_enabled.is_a?(Proc)
83
- !!@is_enabled
84
- end
85
- end
170
+ # View
171
+ class View < WebProfiler::View
172
+ #
173
+ def context
174
+ @context ||= Context.new
175
+ end
86
176
 
87
- # DataStorage
88
- #
89
- # Used to store datas who Collector needs.
90
- #
91
- # @todo do DataStorage compatible with Marshal
92
- class DataStorage
93
- attr_reader :datas
94
-
95
- def initialize
96
- @datas = Hash.new
97
- @status = nil
98
- @show_tab = true
99
- @show_panel = true
100
- end
177
+ # Read a template. Returns file content if template is a file path.
178
+ #
179
+ # @param template [String] template file path or content
180
+ #
181
+ # @return [String]
182
+ def read_template(template)
183
+ unless template.empty?
184
+ return ::File.read(template) if ::File.exist?(template)
185
+ end
186
+ template
187
+ end
101
188
 
102
- # Store a value.
103
- #
104
- # @param k [String, Symbol]
105
- # @param v
106
- def store(k, v)
107
- # @todo check data format (must be compatible with Marshal)
108
- @datas[k.to_sym] = v
109
- end
189
+ # Helpers
190
+ module Helpers
191
+ # @todo comment
192
+ def tab_content
193
+ if block_given?
194
+ @tab_content ||= capture(&Proc.new)
195
+ elsif !@tab_content.nil?
196
+ @tab_content
197
+ end
198
+ end
110
199
 
111
- # Status.
112
- #
113
- # @param v [Symbol, nil]
114
- #
115
- # @return [Symbol, nil]
116
- def status(v = nil)
117
- # @todo check status?
118
- # raise Exception, "" unless [:success, :warning, :error].include?(v)
119
- @status = v.to_sym unless v.nil?
120
- @status
121
- end
200
+ # @todo comment
201
+ def panel_content
202
+ if block_given?
203
+ @panel_content ||= capture(&Proc.new)
204
+ elsif !@panel_content.nil?
205
+ @panel_content
206
+ end
207
+ end
122
208
 
123
- #
124
- #
125
- # @param b [Boolean, nil]
126
- #
127
- # @return [Boolean]
128
- def show_panel(b = nil)
129
- @show_panel = !!b unless b.nil?
130
- @show_panel
131
- end
209
+ # @todo comment
210
+ def data(k, default = nil)
211
+ return nil if @collection.nil?
132
212
 
133
- #
134
- #
135
- # @param b [Boolean, nil]
136
- #
137
- # @return [Boolean]
138
- def show_tab(b = nil)
139
- @show_tab = !!b unless b.nil?
140
- @show_tab
141
- end
213
+ datas = @collection.datas[@collector.identifier.to_sym][:datas]
214
+ return datas[k] if datas.has_key?(k)
142
215
 
143
- # Transform DataStorage to an Hash
144
- #
145
- # @return [Hash<Symbol, Object>]
146
- def to_h
147
- {
148
- datas: @datas,
149
- status: @status,
150
- show_panel: @show_panel,
151
- show_tab: @show_tab,
152
- }
216
+ default
217
+ end
218
+ end
219
+
220
+ # @todo comment
221
+ class Context
222
+ include WebProfiler::View::Helpers::Common
223
+ include Helpers
224
+ end
153
225
  end
154
226
  end
155
227
  end
@@ -0,0 +1,74 @@
1
+ module Rack
2
+ class WebProfiler::Collectors::RackCollector
3
+ include Rack::WebProfiler::Collector::DSL
4
+
5
+ icon <<-'ICON'
6
+ 
7
+ ICON
8
+
9
+ identifier "rack"
10
+ label "Rack"
11
+ position 1
12
+
13
+ collect do |request, _response|
14
+ store :rack_version, Rack.release
15
+ store :rack_env, hash_stringify_values(request.env)
16
+ end
17
+
18
+ template __FILE__, type: :DATA
19
+
20
+ class << self
21
+ def hash_stringify_values(hash)
22
+ return {} unless hash.kind_of?(Hash)
23
+ hash.collect do |k,v|
24
+ v = v.inspect unless v.kind_of?(String)
25
+ [k, v]
26
+ end
27
+ end
28
+ end
29
+ end
30
+ end
31
+
32
+ __END__
33
+ <% tab_content do %>
34
+ <%=h data(:rack_version) %>
35
+ <% end %>
36
+
37
+ <% panel_content do %>
38
+ <div class="block">
39
+ <h3>Rack informations</h3>
40
+
41
+ <table>
42
+ <tbody>
43
+ <tr>
44
+ <th>Version</th>
45
+ <td><%=h data(:rack_version) %></td>
46
+ </tr>
47
+ </tbody>
48
+ </table>
49
+ </div>
50
+
51
+ <div class="block">
52
+ <h3>Env</h3>
53
+ <% if data(:rack_env) && !data(:rack_env).empty? %>
54
+ <table>
55
+ <thead>
56
+ <tr>
57
+ <th>Key</th>
58
+ <th>Value</th>
59
+ </tr>
60
+ <thead>
61
+ <tbody>
62
+ <% data(:rack_env).sort.each do |k, v| %>
63
+ <tr>
64
+ <th><%=h k %></th>
65
+ <td class="code"><%=h v %></td>
66
+ </tr>
67
+ <% end %>
68
+ </tbody>
69
+ </table>
70
+ <% else %>
71
+ <p><span class="text__no-value">No rack env datas</span></p>
72
+ <% end %>
73
+ </div>
74
+ <% end %>
@@ -1,66 +1,52 @@
1
1
  module Rack
2
- class WebProfiler::Collector::Rack::RequestCollector
2
+ class WebProfiler::Collectors::RequestCollector
3
3
  include Rack::WebProfiler::Collector::DSL
4
4
 
5
5
  icon <<-'ICON'
6
6
  
7
7
  ICON
8
8
 
9
- collector_name "rack_request"
10
- # collector_key "rack_request"
11
- # collector_name "Request"
12
- position 2
9
+ identifier "rack.request"
10
+ label "Request"
11
+ position 2
13
12
 
14
13
  collect do |request, response|
15
- store :request_headers, get_request_headers(request)
16
- store :request_path, request.path
17
- store :request_method, request.request_method
18
- store :request_cookies, request.cookies
19
- store :request_get, request.GET
20
- store :request_post, get_request_post(request)
21
- store :request_session, hash_to_s(request.session)
22
- store :request_cookies, request.cookies
23
- store :request_body, get_request_body(request)
24
-
25
- store :rack_env, hash_to_s(request.env)
14
+ store :request_headers, request.http_headers
15
+ store :request_fullpath, request.fullpath
16
+ store :request_method, request.request_method
17
+ store :request_cookies, request.cookies
18
+ store :request_get, request.GET
19
+ store :request_post, request_post(request)
20
+ store :request_session, hash_stringify_values(request.session)
21
+ store :request_cookies, request.cookies
22
+ store :request_body, request.body_string
23
+ store :request_mediatype, request.media_type
24
+ store :request_raw, request.raw
26
25
 
27
26
  store :response_status, response.status
28
27
  store :response_headers, response.headers
28
+ store :response_raw, response.raw
29
29
 
30
30
  if response.successful?
31
31
  status :success
32
32
  elsif response.redirection?
33
33
  status :warning
34
34
  else
35
- status :error
35
+ status :danger
36
36
  end
37
37
  end
38
38
 
39
39
  template __FILE__, type: :DATA
40
40
 
41
41
  class << self
42
- # Get request headers.
43
- #
44
- # @return [Hash]
45
- def get_request_headers(request)
46
- request.env.select {|k,v| k.start_with? 'HTTP_'}
47
- .collect {|k,v| [k.sub(/^HTTP_/, ''), v]}
48
- .collect {|k,v| [k.split('_').collect(&:capitalize).join('-'), v]}
49
- end
50
-
51
- def get_request_post(request)
42
+ def request_post(request)
52
43
  request.POST if request.POST && !request.POST.empty?
53
44
  rescue Exception
54
45
  nil
55
46
  end
56
47
 
57
- def get_request_body(request)
58
- return nil if request.body.nil? || request.get?
59
- body = request.body.dup
60
- body.read
61
- end
62
-
63
- def hash_to_s(hash)
48
+ def hash_stringify_values(hash)
49
+ return {} unless hash.kind_of?(Hash)
64
50
  hash.collect {|k,v| [k, v.to_s]}
65
51
  end
66
52
  end
@@ -70,6 +56,17 @@ end
70
56
  __END__
71
57
  <% tab_content do %>
72
58
  <%=h data(:response_status) %>
59
+
60
+ <div class="details">
61
+ <div class="wrapper">
62
+ <dl>
63
+ <dt>Status</dt>
64
+ <dd><%=h data(:response_status) %> - <%=h Rack::Utils::HTTP_STATUS_CODES[data(:response_status).to_i] %></dd>
65
+ <dt>Path</dt>
66
+ <dd><%=h data(:request_method) %> <%=h data(:request_fullpath) %></dd>
67
+ </dl>
68
+ </div>
69
+ </div>
73
70
  <% end %>
74
71
 
75
72
  <% panel_content do %>
@@ -147,12 +144,22 @@ __END__
147
144
 
148
145
  <div class="block">
149
146
  <h3>Request content</h3>
150
- <% unless data(:request_body).nil? %>
151
- <%=highlight code: data(:request_body) %>
147
+ <% unless data(:request_body).empty? %>
148
+ <%=highlight mimetype: data(:request_mediatype), code: data(:request_body) %>
152
149
  <% else %>
153
150
  <p><span class="text__no-value">No request content</span></p>
154
151
  <% end %>
155
152
  </div>
153
+ </div>
154
+
155
+ <div class="block">
156
+ <h3>Request raw</h3>
157
+ <% unless data(:request_raw).nil? %>
158
+ <%=highlight language: :http, code: data(:request_raw) %>
159
+ <% else %>
160
+ <p><span class="text__no-value">No request raw</span></p>
161
+ <% end %>
162
+ </div>
156
163
 
157
164
  <div class="block">
158
165
  <h3>Response headers</h3>
@@ -225,28 +232,4 @@ __END__
225
232
  <p><span class="text__no-value">No cookies data</span></p>
226
233
  <% end %>
227
234
  </div>
228
-
229
- <div class="block">
230
- <h3>Env</h3>
231
- <% if data(:rack_env) && !data(:rack_env).empty? %>
232
- <table>
233
- <thead>
234
- <tr>
235
- <th>Key</th>
236
- <th>Value</th>
237
- </tr>
238
- <thead>
239
- <tbody>
240
- <% data(:rack_env).sort.each do |k, v| %>
241
- <tr>
242
- <th><%=h k %></th>
243
- <td class="code"><%=h v %></td>
244
- </tr>
245
- <% end %>
246
- </tbody>
247
- </table>
248
- <% else %>
249
- <p><span class="text__no-value">No rack env datas</span></p>
250
- <% end %>
251
- </div>
252
235
  <% end %>
@@ -1,13 +1,14 @@
1
1
  module Rack
2
- class WebProfiler::Collector::RubyCollector
2
+ class WebProfiler::Collectors::RubyCollector
3
3
  include Rack::WebProfiler::Collector::DSL
4
4
 
5
5
  icon <<-'ICON'
6
6
  
7
7
  ICON
8
8
 
9
- collector_name "ruby"
10
- position 0
9
+ identifier "ruby"
10
+ label "Ruby"
11
+ position 0
11
12
 
12
13
  collect do |_request, _response|
13
14
  store :ruby_version, RUBY_VERSION