Skip to content
This repository was archived by the owner on Oct 20, 2023. It is now read-only.

Add search path and collection type to theme-predictive-search #140

Merged
merged 2 commits into from
Dec 15, 2020
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
Original file line number Diff line number Diff line change
@@ -1,6 +1,7 @@
import PredictiveSearch from "@shopify/theme-predictive-search";

var DEFAULT_PREDICTIVE_SEARCH_API_CONFIG = {
search_path: PredictiveSearch.SEARCH_PATH,
resources: {
type: [PredictiveSearch.TYPES.PRODUCT],
options: {
Expand Down
8 changes: 7 additions & 1 deletion packages/theme-predictive-search/README.md
Original file line number Diff line number Diff line change
Expand Up @@ -8,6 +8,11 @@
<th>Type</th>
<th>Description</th>
</tr>
<tr>
<td><code>search_path</code> (optional)</td>
<td>String</td>
<td>Search path to prepend the predictive search route in different locales (e.g. <code>/fr/search</code>). Default to <code>/search</code></td>
</tr>
<tr>
<td><code>resources</code> (required)</td>
<td>Object</td>
Expand All @@ -16,7 +21,7 @@
<tr>
<td><code>type</code> (required)</td>
<td>Array</td>
<td>Specifies the type of results requested. Valid values: <code>product</code>, <code>page</code>, <code>article</code>.</td>
<td>Specifies the type of results requested. Valid values: <code>product</code>, <code>page</code>, <code>article</code>, <code>collection</code>.</td>
</tr>
<tr>
<td><code>limit</code> (optional)</td>
Expand Down Expand Up @@ -50,6 +55,7 @@ See the [help docs for Predictive Search](https://help.shopify.com/en/themes/dev
import PredictiveSearch from "@shopify/theme-predictive-search";

var predictiveSearch = new PredictiveSearch({
search_path: PredictiveSearch.SEARCH_PATH,
resources: {
type: [PredictiveSearch.TYPES.PRODUCT],
limit: 4,
Expand Down
43 changes: 31 additions & 12 deletions packages/theme-predictive-search/__tests__/request.test.js
Original file line number Diff line number Diff line change
Expand Up @@ -21,7 +21,26 @@ describe("request()", () => {
.body(JSON.stringify({ foo: "bar" }))
);

request("config=foo", "foo-200", spyOnSucess);
request("/search", "config=foo", "foo-200", spyOnSucess);

jest.runAllTimers();

expect(spyOnSucess).toHaveBeenNthCalledWith(1, {
query: "foo-200",
foo: "bar"
});
});

it("valid with locale search route", () => {
const spyOnSucess = jest.fn();
xhrMock.get(/^(\/[a-z-]*)?\/search\/suggest\.json/g, (req, res) =>
res
.status(200)
.header("Content-Type", "application/json; charset=utf-8")
.body(JSON.stringify({ foo: "bar" }))
);

request('/pt-pt/search', "config=foo", "foo-200", spyOnSucess);

jest.runAllTimers();

Expand All @@ -40,7 +59,7 @@ describe("request()", () => {
.body(JSON.stringify({ foo: "bar" }))
);

request("config=foo", "foo-200", spyOnSucess);
request("/search", "config=foo", "foo-200", spyOnSucess);

jest.runAllTimers();

Expand All @@ -58,7 +77,7 @@ describe("request()", () => {
.body("boom")
);

request("config=foo", "foo-200-invalid", null, error => {
request("/search", "config=foo", "foo-200-invalid", null, error => {
expect(error).toBeInstanceOf(Error);
expect(error.status).toBe(200);
expect(error.message).toBe(
Expand All @@ -78,7 +97,7 @@ describe("request()", () => {
.body("boom")
);

request("config=foo", "foo-200-invalid", null, error => {
request("/search", "config=foo", "foo-200-invalid", null, error => {
expect(error).toBeInstanceOf(Error);
expect(error.status).toBe(200);
expect(error.message).toBe(
Expand All @@ -95,7 +114,7 @@ describe("request()", () => {
res.status(200).body("boom")
);

request("config=foo", "foo-200-invalid", null, error => {
request("/search", "config=foo", "foo-200-invalid", null, error => {
expect(error).toBeInstanceOf(Error);
expect(error.status).toBe(200);
expect(error.name).toBe("Content-Type error");
Expand All @@ -118,7 +137,7 @@ describe("request()", () => {
.body("boom")
);

request("config=foo", "foo-404", null, error => {
request("/search", "config=foo", "foo-404", null, error => {
expect(error).toBeInstanceOf(Error);
expect(error.status).toBe(404);
expect(error.name).toBe("Not found");
Expand All @@ -144,7 +163,7 @@ describe("request()", () => {
)
);

request("config=foo", "foo-422", null, error => {
request("/search", "config=foo", "foo-422", null, error => {
expect(error).toBeInstanceOf(Error);
expect(error.status).toBe(422);
expect(error.name).toBe("Invalid parameter error");
Expand All @@ -164,7 +183,7 @@ describe("request()", () => {
.body("Invalid parameter error")
);

request("config=foo", "foo-422-invalid", null, error => {
request("/search", "config=foo", "foo-422-invalid", null, error => {
expect(error).toBeInstanceOf(Error);
expect(error.status).toBe(422);
expect(error.name).toBe("JSON parse error");
Expand All @@ -191,7 +210,7 @@ describe("request()", () => {
)
);

request("config=foo", "foo-429", null, error => {
request("/search", "config=foo", "foo-429", null, error => {
expect(error).toBeInstanceOf(Error);
expect(error.status).toBe(429);
expect(error.name).toBe("Throttled");
Expand All @@ -212,7 +231,7 @@ describe("request()", () => {
.body("Throttled")
);

request("config=foo", "foo-429-invalid", null, error => {
request("/search", "config=foo", "foo-429-invalid", null, error => {
expect(error).toBeInstanceOf(Error);
expect(error.status).toBe(429);
expect(error.name).toBe("JSON parse error");
Expand All @@ -238,7 +257,7 @@ describe("request()", () => {
)
);

request("config=foo", "foo-417", null, error => {
request("/search", "config=foo", "foo-417", null, error => {
expect(error).toBeInstanceOf(Error);
expect(error.status).toBe(417);
expect(error.name).toBe("Expectation Failed");
Expand All @@ -255,7 +274,7 @@ describe("request()", () => {
it("head request", done => {
xhrMock.get(/^\/search\/suggest\.json/g, (req, res) => res.status(500));

request("config=foo", "foo-500", null, error => {
request("/search", "config=foo", "foo-500", null, error => {
expect(error).toBeInstanceOf(Error);
expect(error.status).toBe(500);
expect(error.name).toBe("Server error");
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -4,6 +4,7 @@ import searchAsYouTypeTheCallingFixture from "../__fixtures__/search_as_you_type

/* eslint-disable camelcase */
const defaultConfig = {
search_path: PredictiveSearch.SEARCH_PATH,
resources: {
type: [PredictiveSearch.TYPES.PRODUCT],
options: {
Expand Down
5 changes: 3 additions & 2 deletions packages/theme-predictive-search/src/request.js
Original file line number Diff line number Diff line change
Expand Up @@ -9,8 +9,9 @@ import {
GenericError
} from "./utilities/CustomError";

export default function request(configParams, query, onSuccess, onError) {
export default function request(searchPath, configParams, query, onSuccess, onError) {
var xhr = new XMLHttpRequest();
var route = searchPath + '/suggest.json';

xhr.onreadystatechange = function () {
if (xhr.readyState === XMLHttpRequest.DONE) {
Expand Down Expand Up @@ -123,7 +124,7 @@ export default function request(configParams, query, onSuccess, onError) {

xhr.open(
"get",
"/search/suggest.json?q=" + encodeURIComponent(query) + "&" + configParams
route + "?q=" + encodeURIComponent(query) + "&" + configParams
);

xhr.setRequestHeader("Content-Type", "application/json");
Expand Down
17 changes: 15 additions & 2 deletions packages/theme-predictive-search/src/theme-predictive-search.js
Original file line number Diff line number Diff line change
Expand Up @@ -10,18 +10,30 @@ export default function PredictiveSearch(config) {
throw new TypeError("No config object was specified");
}

var configParameters = config;

this._retryAfter = null;
this._currentQuery = null;

this.dispatcher = new Dispatcher();
this.cache = new Cache({ bucketSize: 40 });
this.configParams = objectToQueryParams(config);

this.searchPath = configParameters.search_path || "/search";

if(configParameters.search_path) {
delete configParameters['search_path'];
}

this.configParams = objectToQueryParams(configParameters);
}

PredictiveSearch.SEARCH_PATH = "/search";

PredictiveSearch.TYPES = {
PRODUCT: "product",
PAGE: "page",
ARTICLE: "article"
ARTICLE: "article",
COLLECTION: "collection"
};

PredictiveSearch.FIELDS = {
Expand Down Expand Up @@ -62,6 +74,7 @@ PredictiveSearch.prototype.query = function query(query) {
}

requestDebounced(
this.searchPath,
this.configParams,
query,
function(result) {
Expand Down