constructorio.js

  1. /* eslint-disable camelcase, no-unneeded-ternary, max-len, complexity */
  2. const ConstructorioID = require('@constructor-io/constructorio-id');
  3. // Modules
  4. const Search = require('./modules/search');
  5. const Browse = require('./modules/browse');
  6. const Autocomplete = require('./modules/autocomplete');
  7. const Recommendations = require('./modules/recommendations');
  8. const Tracker = require('./modules/tracker');
  9. const EventDispatcher = require('./utils/event-dispatcher');
  10. const helpers = require('./utils/helpers');
  11. const { default: packageVersion } = require('./version');
  12. const Quizzes = require('./modules/quizzes');
  13. const Assistant = require('./modules/assistant');
  14. // Compute package version string
  15. const computePackageVersion = () => {
  16. const versionPrefix = 'ciojs-client-';
  17. const versionModifiers = [];
  18. if (!helpers.canUseDOM()) {
  19. versionModifiers.push('domless');
  20. }
  21. if (typeof process !== 'undefined' && typeof process.env !== 'undefined' && process.env.BUNDLED) {
  22. versionModifiers.push('bundled');
  23. }
  24. return `${versionPrefix}${versionModifiers.join('-')}${versionModifiers.length ? '-' : ''}${packageVersion}`;
  25. };
  26. /**
  27. * Class to instantiate the ConstructorIO client.
  28. */
  29. class ConstructorIO {
  30. /**
  31. * @param {object} parameters - Parameters for client instantiation
  32. * @param {string} parameters.apiKey - Constructor.io API key
  33. * @param {string} [parameters.serviceUrl='https://ac.cnstrc.com'] - API URL endpoint
  34. * @param {string} [parameters.quizzesServiceUrl='https://quizzes.cnstrc.com'] - Quizzes API URL endpoint
  35. * @param {string} [parameters.assistantServiceUrl='https://assistant.cnstrc.com'] - AI Assistant API URL endpoint
  36. * @param {array} [parameters.segments] - User segments
  37. * @param {object} [parameters.testCells] - User test cells
  38. * @param {string} [parameters.clientId] - Client ID, defaults to value supplied by 'constructorio-id' module
  39. * @param {number} [parameters.sessionId] - Session ID, defaults to value supplied by 'constructorio-id' module
  40. * @param {string} [parameters.userId] - User ID
  41. * @param {function} [parameters.fetch] - If supplied, will be utilized for requests rather than default Fetch API
  42. * @param {number} [parameters.trackingSendDelay=250] - Amount of time to wait before sending tracking events (in ms)
  43. * @param {boolean} [parameters.sendReferrerWithTrackingEvents=true] - Indicates if the referrer is sent with tracking events
  44. * @param {boolean} [parameters.sendTrackingEvents=false] - Indicates if tracking events should be dispatched
  45. * @param {object} [parameters.idOptions] - Options object to be supplied to 'constructorio-id' module
  46. * @param {object} [parameters.eventDispatcher] - Options related to 'EventDispatcher' class
  47. * @param {boolean} [parameters.eventDispatcher.enabled=true] - Determine if events should be dispatched
  48. * @param {boolean} [parameters.eventDispatcher.waitForBeacon=true] - Wait for beacon before dispatching events
  49. * @param {object} [parameters.networkParameters] - Parameters relevant to network requests
  50. * @param {number} [parameters.networkParameters.timeout] - Request timeout (in milliseconds) - may be overridden within individual method calls
  51. * @property {object} search - Interface to {@link module:search}
  52. * @property {object} browse - Interface to {@link module:browse}
  53. * @property {object} autocomplete - Interface to {@link module:autocomplete}
  54. * @property {object} recommendations - Interface to {@link module:recommendations}
  55. * @property {object} tracker - Interface to {@link module:tracker}
  56. * @property {object} quizzes - Interface to {@link module:quizzes}
  57. * @property {object} assistant - Interface to {@link module:assistant}
  58. * @returns {class}
  59. */
  60. constructor(options = {}) {
  61. const {
  62. apiKey,
  63. version: versionFromOptions,
  64. serviceUrl,
  65. quizzesServiceUrl,
  66. assistantServiceUrl,
  67. segments,
  68. testCells,
  69. clientId,
  70. sessionId,
  71. userId,
  72. fetch: fetchFromOptions,
  73. trackingSendDelay,
  74. sendReferrerWithTrackingEvents,
  75. sendTrackingEvents,
  76. eventDispatcher,
  77. idOptions,
  78. beaconMode,
  79. networkParameters,
  80. } = options;
  81. if (!apiKey || typeof apiKey !== 'string') {
  82. throw new Error('API key is a required parameter of type string');
  83. }
  84. let session_id;
  85. let client_id;
  86. const versionFromGlobal = typeof global !== 'undefined' && global.CLIENT_VERSION;
  87. // Initialize ID session if DOM context is available
  88. if (helpers.canUseDOM()) {
  89. ({ session_id, client_id } = new ConstructorioID(idOptions || {}));
  90. } else {
  91. // Validate session ID is provided
  92. if (!sessionId || typeof sessionId !== 'number') {
  93. throw new Error('sessionId is a required user parameter of type number');
  94. }
  95. // Validate client ID is provided
  96. if (!clientId || typeof clientId !== 'string') {
  97. throw new Error('clientId is a required user parameter of type string');
  98. }
  99. }
  100. const normalizedServiceUrl = serviceUrl && serviceUrl.replace(/\/$/, '');
  101. this.options = {
  102. apiKey,
  103. version: versionFromOptions || versionFromGlobal || computePackageVersion(),
  104. serviceUrl: helpers.addHTTPSToString(normalizedServiceUrl) || 'https://ac.cnstrc.com',
  105. quizzesServiceUrl: (quizzesServiceUrl && quizzesServiceUrl.replace(/\/$/, '')) || 'https://quizzes.cnstrc.com',
  106. assistantServiceUrl: (assistantServiceUrl && assistantServiceUrl.replace(/\/$/, '')) || 'https://assistant.cnstrc.com',
  107. sessionId: sessionId || session_id,
  108. clientId: clientId || client_id,
  109. userId,
  110. segments,
  111. testCells,
  112. fetch: fetchFromOptions || fetch,
  113. trackingSendDelay,
  114. sendTrackingEvents,
  115. sendReferrerWithTrackingEvents,
  116. eventDispatcher,
  117. beaconMode: (beaconMode === false) ? false : true, // Defaults to 'true',
  118. networkParameters: networkParameters || {},
  119. };
  120. // Expose global modules
  121. this.search = new Search(this.options);
  122. this.browse = new Browse(this.options);
  123. this.autocomplete = new Autocomplete(this.options);
  124. this.recommendations = new Recommendations(this.options);
  125. this.tracker = new Tracker(this.options);
  126. this.quizzes = new Quizzes(this.options);
  127. this.assistant = new Assistant(this.options);
  128. // Dispatch initialization event
  129. new EventDispatcher(options.eventDispatcher).queue('instantiated', this.options);
  130. }
  131. /**
  132. * Sets the client options
  133. *
  134. * @param {object} options - Client options to update
  135. * @param {string} [options.apiKey] - Constructor.io API key
  136. * @param {array} [options.segments] - User segments
  137. * @param {object} [options.testCells] - User test cells
  138. * @param {number} [options.sessionId] - Session ID - Will only be set in DOM-less environments
  139. * @param {string} [options.userId] - User ID
  140. */
  141. setClientOptions(options) {
  142. if (Object.keys(options).length) {
  143. const { apiKey, segments, testCells, sessionId, userId } = options;
  144. if (apiKey) {
  145. this.options.apiKey = apiKey;
  146. }
  147. if (segments) {
  148. this.options.segments = segments;
  149. }
  150. if (testCells) {
  151. this.options.testCells = testCells;
  152. }
  153. // Set Session ID in dom-less environments only
  154. if (sessionId && !helpers.canUseDOM()) {
  155. this.options.sessionId = sessionId;
  156. }
  157. // If User ID is passed
  158. if ('userId' in options) {
  159. this.options.userId = userId;
  160. }
  161. }
  162. }
  163. }
  164. // Exposed for testing
  165. ConstructorIO.Tracker = Tracker;
  166. // Expose on window object if available
  167. if (helpers.canUseDOM()) {
  168. window.ConstructorioClient = ConstructorIO;
  169. }
  170. module.exports = ConstructorIO;