sql-jarvis 2.1.1 → 2.1.2

Sign up to get free protection for your applications and to get access to all the features.
@@ -0,0 +1,184 @@
1
+ define("ace/theme/ambiance",["require","exports","module","ace/lib/dom"], function(require, exports, module) {
2
+
3
+ exports.isDark = true;
4
+ exports.cssClass = "ace-ambiance";
5
+ exports.cssText = ".ace-ambiance .ace_gutter {\
6
+ background-color: #3d3d3d;\
7
+ background-image: linear-gradient(left, #3D3D3D, #333);\
8
+ background-repeat: repeat-x;\
9
+ border-right: 1px solid #4d4d4d;\
10
+ text-shadow: 0px 1px 1px #4d4d4d;\
11
+ color: #222;\
12
+ }\
13
+ .ace-ambiance .ace_gutter-layer {\
14
+ background: repeat left top;\
15
+ }\
16
+ .ace-ambiance .ace_gutter-active-line {\
17
+ background-color: #3F3F3F;\
18
+ }\
19
+ .ace-ambiance .ace_fold-widget {\
20
+ text-align: center;\
21
+ }\
22
+ .ace-ambiance .ace_fold-widget:hover {\
23
+ color: #777;\
24
+ }\
25
+ .ace-ambiance .ace_fold-widget.ace_start,\
26
+ .ace-ambiance .ace_fold-widget.ace_end,\
27
+ .ace-ambiance .ace_fold-widget.ace_closed{\
28
+ background: none;\
29
+ border: none;\
30
+ box-shadow: none;\
31
+ }\
32
+ .ace-ambiance .ace_fold-widget.ace_start:after {\
33
+ content: '▾'\
34
+ }\
35
+ .ace-ambiance .ace_fold-widget.ace_end:after {\
36
+ content: '▴'\
37
+ }\
38
+ .ace-ambiance .ace_fold-widget.ace_closed:after {\
39
+ content: '‣'\
40
+ }\
41
+ .ace-ambiance .ace_print-margin {\
42
+ border-left: 1px dotted #2D2D2D;\
43
+ right: 0;\
44
+ background: #262626;\
45
+ }\
46
+ .ace-ambiance .ace_scroller {\
47
+ -webkit-box-shadow: inset 0 0 10px black;\
48
+ -moz-box-shadow: inset 0 0 10px black;\
49
+ -o-box-shadow: inset 0 0 10px black;\
50
+ box-shadow: inset 0 0 10px black;\
51
+ }\
52
+ .ace-ambiance {\
53
+ color: #E6E1DC;\
54
+ background-color: #202020;\
55
+ }\
56
+ .ace-ambiance .ace_cursor {\
57
+ border-left: 1px solid #7991E8;\
58
+ }\
59
+ .ace-ambiance .ace_overwrite-cursors .ace_cursor {\
60
+ border: 1px solid #FFE300;\
61
+ background: #766B13;\
62
+ }\
63
+ .ace-ambiance.normal-mode .ace_cursor-layer {\
64
+ z-index: 0;\
65
+ }\
66
+ .ace-ambiance .ace_marker-layer .ace_selection {\
67
+ background: rgba(221, 240, 255, 0.20);\
68
+ }\
69
+ .ace-ambiance .ace_marker-layer .ace_selected-word {\
70
+ border-radius: 4px;\
71
+ border: 8px solid #3f475d;\
72
+ box-shadow: 0 0 4px black;\
73
+ }\
74
+ .ace-ambiance .ace_marker-layer .ace_step {\
75
+ background: rgb(198, 219, 174);\
76
+ }\
77
+ .ace-ambiance .ace_marker-layer .ace_bracket {\
78
+ margin: -1px 0 0 -1px;\
79
+ border: 1px solid rgba(255, 255, 255, 0.25);\
80
+ }\
81
+ .ace-ambiance .ace_marker-layer .ace_active-line {\
82
+ background: rgba(255, 255, 255, 0.031);\
83
+ }\
84
+ .ace-ambiance .ace_invisible {\
85
+ color: #333;\
86
+ }\
87
+ .ace-ambiance .ace_paren {\
88
+ color: #24C2C7;\
89
+ }\
90
+ .ace-ambiance .ace_keyword {\
91
+ color: #cda869;\
92
+ }\
93
+ .ace-ambiance .ace_keyword.ace_operator {\
94
+ color: #fa8d6a;\
95
+ }\
96
+ .ace-ambiance .ace_punctuation.ace_operator {\
97
+ color: #fa8d6a;\
98
+ }\
99
+ .ace-ambiance .ace_identifier {\
100
+ }\
101
+ .ace-ambiance .ace-statement {\
102
+ color: #cda869;\
103
+ }\
104
+ .ace-ambiance .ace_constant {\
105
+ color: #CF7EA9;\
106
+ }\
107
+ .ace-ambiance .ace_constant.ace_language {\
108
+ color: #CF7EA9;\
109
+ }\
110
+ .ace-ambiance .ace_constant.ace_library {\
111
+ }\
112
+ .ace-ambiance .ace_constant.ace_numeric {\
113
+ color: #78CF8A;\
114
+ }\
115
+ .ace-ambiance .ace_invalid {\
116
+ text-decoration: underline;\
117
+ }\
118
+ .ace-ambiance .ace_invalid.ace_illegal {\
119
+ color:#F8F8F8;\
120
+ background-color: rgba(86, 45, 86, 0.75);\
121
+ }\
122
+ .ace-ambiance .ace_invalid,\
123
+ .ace-ambiance .ace_deprecated {\
124
+ text-decoration: underline;\
125
+ font-style: italic;\
126
+ color: #D2A8A1;\
127
+ }\
128
+ .ace-ambiance .ace_support {\
129
+ color: #9B859D;\
130
+ }\
131
+ .ace-ambiance .ace_support.ace_function {\
132
+ color: #DAD085;\
133
+ }\
134
+ .ace-ambiance .ace_function.ace_buildin {\
135
+ color: #9b859d;\
136
+ }\
137
+ .ace-ambiance .ace_string {\
138
+ color: #8f9d6a;\
139
+ }\
140
+ .ace-ambiance .ace_string.ace_regexp {\
141
+ color: #DAD085;\
142
+ }\
143
+ .ace-ambiance .ace_comment {\
144
+ font-style: italic;\
145
+ color: #555;\
146
+ }\
147
+ .ace-ambiance .ace_comment.ace_doc {\
148
+ }\
149
+ .ace-ambiance .ace_comment.ace_doc.ace_tag {\
150
+ color: #666;\
151
+ font-style: normal;\
152
+ }\
153
+ .ace-ambiance .ace_definition,\
154
+ .ace-ambiance .ace_type {\
155
+ color: #aac6e3;\
156
+ }\
157
+ .ace-ambiance .ace_variable {\
158
+ color: #9999cc;\
159
+ }\
160
+ .ace-ambiance .ace_variable.ace_language {\
161
+ color: #9b859d;\
162
+ }\
163
+ .ace-ambiance .ace_xml-pe {\
164
+ color: #494949;\
165
+ }\
166
+ .ace-ambiance .ace_gutter-layer,\
167
+ .ace-ambiance .ace_text-layer {\
168
+ background-image: url(\"\");\
169
+ }\
170
+ .ace-ambiance .ace_indent-guide {\
171
+ background: url(\"\") right repeat-y;\
172
+ }";
173
+
174
+ var dom = require("../lib/dom");
175
+ dom.importCssString(exports.cssText, exports.cssClass);
176
+
177
+ }); (function() {
178
+ window.require(["ace/theme/ambiance"], function(m) {
179
+ if (typeof module == "object" && typeof exports == "object" && module) {
180
+ module.exports = m;
181
+ }
182
+ });
183
+ })();
184
+
@@ -32,6 +32,8 @@
32
32
  $.fn.stupidtable.dir = {ASC: "asc", DESC: "desc"};
33
33
  $.fn.stupidtable.default_sort_fns = {
34
34
  "int": function(a, b) {
35
+ a = a.replace(',', '');
36
+ b = b.replace(',', '');
35
37
  return parseInt(a, 10) - parseInt(b, 10);
36
38
  },
37
39
  "float": function(a, b) {
@@ -61,7 +61,7 @@ input.search:focus {
61
61
  margin-right: 0;
62
62
  }
63
63
 
64
- #editor-container {
64
+ #editor-container, #integration-editor {
65
65
  background-color: #141414;
66
66
  padding-top: 10px;
67
67
  padding-bottom: 10px;
@@ -175,7 +175,7 @@ input.search:focus {
175
175
  }
176
176
 
177
177
  #code {
178
- max-height: 236px;
178
+ max-height: 80vh;
179
179
  overflow: hidden;
180
180
  }
181
181
 
@@ -65,7 +65,11 @@ module Blazer
65
65
  end
66
66
  value = Blazer.transform_variable.call(var, value) if Blazer.transform_variable
67
67
  if value.is_a?(Array)
68
- var_value = value.map{|v| ActiveRecord::Base.connection.quote(v)}.join(', ')
68
+ if value.include?('select-all')
69
+ var_value = Blazer.data_sources[data_source].select_all_variables[var]
70
+ else
71
+ var_value = value.map{|v| ActiveRecord::Base.connection.quote(v)}.join(', ')
72
+ end
69
73
  else
70
74
  var_value = ActiveRecord::Base.connection.quote(value)
71
75
  end
@@ -76,7 +80,9 @@ module Blazer
76
80
 
77
81
  def parse_smart_variables(var, data_source)
78
82
  smart_var_data_source =
79
- ([data_source] + Array(data_source.settings["inherit_smart_settings"]).map { |ds| Blazer.data_sources[ds] }).find { |ds| ds.smart_variables[var] }
83
+ ([data_source] + Array(data_source.settings["inherit_smart_settings"])
84
+ .map { |ds| Blazer.data_sources[ds] })
85
+ .find { |ds| ds.smart_variables[var] }
80
86
 
81
87
  if smart_var_data_source
82
88
  query = smart_var_data_source.smart_variables[var]
@@ -88,6 +94,9 @@ module Blazer
88
94
  elsif query
89
95
  result = smart_var_data_source.run_statement(query)
90
96
  smart_var = result.rows.map { |v| v.reverse }
97
+ if data_source.select_all_variables.keys.include?(var)
98
+ smart_var = [['All', 'select-all']] + smart_var
99
+ end
91
100
  error = result.error if result.error
92
101
  end
93
102
  end
@@ -2,7 +2,7 @@ module Blazer
2
2
  class QueriesController < BaseController
3
3
  before_action :set_query, only: [:show, :edit, :update, :destroy, :refresh]
4
4
  before_action :set_data_source, only: [:new, :edit, :tables, :docs, :schema, :cancel, :columns]
5
- before_action :set_assignees, only: [:new, :create, :show, :edit, :update, :destroy, :refresh]
5
+ before_action :set_accessible, only: [:new, :create, :show, :edit, :update, :destroy, :refresh]
6
6
 
7
7
  def home
8
8
  set_queries(1000)
@@ -85,6 +85,7 @@ module Blazer
85
85
 
86
86
  def run
87
87
  @statement = params[:statement]
88
+ @integration = params[:integration]
88
89
  data_source = params[:data_source]
89
90
  process_vars(@statement, data_source)
90
91
  @only_chart = params[:only_chart]
@@ -139,9 +140,9 @@ module Blazer
139
140
 
140
141
  if @result
141
142
  @data_source.delete_results(@run_id) if @run_id
142
-
143
- @columns = @result.columns
144
- @rows = @result.rows
143
+ @integration_output = Blazer::RunIntegration.new(@result, @integration).call
144
+ @columns = @integration_output&.dig(:columns) || @result.columns
145
+ @rows = @integration_output&.dig(:rows) || @result.rows
145
146
  @error = @result.error
146
147
  @cached_at = @result.cached_at
147
148
  @just_cached = @result.just_cached
@@ -223,179 +224,189 @@ module Blazer
223
224
 
224
225
  private
225
226
 
226
- def set_data_source
227
- @data_source = Blazer.data_sources[params[:data_source]]
227
+ def set_data_source
228
+ @data_source = Blazer.data_sources[params[:data_source]]
229
+ render_forbidden unless Query.new(data_source: @data_source.id).editable?(blazer_user)
230
+ end
231
+
232
+ def continue_run
233
+ render json: {run_id: @run_id, timestamp: @timestamp}, status: :accepted
234
+ end
228
235
 
229
- unless Query.new(data_source: @data_source.id).editable?(blazer_user)
230
- render_forbidden
236
+ def render_run
237
+ @checks = @query ? @query.checks.order(:id) : []
238
+
239
+ @first_row = @rows.first || []
240
+ @column_types = []
241
+ if @rows.any?
242
+ @columns.each_with_index do |_, i|
243
+ @column_types << (
244
+ case @first_row[i]
245
+ when Integer
246
+ "int"
247
+ when Float, BigDecimal
248
+ "float"
249
+ else
250
+ "string-ins"
251
+ end
252
+ )
231
253
  end
232
254
  end
233
255
 
234
- def continue_run
235
- render json: {run_id: @run_id, timestamp: @timestamp}, status: :accepted
236
- end
256
+ @filename = @query.name.parameterize if @query
257
+ @min_width_types = @columns.each_with_index.select { |c, i| @first_row[i].is_a?(Time) || @first_row[i].is_a?(String) || @data_source.smart_columns[c] }.map(&:last)
237
258
 
238
- def render_run
239
- @checks = @query ? @query.checks.order(:id) : []
240
-
241
- @first_row = @rows.first || []
242
- @column_types = []
243
- if @rows.any?
244
- @columns.each_with_index do |_, i|
245
- @column_types << (
246
- case @first_row[i]
247
- when Integer
248
- "int"
249
- when Float, BigDecimal
250
- "float"
251
- else
252
- "string-ins"
253
- end
254
- )
255
- end
256
- end
259
+ @boom = @result.boom if @result
257
260
 
258
- @filename = @query.name.parameterize if @query
259
- @min_width_types = @columns.each_with_index.select { |c, i| @first_row[i].is_a?(Time) || @first_row[i].is_a?(String) || @data_source.smart_columns[c] }.map(&:last)
260
-
261
- @boom = @result.boom if @result
262
-
263
- @linked_columns = @data_source.linked_columns
264
-
265
- @markers = []
266
- [["latitude", "longitude"], ["lat", "lon"], ["lat", "lng"]].each do |keys|
267
- lat_index = @columns.index(keys.first)
268
- lon_index = @columns.index(keys.last)
269
- if lat_index && lon_index
270
- @markers =
271
- @rows.select do |r|
272
- r[lat_index] && r[lon_index]
273
- end.map do |r|
274
- {
275
- title: r.each_with_index.map{ |v, i| i == lat_index || i == lon_index ? nil : "<strong>#{@columns[i]}:</strong> #{v}" }.compact.join("<br />").truncate(140),
276
- latitude: r[lat_index],
277
- longitude: r[lon_index]
278
- }
279
- end
280
- end
281
- end
261
+ @linked_columns = @data_source.linked_columns
282
262
 
283
- filename = @query.try(:name).try(:parameterize).presence || 'query'
284
- respond_to do |format|
285
- format.html do
286
- render layout: false
287
- end
288
- format.xlsx do
289
- parser = ::Blazer::ExcelParser.new(@query, @columns, @rows)
290
- tmp_file = parser.export
291
- send_file tmp_file, type: "application/xlsx; charset=utf-8; header=present", disposition: "attachment; filename=\"#{parser.filename}\""
292
- end
293
- format.csv do
294
- send_data csv_data(@columns, @rows, @data_source), type: "text/csv; charset=utf-8; header=present", disposition: "attachment; filename=\"#{filename}.csv\""
295
- end
263
+ @markers = []
264
+ [["latitude", "longitude"], ["lat", "lon"], ["lat", "lng"]].each do |keys|
265
+ lat_index = @columns.index(keys.first)
266
+ lon_index = @columns.index(keys.last)
267
+ if lat_index && lon_index
268
+ @markers =
269
+ @rows.select do |r|
270
+ r[lat_index] && r[lon_index]
271
+ end.map do |r|
272
+ {
273
+ title: r.each_with_index.map{ |v, i| i == lat_index || i == lon_index ? nil : "<strong>#{@columns[i]}:</strong> #{v}" }.compact.join("<br />").truncate(140),
274
+ latitude: r[lat_index],
275
+ longitude: r[lon_index]
276
+ }
277
+ end
296
278
  end
297
279
  end
298
280
 
299
- def set_queries(limit = nil)
300
- @queries = Blazer::Query.named.select(:id, :name, :creator_id, :statement)
301
- @queries = @queries.includes(:creator) if Blazer.user_class
302
-
303
- if blazer_user && params[:filter] == "mine"
304
- @queries = @queries.where(creator_id: blazer_user.id).reorder(updated_at: :desc)
305
- elsif blazer_user && params[:filter] == "viewed" && Blazer.audit
306
- @queries = queries_by_ids(Blazer::Audit.where(user_id: blazer_user.id).order(created_at: :desc).limit(500).pluck(:query_id).uniq)
307
- else
308
- @queries = @queries.limit(limit) if limit
309
- @queries = @queries.order(:name)
281
+ filename = @query.try(:name).try(:parameterize).presence || 'query'
282
+ respond_to do |format|
283
+ format.html do
284
+ render layout: false
310
285
  end
311
- @queries = @queries.to_a
312
-
313
- @more = limit && @queries.size >= limit
286
+ format.xlsx do
287
+ parser = ::Blazer::ExcelParser.new(@query, @columns, @rows)
288
+ tmp_file = parser.export
289
+ send_file tmp_file, type: "application/xlsx; charset=utf-8; header=present", disposition: "attachment; filename=\"#{parser.filename}\""
290
+ end
291
+ format.csv do
292
+ send_data csv_data(@columns, @rows, @data_source), type: "text/csv; charset=utf-8; header=present", disposition: "attachment; filename=\"#{filename}.csv\""
293
+ end
294
+ end
295
+ end
314
296
 
315
- @queries = @queries.select { |q| !q.name.to_s.start_with?("#") || q.try(:creator).try(:id) == blazer_user.try(:id) }
297
+ def set_queries(limit = nil)
298
+ @queries = Blazer::Query.named.select(:id, :name, :creator_id, :statement)
299
+ @queries = @queries.includes(:creator) if Blazer.user_class
316
300
 
317
- @queries =
318
- @queries.map do |q|
319
- {
320
- id: q.id,
321
- name: q.name,
322
- creator: blazer_user && q.try(:creator) == blazer_user ? "You" : q.try(:creator).try(Blazer.user_name),
323
- vars: q.variables.join(", "),
324
- to_param: q.to_param
325
- }
326
- end
301
+ if blazer_user && params[:filter] == "mine"
302
+ @queries = @queries.where(creator_id: blazer_user.id).reorder(updated_at: :desc)
303
+ elsif blazer_user && params[:filter] == "viewed" && Blazer.audit
304
+ @queries = queries_by_ids(Blazer::Audit.where(user_id: blazer_user.id).order(created_at: :desc).limit(500).pluck(:query_id).uniq)
305
+ else
306
+ @queries = @queries.limit(limit) if limit
307
+ @queries = @queries.order(:name)
327
308
  end
309
+ @queries = @queries.to_a
328
310
 
329
- def queries_by_ids(favorite_query_ids)
330
- queries = Blazer::Query.named.where(id: favorite_query_ids)
331
- queries = queries.includes(:creator) if Blazer.user_class
332
- queries = queries.index_by(&:id)
333
- favorite_query_ids.map { |query_id| queries[query_id] }.compact
334
- end
311
+ @more = limit && @queries.size >= limit
335
312
 
336
- def set_query
337
- @query = Blazer::Query.find(params[:id].to_s.split("-").first)
313
+ @queries = @queries.select { |q| !q.name.to_s.start_with?("#") || q.try(:creator).try(:id) == blazer_user.try(:id) }
338
314
 
339
- unless @query.viewable?(blazer_user)
340
- render_forbidden
315
+ @queries =
316
+ @queries.map do |q|
317
+ {
318
+ id: q.id,
319
+ name: q.name,
320
+ creator: blazer_user && q.try(:creator) == blazer_user ? "You" : q.try(:creator).try(Blazer.user_name),
321
+ vars: q.variables.join(", "),
322
+ to_param: q.to_param
323
+ }
341
324
  end
342
- end
325
+ end
343
326
 
344
- def set_assignees
345
- if Blazer.settings.key?('assignees')
346
- data_source = Blazer.data_sources[params[:data_source]]
347
- statement = Blazer.settings['assignees']
348
- @assignees = Rails.cache.fetch 'jarvis_assignees' do
349
- (Blazer::RunStatement.new.perform(data_source, statement, {}).rows rescue [])
350
- end
351
- else
352
- @assignees = []
353
- end
354
- end
327
+ def queries_by_ids(favorite_query_ids)
328
+ queries = Blazer::Query.named.where(id: favorite_query_ids)
329
+ queries = queries.includes(:creator) if Blazer.user_class
330
+ queries = queries.index_by(&:id)
331
+ favorite_query_ids.map { |query_id| queries[query_id] }.compact
332
+ end
355
333
 
356
- def render_forbidden
357
- render plain: "Access denied", status: :forbidden
358
- end
334
+ def set_query
335
+ @query = Blazer::Query.find(params[:id].to_s.split("-").first)
359
336
 
360
- def query_params
361
- params.require(:query).permit!
337
+ unless @query.viewable?(blazer_user)
338
+ render_forbidden
362
339
  end
340
+ end
363
341
 
364
- def blazer_params
365
- params[:blazer] || {}
342
+ def set_accessible
343
+ @teams = Rails.cache.fetch('jarvis_teams') { get_teams } if Blazer.settings.key?('teams')
344
+ @assignees = Rails.cache.fetch('jarvis_assignees') { get_assignees } if Blazer.settings.key?('assignees')
345
+ ensure
346
+ @teams ||= []
347
+ @assignees ||= []
348
+ end
349
+
350
+ def get_assignees
351
+ Blazer::RunStatement.new.perform(@data_source, Blazer.settings['assignees'], {}).rows.map do |row|
352
+ [row.first, row.last.to_s.titleize]
366
353
  end
354
+ rescue
355
+ []
356
+ end
367
357
 
368
- def csv_data(columns, rows, data_source)
369
- CSV.generate do |csv|
370
- csv << columns
371
- rows.each do |row|
372
- csv << row.each_with_index.map { |v, i| v.is_a?(Time) ? blazer_time_value(data_source, columns[i], v) : v }
373
- end
374
- end
358
+ def get_teams
359
+ Blazer::RunStatement.new.perform(@data_source, Blazer.settings['teams'], {}).rows.map do |row|
360
+ [row.first, row.last.to_s.titleize]
375
361
  end
362
+ rescue
363
+ []
364
+ end
376
365
 
377
- def blazer_time_value(data_source, k, v)
378
- if k.end_with?('_date')
379
- v.in_time_zone(Blazer.time_zone).strftime("%Y/%m/%d")
380
- elsif k.end_with?('_time')
381
- v.in_time_zone(Blazer.time_zone).strftime("%H:%M")
382
- elsif data_source.local_time_suffix.any? { |s| k.ends_with?(s) }
383
- v.to_s.sub(" UTC", "")
384
- else
385
- v.in_time_zone(Blazer.time_zone)
366
+ def render_forbidden
367
+ render plain: "Access denied", status: :forbidden
368
+ end
369
+
370
+ def query_params
371
+ params.require(:query).permit!
372
+ end
373
+
374
+ def blazer_params
375
+ params[:blazer] || {}
376
+ end
377
+
378
+ def csv_data(columns, rows, data_source)
379
+ CSV.generate do |csv|
380
+ csv << columns
381
+ rows.each do |row|
382
+ csv << row.each_with_index.map { |v, i| v.is_a?(Time) ? blazer_time_value(data_source, columns[i], v) : v }
386
383
  end
387
- rescue
388
- return v
389
384
  end
390
- helper_method :blazer_time_value
385
+ end
391
386
 
392
- def blazer_run_id
393
- params[:run_id].to_s.gsub(/[^a-z0-9\-]/i, "")
387
+ def blazer_time_value(data_source, k, v)
388
+ if k.end_with?('_date')
389
+ v.in_time_zone(Blazer.time_zone).strftime("%Y/%m/%d")
390
+ elsif k.end_with?('_time')
391
+ v.in_time_zone(Blazer.time_zone).strftime("%H:%M")
392
+ elsif data_source.local_time_suffix.any? { |s| k.ends_with?(s) }
393
+ v.to_s.sub(" UTC", "")
394
+ else
395
+ v.in_time_zone(Blazer.time_zone)
394
396
  end
397
+ rescue
398
+ return v
399
+ end
400
+ helper_method :blazer_time_value
401
+
402
+ def blazer_run_id
403
+ params[:run_id].to_s.gsub(/[^a-z0-9\-]/i, "")
404
+ end
405
+
406
+ def preview_rows_number
407
+ Blazer.settings['preview_rows_number'] || 365
408
+ end
409
+ helper_method :preview_rows_number
395
410
 
396
- def preview_rows_number
397
- Blazer.settings['preview_rows_number'] || 365
398
- end
399
- helper_method :preview_rows_number
400
411
  end
401
412
  end