feat: add number of searches

Signed-off-by: Phil Berryman <phil@berryman.org.uk>
This commit is contained in:
Phil Berryman
2022-11-22 15:29:56 -08:00
parent db43cd1767
commit 29ebc43a0b
12 changed files with 43 additions and 11 deletions
+5
View File
@@ -0,0 +1,5 @@
---
'@backstage/plugin-search-react': minor
---
The `value` of a search analytics event is now set as the total number of search results (when available)
+8
View File
@@ -0,0 +1,8 @@
---
'@backstage/plugin-search-backend': minor
'@backstage/plugin-search-backend-module-elasticsearch': minor
'@backstage/plugin-search-backend-node': minor
'@backstage/plugin-search-backend-module-pg': minor
---
numberOfResults is now provided alongside the query result
+5
View File
@@ -0,0 +1,5 @@
---
'@backstage/plugin-search-common': minor
---
numberOfResults (total number of results for a given query) can now be provided by each search engine and consumed as part of the search results response
+7 -7
View File
@@ -52,13 +52,13 @@ learn how to contribute the integration yourself!
The following table summarizes events that, depending on the plugins you have
installed, may be captured.
| Action | Subject | Other Notes |
| ---------- | ------------------------------------------------------------------------------------------------------------------------------------------------------------------ | --------------------------------------------------------------------------------------------------- |
| `navigate` | The URL of the page that was navigated to | |
| `click` | The text of the link that was clicked on | The `to` attribute represents the URL clicked to |
| `create` | The `name` of the software being created; if no `name` property is requested by the given Software Template, then the string `new {templateName}` is used instead. | The context holds an `entityRef`, set to the template's ref (e.g. `template:default/template-name`) |
| `search` | The search term entered in any search bar component | The context holds `searchTypes`, representing `types` constraining the search |
| `discover` | The title of the search result that was clicked on | The `value` is the result rank. A `to` attribute is also provided |
| Action | Subject | Other Notes |
| ---------- | ------------------------------------------------------------------------------------------------------------------------------------------------------------------ | ---------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------- |
| `navigate` | The URL of the page that was navigated to | |
| `click` | The text of the link that was clicked on | The `to` attribute represents the URL clicked to |
| `create` | The `name` of the software being created; if no `name` property is requested by the given Software Template, then the string `new {templateName}` is used instead. | The context holds an `entityRef`, set to the template's ref (e.g. `template:default/template-name`) |
| `search` | The search term entered in any search bar component | - The context holds `searchTypes`, representing `types` constraining the search. The `value` represents the total number of search results for the query. This may not be visible if the permission framework is being used. |
| `discover` | The title of the search result that was clicked on | The `value` is the result rank. A `to` attribute is also provided |
If there is an event you'd like to see captured, please [open an
issue][add-event] describing the event you want to see and the questions it
@@ -335,6 +335,7 @@ export class ElasticSearchSearchEngine implements SearchEngine {
),
nextPageCursor,
previousPageCursor,
numberOfResults: result.body.hits.total.value,
};
} catch (error) {
if (error.meta?.body?.error?.type === 'index_not_found_exception') {
@@ -231,6 +231,7 @@ describe('PgSearchEngine', () => {
},
],
nextPageCursor: undefined,
numberOfResults: 1,
});
expect(database.transaction).toHaveBeenCalledTimes(1);
expect(database.query).toHaveBeenCalledWith(tx, {
@@ -288,6 +289,7 @@ describe('PgSearchEngine', () => {
},
})),
nextPageCursor: 'MQ==',
numberOfResults: 30,
});
expect(database.transaction).toHaveBeenCalledTimes(1);
expect(database.query).toHaveBeenCalledWith(tx, {
@@ -348,6 +350,7 @@ describe('PgSearchEngine', () => {
}))
.slice(25),
previousPageCursor: 'MA==',
numberOfResults: 5,
});
expect(database.transaction).toHaveBeenCalledTimes(1);
expect(database.query).toHaveBeenCalledWith(tx, {
@@ -194,6 +194,7 @@ export class PgSearchEngine implements SearchEngine {
const previousPageCursor = hasPreviousPage
? encodePageCursor({ page: page - 1 })
: undefined;
const numberOfResults = rows.length;
const results = pageRows.map(
({ type, document, highlight }, index): IndexableResult => ({
@@ -215,7 +216,7 @@ export class PgSearchEngine implements SearchEngine {
}),
);
return { results, nextPageCursor, previousPageCursor };
return { results, nextPageCursor, previousPageCursor, numberOfResults };
}
}
@@ -238,6 +238,7 @@ export class LunrSearchEngine implements SearchEngine {
}),
},
})),
numberOfResults: results.length,
nextPageCursor,
previousPageCursor,
};
@@ -205,6 +205,7 @@ export class AuthorizedSearchEngine implements SearchEngine {
(nextPageCursor || filteredResults.length > targetResults)
? encodePageCursor({ page: page + 1 })
: undefined,
numberOfResults: undefined,
};
}
+2
View File
@@ -73,6 +73,8 @@ export interface ResultSet<TDocument extends SearchDocument> {
// (undocumented)
nextPageCursor?: string;
// (undocumented)
numberOfResults?: number;
// (undocumented)
previousPageCursor?: string;
// (undocumented)
results: Result<TDocument>[];
+1
View File
@@ -87,6 +87,7 @@ export interface ResultSet<TDocument extends SearchDocument> {
results: Result<TDocument>[];
nextPageCursor?: string;
previousPageCursor?: string;
numberOfResults?: number;
}
/**
@@ -23,14 +23,18 @@ import { useSearch } from '../../context';
*/
export const TrackSearch = ({ children }: { children: React.ReactChild }) => {
const analytics = useAnalytics();
const { term } = useSearch();
const { term, result } = useSearch();
const numberOfResults = result.value?.numberOfResults ?? undefined;
useEffect(() => {
if (term) {
// Capture analytics search event with search term provided as value
analytics.captureEvent('search', term);
analytics.captureEvent('search', term, {
value: numberOfResults,
});
}
}, [analytics, term]);
}, [analytics, term, numberOfResults]);
return <>{children}</>;
};