sql-jarvis 2.1 → 2.1.5
Sign up to get free protection for your applications and to get access to all the features.
- checksums.yaml +4 -4
- data/app/assets/javascripts/blazer/Sortable.js +1489 -1489
- data/app/assets/javascripts/blazer/ace.js +2 -0
- data/app/assets/javascripts/blazer/ace/mode-ruby.js +498 -0
- data/app/assets/javascripts/blazer/ace/theme-ambiance.js +184 -0
- data/app/assets/javascripts/blazer/stupidtable.js +2 -0
- data/app/assets/stylesheets/blazer/application.css +10 -2
- data/app/assets/stylesheets/blazer/main.css +4 -0
- data/app/controllers/blazer/base_controller.rb +11 -2
- data/app/controllers/blazer/queries_controller.rb +173 -142
- data/app/models/blazer/query.rb +9 -0
- data/app/views/blazer/_variables.html.haml +3 -4
- data/app/views/blazer/dashboards/show.html.haml +14 -13
- data/app/views/blazer/queries/_chart.html.haml +52 -0
- data/app/views/blazer/queries/_form.html.haml +59 -15
- data/app/views/blazer/queries/run.html.haml +23 -55
- data/app/views/blazer/queries/show.html.haml +55 -30
- data/lib/blazer.rb +12 -0
- data/lib/blazer/adapters/sql_adapter.rb +3 -3
- data/lib/blazer/data_source.rb +4 -0
- data/lib/blazer/excel_parser.rb +1 -5
- data/lib/blazer/run_integration.rb +24 -0
- data/lib/blazer/version.rb +1 -1
- data/lib/generators/blazer/templates/config.yml.tt +1 -0
- data/lib/generators/blazer/templates/install.rb.tt +2 -0
- metadata +7 -3
@@ -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
|
+
|
@@ -18,6 +18,14 @@ body {
|
|
18
18
|
padding-bottom: 15px;
|
19
19
|
}
|
20
20
|
|
21
|
+
#results-table th {
|
22
|
+
max-width: 200px;
|
23
|
+
}
|
24
|
+
|
25
|
+
#results-table td {
|
26
|
+
word-break: break-word;
|
27
|
+
}
|
28
|
+
|
21
29
|
.results-table th {
|
22
30
|
cursor: pointer;
|
23
31
|
}
|
@@ -61,7 +69,7 @@ input.search:focus {
|
|
61
69
|
margin-right: 0;
|
62
70
|
}
|
63
71
|
|
64
|
-
#editor-container {
|
72
|
+
#editor-container, #integration-editor {
|
65
73
|
background-color: #141414;
|
66
74
|
padding-top: 10px;
|
67
75
|
padding-bottom: 10px;
|
@@ -175,7 +183,7 @@ input.search:focus {
|
|
175
183
|
}
|
176
184
|
|
177
185
|
#code {
|
178
|
-
max-height:
|
186
|
+
max-height: 80vh;
|
179
187
|
overflow: hidden;
|
180
188
|
}
|
181
189
|
|
@@ -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
|
-
|
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"])
|
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 :
|
5
|
+
before_action :set_accessible, only: [:new, :create, :show, :edit, :update, :destroy, :refresh]
|
6
6
|
|
7
7
|
def home
|
8
8
|
set_queries(1000)
|
@@ -61,6 +61,15 @@ module Blazer
|
|
61
61
|
@statement = @query.statement.dup
|
62
62
|
process_vars(@statement, @query.data_source)
|
63
63
|
|
64
|
+
filename = []
|
65
|
+
filename << @query.name.parameterize if @query
|
66
|
+
filename << params[:start_time].to_s.to_date
|
67
|
+
if params[:end_time]
|
68
|
+
filename << 'to'
|
69
|
+
filename << params[:end_time].to_s.to_date
|
70
|
+
end
|
71
|
+
@filename = filename.compact.join('-')
|
72
|
+
|
64
73
|
@smart_vars = {}
|
65
74
|
@sql_errors = []
|
66
75
|
data_source = Blazer.data_sources[@query.data_source]
|
@@ -85,6 +94,7 @@ module Blazer
|
|
85
94
|
|
86
95
|
def run
|
87
96
|
@statement = params[:statement]
|
97
|
+
@integration = params[:integration]
|
88
98
|
data_source = params[:data_source]
|
89
99
|
process_vars(@statement, data_source)
|
90
100
|
@only_chart = params[:only_chart]
|
@@ -139,9 +149,9 @@ module Blazer
|
|
139
149
|
|
140
150
|
if @result
|
141
151
|
@data_source.delete_results(@run_id) if @run_id
|
142
|
-
|
143
|
-
@columns = @result.columns
|
144
|
-
@rows = @result.rows
|
152
|
+
@integration_output = Blazer::RunIntegration.new(@result, @integration).call
|
153
|
+
@columns = @integration_output&.dig(:columns) || @result.columns
|
154
|
+
@rows = @integration_output&.dig(:rows) || @result.rows
|
145
155
|
@error = @result.error
|
146
156
|
@cached_at = @result.cached_at
|
147
157
|
@just_cached = @result.just_cached
|
@@ -223,174 +233,195 @@ module Blazer
|
|
223
233
|
|
224
234
|
private
|
225
235
|
|
226
|
-
|
227
|
-
|
236
|
+
def set_data_source
|
237
|
+
@data_source = Blazer.data_sources[params[:data_source]]
|
238
|
+
render_forbidden unless Query.new(data_source: @data_source.id).editable?(blazer_user)
|
239
|
+
end
|
240
|
+
|
241
|
+
def continue_run
|
242
|
+
render json: {run_id: @run_id, timestamp: @timestamp}, status: :accepted
|
243
|
+
end
|
228
244
|
|
229
|
-
|
230
|
-
|
245
|
+
def render_run
|
246
|
+
@checks = @query ? @query.checks.order(:id) : []
|
247
|
+
|
248
|
+
@first_row = @rows.first || []
|
249
|
+
@column_types = []
|
250
|
+
if @rows.any?
|
251
|
+
@columns.each_with_index do |_, i|
|
252
|
+
@column_types << (
|
253
|
+
case @first_row[i]
|
254
|
+
when Integer
|
255
|
+
"int"
|
256
|
+
when Float, BigDecimal
|
257
|
+
"float"
|
258
|
+
else
|
259
|
+
"string-ins"
|
260
|
+
end
|
261
|
+
)
|
231
262
|
end
|
232
263
|
end
|
233
264
|
|
234
|
-
|
235
|
-
render json: {run_id: @run_id, timestamp: @timestamp}, status: :accepted
|
236
|
-
end
|
265
|
+
@filename = params[:filename] || @query&.name&.parameterize
|
237
266
|
|
238
|
-
|
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
|
267
|
+
@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)
|
257
268
|
|
258
|
-
|
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
|
269
|
+
@boom = @result.boom if @result
|
282
270
|
|
283
|
-
|
284
|
-
|
285
|
-
|
286
|
-
|
287
|
-
|
288
|
-
|
289
|
-
|
290
|
-
|
291
|
-
|
292
|
-
|
293
|
-
|
294
|
-
|
295
|
-
|
271
|
+
@linked_columns = @data_source.linked_columns
|
272
|
+
|
273
|
+
@markers = []
|
274
|
+
[["latitude", "longitude"], ["lat", "lon"], ["lat", "lng"]].each do |keys|
|
275
|
+
lat_index = @columns.index(keys.first)
|
276
|
+
lon_index = @columns.index(keys.last)
|
277
|
+
if lat_index && lon_index
|
278
|
+
@markers =
|
279
|
+
@rows.select do |r|
|
280
|
+
r[lat_index] && r[lon_index]
|
281
|
+
end.map do |r|
|
282
|
+
{
|
283
|
+
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),
|
284
|
+
latitude: r[lat_index],
|
285
|
+
longitude: r[lon_index]
|
286
|
+
}
|
287
|
+
end
|
296
288
|
end
|
297
289
|
end
|
298
290
|
|
299
|
-
|
300
|
-
|
301
|
-
|
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)
|
291
|
+
respond_to do |format|
|
292
|
+
format.html do
|
293
|
+
render layout: false
|
310
294
|
end
|
311
|
-
|
312
|
-
|
313
|
-
|
295
|
+
format.xlsx do
|
296
|
+
parser = ::Blazer::ExcelParser.new(@query, @columns, @rows)
|
297
|
+
tmp_file = parser.export
|
298
|
+
send_file tmp_file,
|
299
|
+
type: 'application/xlsx; charset=utf-8; header=present',
|
300
|
+
disposition: "attachment; filename=\"#{@filename}.xlsx\""
|
301
|
+
end
|
302
|
+
format.csv do
|
303
|
+
send_data csv_data(@columns, @rows, @data_source),
|
304
|
+
type: 'text/csv; charset=utf-8; header=present',
|
305
|
+
disposition: "attachment; filename=\"#{@filename}.csv\""
|
306
|
+
end
|
307
|
+
end
|
308
|
+
end
|
314
309
|
|
315
|
-
|
310
|
+
def set_queries(limit = nil)
|
311
|
+
@queries = Blazer::Query.named.select(:id, :name, :creator_id, :statement)
|
312
|
+
@queries = @queries.includes(:creator) if Blazer.user_class
|
316
313
|
|
317
|
-
|
318
|
-
|
319
|
-
|
320
|
-
|
321
|
-
|
322
|
-
|
323
|
-
|
324
|
-
to_param: q.to_param
|
325
|
-
}
|
326
|
-
end
|
314
|
+
if blazer_user && params[:filter] == "mine"
|
315
|
+
@queries = @queries.where(creator_id: blazer_user.id).reorder(updated_at: :desc)
|
316
|
+
elsif blazer_user && params[:filter] == "viewed" && Blazer.audit
|
317
|
+
@queries = queries_by_ids(Blazer::Audit.where(user_id: blazer_user.id).order(created_at: :desc).limit(500).pluck(:query_id).uniq)
|
318
|
+
else
|
319
|
+
@queries = @queries.limit(limit) if limit
|
320
|
+
@queries = @queries.order(:name)
|
327
321
|
end
|
322
|
+
@queries = @queries.to_a
|
328
323
|
|
329
|
-
|
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
|
324
|
+
@more = limit && @queries.size >= limit
|
335
325
|
|
336
|
-
|
337
|
-
@query = Blazer::Query.find(params[:id].to_s.split("-").first)
|
326
|
+
@queries = @queries.select { |q| !q.name.to_s.start_with?("#") || q.try(:creator).try(:id) == blazer_user.try(:id) }
|
338
327
|
|
339
|
-
|
340
|
-
|
328
|
+
@queries =
|
329
|
+
@queries.map do |q|
|
330
|
+
{
|
331
|
+
id: q.id,
|
332
|
+
name: q.name,
|
333
|
+
creator: blazer_user && q.try(:creator) == blazer_user ? "You" : q.try(:creator).try(Blazer.user_name),
|
334
|
+
vars: q.variables.join(", "),
|
335
|
+
to_param: q.to_param
|
336
|
+
}
|
341
337
|
end
|
342
|
-
|
338
|
+
end
|
343
339
|
|
344
|
-
|
345
|
-
|
346
|
-
|
347
|
-
|
348
|
-
|
349
|
-
|
350
|
-
end
|
351
|
-
else
|
352
|
-
@assignees = []
|
353
|
-
end
|
354
|
-
end
|
340
|
+
def queries_by_ids(favorite_query_ids)
|
341
|
+
queries = Blazer::Query.named.where(id: favorite_query_ids)
|
342
|
+
queries = queries.includes(:creator) if Blazer.user_class
|
343
|
+
queries = queries.index_by(&:id)
|
344
|
+
favorite_query_ids.map { |query_id| queries[query_id] }.compact
|
345
|
+
end
|
355
346
|
|
356
|
-
|
357
|
-
|
358
|
-
end
|
347
|
+
def set_query
|
348
|
+
@query = Blazer::Query.find(params[:id].to_s.split("-").first)
|
359
349
|
|
360
|
-
|
361
|
-
|
350
|
+
unless @query.viewable?(blazer_user)
|
351
|
+
render_forbidden
|
362
352
|
end
|
353
|
+
end
|
354
|
+
|
355
|
+
def set_accessible
|
356
|
+
@teams = get_teams
|
357
|
+
@assignees ||= get_assignees
|
358
|
+
ensure
|
359
|
+
@teams ||= []
|
360
|
+
@assignees ||= []
|
361
|
+
end
|
363
362
|
|
364
|
-
|
365
|
-
|
363
|
+
def get_assignees
|
364
|
+
return [] unless Blazer.settings.key?('assignees')
|
365
|
+
Blazer::RunStatement.new.perform(@data_source, Blazer.settings['assignees'], {}).rows.map do |row|
|
366
|
+
[row.first, row.last.to_s.titleize]
|
366
367
|
end
|
368
|
+
rescue
|
369
|
+
[]
|
370
|
+
end
|
367
371
|
|
368
|
-
|
369
|
-
|
370
|
-
|
371
|
-
|
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
|
372
|
+
def get_teams
|
373
|
+
return [] unless Blazer.settings.key?('teams')
|
374
|
+
Blazer::RunStatement.new.perform(@data_source, Blazer.settings['teams'], {}).rows.map do |row|
|
375
|
+
[row.first, row.last.to_s.titleize]
|
375
376
|
end
|
377
|
+
rescue
|
378
|
+
[]
|
379
|
+
end
|
376
380
|
|
377
|
-
|
378
|
-
|
379
|
-
|
380
|
-
|
381
|
-
|
382
|
-
|
383
|
-
|
384
|
-
|
385
|
-
|
381
|
+
def render_forbidden
|
382
|
+
render plain: "Access denied", status: :forbidden
|
383
|
+
end
|
384
|
+
|
385
|
+
def query_params
|
386
|
+
params.require(:query).permit!
|
387
|
+
end
|
388
|
+
|
389
|
+
def blazer_params
|
390
|
+
params[:blazer] || {}
|
391
|
+
end
|
392
|
+
|
393
|
+
def csv_data(columns, rows, data_source)
|
394
|
+
CSV.generate do |csv|
|
395
|
+
csv << columns
|
396
|
+
rows.each do |row|
|
397
|
+
csv << row.each_with_index.map { |v, i| v.is_a?(Time) ? blazer_time_value(data_source, columns[i], v) : v }
|
386
398
|
end
|
387
|
-
rescue
|
388
|
-
return v
|
389
399
|
end
|
390
|
-
|
400
|
+
end
|
391
401
|
|
392
|
-
|
393
|
-
|
402
|
+
def blazer_time_value(data_source, k, v)
|
403
|
+
if k.end_with?('_date')
|
404
|
+
v.in_time_zone(Blazer.time_zone).strftime("%Y/%m/%d")
|
405
|
+
elsif k.end_with?('_time')
|
406
|
+
v.in_time_zone(Blazer.time_zone).strftime("%H:%M")
|
407
|
+
elsif data_source.local_time_suffix.any? { |s| k.ends_with?(s) }
|
408
|
+
v.to_s.sub(" UTC", "")
|
409
|
+
else
|
410
|
+
v.in_time_zone(Blazer.time_zone)
|
394
411
|
end
|
412
|
+
rescue
|
413
|
+
return v
|
414
|
+
end
|
415
|
+
helper_method :blazer_time_value
|
416
|
+
|
417
|
+
def blazer_run_id
|
418
|
+
params[:run_id].to_s.gsub(/[^a-z0-9\-]/i, "")
|
419
|
+
end
|
420
|
+
|
421
|
+
def preview_rows_number
|
422
|
+
Blazer.settings['preview_rows_number'] || 365
|
423
|
+
end
|
424
|
+
helper_method :preview_rows_number
|
425
|
+
|
395
426
|
end
|
396
427
|
end
|