hssc_surveyor 1.4.1.pre
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.
- data/.gitignore +28 -0
- data/.rspec +1 -0
- data/CHANGELOG.md +198 -0
- data/Gemfile +5 -0
- data/Gemfile.rails_version +19 -0
- data/MIT-LICENSE +20 -0
- data/README.md +186 -0
- data/Rakefile +105 -0
- data/app/controllers/surveyor_controller.rb +6 -0
- data/app/helpers/results_helper.rb +20 -0
- data/app/helpers/survey_form_builder.rb +37 -0
- data/app/helpers/surveyor_helper.rb +3 -0
- data/app/inputs/quiet_input.rb +5 -0
- data/app/inputs/surveyor_check_boxes_input.rb +35 -0
- data/app/inputs/surveyor_radio_input.rb +18 -0
- data/app/models/answer.rb +4 -0
- data/app/models/dependency.rb +4 -0
- data/app/models/dependency_condition.rb +4 -0
- data/app/models/question.rb +4 -0
- data/app/models/question_group.rb +5 -0
- data/app/models/response.rb +5 -0
- data/app/models/response_set.rb +4 -0
- data/app/models/survey.rb +4 -0
- data/app/models/survey_section.rb +5 -0
- data/app/models/survey_section_sweeper.rb +15 -0
- data/app/models/survey_translation.rb +5 -0
- data/app/models/validation.rb +4 -0
- data/app/models/validation_condition.rb +4 -0
- data/app/views/layouts/results.html.erb +13 -0
- data/app/views/layouts/surveyor_default.html.erb +12 -0
- data/app/views/partials/_answer.html.haml +25 -0
- data/app/views/partials/_dependents.html.haml +5 -0
- data/app/views/partials/_question.html.haml +41 -0
- data/app/views/partials/_question_group.html.haml +44 -0
- data/app/views/partials/_section.html.haml +12 -0
- data/app/views/partials/_section_menu.html.haml +12 -0
- data/app/views/surveyor/edit.html.haml +24 -0
- data/app/views/surveyor/export.json.rabl +85 -0
- data/app/views/surveyor/new.html.haml +24 -0
- data/app/views/surveyor/show.html.haml +74 -0
- data/app/views/surveyor/show.json.rabl +15 -0
- data/ci-exec.sh +56 -0
- data/config/routes.rb +8 -0
- data/cucumber.yml +10 -0
- data/doc/REPRESENTATIONS.md +34 -0
- data/doc/api_id_schema.json +7 -0
- data/doc/question types.png +0 -0
- data/doc/response_set_schema.json +54 -0
- data/doc/surveyor question combinations.png +0 -0
- data/doc/surveyor reject or delete decision matrix.png +0 -0
- data/doc/surveyor_models.png +0 -0
- data/doc/surveyor_models2.png +0 -0
- data/doc/surveyor_timestamp_schema.json +9 -0
- data/features/ajax_submissions.feature +140 -0
- data/features/export_to_json.feature +344 -0
- data/features/internationalization.feature +121 -0
- data/features/no_duplicates.feature +110 -0
- data/features/show_survey.feature +71 -0
- data/features/step_definitions/parser_steps.rb +145 -0
- data/features/step_definitions/surveyor_steps.rb +325 -0
- data/features/step_definitions/ui_steps.rb +25 -0
- data/features/step_definitions/web_steps.rb +225 -0
- data/features/support/REDCapDemoDatabase_DataDictionary.csv +127 -0
- data/features/support/database_cleaner.rb +16 -0
- data/features/support/env.rb +56 -0
- data/features/support/hooks.rb +4 -0
- data/features/support/paths.rb +39 -0
- data/features/support/redcap_new_headers.csv +1 -0
- data/features/support/redcap_siblings.csv +1 -0
- data/features/support/redcap_whitespace.csv +1 -0
- data/features/support/selectors.rb +39 -0
- data/features/support/simultaneous_ajax.rb +101 -0
- data/features/support/single_quit_selenium_driver.rb +23 -0
- data/features/support/slow_updates.rb +18 -0
- data/features/surveyor.feature +895 -0
- data/features/surveyor_dependencies.feature +476 -0
- data/features/surveyor_parser.feature +504 -0
- data/features/z_redcap_parser.feature +62 -0
- data/lib/assets/images/surveyor/next.gif +0 -0
- data/lib/assets/images/surveyor/prev.gif +0 -0
- data/lib/assets/images/ui-bg_flat_0_aaaaaa_40x100.png +0 -0
- data/lib/assets/images/ui-bg_flat_75_ffffff_40x100.png +0 -0
- data/lib/assets/images/ui-bg_glass_55_fbf9ee_1x400.png +0 -0
- data/lib/assets/images/ui-bg_glass_65_ffffff_1x400.png +0 -0
- data/lib/assets/images/ui-bg_glass_75_dadada_1x400.png +0 -0
- data/lib/assets/images/ui-bg_glass_75_e6e6e6_1x400.png +0 -0
- data/lib/assets/images/ui-bg_glass_95_fef1ec_1x400.png +0 -0
- data/lib/assets/images/ui-bg_highlight-soft_75_cccccc_1x100.png +0 -0
- data/lib/assets/images/ui-icons_222222_256x240.png +0 -0
- data/lib/assets/images/ui-icons_2e83ff_256x240.png +0 -0
- data/lib/assets/images/ui-icons_454545_256x240.png +0 -0
- data/lib/assets/images/ui-icons_888888_256x240.png +0 -0
- data/lib/assets/images/ui-icons_cd0a0a_256x240.png +0 -0
- data/lib/assets/javascripts/surveyor/jquery-1.9.0.js +9555 -0
- data/lib/assets/javascripts/surveyor/jquery-ui-1.10.0.custom.js +14850 -0
- data/lib/assets/javascripts/surveyor/jquery-ui-timepicker-addon.js +1919 -0
- data/lib/assets/javascripts/surveyor/jquery.maskedinput.js +338 -0
- data/lib/assets/javascripts/surveyor/jquery.selectToUISlider.js +240 -0
- data/lib/assets/javascripts/surveyor/jquery.surveyor.js +156 -0
- data/lib/assets/stylesheets/surveyor/jquery-ui-1.10.0.custom.css +1174 -0
- data/lib/assets/stylesheets/surveyor/jquery-ui-timepicker-addon.css +11 -0
- data/lib/assets/stylesheets/surveyor/reset.css +48 -0
- data/lib/assets/stylesheets/surveyor/results.css +125 -0
- data/lib/assets/stylesheets/surveyor/ui.slider.extras.css +110 -0
- data/lib/assets/stylesheets/surveyor.sass +132 -0
- data/lib/generators/surveyor/custom_generator.rb +18 -0
- data/lib/generators/surveyor/install_generator.rb +102 -0
- data/lib/generators/surveyor/templates/app/assets/javascripts/surveyor_all.js +6 -0
- data/lib/generators/surveyor/templates/app/assets/stylesheets/surveyor_all.css +9 -0
- data/lib/generators/surveyor/templates/app/controllers/surveyor_controller.rb +40 -0
- data/lib/generators/surveyor/templates/app/views/layouts/surveyor_custom.html.erb +13 -0
- data/lib/generators/surveyor/templates/config/locales/surveyor_en.yml +19 -0
- data/lib/generators/surveyor/templates/config/locales/surveyor_es.yml +19 -0
- data/lib/generators/surveyor/templates/config/locales/surveyor_he.yml +19 -0
- data/lib/generators/surveyor/templates/config/locales/surveyor_ko.yml +19 -0
- data/lib/generators/surveyor/templates/db/migrate/add_api_id_to_question_groups.rb +10 -0
- data/lib/generators/surveyor/templates/db/migrate/add_api_ids.rb +14 -0
- data/lib/generators/surveyor/templates/db/migrate/add_api_ids_to_response_sets_and_responses.rb +12 -0
- data/lib/generators/surveyor/templates/db/migrate/add_attachment_to_response.rb +5 -0
- data/lib/generators/surveyor/templates/db/migrate/add_correct_answer_id_to_questions.rb +10 -0
- data/lib/generators/surveyor/templates/db/migrate/add_default_value_to_answers.rb +10 -0
- data/lib/generators/surveyor/templates/db/migrate/add_display_order_to_surveys.rb +10 -0
- data/lib/generators/surveyor/templates/db/migrate/add_display_type_to_answers.rb +14 -0
- data/lib/generators/surveyor/templates/db/migrate/add_index_to_response_sets.rb +10 -0
- data/lib/generators/surveyor/templates/db/migrate/add_index_to_surveys.rb +10 -0
- data/lib/generators/surveyor/templates/db/migrate/add_input_mask_attributes_to_answer.rb +12 -0
- data/lib/generators/surveyor/templates/db/migrate/add_section_id_to_responses.rb +12 -0
- data/lib/generators/surveyor/templates/db/migrate/add_unique_index_on_access_code_and_version_in_surveys.rb +10 -0
- data/lib/generators/surveyor/templates/db/migrate/add_unique_indicies.rb +18 -0
- data/lib/generators/surveyor/templates/db/migrate/add_version_to_surveys.rb +10 -0
- data/lib/generators/surveyor/templates/db/migrate/api_ids_must_be_unique.rb +23 -0
- data/lib/generators/surveyor/templates/db/migrate/create_answers.rb +38 -0
- data/lib/generators/surveyor/templates/db/migrate/create_dependencies.rb +23 -0
- data/lib/generators/surveyor/templates/db/migrate/create_dependency_conditions.rb +30 -0
- data/lib/generators/surveyor/templates/db/migrate/create_question_groups.rb +28 -0
- data/lib/generators/surveyor/templates/db/migrate/create_questions.rb +37 -0
- data/lib/generators/surveyor/templates/db/migrate/create_response_sets.rb +23 -0
- data/lib/generators/surveyor/templates/db/migrate/create_responses.rb +34 -0
- data/lib/generators/surveyor/templates/db/migrate/create_survey_sections.rb +30 -0
- data/lib/generators/surveyor/templates/db/migrate/create_survey_translations.rb +19 -0
- data/lib/generators/surveyor/templates/db/migrate/create_surveys.rb +32 -0
- data/lib/generators/surveyor/templates/db/migrate/create_validation_conditions.rb +33 -0
- data/lib/generators/surveyor/templates/db/migrate/create_validations.rb +21 -0
- data/lib/generators/surveyor/templates/db/migrate/drop_unique_index_on_access_code_in_surveys.rb +10 -0
- data/lib/generators/surveyor/templates/db/migrate/update_blank_api_ids_on_question_group.rb +22 -0
- data/lib/generators/surveyor/templates/db/migrate/update_blank_versions_on_surveys.rb +13 -0
- data/lib/generators/surveyor/templates/surveys/EXTENDING_SURVEYOR.md +52 -0
- data/lib/generators/surveyor/templates/surveys/MODIFYING_SURVEYOR.md +91 -0
- data/lib/generators/surveyor/templates/surveys/date_survey.rb +16 -0
- data/lib/generators/surveyor/templates/surveys/kitchen_sink_survey.rb +284 -0
- data/lib/generators/surveyor/templates/surveys/languages.rb +14 -0
- data/lib/generators/surveyor/templates/surveys/quiz.rb +11 -0
- data/lib/generators/surveyor/templates/surveys/translations/languages.es.yml +18 -0
- data/lib/generators/surveyor/templates/surveys/translations/languages.he.yml +18 -0
- data/lib/generators/surveyor/templates/surveys/translations/languages.ko.yml +18 -0
- data/lib/generators/surveyor/templates/vendor/assets/stylesheets/custom.sass +5 -0
- data/lib/surveyor/acts_as_response.rb +17 -0
- data/lib/surveyor/common.rb +59 -0
- data/lib/surveyor/engine.rb +10 -0
- data/lib/surveyor/helpers/asset_pipeline.rb +13 -0
- data/lib/surveyor/helpers/formtastic_custom_input.rb +13 -0
- data/lib/surveyor/helpers/surveyor_helper_methods.rb +103 -0
- data/lib/surveyor/models/answer_methods.rb +86 -0
- data/lib/surveyor/models/dependency_condition_methods.rb +72 -0
- data/lib/surveyor/models/dependency_methods.rb +60 -0
- data/lib/surveyor/models/question_group_methods.rb +61 -0
- data/lib/surveyor/models/question_methods.rb +115 -0
- data/lib/surveyor/models/response_methods.rb +132 -0
- data/lib/surveyor/models/response_set_methods.rb +196 -0
- data/lib/surveyor/models/survey_methods.rb +103 -0
- data/lib/surveyor/models/survey_section_methods.rb +56 -0
- data/lib/surveyor/models/survey_translation_methods.rb +33 -0
- data/lib/surveyor/models/validation_condition_methods.rb +56 -0
- data/lib/surveyor/models/validation_methods.rb +48 -0
- data/lib/surveyor/mustache_context.rb +11 -0
- data/lib/surveyor/parser.rb +428 -0
- data/lib/surveyor/redcap_parser.rb +289 -0
- data/lib/surveyor/surveyor_controller_methods.rb +237 -0
- data/lib/surveyor/unparser.rb +147 -0
- data/lib/surveyor/version.rb +3 -0
- data/lib/surveyor.rb +13 -0
- data/lib/tasks/surveyor_tasks.rake +88 -0
- data/rails/init.rb +1 -0
- data/spec/controllers/surveyor_controller_spec.rb +307 -0
- data/spec/factories.rb +161 -0
- data/spec/helpers/formtastic_custom_input_spec.rb +15 -0
- data/spec/helpers/surveyor_helper_spec.rb +118 -0
- data/spec/lib/benchmark_spec.rb +22 -0
- data/spec/lib/chinese_survey.rb +14 -0
- data/spec/lib/common_spec.rb +34 -0
- data/spec/lib/parser_spec.rb +204 -0
- data/spec/lib/rake_kitchen_sink.rb +40 -0
- data/spec/lib/redcap_parser_spec.rb +75 -0
- data/spec/lib/tasks_spec.rake +26 -0
- data/spec/lib/unparser_spec.rb +126 -0
- data/spec/models/answer_spec.rb +175 -0
- data/spec/models/dependency_condition_spec.rb +439 -0
- data/spec/models/dependency_spec.rb +101 -0
- data/spec/models/question_group_spec.rb +93 -0
- data/spec/models/question_spec.rb +207 -0
- data/spec/models/response_set_spec.rb +477 -0
- data/spec/models/response_spec.rb +218 -0
- data/spec/models/survey_section_spec.rb +85 -0
- data/spec/models/survey_spec.rb +191 -0
- data/spec/models/validation_condition_spec.rb +113 -0
- data/spec/models/validation_spec.rb +74 -0
- data/spec/rcov.opts +2 -0
- data/spec/spec.opts +4 -0
- data/spec/spec_helper.rb +56 -0
- data/stacktests.sh +65 -0
- data/surveyor.gemspec +45 -0
- metadata +657 -0
@@ -0,0 +1,439 @@
|
|
1
|
+
require File.expand_path(File.dirname(__FILE__) + '/../spec_helper')
|
2
|
+
|
3
|
+
|
4
|
+
describe DependencyCondition do
|
5
|
+
it "should have a list of operators" do
|
6
|
+
%w(== != < > <= >=).each do |operator|
|
7
|
+
DependencyCondition.operators.include?(operator).should be_true
|
8
|
+
end
|
9
|
+
end
|
10
|
+
|
11
|
+
describe "instance" do
|
12
|
+
before(:each) do
|
13
|
+
@dependency_condition = DependencyCondition.new(
|
14
|
+
:dependency_id => 1, :question_id => 45, :operator => "==",
|
15
|
+
:answer_id => 23, :rule_key => "A")
|
16
|
+
end
|
17
|
+
|
18
|
+
it "should be valid" do
|
19
|
+
@dependency_condition.should be_valid
|
20
|
+
end
|
21
|
+
|
22
|
+
it "should be invalid without a parent dependency_id, question_id" do
|
23
|
+
# this causes issues with building and saving
|
24
|
+
# @dependency_condition.dependency_id = nil
|
25
|
+
# @dependency_condition.should have(1).errors_on(:dependency_id)
|
26
|
+
# @dependency_condition.question_id = nil
|
27
|
+
# @dependency_condition.should have(1).errors_on(:question_id)
|
28
|
+
end
|
29
|
+
|
30
|
+
it "should be invalid without an operator" do
|
31
|
+
@dependency_condition.operator = nil
|
32
|
+
@dependency_condition.should have(2).errors_on(:operator)
|
33
|
+
end
|
34
|
+
|
35
|
+
it "should be invalid without a rule_key" do
|
36
|
+
@dependency_condition.should be_valid
|
37
|
+
@dependency_condition.rule_key = nil
|
38
|
+
@dependency_condition.should_not be_valid
|
39
|
+
@dependency_condition.should have(1).errors_on(:rule_key)
|
40
|
+
end
|
41
|
+
|
42
|
+
it "should have unique rule_key within the context of a dependency" do
|
43
|
+
@dependency_condition.should be_valid
|
44
|
+
DependencyCondition.create(
|
45
|
+
:dependency_id => 2, :question_id => 46, :operator => "==",
|
46
|
+
:answer_id => 14, :rule_key => "B")
|
47
|
+
@dependency_condition.rule_key = "B" # rule key uniquness is scoped by dependency_id
|
48
|
+
@dependency_condition.dependency_id = 2
|
49
|
+
@dependency_condition.should_not be_valid
|
50
|
+
@dependency_condition.should have(1).errors_on(:rule_key)
|
51
|
+
end
|
52
|
+
|
53
|
+
it "should have an operator in DependencyCondition.operators" do
|
54
|
+
DependencyCondition.operators.each do |o|
|
55
|
+
@dependency_condition.operator = o
|
56
|
+
@dependency_condition.should have(0).errors_on(:operator)
|
57
|
+
end
|
58
|
+
@dependency_condition.operator = "#"
|
59
|
+
@dependency_condition.should have(1).error_on(:operator)
|
60
|
+
end
|
61
|
+
it "should have a properly formed count operator" do
|
62
|
+
%w(count>1 count<1 count>=1 count<=1 count==1 count!=1).each do |o|
|
63
|
+
@dependency_condition.operator = o
|
64
|
+
@dependency_condition.should have(0).errors_on(:operator)
|
65
|
+
end
|
66
|
+
%w(count> count< count>= count<= count== count!=).each do |o|
|
67
|
+
@dependency_condition.operator = o
|
68
|
+
@dependency_condition.should have(1).errors_on(:operator)
|
69
|
+
end
|
70
|
+
%w(count=1 count><1 count<>1 count!1 count!!1 count=>1 count=<1).each do |o|
|
71
|
+
@dependency_condition.operator = o
|
72
|
+
@dependency_condition.should have(1).errors_on(:operator)
|
73
|
+
end
|
74
|
+
%w(count= count>< count<> count! count!! count=> count=< count> count< count>= count<= count== count!=).each do |o|
|
75
|
+
@dependency_condition.operator = o
|
76
|
+
@dependency_condition.should have(1).errors_on(:operator)
|
77
|
+
end
|
78
|
+
end
|
79
|
+
|
80
|
+
it "should protect timestamps" do
|
81
|
+
saved_attrs = @dependency_condition.attributes
|
82
|
+
if defined? ActiveModel::MassAssignmentSecurity::Error
|
83
|
+
lambda {@dependency_condition.update_attributes(:created_at => 3.days.ago, :updated_at => 3.hours.ago)}.should raise_error(ActiveModel::MassAssignmentSecurity::Error)
|
84
|
+
else
|
85
|
+
@dependency_condition.attributes = {:created_at => 3.days.ago, :updated_at => 3.hours.ago} # automatically protected by Rails
|
86
|
+
end
|
87
|
+
@dependency_condition.attributes.should == saved_attrs
|
88
|
+
end
|
89
|
+
|
90
|
+
end
|
91
|
+
|
92
|
+
it "returns true for != with no responses" do
|
93
|
+
question = Factory(:question)
|
94
|
+
dependency_condition = Factory(:dependency_condition, :rule_key => "C", :question => question)
|
95
|
+
rs = Factory(:response_set)
|
96
|
+
dependency_condition.to_hash(rs).should == {:C => false}
|
97
|
+
end
|
98
|
+
|
99
|
+
|
100
|
+
it "should not assume that Response#as is not nil" do
|
101
|
+
# q_HEIGHT_FT "Portion of height in whole feet (e.g., 5)",
|
102
|
+
# :pick=>:one
|
103
|
+
# a :integer
|
104
|
+
# a_neg_1 "Refused"
|
105
|
+
# a_neg_2 "Don't know"
|
106
|
+
# label "Provided value is outside of the suggested range (4 to 7 feet). This value is admissible, but you may wish to verify."
|
107
|
+
# dependency :rule=>"A or B"
|
108
|
+
# condition_A :q_HEIGHT_FT, "<", {:integer_value => "4"}
|
109
|
+
# condition_B :q_HEIGHT_FT, ">", {:integer_value => "7"}
|
110
|
+
|
111
|
+
answer = Factory(:answer, :response_class => :integer)
|
112
|
+
@dependency_condition = DependencyCondition.new(
|
113
|
+
:dependency => Factory(:dependency),
|
114
|
+
:question => answer.question,
|
115
|
+
:answer => answer,
|
116
|
+
:operator => ">",
|
117
|
+
:integer_value => 4,
|
118
|
+
:rule_key => "A")
|
119
|
+
|
120
|
+
response = Factory(:response, :answer => answer, :question => answer.question)
|
121
|
+
response_set = response.response_set
|
122
|
+
response.integer_value.should == nil
|
123
|
+
|
124
|
+
@dependency_condition.to_hash(response_set).should == {:A => false}
|
125
|
+
end
|
126
|
+
|
127
|
+
describe "evaluate '==' operator" do
|
128
|
+
before(:each) do
|
129
|
+
@a = Factory(:answer, :response_class => "answer")
|
130
|
+
@b = Factory(:answer, :question => @a.question)
|
131
|
+
@r = Factory(:response, :question => @a.question, :answer => @a)
|
132
|
+
@rs = @r.response_set
|
133
|
+
@dc = Factory(:dependency_condition, :question => @a.question, :answer => @a, :operator => "==", :rule_key => "D")
|
134
|
+
@dc.as(:answer).should == @r.as(:answer)
|
135
|
+
end
|
136
|
+
|
137
|
+
it "with checkbox/radio type response" do
|
138
|
+
@dc.to_hash(@rs).should == {:D => true}
|
139
|
+
@dc.answer = @b
|
140
|
+
@dc.to_hash(@rs).should == {:D => false}
|
141
|
+
end
|
142
|
+
|
143
|
+
it "with string value response" do
|
144
|
+
@a.update_attributes(:response_class => "string")
|
145
|
+
@r.update_attributes(:string_value => "hello123")
|
146
|
+
@dc.string_value = "hello123"
|
147
|
+
@dc.to_hash(@rs).should == {:D => true}
|
148
|
+
@r.update_attributes(:string_value => "foo_abc")
|
149
|
+
@dc.to_hash(@rs).should == {:D => false}
|
150
|
+
end
|
151
|
+
|
152
|
+
it "with a text value response" do
|
153
|
+
@a.update_attributes(:response_class => "text")
|
154
|
+
@r.update_attributes(:text_value => "hello this is some text for comparison")
|
155
|
+
@dc.text_value = "hello this is some text for comparison"
|
156
|
+
@dc.to_hash(@rs).should == {:D => true}
|
157
|
+
@r.update_attributes(:text_value => "Not the same text")
|
158
|
+
@dc.to_hash(@rs).should == {:D => false}
|
159
|
+
end
|
160
|
+
|
161
|
+
it "with an integer value response" do
|
162
|
+
@a.update_attributes(:response_class => "integer")
|
163
|
+
@r.update_attributes(:integer_value => 10045)
|
164
|
+
@dc.integer_value = 10045
|
165
|
+
@dc.to_hash(@rs).should == {:D => true}
|
166
|
+
@r.update_attributes(:integer_value => 421)
|
167
|
+
@dc.to_hash(@rs).should == {:D => false}
|
168
|
+
end
|
169
|
+
|
170
|
+
it "with a float value response" do
|
171
|
+
@a.update_attributes(:response_class => "float")
|
172
|
+
@r.update_attributes(:float_value => 121.1)
|
173
|
+
@dc.float_value = 121.1
|
174
|
+
@dc.to_hash(@rs).should == {:D => true}
|
175
|
+
@r.update_attributes(:float_value => 130.123)
|
176
|
+
@dc.to_hash(@rs).should == {:D => false}
|
177
|
+
end
|
178
|
+
end
|
179
|
+
|
180
|
+
describe "evaluate '!=' operator" do
|
181
|
+
before(:each) do
|
182
|
+
@a = Factory(:answer)
|
183
|
+
@b = Factory(:answer, :question => @a.question)
|
184
|
+
@r = Factory(:response, :question => @a.question, :answer => @a)
|
185
|
+
@rs = @r.response_set
|
186
|
+
@dc = Factory(:dependency_condition, :question => @a.question, :answer => @a, :operator => "!=", :rule_key => "E")
|
187
|
+
@dc.as(:answer).should == @r.as(:answer)
|
188
|
+
end
|
189
|
+
|
190
|
+
it "with checkbox/radio type response" do
|
191
|
+
@dc.to_hash(@rs).should == {:E => false}
|
192
|
+
@dc.answer_id = @a.id.to_i+1
|
193
|
+
@dc.to_hash(@rs).should == {:E => true}
|
194
|
+
end
|
195
|
+
|
196
|
+
it "with string value response" do
|
197
|
+
@a.update_attributes(:response_class => "string")
|
198
|
+
@r.update_attributes(:string_value => "hello123")
|
199
|
+
@dc.string_value = "hello123"
|
200
|
+
@dc.to_hash(@rs).should == {:E => false}
|
201
|
+
@r.update_attributes(:string_value => "foo_abc")
|
202
|
+
@dc.to_hash(@rs).should == {:E => true}
|
203
|
+
end
|
204
|
+
|
205
|
+
it "with a text value response" do
|
206
|
+
@a.update_attributes(:response_class => "text")
|
207
|
+
@r.update_attributes(:text_value => "hello this is some text for comparison")
|
208
|
+
@dc.text_value = "hello this is some text for comparison"
|
209
|
+
@dc.to_hash(@rs).should == {:E => false}
|
210
|
+
@r.update_attributes(:text_value => "Not the same text")
|
211
|
+
@dc.to_hash(@rs).should == {:E => true}
|
212
|
+
end
|
213
|
+
|
214
|
+
it "with an integer value response" do
|
215
|
+
@a.update_attributes(:response_class => "integer")
|
216
|
+
@r.update_attributes(:integer_value => 10045)
|
217
|
+
@dc.integer_value = 10045
|
218
|
+
@dc.to_hash(@rs).should == {:E => false}
|
219
|
+
@r.update_attributes(:integer_value => 421)
|
220
|
+
@dc.to_hash(@rs).should == {:E => true}
|
221
|
+
end
|
222
|
+
|
223
|
+
it "with a float value response" do
|
224
|
+
@a.update_attributes(:response_class => "float")
|
225
|
+
@r.update_attributes(:float_value => 121.1)
|
226
|
+
@dc.float_value = 121.1
|
227
|
+
@dc.to_hash(@rs).should == {:E => false}
|
228
|
+
@r.update_attributes(:float_value => 130.123)
|
229
|
+
@dc.to_hash(@rs).should == {:E => true}
|
230
|
+
end
|
231
|
+
end
|
232
|
+
|
233
|
+
describe "evaluate the '<' operator" do
|
234
|
+
before(:each) do
|
235
|
+
@a = Factory(:answer)
|
236
|
+
@b = Factory(:answer, :question => @a.question)
|
237
|
+
@r = Factory(:response, :question => @a.question, :answer => @a)
|
238
|
+
@rs = @r.response_set
|
239
|
+
@dc = Factory(:dependency_condition, :question => @a.question, :answer => @a, :operator => "<", :rule_key => "F")
|
240
|
+
@dc.as(:answer).should == @r.as(:answer)
|
241
|
+
end
|
242
|
+
|
243
|
+
it "with an integer value response" do
|
244
|
+
@a.update_attributes(:response_class => "integer")
|
245
|
+
@r.update_attributes(:integer_value => 50)
|
246
|
+
@dc.integer_value = 100
|
247
|
+
@dc.to_hash(@rs).should == {:F => true}
|
248
|
+
@r.update_attributes(:integer_value => 421)
|
249
|
+
@dc.to_hash(@rs).should == {:F => false}
|
250
|
+
end
|
251
|
+
|
252
|
+
it "with a float value response" do
|
253
|
+
@a.update_attributes(:response_class => "float")
|
254
|
+
@r.update_attributes(:float_value => 5.1)
|
255
|
+
@dc.float_value = 121.1
|
256
|
+
@dc.to_hash(@rs).should == {:F => true}
|
257
|
+
@r.update_attributes(:float_value => 130.123)
|
258
|
+
@dc.to_hash(@rs).should == {:F => false}
|
259
|
+
end
|
260
|
+
end
|
261
|
+
|
262
|
+
describe "evaluate the '<=' operator" do
|
263
|
+
before(:each) do
|
264
|
+
@a = Factory(:answer)
|
265
|
+
@b = Factory(:answer, :question => @a.question)
|
266
|
+
@r = Factory(:response, :question => @a.question, :answer => @a)
|
267
|
+
@rs = @r.response_set
|
268
|
+
@dc = Factory(:dependency_condition, :question => @a.question, :answer => @a, :operator => "<=", :rule_key => "G")
|
269
|
+
@dc.as(:answer).should == @r.as(:answer)
|
270
|
+
end
|
271
|
+
|
272
|
+
it "with an integer value response" do
|
273
|
+
@a.update_attributes(:response_class => "integer")
|
274
|
+
@r.update_attributes(:integer_value => 50)
|
275
|
+
@dc.integer_value = 100
|
276
|
+
@dc.to_hash(@rs).should == {:G => true}
|
277
|
+
@r.update_attributes(:integer_value => 100)
|
278
|
+
@dc.to_hash(@rs).should == {:G => true}
|
279
|
+
@r.update_attributes(:integer_value => 421)
|
280
|
+
@dc.to_hash(@rs).should == {:G => false}
|
281
|
+
end
|
282
|
+
|
283
|
+
it "with a float value response" do
|
284
|
+
@a.update_attributes(:response_class => "float")
|
285
|
+
@r.update_attributes(:float_value => 5.1)
|
286
|
+
@dc.float_value = 121.1
|
287
|
+
@dc.to_hash(@rs).should == {:G => true}
|
288
|
+
@r.update_attributes(:float_value => 121.1)
|
289
|
+
@dc.to_hash(@rs).should == {:G => true}
|
290
|
+
@r.update_attributes(:float_value => 130.123)
|
291
|
+
@dc.to_hash(@rs).should == {:G => false}
|
292
|
+
end
|
293
|
+
|
294
|
+
end
|
295
|
+
|
296
|
+
describe "evaluate the '>' operator" do
|
297
|
+
before(:each) do
|
298
|
+
@a = Factory(:answer)
|
299
|
+
@b = Factory(:answer, :question => @a.question)
|
300
|
+
@r = Factory(:response, :question => @a.question, :answer => @a)
|
301
|
+
@rs = @r.response_set
|
302
|
+
@dc = Factory(:dependency_condition, :question => @a.question, :answer => @a, :operator => ">", :rule_key => "H")
|
303
|
+
@dc.as(:answer).should == @r.as(:answer)
|
304
|
+
end
|
305
|
+
|
306
|
+
it "with an integer value response" do
|
307
|
+
@a.update_attributes(:response_class => "integer")
|
308
|
+
@r.update_attributes(:integer_value => 50)
|
309
|
+
@dc.integer_value = 100
|
310
|
+
@dc.to_hash(@rs).should == {:H => false}
|
311
|
+
@r.update_attributes(:integer_value => 421)
|
312
|
+
@dc.to_hash(@rs).should == {:H => true}
|
313
|
+
end
|
314
|
+
|
315
|
+
it "with a float value response" do
|
316
|
+
@a.update_attributes(:response_class => "float")
|
317
|
+
@r.update_attributes(:float_value => 5.1)
|
318
|
+
@dc.float_value = 121.1
|
319
|
+
@dc.to_hash(@rs).should == {:H => false}
|
320
|
+
@r.update_attributes(:float_value => 130.123)
|
321
|
+
@dc.to_hash(@rs).should == {:H => true}
|
322
|
+
end
|
323
|
+
end
|
324
|
+
|
325
|
+
describe "evaluate the '>=' operator" do
|
326
|
+
before(:each) do
|
327
|
+
@a = Factory(:answer)
|
328
|
+
@b = Factory(:answer, :question => @a.question)
|
329
|
+
@r = Factory(:response, :question => @a.question, :answer => @a)
|
330
|
+
@rs = @r.response_set
|
331
|
+
@dc = Factory(:dependency_condition, :question => @a.question, :answer => @a, :operator => ">=", :rule_key => "I")
|
332
|
+
@dc.as(:answer).should == @r.as(:answer)
|
333
|
+
end
|
334
|
+
|
335
|
+
it "with an integer value response" do
|
336
|
+
@a.update_attributes(:response_class => "integer")
|
337
|
+
@r.update_attributes(:integer_value => 50)
|
338
|
+
@dc.integer_value = 100
|
339
|
+
@dc.to_hash(@rs).should == {:I => false}
|
340
|
+
@r.update_attributes(:integer_value => 100)
|
341
|
+
@dc.to_hash(@rs).should == {:I => true}
|
342
|
+
@r.update_attributes(:integer_value => 421)
|
343
|
+
@dc.to_hash(@rs).should == {:I => true}
|
344
|
+
end
|
345
|
+
|
346
|
+
it "with a float value response" do
|
347
|
+
@a.update_attributes(:response_class => "float")
|
348
|
+
@r.update_attributes(:float_value => 5.1)
|
349
|
+
@dc.float_value = 121.1
|
350
|
+
@dc.to_hash(@rs).should == {:I => false}
|
351
|
+
@r.update_attributes(:float_value => 121.1)
|
352
|
+
@dc.to_hash(@rs).should == {:I => true}
|
353
|
+
@r.update_attributes(:float_value => 130.123)
|
354
|
+
@dc.to_hash(@rs).should == {:I => true}
|
355
|
+
end
|
356
|
+
end
|
357
|
+
|
358
|
+
describe "evaluating with response_class string" do
|
359
|
+
it "should compare answer ids when the dependency condition string_value is nil" do
|
360
|
+
@a = Factory(:answer, :response_class => "string")
|
361
|
+
@b = Factory(:answer, :question => @a.question)
|
362
|
+
@r = Factory(:response, :question => @a.question, :answer => @a, :string_value => "")
|
363
|
+
@rs = @r.response_set
|
364
|
+
@dc = Factory(:dependency_condition, :question => @a.question, :answer => @a, :operator => "==", :rule_key => "J")
|
365
|
+
@dc.to_hash(@rs).should == {:J => true}
|
366
|
+
end
|
367
|
+
|
368
|
+
it "should compare strings when the dependency condition string_value is not nil, even if it is blank" do
|
369
|
+
@a = Factory(:answer, :response_class => "string")
|
370
|
+
@b = Factory(:answer, :question => @a.question)
|
371
|
+
@r = Factory(:response, :question => @a.question, :answer => @a, :string_value => "foo")
|
372
|
+
@rs = @r.response_set
|
373
|
+
@dc = Factory(:dependency_condition, :question => @a.question, :answer => @a, :operator => "==", :rule_key => "K", :string_value => "foo")
|
374
|
+
@dc.to_hash(@rs).should == {:K => true}
|
375
|
+
|
376
|
+
@r.update_attributes(:string_value => "")
|
377
|
+
@dc.string_value = ""
|
378
|
+
@dc.to_hash(@rs).should == {:K => true}
|
379
|
+
end
|
380
|
+
end
|
381
|
+
|
382
|
+
describe "evaluate 'count' operator" do
|
383
|
+
before(:each) do
|
384
|
+
@q = Factory(:question)
|
385
|
+
@dc = DependencyCondition.new(:operator => "count>2", :rule_key => "M", :question => @q)
|
386
|
+
@as = []
|
387
|
+
3.times do
|
388
|
+
@as << Factory(:answer, :question => @q, :response_class => "answer")
|
389
|
+
end
|
390
|
+
@rs = Factory(:response_set)
|
391
|
+
@as.slice(0,2).each do |a|
|
392
|
+
Factory(:response, :question => @q, :answer => a, :response_set => @rs)
|
393
|
+
end
|
394
|
+
@rs.save
|
395
|
+
end
|
396
|
+
|
397
|
+
it "with operator with >" do
|
398
|
+
@dc.to_hash(@rs).should == {:M => false}
|
399
|
+
Factory(:response, :question => @q, :answer => @as.last, :response_set => @rs)
|
400
|
+
@rs.reload.responses.count.should == 3
|
401
|
+
@dc.to_hash(@rs.reload).should == {:M => true}
|
402
|
+
end
|
403
|
+
|
404
|
+
it "with operator with <" do
|
405
|
+
@dc.operator = "count<2"
|
406
|
+
@dc.to_hash(@rs).should == {:M => false}
|
407
|
+
@dc.operator = "count<3"
|
408
|
+
@dc.to_hash(@rs).should == {:M => true}
|
409
|
+
end
|
410
|
+
|
411
|
+
it "with operator with <=" do
|
412
|
+
@dc.operator = "count<=1"
|
413
|
+
@dc.to_hash(@rs).should == {:M => false}
|
414
|
+
@dc.operator = "count<=2"
|
415
|
+
@dc.to_hash(@rs).should == {:M => true}
|
416
|
+
@dc.operator = "count<=3"
|
417
|
+
@dc.to_hash(@rs).should == {:M => true}
|
418
|
+
end
|
419
|
+
|
420
|
+
it "with operator with >=" do
|
421
|
+
@dc.operator = "count>=1"
|
422
|
+
@dc.to_hash(@rs).should == {:M => true}
|
423
|
+
@dc.operator = "count>=2"
|
424
|
+
@dc.to_hash(@rs).should == {:M => true}
|
425
|
+
@dc.operator = "count>=3"
|
426
|
+
@dc.to_hash(@rs).should == {:M => false}
|
427
|
+
end
|
428
|
+
|
429
|
+
it "with operator with !=" do
|
430
|
+
@dc.operator = "count!=1"
|
431
|
+
@dc.to_hash(@rs).should == {:M => true}
|
432
|
+
@dc.operator = "count!=2"
|
433
|
+
@dc.to_hash(@rs).should == {:M => false}
|
434
|
+
@dc.operator = "count!=3"
|
435
|
+
@dc.to_hash(@rs).should == {:M => true}
|
436
|
+
end
|
437
|
+
end
|
438
|
+
|
439
|
+
end
|
@@ -0,0 +1,101 @@
|
|
1
|
+
require File.expand_path(File.dirname(__FILE__) + '/../spec_helper')
|
2
|
+
|
3
|
+
describe Dependency do
|
4
|
+
before(:each) do
|
5
|
+
@dependency = Factory(:dependency)
|
6
|
+
end
|
7
|
+
|
8
|
+
it "should be valid" do
|
9
|
+
@dependency.should be_valid
|
10
|
+
end
|
11
|
+
|
12
|
+
it "should be invalid without a rule" do
|
13
|
+
@dependency.rule = nil
|
14
|
+
@dependency.should have(2).errors_on(:rule)
|
15
|
+
@dependency.rule = " "
|
16
|
+
@dependency.should have(1).errors_on(:rule)
|
17
|
+
end
|
18
|
+
|
19
|
+
it "should be invalid without a question_id" do
|
20
|
+
@dependency.question_id = nil
|
21
|
+
@dependency.should have(1).error_on(:question_id)
|
22
|
+
|
23
|
+
@dependency.question_group_id = 1
|
24
|
+
@dependency.should be_valid
|
25
|
+
|
26
|
+
@dependency.question_id.should be_nil
|
27
|
+
@dependency.question_group_id = nil
|
28
|
+
@dependency.should have(1).error_on(:question_group_id)
|
29
|
+
end
|
30
|
+
|
31
|
+
it "should alias question_id as dependent_question_id" do
|
32
|
+
@dependency.question_id = 19
|
33
|
+
@dependency.dependent_question_id.should == 19
|
34
|
+
@dependency.dependent_question_id = 14
|
35
|
+
@dependency.question_id.should == 14
|
36
|
+
end
|
37
|
+
|
38
|
+
it "should be invalid unless rule composed of only references and operators" do
|
39
|
+
@dependency.rule = "foo"
|
40
|
+
@dependency.should have(1).error_on(:rule)
|
41
|
+
@dependency.rule = "1 to 2"
|
42
|
+
@dependency.should have(1).error_on(:rule)
|
43
|
+
@dependency.rule = "a and b"
|
44
|
+
@dependency.should have(1).error_on(:rule)
|
45
|
+
end
|
46
|
+
it "should protect timestamps" do
|
47
|
+
saved_attrs = @dependency.attributes
|
48
|
+
if defined? ActiveModel::MassAssignmentSecurity::Error
|
49
|
+
lambda {@dependency.update_attributes(:created_at => 3.days.ago, :updated_at => 3.hours.ago)}.should raise_error(ActiveModel::MassAssignmentSecurity::Error)
|
50
|
+
else
|
51
|
+
@dependency.attributes = {:created_at => 3.days.ago, :updated_at => 3.hours.ago} # automatically protected by Rails
|
52
|
+
end
|
53
|
+
@dependency.attributes.should == saved_attrs
|
54
|
+
end
|
55
|
+
|
56
|
+
|
57
|
+
end
|
58
|
+
|
59
|
+
describe Dependency, "when evaluating dependency conditions of a question in a response set" do
|
60
|
+
|
61
|
+
before(:each) do
|
62
|
+
@dep = Dependency.new(:rule => "A", :question_id => 1)
|
63
|
+
@dep2 = Dependency.new(:rule => "A and B", :question_id => 1)
|
64
|
+
@dep3 = Dependency.new(:rule => "A or B", :question_id => 1)
|
65
|
+
@dep4 = Dependency.new(:rule => "!(A and B) and C", :question_id => 1)
|
66
|
+
|
67
|
+
@dep_c = mock_model(DependencyCondition, :id => 1, :rule_key => "A", :to_hash => {:A => true})
|
68
|
+
@dep_c2 = mock_model(DependencyCondition, :id => 2, :rule_key => "B", :to_hash => {:B => false})
|
69
|
+
@dep_c3 = mock_model(DependencyCondition, :id => 3, :rule_key => "C", :to_hash => {:C => true})
|
70
|
+
|
71
|
+
@dep.stub!(:dependency_conditions).and_return([@dep_c])
|
72
|
+
@dep2.stub!(:dependency_conditions).and_return([@dep_c, @dep_c2])
|
73
|
+
@dep3.stub!(:dependency_conditions).and_return([@dep_c, @dep_c2])
|
74
|
+
@dep4.stub!(:dependency_conditions).and_return([@dep_c, @dep_c2, @dep_c3])
|
75
|
+
end
|
76
|
+
|
77
|
+
it "knows if the dependencies are met" do
|
78
|
+
@dep.is_met?(@response_set).should be_true
|
79
|
+
@dep2.is_met?(@response_set).should be_false
|
80
|
+
@dep3.is_met?(@response_set).should be_true
|
81
|
+
@dep4.is_met?(@response_set).should be_true
|
82
|
+
end
|
83
|
+
|
84
|
+
it "returns the proper keyed pairs from the dependency conditions" do
|
85
|
+
@dep.conditions_hash(@response_set).should == {:A => true}
|
86
|
+
@dep2.conditions_hash(@response_set).should == {:A => true, :B => false}
|
87
|
+
@dep3.conditions_hash(@response_set).should == {:A => true, :B => false}
|
88
|
+
@dep4.conditions_hash(@response_set).should == {:A => true, :B => false, :C => true}
|
89
|
+
end
|
90
|
+
end
|
91
|
+
describe Dependency, "with conditions" do
|
92
|
+
it "should destroy conditions when destroyed" do
|
93
|
+
@dependency = Dependency.new(:rule => "A and B and C", :question_id => 1)
|
94
|
+
Factory(:dependency_condition, :dependency => @dependency, :rule_key => "A")
|
95
|
+
Factory(:dependency_condition, :dependency => @dependency, :rule_key => "B")
|
96
|
+
Factory(:dependency_condition, :dependency => @dependency, :rule_key => "C")
|
97
|
+
dc_ids = @dependency.dependency_conditions.map(&:id)
|
98
|
+
@dependency.destroy
|
99
|
+
dc_ids.each{|id| DependencyCondition.find_by_id(id).should == nil}
|
100
|
+
end
|
101
|
+
end
|
@@ -0,0 +1,93 @@
|
|
1
|
+
# encoding: UTF-8
|
2
|
+
require File.expand_path(File.dirname(__FILE__) + '/../spec_helper')
|
3
|
+
|
4
|
+
describe QuestionGroup do
|
5
|
+
let(:question_group){ Factory(:question_group) }
|
6
|
+
let(:dependency){ Factory(:dependency) }
|
7
|
+
let(:response_set){ Factory(:response_set) }
|
8
|
+
|
9
|
+
context "when creating" do
|
10
|
+
it { question_group.should be_valid }
|
11
|
+
it "#display_type = inline by default" do
|
12
|
+
question_group.display_type = "inline"
|
13
|
+
question_group.renderer.should == :inline
|
14
|
+
end
|
15
|
+
it "#renderer == 'default' when #display_type = nil" do
|
16
|
+
question_group.display_type = nil
|
17
|
+
question_group.renderer.should == :default
|
18
|
+
end
|
19
|
+
it "interprets symbolizes #display_type to #renderer" do
|
20
|
+
question_group.display_type = "foo"
|
21
|
+
question_group.renderer.should == :foo
|
22
|
+
end
|
23
|
+
it "reports DOM ready #css_class based on dependencies" do
|
24
|
+
question_group.dependency = dependency
|
25
|
+
dependency.should_receive(:is_met?).and_return(true)
|
26
|
+
question_group.css_class(response_set).should == "g_dependent"
|
27
|
+
|
28
|
+
dependency.should_receive(:is_met?).and_return(false)
|
29
|
+
question_group.css_class(response_set).should == "g_dependent g_hidden"
|
30
|
+
|
31
|
+
question_group.custom_class = "foo bar"
|
32
|
+
dependency.should_receive(:is_met?).and_return(false)
|
33
|
+
question_group.css_class(response_set).should == "g_dependent g_hidden foo bar"
|
34
|
+
end
|
35
|
+
it "protects #api_id" do
|
36
|
+
saved_attrs = question_group.attributes
|
37
|
+
if defined? ActiveModel::MassAssignmentSecurity::Error
|
38
|
+
expect { question_group.update_attributes(:api_id => "NEW") }.to raise_error(ActiveModel::MassAssignmentSecurity::Error)
|
39
|
+
else
|
40
|
+
question_group.attributes = {:api_id => "NEW"} # Rails doesn't return false, but this will be checked in the comparison to saved_attrs
|
41
|
+
end
|
42
|
+
question_group.attributes.should == saved_attrs
|
43
|
+
end
|
44
|
+
it "protects #created_at" do
|
45
|
+
saved_attrs = question_group.attributes
|
46
|
+
if defined? ActiveModel::MassAssignmentSecurity::Error
|
47
|
+
expect { question_group.update_attributes(:created_at => 3.days.ago) }.to raise_error(ActiveModel::MassAssignmentSecurity::Error)
|
48
|
+
else
|
49
|
+
question_group.attributes = {:created_at => 3.days.ago} # Rails doesn't return false, but this will be checked in the comparison to saved_attrs
|
50
|
+
end
|
51
|
+
question_group.attributes.should == saved_attrs
|
52
|
+
end
|
53
|
+
it "protects #updated_at" do
|
54
|
+
saved_attrs = question_group.attributes
|
55
|
+
if defined? ActiveModel::MassAssignmentSecurity::Error
|
56
|
+
expect { question_group.update_attributes(:updated_at => 3.hours.ago) }.to raise_error(ActiveModel::MassAssignmentSecurity::Error)
|
57
|
+
else
|
58
|
+
question_group.attributes = {:updated_at => 3.hours.ago} # Rails doesn't return false, but this will be checked in the comparison to saved_attrs
|
59
|
+
end
|
60
|
+
question_group.attributes.should == saved_attrs
|
61
|
+
end
|
62
|
+
end
|
63
|
+
|
64
|
+
context "with translations" do
|
65
|
+
require 'yaml'
|
66
|
+
let(:survey){ Factory(:survey) }
|
67
|
+
let(:survey_section){ Factory(:survey_section) }
|
68
|
+
let(:survey_translation){
|
69
|
+
Factory(:survey_translation, :locale => :es, :translation => {
|
70
|
+
:question_groups => {
|
71
|
+
:goodbye => {
|
72
|
+
:text => "¡Adios!"
|
73
|
+
}
|
74
|
+
}
|
75
|
+
}.to_yaml)
|
76
|
+
}
|
77
|
+
let(:question){ Factory(:question) }
|
78
|
+
before do
|
79
|
+
question_group.text = "Goodbye"
|
80
|
+
question_group.reference_identifier = "goodbye"
|
81
|
+
question_group.questions = [question]
|
82
|
+
question.survey_section = survey_section
|
83
|
+
survey_section.survey = survey
|
84
|
+
survey.translations << survey_translation
|
85
|
+
end
|
86
|
+
it "returns its own translation" do
|
87
|
+
question_group.translation(:es)[:text].should == "¡Adios!"
|
88
|
+
end
|
89
|
+
it "returns its own default values" do
|
90
|
+
question_group.translation(:de).should == {"text" => "Goodbye", "help_text" => nil}
|
91
|
+
end
|
92
|
+
end
|
93
|
+
end
|