modules/quizzes.js

  1. /* eslint-disable object-curly-newline, no-underscore-dangle */
  2. /* eslint-disable max-len */
  3. /* eslint-disable complexity */
  4. const EventDispatcher = require('../utils/event-dispatcher');
  5. const helpers = require('../utils/helpers');
  6. // Create URL from supplied quizId and parameters
  7. function createQuizUrl(quizId, parameters, options, path) {
  8. const {
  9. apiKey,
  10. clientId,
  11. sessionId,
  12. segments,
  13. userId,
  14. version,
  15. quizzesServiceUrl,
  16. } = options;
  17. let queryParams = { c: version };
  18. let answersParamString = '';
  19. queryParams.key = apiKey;
  20. queryParams.i = clientId;
  21. queryParams.s = sessionId;
  22. // Pull user segments from options
  23. if (segments && segments.length) {
  24. queryParams.us = segments;
  25. }
  26. // Pull user id from options and ensure string
  27. if (userId) {
  28. queryParams.ui = String(userId);
  29. }
  30. // Validate quiz id is provided
  31. if (!quizId || typeof quizId !== 'string') {
  32. throw new Error('quizId is a required parameter of type string');
  33. }
  34. if (path === 'results' && (typeof parameters.answers !== 'object' || !Array.isArray(parameters.answers) || parameters.answers.length === 0)) {
  35. throw new Error('answers is a required parameter of type array');
  36. }
  37. if (parameters) {
  38. const { section, answers, quizSessionId, quizVersionId, page, resultsPerPage, filters, fmtOptions, hiddenFields } = parameters;
  39. // Pull section from parameters
  40. if (section) {
  41. queryParams.section = section;
  42. }
  43. // Pull quiz_version_id from parameters
  44. if (quizVersionId) {
  45. queryParams.quiz_version_id = quizVersionId;
  46. }
  47. // Pull quiz_session_id from parameters
  48. if (quizSessionId) {
  49. queryParams.quiz_session_id = quizSessionId;
  50. }
  51. // Pull a (answers) from parameters and transform
  52. if (answers && answers.length) {
  53. answersParamString = `&${helpers.stringify({ a: answers.map((ans) => [...ans].join(',')) })}`;
  54. }
  55. // Pull page from parameters
  56. if (!helpers.isNil(page)) {
  57. queryParams.page = page;
  58. }
  59. // Pull results per page from parameters
  60. if (!helpers.isNil(resultsPerPage)) {
  61. queryParams.num_results_per_page = resultsPerPage;
  62. }
  63. if (filters) {
  64. queryParams.filters = filters;
  65. }
  66. if (fmtOptions) {
  67. queryParams.fmt_options = fmtOptions;
  68. }
  69. if (hiddenFields) {
  70. if (queryParams.fmt_options) {
  71. queryParams.fmt_options.hidden_fields = hiddenFields;
  72. } else {
  73. queryParams.fmt_options = { hidden_fields: hiddenFields };
  74. }
  75. }
  76. }
  77. queryParams._dt = Date.now();
  78. queryParams = helpers.cleanParams(queryParams);
  79. const queryString = helpers.stringify(queryParams);
  80. return `${quizzesServiceUrl}/v1/quizzes/${encodeURIComponent(quizId)}/${encodeURIComponent(path)}/?${queryString}${answersParamString}`;
  81. }
  82. /**
  83. * Interface to quiz related API calls
  84. *
  85. * @module quizzes
  86. * @inner
  87. * @returns {object}
  88. */
  89. class Quizzes {
  90. constructor(options) {
  91. this.options = options || {};
  92. this.eventDispatcher = new EventDispatcher(options.eventDispatcher);
  93. }
  94. /**
  95. * Retrieve next question from API
  96. *
  97. * @function getQuizNextQuestion
  98. * @description Retrieve next question from Constructor.io API
  99. * @param {string} quizId - The identifier of the quiz
  100. * @param {string} [parameters] - Additional parameters to refine result set
  101. * @param {string} [parameters.section] - Product catalog section
  102. * @param {array} [parameters.answers] - An array of answers in the format [[1,2], [1], ["true"], ["seen"], [""]]. Based on the question type, answers should either be an integer, "true"/"false", "seen" or an empty string ("") if skipped
  103. * @param {string} [parameters.quizVersionId] - Version identifier for the quiz. Version ID will be returned with the first request and it should be passed with subsequent requests. More information can be found: https://docs.constructor.com/reference/configuration-quizzes
  104. * @param {string} [parameters.quizSessionId] - Session identifier for the quiz. Session ID will be returned with the first request and it should be passed with subsequent requests. More information can be found: https://docs.constructor.com/reference/configuration-quizzes
  105. * @param {object} [networkParameters] - Parameters relevant to the network request
  106. * @param {number} [networkParameters.timeout] - Request timeout (in milliseconds)
  107. * @returns {Promise}
  108. * @see https://docs.constructor.com/reference/v1-quizzes-get-quiz-results
  109. * @example
  110. * constructorio.quizzes.getQuizNextQuestion('quizId', {
  111. * answers: [[1,2],[1]],
  112. * section: '123',
  113. * quizVersionId: '123',
  114. * quizSessionId: '1234',
  115. * });
  116. */
  117. getQuizNextQuestion(quizId, parameters, networkParameters = {}) {
  118. let requestUrl;
  119. const { fetch } = this.options;
  120. const controller = new AbortController();
  121. const { signal } = controller;
  122. try {
  123. requestUrl = createQuizUrl(quizId, parameters, this.options, 'next');
  124. } catch (e) {
  125. return Promise.reject(e);
  126. }
  127. // Handle network timeout if specified
  128. helpers.applyNetworkTimeout(this.options, networkParameters, controller);
  129. return fetch(requestUrl, { signal })
  130. .then(helpers.convertResponseToJson)
  131. .then((json) => {
  132. if (json.quiz_version_id) {
  133. this.eventDispatcher.queue('quizzes.getQuizNextQuestion.completed', json);
  134. return json;
  135. }
  136. throw new Error('getQuizNextQuestion response data is malformed');
  137. });
  138. }
  139. /**
  140. * Retrieves filter expression and recommendation URL from given answers
  141. *
  142. * @function getQuizResults
  143. * @description Retrieve quiz recommendation and filter expression from Constructor.io API
  144. * @param {string} quizId - The identifier of the quiz
  145. * @param {string} parameters - Additional parameters to refine result set
  146. * @param {array} parameters.answers - An array of answers in the format [[1,2], [1], ["true"], ["seen"], [""]]. Based on the question type, answers should either be an integer, "true"/"false", "seen" or an empty string ("") if skipped
  147. * @param {string} [parameters.section] - Product catalog section
  148. * @param {string} [parameters.quizVersionId] - Version identifier for the quiz. Version ID will be returned with the first request and it should be passed with subsequent requests. More information can be found: https://docs.constructor.com/reference/configuration-quizzes
  149. * @param {string} [parameters.quizSessionId] - Session identifier for the quiz. Session ID will be returned with the first request and it should be passed with subsequent requests. More information can be found: https://docs.constructor.com/reference/configuration-quizzes
  150. * @param {number} [parameters.page] - The page number of the results
  151. * @param {number} [parameters.resultsPerPage] - The number of results per page to return
  152. * @param {object} [parameters.filters] - Key / value mapping (dictionary) of filters used to refine results
  153. * @param {object} [parameters.fmtOptions] - Key / value mapping (dictionary) of options used for result formatting
  154. * @param {string[]} [parameters.hiddenFields] - Hidden metadata fields to return
  155. * @param {object} [networkParameters] - Parameters relevant to the network request
  156. * @param {number} [networkParameters.timeout] - Request timeout (in milliseconds)
  157. * @returns {Promise}
  158. * @see https://docs.constructor.com/reference/v1-quizzes-get-quiz-results
  159. * @example
  160. * constructorio.quizzes.getQuizResults('quizId', {
  161. * answers: [[1,2],[1]],
  162. * section: '123',
  163. * quizVersionId: '123',
  164. * quizSessionId: '234'
  165. * });
  166. */
  167. getQuizResults(quizId, parameters, networkParameters = {}) {
  168. let requestUrl;
  169. const { fetch } = this.options;
  170. const controller = new AbortController();
  171. const { signal } = controller;
  172. try {
  173. requestUrl = createQuizUrl(quizId, parameters, this.options, 'results');
  174. } catch (e) {
  175. return Promise.reject(e);
  176. }
  177. // Handle network timeout if specified
  178. helpers.applyNetworkTimeout(this.options, networkParameters, controller);
  179. return fetch(requestUrl, { signal })
  180. .then(helpers.convertResponseToJson)
  181. .then((json) => {
  182. if (json.quiz_version_id) {
  183. this.eventDispatcher.queue('quizzes.getQuizResults.completed', json);
  184. return json;
  185. }
  186. throw new Error('getQuizResults response data is malformed');
  187. });
  188. }
  189. /**
  190. * Retrieves configuration for the results page of a particular quiz
  191. *
  192. * @function getQuizResultsConfig
  193. * @description Retrieve quiz results page configuration from Constructor.io API
  194. * @param {string} quizId - The identifier of the quiz
  195. * @param {string} parameters - Additional parameters
  196. * @param {string} [parameters.section] - Product catalog section
  197. * @param {string} [parameters.quizVersionId] - Version identifier for the quiz. Version ID will be returned with the first request and it should be passed with subsequent requests. More information can be found: https://docs.constructor.com/reference/configuration-quizzes
  198. * @param {object} [networkParameters] - Parameters relevant to the network request
  199. * @param {number} [networkParameters.timeout] - Request timeout (in milliseconds)
  200. * @returns {Promise}
  201. * @example
  202. * constructorio.quizzes.getQuizResultsConfig('quizId', {
  203. * quizVersionId: '123',
  204. * });
  205. */
  206. getQuizResultsConfig(quizId, parameters, networkParameters = {}) {
  207. let requestUrl;
  208. const { fetch } = this.options;
  209. const controller = new AbortController();
  210. const { signal } = controller;
  211. try {
  212. requestUrl = createQuizUrl(quizId, parameters, this.options, 'results_config');
  213. } catch (e) {
  214. return Promise.reject(e);
  215. }
  216. // Handle network timeout if specified
  217. helpers.applyNetworkTimeout(this.options, networkParameters, controller);
  218. return fetch(requestUrl, { signal })
  219. .then(helpers.convertResponseToJson)
  220. .then((json) => {
  221. if (json.quiz_version_id) {
  222. this.eventDispatcher.queue('quizzes.getQuizResultsConfig.completed', json);
  223. return json;
  224. }
  225. throw new Error('getQuizResultsConfig response data is malformed');
  226. });
  227. }
  228. }
  229. module.exports = Quizzes;