angular-instantsearch
Version:
Lightning-fast search for Angular apps, by Algolia.
96 lines • 16.1 kB
JavaScript
import * as algoliasearchProxy from 'algoliasearch/lite';
import * as encodeProxy from 'querystring-es3/encode';
import { VERSION as AngularVersion } from '@angular/core';
import { VERSION } from './version';
// compatibility with different typescript settings:
// - esModuleInterop
// - allowSyntheticDefaultImports
const algoliasearch = (typeof algoliasearchProxy.default === 'function'
? algoliasearchProxy.default
: algoliasearchProxy);
const encode = encodeProxy.default || encodeProxy;
export function createSSRSearchClient({ appId, apiKey, httpClient, HttpHeaders, transferState, makeStateKey, options = {}, }) {
// A custom network request needs to be done, using TransferState of Angular.
// This is done to make sure the request done backend for SSR doesn't get
// made again frontend during hydration.
// For compatibility with both v3 and v4 of algoliasearch, we are overriding the
// network request function in two places:
// v4: custom "requester"
// v3: override "_request" on the prototype
// since neither v3 uses the requester argument, and v4 use the _request, we
// can safely do this without checking the version
const searchClient = algoliasearch(appId, apiKey, Object.assign(Object.assign({}, options), { requester: {
send({ headers, method, url, data }) {
const transferStateKey = makeStateKey(`ngais(${data})`);
if (transferState.hasKey(transferStateKey)) {
const response = JSON.parse(transferState.get(transferStateKey, JSON.stringify({})));
return Promise.resolve({
status: response.status,
content: JSON.stringify(response.body),
isTimedOut: false,
});
}
return new Promise((resolve, reject) => {
httpClient
.request(method, url, {
headers,
body: data,
observe: 'response',
})
.subscribe(response => {
transferState.set(transferStateKey, JSON.stringify(response));
resolve({
status: response.status,
content: JSON.stringify(response.body),
isTimedOut: false,
});
}, response => reject({
status: response.status,
body: response.body,
}));
});
},
} }));
searchClient.addAlgoliaAgent(`angular (${AngularVersion.full})`);
searchClient.addAlgoliaAgent(`angular-instantsearch (${VERSION})`);
searchClient.addAlgoliaAgent(`angular-instantsearch-server (${VERSION})`);
searchClient._request = (rawUrl, options) => {
let headers = new HttpHeaders();
headers = headers.set('content-type', options.method === 'POST'
? 'application/x-www-form-urlencoded'
: 'application/json');
headers = headers.set('accept', 'application/json');
const url = rawUrl + (rawUrl.includes('?') ? '&' : '?') + encode(options.headers);
const transferStateKey = makeStateKey(`ngais(${options.body})`);
if (transferState.hasKey(transferStateKey)) {
const response = JSON.parse(transferState.get(transferStateKey, JSON.stringify({})));
return Promise.resolve({
statusCode: response.status,
body: response.body,
headers: response.headers,
});
}
return new Promise((resolve, reject) => {
httpClient
.request(options.method, url, {
headers,
body: options.body,
observe: 'response',
})
.subscribe(response => {
transferState.set(transferStateKey, JSON.stringify(response));
resolve({
statusCode: response.status,
body: response.body,
headers: response.headers,
});
}, response => reject({
statusCode: response.status,
body: response.body,
headers: response.headers,
}));
});
};
return searchClient;
}
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"create-ssr-search-client.js","sourceRoot":"","sources":["../../src/create-ssr-search-client.ts"],"names":[],"mappings":"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","sourcesContent":["import * as algoliasearchProxy from 'algoliasearch/lite';\nimport * as encodeProxy from 'querystring-es3/encode';\nimport { VERSION as AngularVersion } from '@angular/core';\nimport { HttpClient, HttpHeaders } from '@angular/common/http';\nimport { TransferState, StateKey } from '@angular/platform-browser';\nimport { VERSION } from './version';\n\ntype SSRSearchClientOptions = {\n  appId: string;\n  apiKey: string;\n  httpClient: HttpClient;\n  HttpHeaders: typeof HttpHeaders;\n  transferState: TransferState;\n  options?: object;\n  makeStateKey<T = void>(key: string): StateKey<T>;\n};\n\ntype RequestOptions = {\n  // Algolia only uses GET and POST methods for searching.\n  // See: https://www.algolia.com/doc/rest-api/search/#search-endpoints\n  method: 'GET' | 'POST';\n  headers: string;\n  body: string;\n};\n\n// compatibility with different typescript settings:\n// - esModuleInterop\n// - allowSyntheticDefaultImports\nconst algoliasearch = (typeof algoliasearchProxy.default === 'function'\n  ? algoliasearchProxy.default\n  : algoliasearchProxy) as typeof algoliasearchProxy.default extends Function\n  ? typeof algoliasearchProxy.default\n  : typeof algoliasearchProxy;\n\nconst encode = encodeProxy.default || encodeProxy;\n\nexport function createSSRSearchClient({\n  appId,\n  apiKey,\n  httpClient,\n  HttpHeaders,\n  transferState,\n  makeStateKey,\n  options = {},\n}: SSRSearchClientOptions) {\n  // A custom network request needs to be done, using TransferState of Angular.\n  // This is done to make sure the request done backend for SSR doesn't get\n  // made again frontend during hydration.\n  // For compatibility with both v3 and v4 of algoliasearch, we are overriding the\n  // network request function in two places:\n  // v4: custom \"requester\"\n  // v3: override \"_request\" on the prototype\n  // since neither v3 uses the requester argument, and v4 use the _request, we\n  // can safely do this without checking the version\n  const searchClient = algoliasearch(appId, apiKey, {\n    ...options,\n    requester: {\n      send({ headers, method, url, data }) {\n        const transferStateKey = makeStateKey<string>(`ngais(${data})`);\n\n        if (transferState.hasKey(transferStateKey)) {\n          const response = JSON.parse(\n            transferState.get(transferStateKey, JSON.stringify({}))\n          );\n\n          return Promise.resolve({\n            status: response.status,\n            content: JSON.stringify(response.body),\n            isTimedOut: false,\n          });\n        }\n\n        return new Promise((resolve, reject) => {\n          httpClient\n            .request(method, url, {\n              headers,\n              body: data,\n              observe: 'response',\n            })\n            .subscribe(\n              response => {\n                transferState.set(transferStateKey, JSON.stringify(response));\n\n                resolve({\n                  status: response.status,\n                  content: JSON.stringify(response.body),\n                  isTimedOut: false,\n                });\n              },\n              response =>\n                reject({\n                  status: response.status,\n                  body: response.body,\n                })\n            );\n        });\n      },\n    },\n  });\n\n  searchClient.addAlgoliaAgent(`angular (${AngularVersion.full})`);\n  searchClient.addAlgoliaAgent(`angular-instantsearch (${VERSION})`);\n  searchClient.addAlgoliaAgent(`angular-instantsearch-server (${VERSION})`);\n\n  (searchClient as any)._request = (\n    rawUrl: string,\n    options: RequestOptions\n  ) => {\n    let headers = new HttpHeaders();\n\n    headers = headers.set(\n      'content-type',\n      options.method === 'POST'\n        ? 'application/x-www-form-urlencoded'\n        : 'application/json'\n    );\n\n    headers = headers.set('accept', 'application/json');\n\n    const url =\n      rawUrl + (rawUrl.includes('?') ? '&' : '?') + encode(options.headers);\n\n    const transferStateKey = makeStateKey<string>(`ngais(${options.body})`);\n\n    if (transferState.hasKey(transferStateKey)) {\n      const response = JSON.parse(\n        transferState.get(transferStateKey, JSON.stringify({}))\n      );\n\n      return Promise.resolve({\n        statusCode: response.status,\n        body: response.body,\n        headers: response.headers,\n      });\n    }\n\n    return new Promise((resolve, reject) => {\n      httpClient\n        .request(options.method, url, {\n          headers,\n          body: options.body,\n          observe: 'response',\n        })\n        .subscribe(\n          response => {\n            transferState.set(transferStateKey, JSON.stringify(response));\n\n            resolve({\n              statusCode: response.status,\n              body: response.body,\n              headers: response.headers,\n            });\n          },\n          response =>\n            reject({\n              statusCode: response.status,\n              body: response.body,\n              headers: response.headers,\n            })\n        );\n    });\n  };\n\n  return searchClient;\n}\n"]}