quby-compiler 0.5.14 → 0.5.16
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.
- checksums.yaml +4 -4
- data/CHANGELOG.md +17 -0
- data/lib/quby/compiler/dsl/questionnaire_builder.rb +5 -0
- data/lib/quby/compiler/dsl/questions/base.rb +5 -0
- data/lib/quby/compiler/dsl/questions/integer_question_builder.rb +10 -0
- data/lib/quby/compiler/dsl/sexp_variable_builder.rb +57 -0
- data/lib/quby/compiler/entities/panel.rb +0 -18
- data/lib/quby/compiler/entities/question.rb +7 -24
- data/lib/quby/compiler/entities/question_option.rb +2 -34
- data/lib/quby/compiler/entities/questionnaire.rb +7 -0
- data/lib/quby/compiler/entities/questions/checkbox_question.rb +0 -8
- data/lib/quby/compiler/entities/questions/concerns/slider.rb +0 -14
- data/lib/quby/compiler/entities/questions/concerns/split_to_units.rb +58 -0
- data/lib/quby/compiler/entities/questions/date_question.rb +0 -13
- data/lib/quby/compiler/entities/questions/deprecated_question.rb +0 -4
- data/lib/quby/compiler/entities/questions/float_question.rb +0 -7
- data/lib/quby/compiler/entities/questions/integer_question.rb +2 -7
- data/lib/quby/compiler/entities/questions/radio_question.rb +0 -3
- data/lib/quby/compiler/entities/questions/select_question.rb +0 -6
- data/lib/quby/compiler/entities/questions/string_question.rb +0 -6
- data/lib/quby/compiler/entities/questions/text_question.rb +0 -3
- data/lib/quby/compiler/entities/sexp_variable.rb +50 -0
- data/lib/quby/compiler/entities/sexp_variables.rb +41 -0
- data/lib/quby/compiler/outputs/quby_frontend_v2_serializer.rb +230 -4
- data/lib/quby/compiler/services/definition_validator.rb +7 -0
- data/lib/quby/compiler/version.rb +1 -1
- data/lib/quby/compiler.rb +1 -0
- data/lib/quby/inspect_except.rb +12 -0
- metadata +7 -2
checksums.yaml
CHANGED
@@ -1,7 +1,7 @@
|
|
1
1
|
---
|
2
2
|
SHA256:
|
3
|
-
metadata.gz:
|
4
|
-
data.tar.gz:
|
3
|
+
metadata.gz: 5fbd126d732f2ab8201f325d09041cb564138354150fbedc83f3f7e9f68a755f
|
4
|
+
data.tar.gz: df750a8516c8596596cf11a03f3ce758bdb745de607068d4b8b753ae62a95e17
|
5
5
|
SHA512:
|
6
|
-
metadata.gz:
|
7
|
-
data.tar.gz:
|
6
|
+
metadata.gz: ddd7e5767209991dd94f35b086294903cc2616d76104cbdaf776181e661e0cefe501393033abc41674aed88f14a7849a3f275c81330c72c68384dd42cf9052a3
|
7
|
+
data.tar.gz: 1ba99e0f79496a7d64d95952739bd5fcc1f0ef6f1927c2cd5ce211c6bddd134dd37f617400dd3201344d35a58503e7b14503856c274404f41d6e90391b07ace3
|
data/CHANGELOG.md
CHANGED
@@ -1,3 +1,20 @@
|
|
1
|
+
# 0.5.16
|
2
|
+
|
3
|
+
* add integer as split_to_units option, with units and conversions as attributes
|
4
|
+
* add sexp_variables to do calculations using s expressions, for now in quby2, but later in the backend.
|
5
|
+
* test that select/radio options are always numeric
|
6
|
+
* quby2.json
|
7
|
+
* add split_to_units question, that saves a integer.
|
8
|
+
* Calculate sexpr variables and allow strings to interpolate them {{calculation.some_var}}
|
9
|
+
|
10
|
+
# 0.5.15
|
11
|
+
|
12
|
+
* Add context_description to questions, to have a text item that is hidden together with the question.
|
13
|
+
* quby2.json
|
14
|
+
* Moved quby2 serialization to the serializer, compact everything, no markdown for v2 title/descriptions.
|
15
|
+
* Sanitize Quby2 html in v2.
|
16
|
+
* Add contextDescription to questions.
|
17
|
+
|
1
18
|
# 0.5.14
|
2
19
|
|
3
20
|
* Support for activemodel 7.1 and 7.2
|
@@ -8,6 +8,7 @@ require 'quby/compiler/dsl/charting/line_chart_builder'
|
|
8
8
|
require 'quby/compiler/dsl/charting/radar_chart_builder'
|
9
9
|
require 'quby/compiler/dsl/charting/bar_chart_builder'
|
10
10
|
require 'quby/compiler/dsl/charting/overview_chart_builder'
|
11
|
+
require 'quby/compiler/dsl/sexp_variable_builder'
|
11
12
|
|
12
13
|
require_relative 'standardized_panel_generators'
|
13
14
|
|
@@ -200,6 +201,10 @@ module Quby
|
|
200
201
|
end
|
201
202
|
end
|
202
203
|
|
204
|
+
def sexp_variable(key, &block)
|
205
|
+
@questionnaire.add_sexp_variable(key, SexpVariableBuilder.new(key, &block).build)
|
206
|
+
end
|
207
|
+
|
203
208
|
# variable :totaal do
|
204
209
|
# # Plain old Ruby code here, executed in the scope of the answer
|
205
210
|
# # variables are private to the score calculation
|
@@ -12,6 +12,7 @@ module Quby
|
|
12
12
|
end
|
13
13
|
|
14
14
|
def build
|
15
|
+
@question.after_build
|
15
16
|
@question
|
16
17
|
end
|
17
18
|
|
@@ -27,6 +28,10 @@ module Quby
|
|
27
28
|
@question.description = value
|
28
29
|
end
|
29
30
|
|
31
|
+
def context_description(value)
|
32
|
+
@question.context_description = value
|
33
|
+
end
|
34
|
+
|
30
35
|
def presentation(value)
|
31
36
|
@question.presentation = value
|
32
37
|
end
|
@@ -14,6 +14,16 @@ module Quby
|
|
14
14
|
super
|
15
15
|
@question = Entities::Questions::IntegerQuestion.new(key, options)
|
16
16
|
end
|
17
|
+
|
18
|
+
# as split_to_units
|
19
|
+
def units(*values)
|
20
|
+
@question.units = values
|
21
|
+
end
|
22
|
+
|
23
|
+
# as split_to_units
|
24
|
+
def conversions(value)
|
25
|
+
@question.conversions = value
|
26
|
+
end
|
17
27
|
end
|
18
28
|
end
|
19
29
|
end
|
@@ -0,0 +1,57 @@
|
|
1
|
+
# frozen_string_literal: true
|
2
|
+
|
3
|
+
require 'quby/compiler/entities/sexp_variable'
|
4
|
+
module Quby::Compiler
|
5
|
+
module DSL
|
6
|
+
# sexp_variable :myvar do
|
7
|
+
# sum(number_values(:v_1, :v_2))
|
8
|
+
# end
|
9
|
+
class ::SexpVariableBuilder
|
10
|
+
attr_reader :calculation, :key
|
11
|
+
|
12
|
+
def initialize(key, &block)
|
13
|
+
@key = key
|
14
|
+
@calculation = instance_eval(&block)
|
15
|
+
end
|
16
|
+
|
17
|
+
def string_value(key)
|
18
|
+
Entities::SexpVariables::StringValue.new(op: :string_value, key: key)
|
19
|
+
end
|
20
|
+
|
21
|
+
def number_value(key)
|
22
|
+
Entities::SexpVariables::NumberValue.new(op: :number_value, key: key)
|
23
|
+
end
|
24
|
+
|
25
|
+
def number_values(*keys)
|
26
|
+
keys.map { |key| number_value(key) }
|
27
|
+
end
|
28
|
+
|
29
|
+
%i[sum subtract multiply divide].each do |op|
|
30
|
+
define_method(op) do |*values|
|
31
|
+
Entities::SexpVariables::NumberReducer.new(op:, values: wrap_and_flatten(values))
|
32
|
+
end
|
33
|
+
end
|
34
|
+
|
35
|
+
def round(value)
|
36
|
+
Entities::SexpVariables::NumberMethod.new(op: :round, value: value)
|
37
|
+
end
|
38
|
+
|
39
|
+
def build
|
40
|
+
Entities::SexpVariable.new(key:, calculation:)
|
41
|
+
end
|
42
|
+
|
43
|
+
private
|
44
|
+
|
45
|
+
def wrap_and_flatten(values)
|
46
|
+
values.flat_map { |value|
|
47
|
+
case value
|
48
|
+
when Numeric
|
49
|
+
Entities::SexpVariables::Number.new(op: :number, value: value)
|
50
|
+
else
|
51
|
+
value
|
52
|
+
end
|
53
|
+
}
|
54
|
+
end
|
55
|
+
end
|
56
|
+
end
|
57
|
+
end
|
@@ -16,10 +16,6 @@ module Quby
|
|
16
16
|
@items = options[:items] || []
|
17
17
|
end
|
18
18
|
|
19
|
-
def as_json(options = {})
|
20
|
-
super.merge(title: title, index: index, items: json_items)
|
21
|
-
end
|
22
|
-
|
23
19
|
def index
|
24
20
|
@questionnaire.panels.index(self)
|
25
21
|
end
|
@@ -44,20 +40,6 @@ module Quby
|
|
44
40
|
end
|
45
41
|
end
|
46
42
|
|
47
|
-
def json_items
|
48
|
-
items.map do |item|
|
49
|
-
case item
|
50
|
-
when Text
|
51
|
-
{ type: 'html', html: item.html }
|
52
|
-
when Question
|
53
|
-
next if item.table # things inside a table are added to the table, AND ALSO to the panel. skip them.
|
54
|
-
{ type: 'question', key: item.key }
|
55
|
-
when Table
|
56
|
-
{ type: "table" }
|
57
|
-
end
|
58
|
-
end.compact
|
59
|
-
end
|
60
|
-
|
61
43
|
def validations
|
62
44
|
vals = {}
|
63
45
|
items.each do |item|
|
@@ -6,6 +6,8 @@ module Quby
|
|
6
6
|
module Compiler
|
7
7
|
module Entities
|
8
8
|
class Question < Item
|
9
|
+
include ::Quby::InspectExcept.new(*%i[@dependencies @options @questionnaire @table @validations @parent])
|
10
|
+
|
9
11
|
MARKDOWN_ATTRIBUTES = %w(description title).freeze
|
10
12
|
|
11
13
|
set_callback :after_dsl_enhance, :expand_depends_on_input_keys
|
@@ -18,6 +20,7 @@ module Quby
|
|
18
20
|
attr_accessor :title
|
19
21
|
attr_accessor :context_free_title
|
20
22
|
attr_accessor :description
|
23
|
+
attr_accessor :context_description
|
21
24
|
|
22
25
|
attr_accessor :labels
|
23
26
|
|
@@ -205,6 +208,10 @@ module Quby
|
|
205
208
|
end
|
206
209
|
# rubocop:enable CyclomaticComplexity, Metrics/MethodLength
|
207
210
|
|
211
|
+
# called after DSL has instance_evalled everything within the question block.
|
212
|
+
def after_build
|
213
|
+
end
|
214
|
+
|
208
215
|
def context_free_title_or_title
|
209
216
|
context_free_title || title
|
210
217
|
end
|
@@ -229,30 +236,6 @@ module Quby
|
|
229
236
|
options.length > 0 && type != :select ? options.length : @col_span
|
230
237
|
end
|
231
238
|
|
232
|
-
def as_json(options = {})
|
233
|
-
# rubocop:disable SymbolName
|
234
|
-
super.merge(
|
235
|
-
key: key,
|
236
|
-
title: Quby::Compiler::MarkdownParser.new(title).to_html,
|
237
|
-
description: Quby::Compiler::MarkdownParser.new(description).to_html,
|
238
|
-
type: type,
|
239
|
-
size: size.presence && Integer(size), # 2022-11: 4k string and 7k integer
|
240
|
-
hidden: hidden?,
|
241
|
-
displayModes: display_modes,
|
242
|
-
defaultInvisible: default_invisible,
|
243
|
-
viewSelector: view_selector,
|
244
|
-
parentKey: parent&.key,
|
245
|
-
parentOptionKey: parent_option_key,
|
246
|
-
deselectable: deselectable,
|
247
|
-
presentation: presentation,
|
248
|
-
as: as || type, # default to type so typescript can narrow on it.
|
249
|
-
questionGroup: question_group
|
250
|
-
).tap do |json|
|
251
|
-
json[:unit] = unit if %i[integer float string].include?(type) && as != :slider
|
252
|
-
json[:showValues] = [true, :all].include?(show_values) if %i[radio scale].include?(type) && show_values
|
253
|
-
end
|
254
|
-
end
|
255
|
-
|
256
239
|
def title_question?
|
257
240
|
presentation == :next_to_title
|
258
241
|
end
|
@@ -5,11 +5,13 @@ module Quby
|
|
5
5
|
module Entities
|
6
6
|
class QuestionOption
|
7
7
|
include ActiveModel::Validations
|
8
|
+
include ::Quby::InspectExcept.new(:@question, :@questions)
|
8
9
|
|
9
10
|
MARKDOWN_ATTRIBUTES = %w(description).freeze
|
10
11
|
|
11
12
|
attr_reader :key
|
12
13
|
attr_reader :value
|
14
|
+
validates :value, numericality: {allow_nil: true} # nil for checkbox questions.
|
13
15
|
attr_reader :description, :context_free_description
|
14
16
|
attr_reader :questions
|
15
17
|
# for scale/radio/checbox questions, piece of of html that is rendered between the options
|
@@ -53,40 +55,6 @@ module Quby
|
|
53
55
|
@questions.each { |q| return true if q.key_in_use?(k) }
|
54
56
|
false
|
55
57
|
end
|
56
|
-
|
57
|
-
def as_json(options = {})
|
58
|
-
if inner_title
|
59
|
-
inner_title_as_json(options)
|
60
|
-
elsif placeholder
|
61
|
-
nil # placeholder attr on question.
|
62
|
-
else
|
63
|
-
option_as_json(options)
|
64
|
-
end
|
65
|
-
end
|
66
|
-
|
67
|
-
def inner_title_as_json(options = {})
|
68
|
-
{
|
69
|
-
type: 'html',
|
70
|
-
key: SecureRandom.uuid,
|
71
|
-
html: Quby::Compiler::MarkdownParser.new(description).to_html
|
72
|
-
}
|
73
|
-
end
|
74
|
-
|
75
|
-
def option_as_json(options = {})
|
76
|
-
{
|
77
|
-
type: 'option',
|
78
|
-
key: key,
|
79
|
-
value: value,
|
80
|
-
description: question.type == :select \
|
81
|
-
? description
|
82
|
-
: Quby::Compiler::MarkdownParser.new(description).to_html,
|
83
|
-
questions: questions,
|
84
|
-
hidesQuestions: hides_questions,
|
85
|
-
showsQuestions: shows_questions,
|
86
|
-
hidden: hidden,
|
87
|
-
viewId: view_id
|
88
|
-
}
|
89
|
-
end
|
90
58
|
end
|
91
59
|
end
|
92
60
|
end
|
@@ -17,6 +17,7 @@ module Quby
|
|
17
17
|
class Questionnaire
|
18
18
|
extend ActiveModel::Naming
|
19
19
|
include ActiveModel::Validations
|
20
|
+
include ::Quby::InspectExcept.new(*%i[@attributes @charts @fields @flags @panels @questions @sourcecode @score_calculations @score_schemas @textvars])
|
20
21
|
|
21
22
|
class ValidationError < StandardError; end
|
22
23
|
class UnknownInputKey < ValidationError; end
|
@@ -52,6 +53,7 @@ module Quby
|
|
52
53
|
@outcome_tables = []
|
53
54
|
@check_score_keys_consistency = true
|
54
55
|
@lookup_tables = {}
|
56
|
+
@sexp_variables = {}
|
55
57
|
@versions = []
|
56
58
|
@seeds_patch = {}
|
57
59
|
@anonymous_conditions = Entities::AnonymousConditions.new
|
@@ -101,6 +103,7 @@ module Quby
|
|
101
103
|
|
102
104
|
attr_accessor :outcome_tables
|
103
105
|
attr_accessor :score_schemas
|
106
|
+
attr_accessor :sexp_variables
|
104
107
|
attr_accessor :lookup_tables
|
105
108
|
attr_accessor :anonymous_conditions
|
106
109
|
|
@@ -361,6 +364,10 @@ module Quby
|
|
361
364
|
end
|
362
365
|
end
|
363
366
|
|
367
|
+
def add_sexp_variable(key, sexp_variable)
|
368
|
+
sexp_variables[key] = sexp_variable
|
369
|
+
end
|
370
|
+
|
364
371
|
def add_outcome_table(outcome_table_options)
|
365
372
|
outcome_tables << OutcomeTable.new(**outcome_table_options, questionnaire: self)
|
366
373
|
end
|
@@ -58,14 +58,6 @@ module Quby
|
|
58
58
|
# Some options don't have a key (inner_title), they are stripped.
|
59
59
|
options.map { |opt| opt.input_key }.compact
|
60
60
|
end
|
61
|
-
|
62
|
-
def as_json(options = {})
|
63
|
-
super.tap do |json|
|
64
|
-
json[:children] = @options.as_json
|
65
|
-
json[:checkAllOption] = check_all_option if check_all_option.present?
|
66
|
-
json[:uncheckAllOption] = uncheck_all_option if uncheck_all_option.present?
|
67
|
-
end
|
68
|
-
end
|
69
61
|
end
|
70
62
|
end
|
71
63
|
end
|
@@ -5,19 +5,5 @@ module Quby::Compiler::Entities::Questions::Concerns
|
|
5
5
|
included do
|
6
6
|
validates :minimum, :maximum, presence: true, if: -> { as == :slider }
|
7
7
|
end
|
8
|
-
|
9
|
-
def as_json(options = {})
|
10
|
-
if as == :slider
|
11
|
-
super.merge(
|
12
|
-
step: step,
|
13
|
-
defaultPosition: default_position.is_a?(Numeric) ? default_position : minimum,
|
14
|
-
startThumbHidden: default_position == :hidden,
|
15
|
-
valueTooltip: input_data[:value_tooltip] || false,
|
16
|
-
labels: labels
|
17
|
-
)
|
18
|
-
else
|
19
|
-
super
|
20
|
-
end
|
21
|
-
end
|
22
8
|
end
|
23
9
|
end
|
@@ -0,0 +1,58 @@
|
|
1
|
+
module Quby::Compiler::Entities::Questions::Concerns
|
2
|
+
module SplitToUnits
|
3
|
+
extend ActiveSupport::Concern
|
4
|
+
DEFAULT_SPLIT_TO_UNIT_CONVERSIONS = {
|
5
|
+
minutes: {
|
6
|
+
hours: 60,
|
7
|
+
days: 1440,
|
8
|
+
weeks: 10080
|
9
|
+
},
|
10
|
+
seconds: {
|
11
|
+
minutes: 60,
|
12
|
+
hours: 3600,
|
13
|
+
days: 86400
|
14
|
+
},
|
15
|
+
m: {
|
16
|
+
km: 1000
|
17
|
+
},
|
18
|
+
cm: {
|
19
|
+
m: 100
|
20
|
+
},
|
21
|
+
mm: {
|
22
|
+
cm: 10,
|
23
|
+
m: 1000
|
24
|
+
},
|
25
|
+
g: {
|
26
|
+
kg: 1000
|
27
|
+
}
|
28
|
+
}.freeze
|
29
|
+
|
30
|
+
included do
|
31
|
+
attr_accessor :units, :conversions
|
32
|
+
|
33
|
+
validates :units, :conversions, presence: true, if: -> { as == :split_to_units }
|
34
|
+
validate :validate_split_to_units, if: -> { as == :split_to_units }
|
35
|
+
end
|
36
|
+
|
37
|
+
def after_build
|
38
|
+
super
|
39
|
+
return unless as == :split_to_units
|
40
|
+
@unit = units&.last
|
41
|
+
(@conversions ||= {}).reverse_merge!(default_split_to_units_conversions)
|
42
|
+
end
|
43
|
+
|
44
|
+
def default_split_to_units_conversions
|
45
|
+
(DEFAULT_SPLIT_TO_UNIT_CONVERSIONS[unit] || {}).slice(*units)
|
46
|
+
end
|
47
|
+
|
48
|
+
def validate_split_to_units
|
49
|
+
return unless units.present? && conversions.present?
|
50
|
+
|
51
|
+
(units - [unit]).each do |unit_to_convert|
|
52
|
+
if conversions[unit_to_convert].nil?
|
53
|
+
errors.add(:conversions, "should contain a conversion for unit #{unit_to_convert}")
|
54
|
+
end
|
55
|
+
end
|
56
|
+
end
|
57
|
+
end
|
58
|
+
end
|
@@ -53,19 +53,6 @@ module Quby
|
|
53
53
|
send("#{component}_key").to_sym
|
54
54
|
end
|
55
55
|
end
|
56
|
-
|
57
|
-
def as_json(options = {})
|
58
|
-
super.merge(
|
59
|
-
type: 'date_parts',
|
60
|
-
as: 'date_parts',
|
61
|
-
dateParts: components.map { |component|
|
62
|
-
{
|
63
|
-
part: component,
|
64
|
-
key: send("#{component}_key")
|
65
|
-
}
|
66
|
-
}
|
67
|
-
)
|
68
|
-
end
|
69
56
|
end
|
70
57
|
end
|
71
58
|
end
|
@@ -1,6 +1,7 @@
|
|
1
1
|
# frozen_string_literal: true
|
2
2
|
|
3
3
|
require_relative 'concerns/slider'
|
4
|
+
require_relative 'concerns/split_to_units'
|
4
5
|
|
5
6
|
module Quby
|
6
7
|
module Compiler
|
@@ -8,13 +9,7 @@ module Quby
|
|
8
9
|
module Questions
|
9
10
|
class IntegerQuestion < Question
|
10
11
|
include Concerns::Slider
|
11
|
-
|
12
|
-
def as_json(options = {})
|
13
|
-
super.merge(
|
14
|
-
minimum: minimum,
|
15
|
-
maximum: maximum
|
16
|
-
)
|
17
|
-
end
|
12
|
+
include Concerns::SplitToUnits
|
18
13
|
|
19
14
|
def size
|
20
15
|
@size || 30
|
@@ -5,12 +5,6 @@ module Quby
|
|
5
5
|
module Entities
|
6
6
|
module Questions
|
7
7
|
class SelectQuestion < Question
|
8
|
-
def as_json(options = {})
|
9
|
-
super.merge(
|
10
|
-
children: @options.as_json.compact, # for now just options, but we'll add optgroups later.
|
11
|
-
placeholder: @options.find { _1.placeholder }&.description # nil for no placeholder
|
12
|
-
)
|
13
|
-
end
|
14
8
|
end
|
15
9
|
end
|
16
10
|
end
|
@@ -5,12 +5,6 @@ module Quby
|
|
5
5
|
module Entities
|
6
6
|
module Questions
|
7
7
|
class StringQuestion < Question
|
8
|
-
def as_json(options = {})
|
9
|
-
super.merge(autocomplete: @autocomplete)
|
10
|
-
super.merge(autocomplete: @autocomplete).tap do |json|
|
11
|
-
json[:setsTextvar] = sets_textvar if sets_textvar
|
12
|
-
end
|
13
|
-
end
|
14
8
|
end
|
15
9
|
end
|
16
10
|
end
|
@@ -0,0 +1,50 @@
|
|
1
|
+
require 'quby/compiler/entities/sexp_variables'
|
2
|
+
|
3
|
+
module Quby::Compiler::Entities
|
4
|
+
class SexpVariable
|
5
|
+
attr_reader :calculation, :key
|
6
|
+
|
7
|
+
def initialize(key:, calculation:)
|
8
|
+
@key = key
|
9
|
+
@calculation = calculation
|
10
|
+
end
|
11
|
+
|
12
|
+
# Called by DefinitionValidator.
|
13
|
+
def validate(questionnaire)
|
14
|
+
case calculation
|
15
|
+
when SexpVariables::NumberValue
|
16
|
+
validate_question_exist(questionnaire, calculation)
|
17
|
+
validate_value_is_number(questionnaire, calculation)
|
18
|
+
when SexpVariables::StringValue
|
19
|
+
validate_question_exist(questionnaire, calculation)
|
20
|
+
validate_value_is_string(questionnaire, calculation)
|
21
|
+
end
|
22
|
+
end
|
23
|
+
|
24
|
+
def validate_question_exist(questionnaire, sexp)
|
25
|
+
return if questionnaire.question_hash.key?(sexp.key)
|
26
|
+
|
27
|
+
fail "sexp_variable #{key} uses nonexistent question #{sexp.key}."
|
28
|
+
end
|
29
|
+
|
30
|
+
def validate_value_is_number(questionnaire, sexp)
|
31
|
+
question = questionnaire.question_hash[sexp.key]
|
32
|
+
case question
|
33
|
+
when Questions::IntegerQuestion, Questions::FloatQuestion, Questions::SelectQuestion, Questions::RadioQuestion
|
34
|
+
return
|
35
|
+
end
|
36
|
+
|
37
|
+
fail "sexp_variable #{key} uses non-numeric question #{sexp.key} for number_value."
|
38
|
+
end
|
39
|
+
|
40
|
+
def validate_value_is_string(questionnaire, sexp)
|
41
|
+
question = questionnaire.question_hash[sexp.key]
|
42
|
+
case question
|
43
|
+
when Questions::StringQuestion, Questions::TextareaQuestion
|
44
|
+
return
|
45
|
+
end
|
46
|
+
|
47
|
+
fail "sexp_variable #{key} uses non-string question #{sexp.key} for string_value."
|
48
|
+
end
|
49
|
+
end
|
50
|
+
end
|
@@ -0,0 +1,41 @@
|
|
1
|
+
module Quby::Compiler::Entities
|
2
|
+
module SexpVariables
|
3
|
+
class Base < Dry::Struct
|
4
|
+
end
|
5
|
+
|
6
|
+
# recursive dependencies, we so we define the classes first.
|
7
|
+
class NumberValue < Base; end
|
8
|
+
class Number < Base; end
|
9
|
+
class NumberReducer < Base; end
|
10
|
+
class NumberMethod < Base; end
|
11
|
+
|
12
|
+
NumericType = Number | NumberReducer | NumberMethod | NumberValue
|
13
|
+
|
14
|
+
class NumberValue
|
15
|
+
attribute :op, Quby::Types::Symbol.enum(:number_value)
|
16
|
+
attribute :key, Quby::Types::Symbol
|
17
|
+
end
|
18
|
+
|
19
|
+
class StringValue < Base
|
20
|
+
attribute :op, Quby::Types::Symbol.enum(:string_value)
|
21
|
+
attribute :key, Quby::Types::Symbol
|
22
|
+
end
|
23
|
+
|
24
|
+
class Number
|
25
|
+
attribute :op, Quby::Types::Symbol.enum(:value)
|
26
|
+
attribute :value, Quby::Types::Integer | Quby::Types::Float
|
27
|
+
end
|
28
|
+
|
29
|
+
# returning a Number
|
30
|
+
class NumberReducer
|
31
|
+
attribute :op, Quby::Types::Symbol.enum(:sum, :subtract, :multiply, :divide)
|
32
|
+
attribute :values, Quby::Types::Array.of(NumericType)
|
33
|
+
end
|
34
|
+
|
35
|
+
# returning a Number
|
36
|
+
class NumberMethod
|
37
|
+
attribute :op, Quby::Types::Symbol.enum(:round)
|
38
|
+
attribute :value, NumericType
|
39
|
+
end
|
40
|
+
end
|
41
|
+
end
|
@@ -19,12 +19,217 @@ module Quby
|
|
19
19
|
description: description,
|
20
20
|
shortDescription: short_description,
|
21
21
|
defaultAnswerValue: Services::TransformQuby1ValuesIntoQuby2Values.run!(@questionnaire, default_answer_value),
|
22
|
-
panels: panels,
|
23
|
-
questions:
|
24
|
-
flags: flags,
|
22
|
+
panels: panels.map { panel(_1) },
|
23
|
+
questions: questions,
|
25
24
|
textvars: textvars,
|
26
25
|
validations: validations,
|
27
|
-
visibilityRules: visibility_rules
|
26
|
+
visibilityRules: visibility_rules.as_json,
|
27
|
+
sexpVariables: sexp_variables,
|
28
|
+
}
|
29
|
+
end
|
30
|
+
|
31
|
+
def panel(panel)
|
32
|
+
{
|
33
|
+
title: panel.title,
|
34
|
+
items: panel.items.map { panel_item(_1) }.compact,
|
35
|
+
}.compact
|
36
|
+
end
|
37
|
+
|
38
|
+
def panel_item(item)
|
39
|
+
case item
|
40
|
+
when Quby::Compiler::Entities::Text
|
41
|
+
{ type: 'html', html: handle_html(item.html, type: :prose, v1_markdown: false) }
|
42
|
+
when Quby::Compiler::Entities::Question
|
43
|
+
return if item.table # things inside a table are added to the table, AND ALSO to the panel. skip them.
|
44
|
+
{ type: 'question', key: item.key }
|
45
|
+
when Quby::Compiler::Entities::Table
|
46
|
+
{ type: "table" }
|
47
|
+
end
|
48
|
+
end
|
49
|
+
|
50
|
+
def questions
|
51
|
+
fields.question_hash \
|
52
|
+
.to_h { |k, question| [k, question(question)] } \
|
53
|
+
.compact
|
54
|
+
end
|
55
|
+
|
56
|
+
def question(question)
|
57
|
+
send(:"#{question_type(question)}_question", question)
|
58
|
+
end
|
59
|
+
|
60
|
+
def check_box_question(question)
|
61
|
+
{
|
62
|
+
**base_question(question),
|
63
|
+
children: children(question),
|
64
|
+
checkAllOption: question.check_all_option,
|
65
|
+
uncheckAllOption: question.uncheck_all_option,
|
66
|
+
}.compact
|
67
|
+
end
|
68
|
+
|
69
|
+
def date_parts_question(question)
|
70
|
+
{
|
71
|
+
**base_question(question),
|
72
|
+
dateParts: question.components.map { |component|
|
73
|
+
{
|
74
|
+
part: component,
|
75
|
+
key: question.send("#{component}_key"),
|
76
|
+
}
|
77
|
+
},
|
78
|
+
}.compact
|
79
|
+
end
|
80
|
+
|
81
|
+
# deprecated
|
82
|
+
def hidden_question(question)
|
83
|
+
nil
|
84
|
+
end
|
85
|
+
|
86
|
+
def float_question(question)
|
87
|
+
number_question(question)
|
88
|
+
end
|
89
|
+
|
90
|
+
def integer_question(question)
|
91
|
+
{
|
92
|
+
**number_question(question),
|
93
|
+
**split_to_units_question(question),
|
94
|
+
}.compact
|
95
|
+
end
|
96
|
+
|
97
|
+
def number_question(question)
|
98
|
+
{
|
99
|
+
**base_question(question),
|
100
|
+
**slider_question(question),
|
101
|
+
minimum: question.minimum,
|
102
|
+
maximum: question.maximum,
|
103
|
+
size: size(question),
|
104
|
+
unit: question.as != :slider && question.unit,
|
105
|
+
}.compact
|
106
|
+
end
|
107
|
+
|
108
|
+
def radio_question(question)
|
109
|
+
{
|
110
|
+
**base_question(question),
|
111
|
+
children: children(question),
|
112
|
+
showValues: [true, :all].include?(question.show_values),
|
113
|
+
}.compact
|
114
|
+
end
|
115
|
+
|
116
|
+
def scale_question(question)
|
117
|
+
radio_question(question)
|
118
|
+
end
|
119
|
+
|
120
|
+
def select_question(question)
|
121
|
+
{
|
122
|
+
**base_question(question),
|
123
|
+
children: children(question),
|
124
|
+
placeholder: question.options.find { _1.placeholder }&.description,
|
125
|
+
}.compact
|
126
|
+
end
|
127
|
+
|
128
|
+
def string_question(question)
|
129
|
+
{
|
130
|
+
**base_question(question),
|
131
|
+
setsTextvar: question.sets_textvar,
|
132
|
+
size: size(question),
|
133
|
+
unit: question.as != :slider && question.unit,
|
134
|
+
}.compact
|
135
|
+
end
|
136
|
+
|
137
|
+
def textarea_question(question)
|
138
|
+
{
|
139
|
+
**base_question(question),
|
140
|
+
autocomplete: question.autocomplete,
|
141
|
+
lines: question.lines,
|
142
|
+
}.compact
|
143
|
+
end
|
144
|
+
|
145
|
+
def base_question(question)
|
146
|
+
{
|
147
|
+
key: question.key,
|
148
|
+
title: handle_html(question.title),
|
149
|
+
description: handle_html(question.description, type: :question_description),
|
150
|
+
contextDescription: handle_html(question.context_description, type: :prose, v1_markdown: false),
|
151
|
+
type: question_type(question),
|
152
|
+
hidden: question.hidden?,
|
153
|
+
displayModes: question.display_modes,
|
154
|
+
viewSelector: question.view_selector,
|
155
|
+
parentKey: question.parent&.key,
|
156
|
+
parentOptionKey: question.parent_option_key,
|
157
|
+
deselectable: question.deselectable,
|
158
|
+
presentation: question.presentation,
|
159
|
+
as: question.as || question_type(question), # default to type so typescript can narrow on it.
|
160
|
+
questionGroup: question.question_group,
|
161
|
+
}
|
162
|
+
end
|
163
|
+
|
164
|
+
def slider_question(question)
|
165
|
+
return {} unless question.as == :slider
|
166
|
+
|
167
|
+
{
|
168
|
+
step: question.step,
|
169
|
+
defaultPosition: question.default_position.is_a?(Numeric) ? question.default_position : question.minimum,
|
170
|
+
startThumbHidden: question.default_position == :hidden,
|
171
|
+
valueTooltip: question.input_data[:value_tooltip] || false,
|
172
|
+
labels: question.labels,
|
173
|
+
}
|
174
|
+
end
|
175
|
+
|
176
|
+
def split_to_units_question(question)
|
177
|
+
{
|
178
|
+
units: question.units,
|
179
|
+
conversions: question.conversions,
|
180
|
+
}
|
181
|
+
end
|
182
|
+
|
183
|
+
def question_type(question)
|
184
|
+
{
|
185
|
+
date: 'date_parts',
|
186
|
+
}[question.type] || question.type
|
187
|
+
end
|
188
|
+
|
189
|
+
def size(question)
|
190
|
+
question.size.presence && Integer(question.size) # 2022-11: 4k string and 7k integer
|
191
|
+
end
|
192
|
+
|
193
|
+
def children(question)
|
194
|
+
question.options.map { |child|
|
195
|
+
if child.inner_title
|
196
|
+
inner_title_as_json(child)
|
197
|
+
elsif child.placeholder
|
198
|
+
nil # placeholder attr on question.
|
199
|
+
else
|
200
|
+
option_as_json(child)
|
201
|
+
end
|
202
|
+
}.compact
|
203
|
+
end
|
204
|
+
|
205
|
+
def inner_title_as_json(option)
|
206
|
+
{
|
207
|
+
type: 'html',
|
208
|
+
key: SecureRandom.uuid,
|
209
|
+
html: handle_html(option.description)
|
210
|
+
}
|
211
|
+
end
|
212
|
+
|
213
|
+
def option_as_json(option)
|
214
|
+
{
|
215
|
+
type: 'option',
|
216
|
+
key: option.key,
|
217
|
+
value: option.question.type != :check_box && option.value,
|
218
|
+
description: option.question.type == :select ? option.description : handle_html(option.description),
|
219
|
+
questions: option.question.type != :select && option.questions.map{ question(_1) },
|
220
|
+
viewId: option.view_id
|
221
|
+
}.compact
|
222
|
+
end
|
223
|
+
|
224
|
+
def textvars
|
225
|
+
@questionnaire.textvars.to_h { |key, textvar|
|
226
|
+
[
|
227
|
+
key,
|
228
|
+
{
|
229
|
+
key: textvar.key,
|
230
|
+
default: textvar.default,
|
231
|
+
}
|
232
|
+
]
|
28
233
|
}
|
29
234
|
end
|
30
235
|
|
@@ -40,6 +245,27 @@ module Quby
|
|
40
245
|
end.as_json
|
41
246
|
end
|
42
247
|
end
|
248
|
+
|
249
|
+
def handle_html(html, type: :simple, v1_markdown: true)
|
250
|
+
if layout_version == :v2
|
251
|
+
case type
|
252
|
+
when :simple
|
253
|
+
html_sanitizer.sanitize(html, tags: %w[strong em sup sub br span], attributes: %w[class])
|
254
|
+
when :question_description
|
255
|
+
html_sanitizer.sanitize(html, tags: %w[strong em b i u sup sub p span br ul ol li], attributes: %w[class])
|
256
|
+
when :prose
|
257
|
+
html_sanitizer.sanitize(html, tags: %w[strong em b i u sup sub pre blockquote p span br ul ol li a h1 h2 h3 h4], attributes: %w[href class target])
|
258
|
+
end
|
259
|
+
elsif v1_markdown
|
260
|
+
Quby::Compiler::MarkdownParser.new(html).to_html
|
261
|
+
else
|
262
|
+
html
|
263
|
+
end
|
264
|
+
end
|
265
|
+
|
266
|
+
def html_sanitizer
|
267
|
+
@html_sanitize ||= Rails::HTML5::SafeListSanitizer.new
|
268
|
+
end
|
43
269
|
end
|
44
270
|
end
|
45
271
|
end
|
@@ -26,6 +26,7 @@ module Quby
|
|
26
26
|
validate_outcome_tables(questionnaire)
|
27
27
|
validate_markdown_fields(questionnaire) if questionnaire.validate_html
|
28
28
|
validate_raw_content_items(questionnaire) if questionnaire.validate_html
|
29
|
+
validate_sexp_variables(questionnaire)
|
29
30
|
# Some compilation errors are Exceptions (pure syntax errors) and some StandardErrors (NameErrors)
|
30
31
|
rescue Exception => exception # rubocop:disable Lint/RescueException
|
31
32
|
definition.errors.add(:sourcecode, message: "Questionnaire error: #{definition.key}\n" \
|
@@ -364,6 +365,12 @@ scores_schema tables to the resulting seed."
|
|
364
365
|
fail "#{key || html} contains invalid html: #{fragment.errors.map(&:to_s).join(', ')}."
|
365
366
|
end
|
366
367
|
|
368
|
+
def validate_sexp_variables(questionnaire)
|
369
|
+
questionnaire.sexp_variables.each_value do |sexp_variable|
|
370
|
+
sexp_variable.validate(questionnaire)
|
371
|
+
end
|
372
|
+
end
|
373
|
+
|
367
374
|
def delete_prefix(key, questionnaire)
|
368
375
|
key.delete_prefix("#{questionnaire.key}_")
|
369
376
|
end
|
data/lib/quby/compiler.rb
CHANGED
@@ -0,0 +1,12 @@
|
|
1
|
+
class Quby::InspectExcept < Module
|
2
|
+
def initialize(*excepts)
|
3
|
+
define_method :inspect do
|
4
|
+
prefix = "#<#{self.class}:0x#{self.__id__.to_s(16)}"
|
5
|
+
|
6
|
+
parts = (instance_variables - excepts).map do |var|
|
7
|
+
"#{var}=#{instance_variable_get(var).inspect}"
|
8
|
+
end
|
9
|
+
"#{prefix}\n #{parts.join(", ")}>"
|
10
|
+
end
|
11
|
+
end
|
12
|
+
end
|
metadata
CHANGED
@@ -1,13 +1,13 @@
|
|
1
1
|
--- !ruby/object:Gem::Specification
|
2
2
|
name: quby-compiler
|
3
3
|
version: !ruby/object:Gem::Version
|
4
|
-
version: 0.5.
|
4
|
+
version: 0.5.16
|
5
5
|
platform: ruby
|
6
6
|
authors:
|
7
7
|
- Marten Veldthuis
|
8
8
|
bindir: exe
|
9
9
|
cert_chain: []
|
10
|
-
date: 2025-
|
10
|
+
date: 2025-03-23 00:00:00.000000000 Z
|
11
11
|
dependencies:
|
12
12
|
- !ruby/object:Gem::Dependency
|
13
13
|
name: activemodel
|
@@ -165,6 +165,7 @@ files:
|
|
165
165
|
- lib/quby/compiler/dsl/questions/text_question_builder.rb
|
166
166
|
- lib/quby/compiler/dsl/score_builder.rb
|
167
167
|
- lib/quby/compiler/dsl/score_schema_builder.rb
|
168
|
+
- lib/quby/compiler/dsl/sexp_variable_builder.rb
|
168
169
|
- lib/quby/compiler/dsl/standardized_panel_generators.rb
|
169
170
|
- lib/quby/compiler/dsl/table_builder.rb
|
170
171
|
- lib/quby/compiler/entities.rb
|
@@ -188,6 +189,7 @@ files:
|
|
188
189
|
- lib/quby/compiler/entities/questionnaire.rb
|
189
190
|
- lib/quby/compiler/entities/questions/checkbox_question.rb
|
190
191
|
- lib/quby/compiler/entities/questions/concerns/slider.rb
|
192
|
+
- lib/quby/compiler/entities/questions/concerns/split_to_units.rb
|
191
193
|
- lib/quby/compiler/entities/questions/date_question.rb
|
192
194
|
- lib/quby/compiler/entities/questions/deprecated_question.rb
|
193
195
|
- lib/quby/compiler/entities/questions/float_question.rb
|
@@ -198,6 +200,8 @@ files:
|
|
198
200
|
- lib/quby/compiler/entities/questions/text_question.rb
|
199
201
|
- lib/quby/compiler/entities/score_calculation.rb
|
200
202
|
- lib/quby/compiler/entities/score_schema.rb
|
203
|
+
- lib/quby/compiler/entities/sexp_variable.rb
|
204
|
+
- lib/quby/compiler/entities/sexp_variables.rb
|
201
205
|
- lib/quby/compiler/entities/subscore_schema.rb
|
202
206
|
- lib/quby/compiler/entities/table.rb
|
203
207
|
- lib/quby/compiler/entities/text.rb
|
@@ -220,6 +224,7 @@ files:
|
|
220
224
|
- lib/quby/compiler/services/transform_quby1_values_into_quby2_values.rb
|
221
225
|
- lib/quby/compiler/type_validator.rb
|
222
226
|
- lib/quby/compiler/version.rb
|
227
|
+
- lib/quby/inspect_except.rb
|
223
228
|
- lib/quby/range_categories.rb
|
224
229
|
- lib/quby/settings.rb
|
225
230
|
- lib/quby/text_transformation.rb
|