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.
Files changed (29) hide show
  1. checksums.yaml +4 -4
  2. data/CHANGELOG.md +17 -0
  3. data/lib/quby/compiler/dsl/questionnaire_builder.rb +5 -0
  4. data/lib/quby/compiler/dsl/questions/base.rb +5 -0
  5. data/lib/quby/compiler/dsl/questions/integer_question_builder.rb +10 -0
  6. data/lib/quby/compiler/dsl/sexp_variable_builder.rb +57 -0
  7. data/lib/quby/compiler/entities/panel.rb +0 -18
  8. data/lib/quby/compiler/entities/question.rb +7 -24
  9. data/lib/quby/compiler/entities/question_option.rb +2 -34
  10. data/lib/quby/compiler/entities/questionnaire.rb +7 -0
  11. data/lib/quby/compiler/entities/questions/checkbox_question.rb +0 -8
  12. data/lib/quby/compiler/entities/questions/concerns/slider.rb +0 -14
  13. data/lib/quby/compiler/entities/questions/concerns/split_to_units.rb +58 -0
  14. data/lib/quby/compiler/entities/questions/date_question.rb +0 -13
  15. data/lib/quby/compiler/entities/questions/deprecated_question.rb +0 -4
  16. data/lib/quby/compiler/entities/questions/float_question.rb +0 -7
  17. data/lib/quby/compiler/entities/questions/integer_question.rb +2 -7
  18. data/lib/quby/compiler/entities/questions/radio_question.rb +0 -3
  19. data/lib/quby/compiler/entities/questions/select_question.rb +0 -6
  20. data/lib/quby/compiler/entities/questions/string_question.rb +0 -6
  21. data/lib/quby/compiler/entities/questions/text_question.rb +0 -3
  22. data/lib/quby/compiler/entities/sexp_variable.rb +50 -0
  23. data/lib/quby/compiler/entities/sexp_variables.rb +41 -0
  24. data/lib/quby/compiler/outputs/quby_frontend_v2_serializer.rb +230 -4
  25. data/lib/quby/compiler/services/definition_validator.rb +7 -0
  26. data/lib/quby/compiler/version.rb +1 -1
  27. data/lib/quby/compiler.rb +1 -0
  28. data/lib/quby/inspect_except.rb +12 -0
  29. metadata +7 -2
checksums.yaml CHANGED
@@ -1,7 +1,7 @@
1
1
  ---
2
2
  SHA256:
3
- metadata.gz: 35bdecff443479812941ad2ced1e6585228fb03881c4ba9c48af3468b48c544f
4
- data.tar.gz: 3cf5faea78f741c241c0f57d873480fcc3b028aa91942ff020e9970eaf332530
3
+ metadata.gz: 5fbd126d732f2ab8201f325d09041cb564138354150fbedc83f3f7e9f68a755f
4
+ data.tar.gz: df750a8516c8596596cf11a03f3ce758bdb745de607068d4b8b753ae62a95e17
5
5
  SHA512:
6
- metadata.gz: d0b431d1f495544c5541bc5b2e5ceb57ece1d6d5782cabf6bfa80b9d7bb251a33e9b13764ebc6f0ac726523883eab6ee04585d4e4b102cd10ee419a3466cf1e7
7
- data.tar.gz: 4f22422b8789dbad63cb5cc57550582971ef47c2958b829bae72e9d140081a4378797534c5d5ae7a9888f562f3cfd87ad87bac6cc0c620032610264cc55b46e2
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
@@ -8,10 +8,6 @@ module Quby
8
8
  def hidden?
9
9
  true
10
10
  end
11
-
12
- def as_json(options = {})
13
- super.merge(options: @options.as_json)
14
- end
15
11
  end
16
12
  end
17
13
  end
@@ -9,13 +9,6 @@ module Quby
9
9
  class FloatQuestion < Question
10
10
  include Concerns::Slider
11
11
 
12
- def as_json(options = {})
13
- super.merge(
14
- minimum: minimum,
15
- maximum: maximum
16
- )
17
- end
18
-
19
12
  def size
20
13
  @size || 30
21
14
  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,9 +5,6 @@ module Quby
5
5
  module Entities
6
6
  module Questions
7
7
  class RadioQuestion < Question
8
- def as_json(options = {})
9
- super.merge(children: @options.as_json)
10
- end
11
8
  end
12
9
  end
13
10
  end
@@ -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
@@ -5,9 +5,6 @@ module Quby
5
5
  module Entities
6
6
  module Questions
7
7
  class TextQuestion < Question
8
- def as_json(options = {})
9
- super.merge(autocomplete: @autocomplete, lines: lines)
10
- end
11
8
  end
12
9
  end
13
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: fields.question_hash.as_json,
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
@@ -1,5 +1,5 @@
1
1
  module Quby
2
2
  module Compiler
3
- VERSION = "0.5.14"
3
+ VERSION = "0.5.16"
4
4
  end
5
5
  end
data/lib/quby/compiler.rb CHANGED
@@ -15,6 +15,7 @@ module Quby
15
15
  end
16
16
  end
17
17
 
18
+ require 'quby/inspect_except'
18
19
  require 'quby/compiler/markdown_parser'
19
20
  require 'quby/range_categories'
20
21
  require 'quby/compiler/type_validator'
@@ -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.14
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-02-13 00:00:00.000000000 Z
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