modules/search.js

  1. /* eslint-disable complexity */
  2. /* eslint-disable max-len */
  3. /* eslint-disable object-curly-newline, no-underscore-dangle */
  4. const EventDispatcher = require('../utils/event-dispatcher');
  5. const helpers = require('../utils/helpers');
  6. // Create URL from supplied query (term) and parameters
  7. function createSearchUrl(query, parameters, options, isVoiceSearch = false) {
  8. const {
  9. apiKey,
  10. version,
  11. serviceUrl,
  12. sessionId,
  13. clientId,
  14. userId,
  15. segments,
  16. testCells,
  17. } = options;
  18. let queryParams = { c: version };
  19. queryParams.key = apiKey;
  20. queryParams.i = clientId;
  21. queryParams.s = sessionId;
  22. // Validate query (term) is provided
  23. if (!query || typeof query !== 'string') {
  24. throw new Error('query is a required parameter of type string');
  25. }
  26. // Pull test cells from options
  27. if (testCells) {
  28. Object.keys(testCells).forEach((testCellKey) => {
  29. queryParams[`ef-${testCellKey}`] = testCells[testCellKey];
  30. });
  31. }
  32. // Pull user segments from options
  33. if (segments && segments.length) {
  34. queryParams.us = segments;
  35. }
  36. // Pull user id from options and ensure string
  37. if (userId) {
  38. queryParams.ui = String(userId);
  39. }
  40. if (parameters) {
  41. const { offset, page, resultsPerPage, filters, sortBy, sortOrder, section, fmtOptions, hiddenFields, hiddenFacets, variationsMap, qsParam, preFilterExpression, filterMatchTypes } = parameters;
  42. // Pull offset from parameters
  43. if (!helpers.isNil(offset)) {
  44. queryParams.offset = offset;
  45. }
  46. // Pull page from parameters
  47. if (!helpers.isNil(page)) {
  48. queryParams.page = page;
  49. }
  50. // Pull results per page from parameters
  51. if (!helpers.isNil(resultsPerPage)) {
  52. queryParams.num_results_per_page = resultsPerPage;
  53. }
  54. // Pull filters from parameters
  55. if (filters) {
  56. queryParams.filters = filters;
  57. }
  58. if (filterMatchTypes) {
  59. queryParams.filter_match_types = filterMatchTypes;
  60. }
  61. // Pull sort by from parameters
  62. if (sortBy) {
  63. queryParams.sort_by = sortBy;
  64. }
  65. // Pull sort order from parameters
  66. if (sortOrder) {
  67. queryParams.sort_order = sortOrder;
  68. }
  69. // Pull section from parameters
  70. if (section) {
  71. queryParams.section = section;
  72. }
  73. // Pull format options from parameters
  74. if (fmtOptions) {
  75. queryParams.fmt_options = fmtOptions;
  76. }
  77. // Pull hidden fields from parameters
  78. if (hiddenFields) {
  79. if (queryParams.fmt_options) {
  80. queryParams.fmt_options.hidden_fields = hiddenFields;
  81. } else {
  82. queryParams.fmt_options = { hidden_fields: hiddenFields };
  83. }
  84. }
  85. // Pull hidden facets from parameters
  86. if (hiddenFacets) {
  87. if (queryParams.fmt_options) {
  88. queryParams.fmt_options.hidden_facets = hiddenFacets;
  89. } else {
  90. queryParams.fmt_options = { hidden_facets: hiddenFacets };
  91. }
  92. }
  93. // Pull variations map from parameters
  94. if (variationsMap) {
  95. queryParams.variations_map = JSON.stringify(variationsMap);
  96. }
  97. // Pull pre_filter_expression from parameters
  98. if (preFilterExpression) {
  99. queryParams.pre_filter_expression = JSON.stringify(preFilterExpression);
  100. }
  101. // pull qs param from parameters
  102. if (qsParam) {
  103. queryParams.qs = JSON.stringify(qsParam);
  104. }
  105. }
  106. queryParams._dt = Date.now();
  107. queryParams = helpers.cleanParams(queryParams);
  108. const queryString = helpers.stringify(queryParams);
  109. const searchUrl = isVoiceSearch ? 'search/natural_language' : 'search';
  110. return `${serviceUrl}/${searchUrl}/${helpers.encodeURIComponentRFC3986(helpers.trimNonBreakingSpaces(query))}?${queryString}`;
  111. }
  112. /**
  113. * Interface to search related API calls
  114. *
  115. * @module search
  116. * @inner
  117. * @returns {object}
  118. */
  119. class Search {
  120. constructor(options) {
  121. this.options = options || {};
  122. this.eventDispatcher = new EventDispatcher(options.eventDispatcher);
  123. }
  124. /**
  125. * Retrieve search results from API
  126. *
  127. * @function getSearchResults
  128. * @description Retrieve search results from Constructor.io API
  129. * @param {string} query - Term to use to perform a search
  130. * @param {object} [parameters] - Additional parameters to refine result set
  131. * @param {number} [parameters.page] - The page number of the results(Can't be used together with offset)
  132. * @param {number} [parameters.offset] - The number of results to skip from the beginning (Can't be used together with page)
  133. * @param {number} [parameters.resultsPerPage] - The number of results per page to return
  134. * @param {object} [parameters.filters] - Key / value mapping (dictionary) of filters used to refine results
  135. * @param {string} [parameters.sortBy='relevance'] - The sort method for results
  136. * @param {string} [parameters.sortOrder='descending'] - The sort order for results
  137. * @param {string} [parameters.section='Products'] - The section name for results
  138. * @param {object} [parameters.fmtOptions] - The format options used to refine result groups. Please refer to https://docs.constructor.com/reference/search-search-resultsqueries for details
  139. * @param {object} [parameters.preFilterExpression] - Faceting expression to scope search results. Please refer to https://docs.constructor.com/reference/configuration-collections
  140. * @param {string[]} [parameters.hiddenFields] - Hidden metadata fields to return
  141. * @param {string[]} [parameters.hiddenFacets] - Hidden facets to return
  142. * @param {object} [parameters.variationsMap] - The variations map object to aggregate variations. Please refer to https://docs.constructor.com/reference/shared-variations-mapping for details
  143. * @param {object} [parameters.qsParam] - Parameters listed above can be serialized into a JSON object and parsed through this parameter. Please refer to https://docs.constructor.com/reference/v1-search-get-search-results
  144. * @param {object} [parameters.filterMatchTypes] - An object specifying whether results must match `all`, `any` or `none` of a given filter. Please refer to https://docs.constructor.com/reference/v1-search-get-search-results
  145. * @param {object} [networkParameters] - Parameters relevant to the network request
  146. * @param {number} [networkParameters.timeout] - Request timeout (in milliseconds)
  147. * @returns {Promise}
  148. * @see https://docs.constructor.com/reference/search-search-results
  149. * @example
  150. * constructorio.search.getSearchResults('t-shirt', {
  151. * resultsPerPage: 40,
  152. * filters: {
  153. * size: 'medium'
  154. * },
  155. * filterMatchTypes: {
  156. * size: 'all'
  157. * }
  158. * });
  159. */
  160. getSearchResults(query, parameters, networkParameters = {}) {
  161. let requestUrl;
  162. const { fetch } = this.options;
  163. let signal;
  164. if (typeof AbortController === 'function') {
  165. const controller = new AbortController();
  166. signal = controller && controller.signal;
  167. // Handle network timeout if specified
  168. helpers.applyNetworkTimeout(this.options, networkParameters, controller);
  169. }
  170. try {
  171. requestUrl = createSearchUrl(query, parameters, this.options);
  172. } catch (e) {
  173. return Promise.reject(e);
  174. }
  175. return fetch(requestUrl, { signal })
  176. .then(helpers.convertResponseToJson)
  177. .then((json) => {
  178. // Search results
  179. if (json.response && json.response.results) {
  180. if (json.result_id) {
  181. // Append `result_id` to each result item
  182. json.response.results.forEach((result) => {
  183. // eslint-disable-next-line no-param-reassign
  184. result.result_id = json.result_id;
  185. });
  186. }
  187. this.eventDispatcher.queue('search.getSearchResults.completed', json);
  188. return json;
  189. }
  190. // Redirect rules
  191. if (json.response && json.response.redirect) {
  192. this.eventDispatcher.queue('search.getSearchResults.completed', json);
  193. return json;
  194. }
  195. throw new Error('getSearchResults response data is malformed');
  196. });
  197. }
  198. /**
  199. * Retrieve voice search results from API
  200. *
  201. * @function getVoiceSearchResults
  202. * @description Retrieve voice search results from Constructor.io API
  203. * @param {string} query - Term to use to perform a voice search
  204. * @param {object} [parameters] - Additional parameters to refine result set
  205. * @param {number} [parameters.page] - The page number of the results (Can't be used together with offset)
  206. * @param {number} [parameters.offset] - The number of results to skip from the beginning (Can't be used together with page)
  207. * @param {number} [parameters.resultsPerPage] - The number of results per page to return
  208. * @param {string} [parameters.section='Products'] - The section name for results
  209. * @param {object} [parameters.fmtOptions] - The format options used to refine result groups. Please refer to https://docs.constructor.com/reference/search-search-resultsqueries for details
  210. * @param {object} [parameters.preFilterExpression] - Faceting expression to scope search results. Please refer to https://docs.constructor.com/reference/configuration-collections
  211. * @param {object} [parameters.variationsMap] - The variations map object to aggregate variations. Please refer to https://docs.constructor.com/reference/shared-variations-mapping for details
  212. * @param {string[]} [parameters.hiddenFields] - Hidden metadata fields to return
  213. * @param {string[]} [parameters.hiddenFacets] - Hidden facets to return
  214. * @param {object} [parameters.qsParam] - Parameters listed above can be serialized into a JSON object and parsed through this parameter. Please refer to https://docs.constructor.com/reference/v1-search-get-search-results
  215. * @param {object} [networkParameters] - Parameters relevant to the network request
  216. * @param {number} [networkParameters.timeout] - Request timeout (in milliseconds)
  217. * @returns {Promise}
  218. * @see https://docs.constructor.com/docs/products-search-learn-about-search
  219. * @example
  220. * constructorio.search.getVoiceSearchResults('show me lipstick');
  221. */
  222. getVoiceSearchResults(query, parameters, networkParameters = {}) {
  223. let requestUrl;
  224. const { fetch } = this.options;
  225. let signal;
  226. if (typeof AbortController === 'function') {
  227. const controller = new AbortController();
  228. signal = controller && controller.signal;
  229. // Handle network timeout if specified
  230. helpers.applyNetworkTimeout(this.options, networkParameters, controller);
  231. }
  232. try {
  233. const isVoiceSearch = true;
  234. requestUrl = createSearchUrl(query, parameters, this.options, isVoiceSearch);
  235. } catch (e) {
  236. return Promise.reject(e);
  237. }
  238. return fetch(requestUrl, { signal })
  239. .then(helpers.convertResponseToJson)
  240. .then((json) => {
  241. // Search results
  242. if (json.response && json.response.results) {
  243. if (json.result_id) {
  244. // Append `result_id` to each result item
  245. json.response.results.forEach((result) => {
  246. // eslint-disable-next-line no-param-reassign
  247. result.result_id = json.result_id;
  248. });
  249. }
  250. this.eventDispatcher.queue('search.getVoiceSearchResults.completed', json);
  251. return json;
  252. }
  253. // Redirect rules
  254. if (json.response && json.response.redirect) {
  255. this.eventDispatcher.queue('search.getVoiceSearchResults.completed', json);
  256. return json;
  257. }
  258. throw new Error('getVoiceSearchResults response data is malformed');
  259. });
  260. }
  261. }
  262. module.exports = Search;