quby 4.0.4 → 5.0.0
Sign up to get free protection for your applications and to get access to all the features.
- checksums.yaml +4 -4
- data/README.markdown +0 -1
- data/lib/quby.rb +2 -10
- data/lib/quby/answers/entities/answer.rb +4 -2
- data/lib/quby/answers/services/score_calculator.rb +1 -3
- data/lib/quby/engine.rb +0 -1
- data/lib/quby/questionnaires.rb +1 -0
- data/lib/quby/questionnaires/api.rb +5 -1
- data/lib/quby/questionnaires/deserializer.rb +441 -0
- data/lib/quby/questionnaires/dsl.rb +12 -15
- data/lib/quby/questionnaires/entities.rb +1 -0
- data/lib/quby/questionnaires/entities/charting/line_chart.rb +23 -0
- data/lib/quby/questionnaires/entities/charting/overview_chart.rb +3 -1
- data/lib/quby/questionnaires/entities/definition.rb +3 -5
- data/lib/quby/questionnaires/entities/fields.rb +0 -15
- data/lib/quby/questionnaires/entities/question.rb +9 -32
- data/lib/quby/questionnaires/entities/questionnaire.rb +4 -15
- data/lib/quby/questionnaires/entities/questions/checkbox_question.rb +0 -24
- data/lib/quby/questionnaires/entities/questions/date_question.rb +0 -8
- data/lib/quby/questionnaires/entities/score_calculation.rb +40 -3
- data/lib/quby/questionnaires/repos.rb +1 -0
- data/lib/quby/questionnaires/repos/bundle_disk_repo.rb +51 -0
- data/lib/quby/table_backend/range_tree.rb +0 -51
- data/lib/quby/version.rb +1 -1
- data/spec/features/tables_spec.rb +40 -0
- data/spec/fixtures/score_test.rb +4 -0
- data/spec/internal/log/test-events.log +183 -0
- data/spec/internal/log/test.log +5522 -0
- data/spec/internal/tmp/cache/assets/sprockets/v3.0/00/00nglkSepN01iGiDa7URmJtWQP-xSWffZb9wJgBqUr4.cache +2 -0
- data/spec/internal/tmp/cache/assets/sprockets/v3.0/1V/1VQnRlqKBL0MybpMmC3EVtA2s6-ZX-GSGs2Cza3g3kI.cache +1 -0
- data/spec/internal/tmp/cache/assets/sprockets/v3.0/1_/1_J_3havMjJDCz3iAQXAXXBh6tBQRJ3OpdWm3W46ss8.cache +1 -0
- data/spec/internal/tmp/cache/assets/sprockets/v3.0/4G/4gcYvPHmPX3fZw9xr6plMl5Hfbo2xTWwcD7T7lIlD4Y.cache +2 -0
- data/spec/internal/tmp/cache/assets/sprockets/v3.0/4_/4_FK6PMcF1eyIQx7RtVQ5t8c6X7smPQmJWNN5YTImVc.cache +1 -0
- data/spec/internal/tmp/cache/assets/sprockets/v3.0/4d/4Dd15Upcg8_kiEsx_5wpQNrjM1sCk0tGTtWGhtmCKpg.cache +1 -0
- data/spec/internal/tmp/cache/assets/sprockets/v3.0/4h/4hd8RzwlWb2IEeLLZN0s-3_sdJXtw6nu2k5baLb9vNs.cache +0 -0
- data/spec/internal/tmp/cache/assets/sprockets/v3.0/5X/5xFJleOz2sGyE9gbs8KPkToDiv3s9bwFzroBRcjPELY.cache +0 -0
- data/spec/internal/tmp/cache/assets/sprockets/v3.0/6Y/6y1tV_fKJP9YIQn-dF847aZ9kza1O7iyTEUht42D7nc.cache +0 -0
- data/spec/internal/tmp/cache/assets/sprockets/v3.0/6d/6dfYZY_DkQe892FZVDUbBbrINslzVUkLZePdnVxO3Ps.cache +1 -0
- data/spec/internal/tmp/cache/assets/sprockets/v3.0/6w/6WcYsWQCgw0z_URAWGhOsbiKXcqVxZ--ImazY5S-2Gk.cache +2 -0
- data/spec/internal/tmp/cache/assets/sprockets/v3.0/77/77EJjFGVCRzLbhJy6KZZUYe5w0LxsqZQcmFcHAbxKeQ.cache +1 -0
- data/spec/internal/tmp/cache/assets/sprockets/v3.0/7f/7fvhJnfiJknq6sA-MPtumXlL5qB-rfH__Heye13NyHc.cache +1 -0
- data/spec/internal/tmp/cache/assets/sprockets/v3.0/8W/8W6AwDwFqo6egmWyt6Cs0vf_XJbT2YxPs2sqbo0Q9kk.cache +2 -0
- data/spec/internal/tmp/cache/assets/sprockets/v3.0/8c/8cQ8uBhUhrouKwor0Z87dDCxyPbqbLvK8KEFEgjHxqM.cache +0 -0
- data/spec/internal/tmp/cache/assets/sprockets/v3.0/9L/9LJo1U9wQdfWrrn3HkNZTgIZc6jeHvWbGk8tJ4m_Qec.cache +1 -0
- data/spec/internal/tmp/cache/assets/sprockets/v3.0/9o/9oXJXZnA6F49i4EflydD0Ivx6dhSxGGDD69G-N5NSCg.cache +0 -0
- data/spec/internal/tmp/cache/assets/sprockets/v3.0/A1/a1jgtYtDHIb04Og07H6_XNmhXppaAnFiO88kgq80edQ.cache +1 -0
- data/spec/internal/tmp/cache/assets/sprockets/v3.0/AC/ACV6t4unw39Kpn2DnbBtHeJYLoAHyyBN1DTRXs6a2ik.cache +1 -0
- data/spec/internal/tmp/cache/assets/sprockets/v3.0/BO/BoVO8e1GTTGHfnJTOByDxMlvVi9fNGyOdSubqCbBITI.cache +2 -0
- data/spec/internal/tmp/cache/assets/sprockets/v3.0/Ba/babjk3aW-jBoEw_ERhmT5ZQprEqilJCWmImrvVskWk8.cache +0 -0
- data/spec/internal/tmp/cache/assets/sprockets/v3.0/Cg/cgb1vvOogMD0V1lqQ5k_mny7Z38rbqBdIumlGh5sZ_E.cache +0 -0
- data/spec/internal/tmp/cache/assets/sprockets/v3.0/Cw/CwjNk_RYiC93RW8vRs7fWsHcwKTg7uE9OLifRm3Y6WA.cache +2 -0
- data/spec/internal/tmp/cache/assets/sprockets/v3.0/DY/Dy1qsY0W-eyFR6RVKSCCli-XsitpaXD4DwTjXL86t2M.cache +1 -0
- data/spec/internal/tmp/cache/assets/sprockets/v3.0/Dk/DkGOwkJNeDdljv4b8QbwtPVrsghwYuA6EOxx1goEQg0.cache +1 -0
- data/spec/internal/tmp/cache/assets/sprockets/v3.0/FG/fGmVSU-v7aZV9-CLG5U2OmteGgWTuAotlFcwFyx4Eqs.cache +1 -0
- data/spec/internal/tmp/cache/assets/sprockets/v3.0/Fd/fDidWZjSZsPN4uAEyVYABV2WFCSR7AZE0l8U-9XOw7U.cache +0 -0
- data/spec/internal/tmp/cache/assets/sprockets/v3.0/G1/G1RRwpldS5MGQ5x-WHyPWzXJWDOd2qyWMMFumuU4-Ms.cache +1 -0
- data/spec/internal/tmp/cache/assets/sprockets/v3.0/GG/ggGfy_gZryUPmYwgenN3A4jv0q-buxyjCGdLB6wfHxQ.cache +1 -0
- data/spec/internal/tmp/cache/assets/sprockets/v3.0/GY/gYwYoJrifknCG-a0FXlM1NU6tUj1JXxm27s-a0uP_IE.cache +1 -0
- data/spec/internal/tmp/cache/assets/sprockets/v3.0/I6/i64Ysik0cyoTrYF0-O63B7sN3AdIgeoJ7zLh9pMC2u4.cache +0 -0
- data/spec/internal/tmp/cache/assets/sprockets/v3.0/Ic/ICKeOd4yOff3JGsfU1XliLgXfJtTPKSbXsqNVZAb1-Y.cache +1 -0
- data/spec/internal/tmp/cache/assets/sprockets/v3.0/J7/J75M1l--IMcCn-Wy7uWv17SWaj0DFaGiiwac63pENPQ.cache +1 -0
- data/spec/internal/tmp/cache/assets/sprockets/v3.0/Jh/JHPqsKZsrkvQg23ukphinYoViiIw3askwCMP8VrHTMk.cache +1 -0
- data/spec/internal/tmp/cache/assets/sprockets/v3.0/K9/k9tOCRdX5tsxa5LImq7MZk0WzC692t8tx5LzLln0l4Q.cache +0 -0
- data/spec/internal/tmp/cache/assets/sprockets/v3.0/LY/ly4TNAIbW9ghgXz8mxfmskR_6AkuiMiG-RIFGhewgQ0.cache +1 -0
- data/spec/internal/tmp/cache/assets/sprockets/v3.0/Lq/LQ9fP0wIqVc5s-AtP3ESWJr0qpkDe-VpTxpgYl-9FfU.cache +1 -0
- data/spec/internal/tmp/cache/assets/sprockets/v3.0/MO/MOmnpcaRW98Ke-zTOOCSZ0ODnnQntKhQr0vFc_un1_A.cache +1 -0
- data/spec/internal/tmp/cache/assets/sprockets/v3.0/OX/OxOBsUA5EpScoZgEt_UhsHzhu-GGmxKnHpCwnhbTwiM.cache +1 -0
- data/spec/internal/tmp/cache/assets/sprockets/v3.0/Ot/oTVAc3-iuSlg7XAM-YIAhw3cdc5yhgL2-tK3N7sfOfU.cache +0 -0
- data/spec/internal/tmp/cache/assets/sprockets/v3.0/PK/pKNcBOgAbCyIwsWIvnlmiPiFxrw6nUo_nTz7wWFDr80.cache +1 -0
- data/spec/internal/tmp/cache/assets/sprockets/v3.0/QB/QBMvSW7SYbYcm5lSjybgO7fhHDFIQjJWOvefH0MQC-A.cache +0 -0
- data/spec/internal/tmp/cache/assets/sprockets/v3.0/Qs/QSbtm1y8JKPV0MQkpiC0hEisaSjKZ3tp2AWHSrhO30Y.cache +1 -0
- data/spec/internal/tmp/cache/assets/sprockets/v3.0/Sp/SpyGDguFy3tp0sim1miZOehEDpE97oueO7wl3oUFYcE.cache +1 -0
- data/spec/internal/tmp/cache/assets/sprockets/v3.0/TW/TwtJPRSE9lmwTuSP0lezWSRSBp2u43zuF196FQcHY-c.cache +1 -0
- data/spec/internal/tmp/cache/assets/sprockets/v3.0/VX/Vxn9zv60MCKUBfRhrjefo-MEzFej81PPEutxTPQ4VLQ.cache +0 -0
- data/spec/internal/tmp/cache/assets/sprockets/v3.0/Vr/vrqPEzlsuORocHY0EbtGdFomRmrcoAa6wWSDMx642-Q.cache +2 -0
- data/spec/internal/tmp/cache/assets/sprockets/v3.0/WH/WHVQ0GBe51hbYqv2g7YYo7EIw9PSzt9-RqU7hKP2dc4.cache +1 -0
- data/spec/internal/tmp/cache/assets/sprockets/v3.0/_B/_B796ezRHfk7H8YKj7zr3-gRy6Mw2IVn2NbHu-k20ug.cache +1 -0
- data/spec/internal/tmp/cache/assets/sprockets/v3.0/_W/_Wh-7b1t-Xm19wg3KPtkXq7DOpohaL5OhnLggzJLpj4.cache +0 -0
- data/spec/internal/tmp/cache/assets/sprockets/v3.0/b2/B2y-2qTbQYKRm6_sCQ6qfUeVfl9rKXUvotTDm56w4bs.cache +1 -0
- data/spec/internal/tmp/cache/assets/sprockets/v3.0/bw/bwFR2MLpvJvPSAPJzSLWk6TUDCw5TlBFUUuq7z_Y9vg.cache +1 -0
- data/spec/internal/tmp/cache/assets/sprockets/v3.0/c7/c7ubvrHhbhAYh-K4srvM9EqwWdbWTLrsLMgjG5XQDbY.cache +1 -0
- data/spec/internal/tmp/cache/assets/sprockets/v3.0/cX/CXpMzZYfJy6HIOVXN0IyYilvy6J4CDgFZIwqKm8OQEU.cache +1 -0
- data/spec/internal/tmp/cache/assets/sprockets/v3.0/eN/EN0enu5dkKEnIz-DVCpbhkfeph7wImeXnlqb7SXMrjE.cache +1 -0
- data/spec/internal/tmp/cache/assets/sprockets/v3.0/fu/fuP9zP6C6rj650HLkypoTfYN2KD1ecB9HdtlbhbA2cg.cache +1 -0
- data/spec/internal/tmp/cache/assets/sprockets/v3.0/gr/gRPXI3EsVPwysy9YKW_6yDxaqRRr4Idxtexf7zCHECw.cache +1 -0
- data/spec/internal/tmp/cache/assets/sprockets/v3.0/h0/h0P63ZZdlSbtztDOe0V9r7jtCoy0L2bAf_iDNKLJ0nk.cache +1 -0
- data/spec/internal/tmp/cache/assets/sprockets/v3.0/i-/i-Or0Xuxbbm3-HdE_xGIf6sAx2vhUa3eVMpbT37lyEo.cache +1 -0
- data/spec/internal/tmp/cache/assets/sprockets/v3.0/iF/iFP67If98izMEAAMKnAFi2dWAb8TNlHNayZKNjXdAMs.cache +1 -0
- data/spec/internal/tmp/cache/assets/sprockets/v3.0/iH/Ih2svq5ul-xibfRUfd1fxigouli6U19Rf80FrTSoMak.cache +1 -0
- data/spec/internal/tmp/cache/assets/sprockets/v3.0/ka/Ka_YEbZNQxpS8XAA7xZturgPr2vLYQq60KK1uwou4Lo.cache +1 -0
- data/spec/internal/tmp/cache/assets/sprockets/v3.0/kq/KqkIGfAdJjTcr53lFl1aDAMUeqhhKdXGyTlHVlCz0R4.cache +1 -0
- data/spec/internal/tmp/cache/assets/sprockets/v3.0/mR/mRDPkwomGSfjpIWEKtPPBSx5cBHssvH8v0gCSI33P3Y.cache +1 -0
- data/spec/internal/tmp/cache/assets/sprockets/v3.0/n4/n4qaNjjNvKB0VAbVJlP_BjFlcLf70TsBzcIMc7VO9dw.cache +2 -0
- data/spec/internal/tmp/cache/assets/sprockets/v3.0/nY/nykt5oy2ExUYa8g1phLCDRqwb55BuUot5ZBbRdNuCCw.cache +2 -0
- data/spec/internal/tmp/cache/assets/sprockets/v3.0/nv/nvmgm4bOtxCbD6EA14ZIwDcYO8fxXOoEk_s7V-uRMMM.cache +1 -0
- data/spec/internal/tmp/cache/assets/sprockets/v3.0/oG/og5ypFjbFAX_nH7oxK6Bn5Ejh81BMDBSMxbGcL5ZbQs.cache +1 -0
- data/spec/internal/tmp/cache/assets/sprockets/v3.0/oO/oO4iBwOm5E7uLH39zeR8zun4eBM-WxZKvEMpIYKx6SU.cache +1 -0
- data/spec/internal/tmp/cache/assets/sprockets/v3.0/pN/PnQ-dEclV6jTDE4kkqw4G9dBpHArAgUnATOpRrQAsGc.cache +1 -0
- data/spec/internal/tmp/cache/assets/sprockets/v3.0/pe/pet4fFbc2Cm2SD_ZBO_s83Gey29EXE26u-7lwn2O8gM.cache +1 -0
- data/spec/internal/tmp/cache/assets/sprockets/v3.0/qo/Qoqb-JjTe1BiY4rICKHm5e6bsPt-DHXZc4ADXESjw2I.cache +1 -0
- data/spec/internal/tmp/cache/assets/sprockets/v3.0/qx/QXICLj9xx1EI9rBeXLXpKOeHGll_LBWHRWlpb1DK424.cache +1 -0
- data/spec/internal/tmp/cache/assets/sprockets/v3.0/rM/rMF4Y2UGkmBABLlC1eSSqb6y6g_Z3uUCLBtFvz_LoyI.cache +1 -0
- data/spec/internal/tmp/cache/assets/sprockets/v3.0/ty/ty1qAnpL07anYdhIJoiPYLcsWKejuzy6JaxU9mI7egw.cache +1 -0
- data/spec/internal/tmp/cache/assets/sprockets/v3.0/uH/UHs9nE45XS7v4QlMjSjhwHbqYIKC06FsEjMSMIfKanU.cache +1 -0
- data/spec/internal/tmp/cache/assets/sprockets/v3.0/v8/v8VXJ3AQom-JqxBYAB7tOjV-tt4IdDb6YyFEzlsWV8c.cache +1 -0
- data/spec/internal/tmp/cache/assets/sprockets/v3.0/vD/VDI2CxkTg7-DuMcLmofLvch0yukSl88oI_AjGgN-hQ8.cache +1 -0
- data/spec/internal/tmp/cache/assets/sprockets/v3.0/wq/wqhk46aZc1Bbb8gTCOZKxTvwTV0ANLSz8VdQYACo03E.cache +1 -0
- data/spec/internal/tmp/cache/assets/sprockets/v3.0/xy/Xy8Yk--iYZNupHMjUtYX1AQ0BppsvThCh_ZAizblCb0.cache +1 -0
- data/spec/internal/tmp/cache/assets/sprockets/v3.0/zu/zUCfHRTHprn06F5WxH6wp_BLs9-yhs-foc0j0Y7BYbg.cache +1 -0
- data/spec/internal/tmp/cache/assets/sprockets/v3.0/zx/ZX2IY3eKQjBy93tlH7_Xg7zGZKYuaQ-E8YPvtPT_4vM.cache +1 -0
- data/spec/internal/tmp/capybara/screenshot_2020-11-17-14-27-35.705.html +1 -0
- data/spec/internal/tmp/capybara/screenshot_2020-11-17-14-27-35.705.png +0 -0
- data/spec/internal/tmp/capybara/screenshot_2020-11-17-14-27-36.005.html +1 -0
- data/spec/internal/tmp/capybara/screenshot_2020-11-17-14-27-36.005.png +0 -0
- data/spec/internal/tmp/capybara/screenshot_2020-11-17-14-27-36.276.html +1 -0
- data/spec/internal/tmp/capybara/screenshot_2020-11-17-14-27-36.276.png +0 -0
- data/spec/internal/tmp/capybara/screenshot_2020-11-17-14-27-36.556.html +1 -0
- data/spec/internal/tmp/capybara/screenshot_2020-11-17-14-27-36.556.png +0 -0
- data/spec/internal/tmp/capybara/screenshot_2020-11-17-14-27-36.818.html +1 -0
- data/spec/internal/tmp/capybara/screenshot_2020-11-17-14-27-36.818.png +0 -0
- data/spec/internal/tmp/capybara/screenshot_2020-11-17-14-27-37.083.html +1 -0
- data/spec/internal/tmp/capybara/screenshot_2020-11-17-14-27-37.083.png +0 -0
- data/spec/internal/tmp/capybara/screenshot_2020-11-17-14-27-37.350.html +1 -0
- data/spec/internal/tmp/capybara/screenshot_2020-11-17-14-27-37.350.png +0 -0
- data/spec/internal/tmp/capybara/screenshot_2020-11-17-14-27-37.620.html +1 -0
- data/spec/internal/tmp/capybara/screenshot_2020-11-17-14-27-37.620.png +0 -0
- data/spec/internal/tmp/capybara/screenshot_2020-11-17-14-27-37.885.html +1 -0
- data/spec/internal/tmp/capybara/screenshot_2020-11-17-14-27-37.885.png +0 -0
- data/spec/internal/tmp/capybara/screenshot_2020-11-17-14-27-38.152.html +1 -0
- data/spec/internal/tmp/capybara/screenshot_2020-11-17-14-27-38.152.png +0 -0
- data/spec/internal/tmp/capybara/screenshot_2020-11-17-14-28-14.063.html +207 -0
- data/spec/internal/tmp/capybara/screenshot_2020-11-17-14-28-14.063.png +0 -0
- data/spec/quby/answers/entities/score_spec.rb +4 -0
- data/spec/quby/answers/services/answer_validations_spec.rb +8 -8
- data/spec/quby/answers/services/score_calculator_spec.rb +4 -14
- data/spec/quby/questionnaires/deserializer/questionnaire_spec.rb +237 -0
- data/spec/quby/questionnaires/dsl_spec.rb +0 -9
- data/spec/quby/questionnaires/entities/fields_spec.rb +3 -3
- data/spec/quby/questionnaires/entities/question_spec.rb +0 -8
- data/spec/quby/questionnaires/entities/questionnaire_spec.rb +2 -26
- data/spec/quby/table_backend/range_tree_spec.rb +46 -13
- data/spec/spec_helper.rb +1 -1
- metadata +230 -53
- data/lib/quby/lookup_table.rb +0 -29
- data/lib/quby/lookup_table_repo.rb +0 -24
- data/lib/quby/questionnaires/dsl/base.rb +0 -20
- data/lib/quby/questionnaires/dsl/calls_custom_methods.rb +0 -29
- data/lib/quby/questionnaires/dsl/charting/bar_chart_builder.rb +0 -18
- data/lib/quby/questionnaires/dsl/charting/chart_builder.rb +0 -91
- data/lib/quby/questionnaires/dsl/charting/line_chart_builder.rb +0 -57
- data/lib/quby/questionnaires/dsl/charting/overview_chart_builder.rb +0 -31
- data/lib/quby/questionnaires/dsl/charting/radar_chart_builder.rb +0 -18
- data/lib/quby/questionnaires/dsl/helpers.rb +0 -51
- data/lib/quby/questionnaires/dsl/panel_builder.rb +0 -80
- data/lib/quby/questionnaires/dsl/question_builder.rb +0 -40
- data/lib/quby/questionnaires/dsl/questionnaire_builder.rb +0 -260
- data/lib/quby/questionnaires/dsl/questions/base.rb +0 -179
- data/lib/quby/questionnaires/dsl/questions/checkbox_question_builder.rb +0 -20
- data/lib/quby/questionnaires/dsl/questions/date_question_builder.rb +0 -18
- data/lib/quby/questionnaires/dsl/questions/deprecated_question_builder.rb +0 -18
- data/lib/quby/questionnaires/dsl/questions/float_question_builder.rb +0 -21
- data/lib/quby/questionnaires/dsl/questions/integer_question_builder.rb +0 -21
- data/lib/quby/questionnaires/dsl/questions/radio_question_builder.rb +0 -20
- data/lib/quby/questionnaires/dsl/questions/select_question_builder.rb +0 -18
- data/lib/quby/questionnaires/dsl/questions/string_question_builder.rb +0 -20
- data/lib/quby/questionnaires/dsl/questions/text_question_builder.rb +0 -22
- data/lib/quby/questionnaires/dsl/score_builder.rb +0 -22
- data/lib/quby/questionnaires/dsl/standardized_panel_generators.rb +0 -33
- data/lib/quby/questionnaires/dsl/table_builder.rb +0 -48
- data/lib/quby/questionnaires/services/definition_validator.rb +0 -298
- data/spec/benchmarks/load_normscore_csv.rb +0 -18
- data/spec/quby/lookup_table_repo_spec.rb +0 -20
- data/spec/quby/lookup_table_spec.rb +0 -38
- data/spec/quby/questionnaires/dsl/calls_custom_methods_spec.rb +0 -38
- data/spec/quby/questionnaires/dsl/charting/bar_chart_builder_spec.rb +0 -41
- data/spec/quby/questionnaires/dsl/charting/chart_builder_spec.rb +0 -127
- data/spec/quby/questionnaires/dsl/charting/line_chart_builder_spec.rb +0 -66
- data/spec/quby/questionnaires/dsl/charting/radar_chart_builder_spec.rb +0 -41
- data/spec/quby/questionnaires/dsl/helpers_spec.rb +0 -80
- data/spec/quby/questionnaires/dsl/questionnaire_builder_spec.rb +0 -480
- data/spec/quby/questionnaires/services/definition_validator_spec.rb +0 -793
- data/spec/support/examples_for_chart_builders.rb +0 -59
data/lib/quby/lookup_table.rb
DELETED
@@ -1,29 +0,0 @@
|
|
1
|
-
# frozen_string_literal: true
|
2
|
-
|
3
|
-
require 'quby/table_backend/range_tree'
|
4
|
-
|
5
|
-
module Quby
|
6
|
-
class LookupTable
|
7
|
-
attr_accessor :key
|
8
|
-
|
9
|
-
def initialize(key)
|
10
|
-
@key = key
|
11
|
-
end
|
12
|
-
|
13
|
-
def backing
|
14
|
-
return @backing if @backing.present?
|
15
|
-
all_data = data
|
16
|
-
headers = all_data.shift
|
17
|
-
compare = all_data.shift
|
18
|
-
@backing = Quby::TableBackend::RangeTree.from_csv(levels: headers, compare: compare, data: all_data)
|
19
|
-
end
|
20
|
-
|
21
|
-
def lookup(parameters)
|
22
|
-
backing.lookup(parameters)
|
23
|
-
end
|
24
|
-
|
25
|
-
def data
|
26
|
-
Quby.lookup_table_repo.retrieve(key)
|
27
|
-
end
|
28
|
-
end
|
29
|
-
end
|
@@ -1,24 +0,0 @@
|
|
1
|
-
require 'pathname'
|
2
|
-
require 'csv'
|
3
|
-
|
4
|
-
module Quby
|
5
|
-
module LookupTableRepo
|
6
|
-
def self.validate_key(key)
|
7
|
-
raise 'invalid key' unless key =~ /\A[a-z][a-z_0-9]{1,50}\z/
|
8
|
-
end
|
9
|
-
|
10
|
-
class Disk
|
11
|
-
attr_reader :disk_table_root
|
12
|
-
|
13
|
-
def initialize(disk_table_root)
|
14
|
-
@disk_table_root = Pathname.new(disk_table_root)
|
15
|
-
end
|
16
|
-
|
17
|
-
def retrieve(key)
|
18
|
-
Quby::LookupTableRepo.validate_key(key)
|
19
|
-
path = disk_table_root.join(key + '.csv')
|
20
|
-
CSV.read(path, col_sep: ';', skip_blanks: true)
|
21
|
-
end
|
22
|
-
end
|
23
|
-
end
|
24
|
-
end
|
@@ -1,20 +0,0 @@
|
|
1
|
-
# frozen_string_literal: true
|
2
|
-
|
3
|
-
require 'quby/questionnaires/dsl/helpers'
|
4
|
-
require 'quby/questionnaires/dsl/calls_custom_methods'
|
5
|
-
|
6
|
-
module Quby
|
7
|
-
module Questionnaires
|
8
|
-
module DSL
|
9
|
-
class Base
|
10
|
-
include Helpers
|
11
|
-
|
12
|
-
def self.build(*args, &block)
|
13
|
-
builder = new(*args)
|
14
|
-
builder.instance_eval(&block) if block
|
15
|
-
builder.build
|
16
|
-
end
|
17
|
-
end
|
18
|
-
end
|
19
|
-
end
|
20
|
-
end
|
@@ -1,29 +0,0 @@
|
|
1
|
-
# frozen_string_literal: true
|
2
|
-
|
3
|
-
module Quby
|
4
|
-
module Questionnaires
|
5
|
-
module DSL
|
6
|
-
module CallsCustomMethods
|
7
|
-
attr_reader :custom_methods
|
8
|
-
|
9
|
-
def initialize(*args)
|
10
|
-
options = args.last.is_a?(::Hash) ? args.last : {}
|
11
|
-
@custom_methods = options[:custom_methods] || {}
|
12
|
-
super
|
13
|
-
end
|
14
|
-
|
15
|
-
def method_missing(method_sym, *args, &block)
|
16
|
-
if @custom_methods.key? method_sym
|
17
|
-
instance_exec(*args, &custom_methods[method_sym])
|
18
|
-
else
|
19
|
-
super
|
20
|
-
end
|
21
|
-
end
|
22
|
-
|
23
|
-
def respond_to_missing?(method_name, include_private = false)
|
24
|
-
custom_methods.key?(method_name) || super
|
25
|
-
end
|
26
|
-
end
|
27
|
-
end
|
28
|
-
end
|
29
|
-
end
|
@@ -1,18 +0,0 @@
|
|
1
|
-
# frozen_string_literal: true
|
2
|
-
|
3
|
-
require 'quby/questionnaires/entities/charting/bar_chart'
|
4
|
-
require_relative 'chart_builder'
|
5
|
-
|
6
|
-
module Quby
|
7
|
-
module Questionnaires
|
8
|
-
module DSL
|
9
|
-
class BarChartBuilder < ChartBuilder
|
10
|
-
set_chart_class(Entities::Charting::BarChart)
|
11
|
-
|
12
|
-
def plotline(value, color)
|
13
|
-
@chart.plotlines << {value: value, color: color, width: 1, zIndex: 3}
|
14
|
-
end
|
15
|
-
end
|
16
|
-
end
|
17
|
-
end
|
18
|
-
end
|
@@ -1,91 +0,0 @@
|
|
1
|
-
# frozen_string_literal: true
|
2
|
-
|
3
|
-
require 'quby/questionnaires/entities'
|
4
|
-
|
5
|
-
module Quby
|
6
|
-
module Questionnaires
|
7
|
-
module DSL
|
8
|
-
class ChartBuilder
|
9
|
-
# rubocop:disable AccessorMethodName
|
10
|
-
def self.set_chart_class(chart_class)
|
11
|
-
@chart_class = chart_class
|
12
|
-
end
|
13
|
-
# rubocop:enable AccessorMethodName
|
14
|
-
|
15
|
-
def self.chart_class
|
16
|
-
@chart_class
|
17
|
-
end
|
18
|
-
|
19
|
-
def initialize(questionnaire, key, options = {})
|
20
|
-
@chart = self.class.chart_class.new(key, options)
|
21
|
-
@questionnaire = questionnaire
|
22
|
-
end
|
23
|
-
|
24
|
-
def title(title)
|
25
|
-
@chart.title = title
|
26
|
-
end
|
27
|
-
|
28
|
-
def range(range)
|
29
|
-
@chart.y_range = range
|
30
|
-
end
|
31
|
-
|
32
|
-
def tick_interval(tick_interval)
|
33
|
-
@chart.tick_interval = tick_interval
|
34
|
-
end
|
35
|
-
|
36
|
-
def y_categories(y_categories)
|
37
|
-
@chart.y_categories = y_categories
|
38
|
-
end
|
39
|
-
|
40
|
-
def y_range_categories(*y_range_categories)
|
41
|
-
@chart.y_range_categories = RangeCategories.new(*y_range_categories).as_range_hash
|
42
|
-
end
|
43
|
-
|
44
|
-
def plotband(from, to, color)
|
45
|
-
@chart.plotbands << {from: from, to: to, color: color}
|
46
|
-
end
|
47
|
-
|
48
|
-
def plot(key, options = {})
|
49
|
-
unless plottable = @questionnaire.find_plottable(key)
|
50
|
-
fail "Questionnaire #{@questionnaire.key} chart #{@chart.key} references unknown score or question #{key}"
|
51
|
-
end
|
52
|
-
|
53
|
-
configure_options plottable, options
|
54
|
-
|
55
|
-
@chart.plottables << Entities::Charting::Plottable.new(plottable.key, options)
|
56
|
-
end
|
57
|
-
|
58
|
-
def chart_type(type)
|
59
|
-
@chart.chart_type = type
|
60
|
-
end
|
61
|
-
|
62
|
-
def build(&block)
|
63
|
-
instance_eval(&block)
|
64
|
-
validate!
|
65
|
-
@chart
|
66
|
-
end
|
67
|
-
|
68
|
-
def validate!
|
69
|
-
if @chart.y_categories.present? && @chart.y_range.present? &&
|
70
|
-
@chart.y_range != (0..(@chart.y_categories.count - 1))
|
71
|
-
fail ArgumentError, 'Y_categories size and range do not match'
|
72
|
-
end
|
73
|
-
|
74
|
-
true
|
75
|
-
end
|
76
|
-
|
77
|
-
private
|
78
|
-
|
79
|
-
def configure_options(plottable, options)
|
80
|
-
case plottable
|
81
|
-
when Entities::ScoreCalculation
|
82
|
-
options.reverse_merge! plottable.options
|
83
|
-
when Entities::Question
|
84
|
-
options[:label] ||= plottable.title
|
85
|
-
end
|
86
|
-
options[:questionnaire_key] = @questionnaire.key
|
87
|
-
end
|
88
|
-
end
|
89
|
-
end
|
90
|
-
end
|
91
|
-
end
|
@@ -1,57 +0,0 @@
|
|
1
|
-
# frozen_string_literal: true
|
2
|
-
|
3
|
-
require 'quby/questionnaires/entities/charting/line_chart'
|
4
|
-
require_relative 'chart_builder'
|
5
|
-
|
6
|
-
module Quby
|
7
|
-
module Questionnaires
|
8
|
-
module DSL
|
9
|
-
class LineChartBuilder < ChartBuilder
|
10
|
-
set_chart_class(Entities::Charting::LineChart)
|
11
|
-
|
12
|
-
def y_axis_label(label)
|
13
|
-
@chart.y_label = label
|
14
|
-
end
|
15
|
-
|
16
|
-
def tonality(value)
|
17
|
-
@chart.tonality = value
|
18
|
-
end
|
19
|
-
|
20
|
-
# value - Number or a hash `{from_age..to_age => {female: 5, default: 4}, .., default: { default: 3 }}`
|
21
|
-
# block - deprecated.
|
22
|
-
def baseline(value = nil, &block)
|
23
|
-
unless value.nil? ^ block.nil?
|
24
|
-
fail ArgumentError, "Must give either value or a block"
|
25
|
-
end
|
26
|
-
|
27
|
-
if value && value.is_a?(Hash)
|
28
|
-
@chart.baseline = ->(age, gender) {
|
29
|
-
age_match = value.find { |age_range, _v| age && age_range === age }
|
30
|
-
hash_by_gender = (age_match&.last || value.stringify_keys["default"])
|
31
|
-
|
32
|
-
gender_match = hash_by_gender.find {|gender_key, _v| gender && gender_key.to_s == gender.to_s }
|
33
|
-
gender_match&.last || hash_by_gender.stringify_keys['default']
|
34
|
-
}
|
35
|
-
elsif value
|
36
|
-
@chart.baseline = ->(age, gender) { value }
|
37
|
-
end
|
38
|
-
|
39
|
-
if block
|
40
|
-
if block.arity != 2
|
41
|
-
fail ArgumentError, "Given block must take two arguments"
|
42
|
-
end
|
43
|
-
@chart.baseline = block
|
44
|
-
end
|
45
|
-
end
|
46
|
-
|
47
|
-
def clinically_relevant_change(value)
|
48
|
-
@chart.clinically_relevant_change = value
|
49
|
-
end
|
50
|
-
|
51
|
-
def validate!
|
52
|
-
fail "Chart #{@chart.key} has no range specified" unless @chart.y_range
|
53
|
-
end
|
54
|
-
end
|
55
|
-
end
|
56
|
-
end
|
57
|
-
end
|
@@ -1,31 +0,0 @@
|
|
1
|
-
require 'quby/questionnaires/entities/charting/overview_chart'
|
2
|
-
require_relative 'chart_builder'
|
3
|
-
|
4
|
-
module Quby
|
5
|
-
module Questionnaires
|
6
|
-
module DSL
|
7
|
-
class OverviewChartBuilder < ChartBuilder
|
8
|
-
set_chart_class(Entities::Charting::OverviewChart)
|
9
|
-
|
10
|
-
def initialize(questionnaire, options = {})
|
11
|
-
@questionnaire = questionnaire
|
12
|
-
@chart = self.class.chart_class.new
|
13
|
-
end
|
14
|
-
|
15
|
-
def subscore(key)
|
16
|
-
@chart.subscore = key
|
17
|
-
end
|
18
|
-
|
19
|
-
def y_max(value)
|
20
|
-
@chart.y_max = value
|
21
|
-
end
|
22
|
-
|
23
|
-
def validate!
|
24
|
-
fail ArgumentError, "subscore not specified" unless @chart.subscore.present?
|
25
|
-
fail ArgumentError, "y_max not specified" unless @chart.y_max.present?
|
26
|
-
true
|
27
|
-
end
|
28
|
-
end
|
29
|
-
end
|
30
|
-
end
|
31
|
-
end
|
@@ -1,18 +0,0 @@
|
|
1
|
-
# frozen_string_literal: true
|
2
|
-
|
3
|
-
require 'quby/questionnaires/entities/charting/radar_chart'
|
4
|
-
require_relative 'chart_builder'
|
5
|
-
|
6
|
-
module Quby
|
7
|
-
module Questionnaires
|
8
|
-
module DSL
|
9
|
-
class RadarChartBuilder < ChartBuilder
|
10
|
-
set_chart_class(Entities::Charting::RadarChart)
|
11
|
-
|
12
|
-
def plotline(value, color)
|
13
|
-
@chart.plotlines << {value: value, color: color, width: 1, zIndex: 3}
|
14
|
-
end
|
15
|
-
end
|
16
|
-
end
|
17
|
-
end
|
18
|
-
end
|
@@ -1,51 +0,0 @@
|
|
1
|
-
# frozen_string_literal: true
|
2
|
-
|
3
|
-
module Quby
|
4
|
-
module Questionnaires
|
5
|
-
module DSL
|
6
|
-
module Helpers
|
7
|
-
def image_tag(*args)
|
8
|
-
options = args.extract_options!
|
9
|
-
|
10
|
-
ActionController::Base.helpers.image_tag(
|
11
|
-
*args, options.reverse_merge(alt: image_alt(args.first))
|
12
|
-
)
|
13
|
-
end
|
14
|
-
|
15
|
-
# Copied from ActionController::Base.helpers.image_alt, because it will be removed from Rails 6.0, but we want
|
16
|
-
# to keep using this functionality
|
17
|
-
def image_alt(source)
|
18
|
-
File.basename(source, ".*").sub(/-[[:xdigit:]]{32,64}\z/, "").tr("-_", " ").capitalize
|
19
|
-
end
|
20
|
-
|
21
|
-
def check_question_keys_uniqueness(key, options, questionnaire)
|
22
|
-
keys = QuestionBuilder.build(key, options).claimed_keys
|
23
|
-
if keys.any? { |k| questionnaire.key_in_use? k }
|
24
|
-
fail "#{questionnaire.key}:#{key}: A question or option with input key #{key} is already defined."
|
25
|
-
end
|
26
|
-
end
|
27
|
-
|
28
|
-
def video_tag(*urls, poster: nil, autoplay: false, loop: false)
|
29
|
-
# assume the file extension is the video format
|
30
|
-
# otherwise, the host's declared mime type is used to figure out compatibility
|
31
|
-
# which is usually wrong
|
32
|
-
sources = urls.map { |url| [url, url.match(/[^\.]*\z/)&.[](0).downcase] }
|
33
|
-
sources.each { |_url, ext| raise 'unknown video url file extension' unless %w[mp4 webm].include?(ext) }
|
34
|
-
|
35
|
-
ActionController::Base.helpers.tag.video\
|
36
|
-
width: '100%',
|
37
|
-
preload: 'none',
|
38
|
-
poster: poster,
|
39
|
-
loop: loop,
|
40
|
-
autoplay: autoplay,
|
41
|
-
muted: autoplay,
|
42
|
-
controls: '' do |tag|
|
43
|
-
sources.map { |url, ext| tag.source src: url, type: "video/#{ext}" }
|
44
|
-
.append(I18n.t('video_not_supported'))
|
45
|
-
.yield_self(&tag.method(:safe_join))
|
46
|
-
end
|
47
|
-
end
|
48
|
-
end
|
49
|
-
end
|
50
|
-
end
|
51
|
-
end
|
@@ -1,80 +0,0 @@
|
|
1
|
-
# frozen_string_literal: true
|
2
|
-
|
3
|
-
require 'quby/questionnaires/entities'
|
4
|
-
|
5
|
-
module Quby
|
6
|
-
module Questionnaires
|
7
|
-
module DSL
|
8
|
-
class PanelBuilder < Base
|
9
|
-
attr_reader :title
|
10
|
-
attr_reader :questionnaire
|
11
|
-
|
12
|
-
def initialize(title, options = {})
|
13
|
-
@panel = Entities::Panel.new(options.merge(title: title, items: []))
|
14
|
-
@default_question_options = options[:default_question_options] || {}
|
15
|
-
@questionnaire = options[:questionnaire]
|
16
|
-
@custom_methods = options[:custom_methods] || {}
|
17
|
-
end
|
18
|
-
|
19
|
-
def build
|
20
|
-
@panel
|
21
|
-
end
|
22
|
-
|
23
|
-
def title(value)
|
24
|
-
@panel.title = value
|
25
|
-
end
|
26
|
-
|
27
|
-
def text(value, options = {})
|
28
|
-
@panel.items << Entities::Text.new(value.to_s, options)
|
29
|
-
end
|
30
|
-
|
31
|
-
def html(value)
|
32
|
-
@panel.items << Entities::Text.new('', html_content: value.to_s)
|
33
|
-
end
|
34
|
-
|
35
|
-
def raw_html(value)
|
36
|
-
@panel.items << Entities::Text.new('', raw_content: value.to_s)
|
37
|
-
end
|
38
|
-
|
39
|
-
def video(*urls, **options)
|
40
|
-
video_html = video_tag *urls, **options
|
41
|
-
@panel.items << Entities::Text.new('', raw_content: video_html)
|
42
|
-
end
|
43
|
-
|
44
|
-
def default_question_options(options = {})
|
45
|
-
@default_question_options = @default_question_options.merge(options)
|
46
|
-
end
|
47
|
-
|
48
|
-
def question(key, options = {}, &block)
|
49
|
-
options = @default_question_options.merge(options).merge(questionnaire: @panel.questionnaire)
|
50
|
-
|
51
|
-
check_question_keys_uniqueness key, options, @questionnaire
|
52
|
-
|
53
|
-
question = QuestionBuilder.build(key, options, &block)
|
54
|
-
|
55
|
-
@questionnaire.register_question(question)
|
56
|
-
@panel.items << question
|
57
|
-
end
|
58
|
-
|
59
|
-
def table(options = {}, &block)
|
60
|
-
table_builder = TableBuilder.new(@panel, options.merge(questionnaire: @panel.questionnaire,
|
61
|
-
default_question_options: @default_question_options,
|
62
|
-
custom_methods: @custom_methods))
|
63
|
-
table_builder.instance_eval(&block) if block
|
64
|
-
end
|
65
|
-
|
66
|
-
def method_missing(method_sym, *args, &block)
|
67
|
-
if @custom_methods.key? method_sym
|
68
|
-
instance_exec(*args, &@custom_methods[method_sym])
|
69
|
-
else
|
70
|
-
super
|
71
|
-
end
|
72
|
-
end
|
73
|
-
|
74
|
-
def respond_to_missing?(method_name, include_private = false)
|
75
|
-
@custom_methods.key?(method_name) || super
|
76
|
-
end
|
77
|
-
end
|
78
|
-
end
|
79
|
-
end
|
80
|
-
end
|