@internetarchive/collection-browser 2.7.5-alpha.0 → 2.7.5-alpha.2
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
|
@@ -179,7 +179,7 @@ export class CollectionBrowserDataSource {
|
|
|
179
179
|
// We should only reset if either:
|
|
180
180
|
// (a) our state permits a valid search, or
|
|
181
181
|
// (b) we have a blank query that we want to show empty results for
|
|
182
|
-
const shouldShowEmptyQueryResults = this.host.
|
|
182
|
+
const shouldShowEmptyQueryResults = !this.host.baseQuery;
|
|
183
183
|
if (!(this.canPerformSearch || shouldShowEmptyQueryResults))
|
|
184
184
|
return;
|
|
185
185
|
if (this.activeOnHost)
|
|
@@ -356,19 +356,26 @@ export class CollectionBrowserDataSource {
|
|
|
356
356
|
*/
|
|
357
357
|
get canFetchFacets() {
|
|
358
358
|
var _a;
|
|
359
|
+
console.log('in canFetchFacets');
|
|
360
|
+
console.log(`facetLoadStrategy is ${this.host.facetLoadStrategy}`);
|
|
361
|
+
console.log(`profileElement is ${this.host.profileElement}`);
|
|
359
362
|
// Don't fetch facets if they are suppressed entirely or not required for the current profile page element
|
|
360
363
|
if (this.host.facetLoadStrategy === 'off')
|
|
361
364
|
return false;
|
|
362
365
|
if (FACETLESS_PAGE_ELEMENTS.includes(this.host.profileElement))
|
|
363
366
|
return false;
|
|
367
|
+
console.log(`facetsReadyToLoad: ${this.facetsReadyToLoad}`);
|
|
364
368
|
// If facets are to be lazy-loaded, don't fetch them if they are not going to be visible anyway
|
|
365
369
|
// (wait until they become visible instead)
|
|
366
370
|
if (this.host.facetLoadStrategy !== 'eager' && !this.facetsReadyToLoad)
|
|
367
371
|
return false;
|
|
368
372
|
// Don't fetch facets again if they are already fetched or pending
|
|
369
373
|
const facetsAlreadyFetched = Object.keys((_a = this.aggregations) !== null && _a !== void 0 ? _a : {}).length > 0;
|
|
374
|
+
console.log(`facetsAlreadyFetched: ${facetsAlreadyFetched}`);
|
|
375
|
+
console.log(`facetsLoading: ${this.facetsLoading}`);
|
|
370
376
|
if (this.facetsLoading || facetsAlreadyFetched)
|
|
371
377
|
return false;
|
|
378
|
+
console.log('canFetchFacets = true');
|
|
372
379
|
return true;
|
|
373
380
|
}
|
|
374
381
|
/**
|
|
@@ -411,6 +418,8 @@ export class CollectionBrowserDataSource {
|
|
|
411
418
|
*/
|
|
412
419
|
get canPerformSearch() {
|
|
413
420
|
var _a;
|
|
421
|
+
console.log('in canPerformSearch');
|
|
422
|
+
console.log(`has search service: ${!!this.host.searchService}`);
|
|
414
423
|
if (!this.host.searchService)
|
|
415
424
|
return false;
|
|
416
425
|
const trimmedQuery = (_a = this.host.baseQuery) === null || _a === void 0 ? void 0 : _a.trim();
|
|
@@ -419,6 +428,11 @@ export class CollectionBrowserDataSource {
|
|
|
419
428
|
const isProfileSearch = !!this.host.withinProfile;
|
|
420
429
|
const hasProfileElement = !!this.host.profileElement;
|
|
421
430
|
const isMetadataSearch = this.host.searchType === SearchType.METADATA;
|
|
431
|
+
console.log(`hasNonEmptyQuery: ${hasNonEmptyQuery}
|
|
432
|
+
isCollectionSearch: ${isCollectionSearch}
|
|
433
|
+
isProfileSearch: ${isProfileSearch}
|
|
434
|
+
isMetadataSearch: ${isMetadataSearch}
|
|
435
|
+
hasProfileElement: ${hasProfileElement}`);
|
|
422
436
|
// Metadata searches within a collection/profile are allowed to have no query.
|
|
423
437
|
// Otherwise, a non-empty query must be set.
|
|
424
438
|
return (hasNonEmptyQuery ||
|
|
@@ -430,6 +444,7 @@ export class CollectionBrowserDataSource {
|
|
|
430
444
|
* current query is loading
|
|
431
445
|
*/
|
|
432
446
|
setSearchResultsLoading(loading) {
|
|
447
|
+
console.log(`setSearchResultsLoading: ${loading} (active on host: ${this.activeOnHost})`);
|
|
433
448
|
this.searchResultsLoading = loading;
|
|
434
449
|
if (this.activeOnHost) {
|
|
435
450
|
this.host.setSearchResultsLoading(loading);
|
|
@@ -439,6 +454,7 @@ export class CollectionBrowserDataSource {
|
|
|
439
454
|
* Sets the state for whether the facets for a query is loading
|
|
440
455
|
*/
|
|
441
456
|
setFacetsLoading(loading) {
|
|
457
|
+
console.log(`setFacetsLoading: ${loading} (active on host: ${this.activeOnHost})`);
|
|
442
458
|
this.facetsLoading = loading;
|
|
443
459
|
if (this.activeOnHost) {
|
|
444
460
|
this.host.setFacetsLoading(loading);
|
|
@@ -704,10 +720,14 @@ export class CollectionBrowserDataSource {
|
|
|
704
720
|
*/
|
|
705
721
|
async fetchFacets() {
|
|
706
722
|
var _a, _b, _c, _d, _e, _f, _g, _h, _j;
|
|
723
|
+
console.log('in fetchFacets');
|
|
724
|
+
console.log(`can perform search? ${this.canPerformSearch}`);
|
|
707
725
|
const trimmedQuery = (_a = this.host.baseQuery) === null || _a === void 0 ? void 0 : _a.trim();
|
|
708
726
|
if (!this.canPerformSearch)
|
|
709
727
|
return;
|
|
710
728
|
const { facetFetchQueryKey } = this;
|
|
729
|
+
console.log(`facetFetchQueryKey: ${facetFetchQueryKey}`);
|
|
730
|
+
console.log(`fetch in progress? ${this.fetchesInProgress.has(facetFetchQueryKey)}`);
|
|
711
731
|
if (this.fetchesInProgress.has(facetFetchQueryKey))
|
|
712
732
|
return;
|
|
713
733
|
this.fetchesInProgress.add(facetFetchQueryKey);
|
|
@@ -724,6 +744,7 @@ export class CollectionBrowserDataSource {
|
|
|
724
744
|
// The default aggregations for the search_results page type should be what we need here.
|
|
725
745
|
};
|
|
726
746
|
params.uid = await this.requestUID({ ...params, sort: sortParams }, 'aggregations');
|
|
747
|
+
console.log('sending facet request');
|
|
727
748
|
const searchResponse = await ((_b = this.host.searchService) === null || _b === void 0 ? void 0 : _b.search(params, this.host.searchType));
|
|
728
749
|
const success = searchResponse === null || searchResponse === void 0 ? void 0 : searchResponse.success;
|
|
729
750
|
// This is checking to see if the query has changed since the data was fetched.
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"collection-browser-data-source.js","sourceRoot":"","sources":["../../../src/data-source/collection-browser-data-source.ts"],"names":[],"mappings":"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","sourcesContent":["import type { ReactiveControllerHost } from 'lit';\nimport {\n AccountExtraInfo,\n Aggregation,\n Bucket,\n CollectionExtraInfo,\n FilterConstraint,\n FilterMap,\n FilterMapBuilder,\n PageElementMap,\n SearchParams,\n SearchResponseSessionContext,\n SearchResult,\n SearchType,\n} from '@internetarchive/search-service';\nimport {\n prefixFilterAggregationKeys,\n type FacetBucket,\n type PrefixFilterType,\n TileModel,\n PrefixFilterCounts,\n RequestKind,\n SortField,\n SORT_OPTIONS,\n} from '../models';\nimport { FACETLESS_PAGE_ELEMENTS, type PageSpecifierParams } from './models';\nimport type { CollectionBrowserDataSourceInterface } from './collection-browser-data-source-interface';\nimport type { CollectionBrowserSearchInterface } from './collection-browser-query-state';\nimport { sha1 } from '../utils/sha1';\nimport { log } from '../utils/log';\n\nexport class CollectionBrowserDataSource\n implements CollectionBrowserDataSourceInterface\n{\n /**\n * All pages of tile models that have been fetched so far, indexed by their page\n * number (with the first being page 1).\n */\n private pages: Record<string, TileModel[]> = {};\n\n /**\n * Tile offset to apply when looking up tiles in the pages, in order to maintain\n * page alignment after tiles are removed.\n */\n private offset = 0;\n\n /**\n * Total number of tile models stored in this data source's pages\n */\n private numTileModels = 0;\n\n /**\n * A set of fetch IDs that are valid for the current query state\n */\n private fetchesInProgress = new Set<string>();\n\n /**\n * A record of the query key used for the last search.\n * If this changes, we need to load new results.\n */\n private previousQueryKey: string = '';\n\n /**\n * Whether the initial page of search results for the current query state\n * is presently being fetched.\n */\n private searchResultsLoading = false;\n\n /**\n * Whether the facets (aggregations) for the current query state are\n * presently being fetched.\n */\n private facetsLoading = false;\n\n /**\n * Whether the facets are actually visible -- if not, then we can delay any facet\n * fetches until they become visible.\n */\n private facetsReadyToLoad = false;\n\n /**\n * Whether further query changes should be ignored and not trigger fetches\n */\n private suppressFetches = false;\n\n /**\n * @inheritdoc\n */\n totalResults = 0;\n\n /**\n * @inheritdoc\n */\n endOfDataReached = false;\n\n /**\n * @inheritdoc\n */\n queryInitialized = false;\n\n /**\n * @inheritdoc\n */\n aggregations?: Record<string, Aggregation>;\n\n /**\n * @inheritdoc\n */\n yearHistogramAggregation?: Aggregation;\n\n /**\n * @inheritdoc\n */\n collectionTitles = new Map<string, string>();\n\n /**\n * @inheritdoc\n */\n collectionExtraInfo?: CollectionExtraInfo;\n\n /**\n * @inheritdoc\n */\n accountExtraInfo?: AccountExtraInfo;\n\n /**\n * @inheritdoc\n */\n sessionContext?: SearchResponseSessionContext;\n\n /**\n * @inheritdoc\n */\n pageElements?: PageElementMap;\n\n /**\n * @inheritdoc\n */\n parentCollections?: string[] = [];\n\n /**\n * @inheritdoc\n */\n prefixFilterCountMap: Partial<Record<PrefixFilterType, PrefixFilterCounts>> =\n {};\n\n /**\n * @inheritdoc\n */\n queryErrorMessage?: string;\n\n /**\n * Internal property to store the `resolve` function for the most recent\n * `initialSearchComplete` promise, allowing us to resolve it at the appropriate time.\n */\n private _initialSearchCompleteResolver!: (val: boolean) => void;\n\n /**\n * Internal property to store the private value backing the `initialSearchComplete` getter.\n */\n private _initialSearchCompletePromise: Promise<boolean> = new Promise(res => {\n this._initialSearchCompleteResolver = res;\n });\n\n /**\n * @inheritdoc\n */\n get initialSearchComplete(): Promise<boolean> {\n return this._initialSearchCompletePromise;\n }\n\n // eslint-disable-next-line no-useless-constructor\n constructor(\n /** The host element to which this controller should attach listeners */\n private readonly host: ReactiveControllerHost &\n CollectionBrowserSearchInterface,\n /** Default size of result pages */\n private pageSize: number = 50\n ) {\n // Just setting some property values\n }\n\n hostConnected(): void {\n this.setSearchResultsLoading(this.searchResultsLoading);\n this.setFacetsLoading(this.facetsLoading);\n }\n\n hostUpdate(): void {\n // This reactive controller hook is run whenever the host component (collection-browser) performs an update.\n // We check whether the host's state has changed in a way which should trigger a reset & new results fetch.\n\n // Only the currently-installed data source should react to the update\n if (!this.activeOnHost) return;\n\n // Copy loading states onto the host\n this.setSearchResultsLoading(this.searchResultsLoading);\n this.setFacetsLoading(this.facetsLoading);\n\n // Can't perform searches without a search service\n if (!this.host.searchService) return;\n\n // We should only reset if part of the full query state has changed\n const queryKeyChanged = this.pageFetchQueryKey !== this.previousQueryKey;\n if (!queryKeyChanged) return;\n\n // We should only reset if either:\n // (a) our state permits a valid search, or\n // (b) we have a blank query that we want to show empty results for\n const shouldShowEmptyQueryResults =\n this.host.clearResultsOnEmptyQuery && this.host.baseQuery === '';\n if (!(this.canPerformSearch || shouldShowEmptyQueryResults)) return;\n\n if (this.activeOnHost) this.host.emitQueryStateChanged();\n this.handleQueryChange();\n }\n\n /**\n * Returns whether this data source is the one currently installed on the host component.\n */\n private get activeOnHost(): boolean {\n return this.host.dataSource === this;\n }\n\n /**\n * @inheritdoc\n */\n get size(): number {\n return this.numTileModels;\n }\n\n /**\n * @inheritdoc\n */\n reset(): void {\n log('Resetting CB data source');\n this.pages = {};\n this.aggregations = {};\n this.yearHistogramAggregation = undefined;\n this.pageElements = undefined;\n this.parentCollections = [];\n this.queryErrorMessage = undefined;\n\n this.offset = 0;\n this.numTileModels = 0;\n this.endOfDataReached = false;\n this.queryInitialized = false;\n this.facetsLoading = false;\n\n // Invalidate any fetches in progress\n this.fetchesInProgress.clear();\n\n this.setTotalResultCount(0);\n this.requestHostUpdate();\n }\n\n /**\n * @inheritdoc\n */\n addPage(pageNum: number, pageTiles: TileModel[]): void {\n this.pages[pageNum] = pageTiles;\n this.numTileModels += pageTiles.length;\n this.requestHostUpdate();\n }\n\n /**\n * @inheritdoc\n */\n addMultiplePages(firstPageNum: number, tiles: TileModel[]): void {\n const numPages = Math.ceil(tiles.length / this.pageSize);\n for (let i = 0; i < numPages; i += 1) {\n const pageStartIndex = this.pageSize * i;\n this.addPage(\n firstPageNum + i,\n tiles.slice(pageStartIndex, pageStartIndex + this.pageSize)\n );\n }\n\n const visiblePages = this.host.currentVisiblePageNumbers;\n const needsReload = visiblePages.some(\n page => page >= firstPageNum && page <= firstPageNum + numPages\n );\n if (needsReload) {\n this.refreshVisibleResults();\n }\n }\n\n /**\n * @inheritdoc\n */\n getPage(pageNum: number): TileModel[] {\n return this.pages[pageNum];\n }\n\n /**\n * @inheritdoc\n */\n getAllPages(): Record<string, TileModel[]> {\n return this.pages;\n }\n\n /**\n * @inheritdoc\n */\n hasPage(pageNum: number): boolean {\n return !!this.pages[pageNum];\n }\n\n /**\n * @inheritdoc\n */\n getTileModelAt(index: number): TileModel | undefined {\n const offsetIndex = index + this.offset;\n const expectedPageNum = Math.floor(offsetIndex / this.pageSize) + 1;\n const expectedIndexOnPage = offsetIndex % this.pageSize;\n\n let page = 1;\n let tilesSeen = 0;\n while (tilesSeen <= offsetIndex) {\n if (!this.pages[page]) {\n // If we encounter a missing page, either we're past all the results or the page data is sparse.\n // So just return the tile at the expected position if it exists.\n return this.pages[expectedPageNum]?.[expectedIndexOnPage];\n }\n\n if (tilesSeen + this.pages[page].length > offsetIndex) {\n return this.pages[page][offsetIndex - tilesSeen];\n }\n\n tilesSeen += this.pages[page].length;\n page += 1;\n }\n\n return this.pages[expectedPageNum]?.[expectedIndexOnPage];\n }\n\n /**\n * @inheritdoc\n */\n indexOf(tile: TileModel): number {\n return Object.values(this.pages).flat().indexOf(tile) - this.offset;\n }\n\n /**\n * @inheritdoc\n */\n getPageSize(): number {\n return this.pageSize;\n }\n\n /**\n * @inheritdoc\n */\n setPageSize(pageSize: number): void {\n this.reset();\n this.pageSize = pageSize;\n }\n\n /**\n * @inheritdoc\n */\n setTotalResultCount(count: number): void {\n this.totalResults = count;\n if (this.activeOnHost) {\n this.host.setTotalResultCount(count);\n }\n }\n\n /**\n * @inheritdoc\n */\n setFetchesSuppressed(suppressed: boolean): void {\n this.suppressFetches = suppressed;\n }\n\n /**\n * @inheritdoc\n */\n async handleQueryChange(): Promise<void> {\n // Don't react to the change if fetches are suppressed for this data source\n if (this.suppressFetches) return;\n\n this.reset();\n\n // Reset the `initialSearchComplete` promise with a new value for the imminent search\n this._initialSearchCompletePromise = new Promise(res => {\n this._initialSearchCompleteResolver = res;\n });\n\n // Fire the initial page & facet requests\n this.queryInitialized = true;\n await Promise.all([\n this.doInitialPageFetch(),\n this.canFetchFacets ? this.fetchFacets() : null,\n ]);\n\n // Resolve the `initialSearchComplete` promise for this search\n this._initialSearchCompleteResolver(true);\n }\n\n /**\n * @inheritdoc\n */\n async handleFacetReadinessChange(ready: boolean): Promise<void> {\n const facetsBecameReady = !this.facetsReadyToLoad && ready;\n this.facetsReadyToLoad = ready;\n\n const needsFetch = facetsBecameReady && this.canFetchFacets;\n if (needsFetch) {\n this.fetchFacets();\n }\n }\n\n /**\n * Whether the data source & its host are in a state where a facet request should be fired.\n * (i.e., they aren't suppressed or already loading, etc.)\n */\n private get canFetchFacets(): boolean {\n // Don't fetch facets if they are suppressed entirely or not required for the current profile page element\n if (this.host.facetLoadStrategy === 'off') return false;\n if (FACETLESS_PAGE_ELEMENTS.includes(this.host.profileElement!))\n return false;\n\n // If facets are to be lazy-loaded, don't fetch them if they are not going to be visible anyway\n // (wait until they become visible instead)\n if (this.host.facetLoadStrategy !== 'eager' && !this.facetsReadyToLoad)\n return false;\n\n // Don't fetch facets again if they are already fetched or pending\n const facetsAlreadyFetched =\n Object.keys(this.aggregations ?? {}).length > 0;\n if (this.facetsLoading || facetsAlreadyFetched) return false;\n\n return true;\n }\n\n /**\n * @inheritdoc\n */\n map(\n callback: (model: TileModel, index: number, array: TileModel[]) => TileModel\n ): void {\n this.pages = Object.fromEntries(\n Object.entries(this.pages).map(([page, tileModels]) => [\n page,\n tileModels.map((model, index, array) =>\n model ? callback(model, index, array) : model\n ),\n ])\n );\n this.requestHostUpdate();\n this.refreshVisibleResults();\n }\n\n /**\n * @inheritdoc\n */\n checkAllTiles = (): void => {\n this.map(model => {\n const cloned = model.clone();\n cloned.checked = true;\n return cloned;\n });\n };\n\n /**\n * @inheritdoc\n */\n uncheckAllTiles = (): void => {\n this.map(model => {\n const cloned = model.clone();\n cloned.checked = false;\n return cloned;\n });\n };\n\n /**\n * @inheritdoc\n */\n removeCheckedTiles = (): void => {\n // To make sure our data source remains page-aligned, we will offset our data source by\n // the number of removed tiles, so that we can just add the offset when the infinite\n // scroller queries for cell contents.\n // This only matters while we're still viewing the same set of results. If the user changes\n // their query/filters/sort, then the data source is overwritten and the offset cleared.\n const { checkedTileModels, uncheckedTileModels } = this;\n const numChecked = checkedTileModels.length;\n if (numChecked === 0) return;\n this.offset += numChecked;\n const newPages: typeof this.pages = {};\n\n // Which page the remaining tile models start on, post-offset\n let offsetPageNumber = Math.floor(this.offset / this.pageSize) + 1;\n let indexOnPage = this.offset % this.pageSize;\n\n // Fill the pages up to that point with empty models\n for (let page = 1; page <= offsetPageNumber; page += 1) {\n const remainingHidden = this.offset - this.pageSize * (page - 1);\n const offsetCellsOnPage = Math.min(this.pageSize, remainingHidden);\n newPages[page] = Array(offsetCellsOnPage).fill(undefined);\n }\n\n // Shift all the remaining tiles into their new positions in the data source\n for (const model of uncheckedTileModels) {\n if (!newPages[offsetPageNumber]) newPages[offsetPageNumber] = [];\n newPages[offsetPageNumber].push(model);\n indexOnPage += 1;\n if (indexOnPage >= this.pageSize) {\n offsetPageNumber += 1;\n indexOnPage = 0;\n }\n }\n\n // Swap in the new pages\n this.pages = newPages;\n this.numTileModels -= numChecked;\n this.totalResults -= numChecked;\n this.host.setTileCount(this.size);\n this.host.setTotalResultCount(this.totalResults);\n this.requestHostUpdate();\n this.refreshVisibleResults();\n };\n\n /**\n * @inheritdoc\n */\n get checkedTileModels(): TileModel[] {\n return this.getFilteredTileModels(model => model.checked);\n }\n\n /**\n * @inheritdoc\n */\n get uncheckedTileModels(): TileModel[] {\n return this.getFilteredTileModels(model => !model.checked);\n }\n\n /**\n * Returns a flattened, filtered array of all the tile models in the data source\n * for which the given predicate returns a truthy value.\n *\n * @param predicate A callback function to apply on each tile model, as with Array.filter\n * @returns A filtered array of tile models satisfying the predicate\n */\n private getFilteredTileModels(\n predicate: (model: TileModel, index: number, array: TileModel[]) => unknown\n ): TileModel[] {\n return Object.values(this.pages)\n .flat()\n .filter((model, index, array) =>\n model ? predicate(model, index, array) : false\n );\n }\n\n /**\n * @inheritdoc\n */\n get canPerformSearch(): boolean {\n if (!this.host.searchService) return false;\n\n const trimmedQuery = this.host.baseQuery?.trim();\n const hasNonEmptyQuery = !!trimmedQuery;\n const isCollectionSearch = !!this.host.withinCollection;\n const isProfileSearch = !!this.host.withinProfile;\n const hasProfileElement = !!this.host.profileElement;\n const isMetadataSearch = this.host.searchType === SearchType.METADATA;\n\n // Metadata searches within a collection/profile are allowed to have no query.\n // Otherwise, a non-empty query must be set.\n return (\n hasNonEmptyQuery ||\n (isCollectionSearch && isMetadataSearch) ||\n (isProfileSearch && hasProfileElement && isMetadataSearch)\n );\n }\n\n /**\n * Sets the state for whether the initial set of search results for the\n * current query is loading\n */\n private setSearchResultsLoading(loading: boolean): void {\n this.searchResultsLoading = loading;\n if (this.activeOnHost) {\n this.host.setSearchResultsLoading(loading);\n }\n }\n\n /**\n * Sets the state for whether the facets for a query is loading\n */\n private setFacetsLoading(loading: boolean): void {\n this.facetsLoading = loading;\n if (this.activeOnHost) {\n this.host.setFacetsLoading(loading);\n }\n }\n\n /**\n * Requests that the host perform an update, provided this data\n * source is actively installed on it.\n */\n private requestHostUpdate(): void {\n if (this.activeOnHost) {\n this.host.requestUpdate();\n }\n }\n\n /**\n * Requests that the host refresh its visible tiles, provided this\n * data source is actively installed on it.\n */\n private refreshVisibleResults(): void {\n if (this.activeOnHost) {\n this.host.refreshVisibleResults();\n }\n }\n\n /**\n * The query key is a string that uniquely identifies the current search.\n * It consists of:\n * - The current base query\n * - The current collection/profile target & page element\n * - The current search type\n * - Any currently-applied facets\n * - Any currently-applied date range\n * - Any currently-applied prefix filters\n * - The current sort options\n *\n * This lets us internally keep track of queries so we don't persist data that's\n * no longer relevant. Not meant to be human-readable.\n */\n get pageFetchQueryKey(): string {\n const profileKey = `pf;${this.host.withinProfile}--pe;${this.host.profileElement}`;\n const pageTarget = this.host.withinCollection ?? profileKey;\n const sortField = this.host.selectedSort ?? 'none';\n const sortDirection = this.host.sortDirection ?? 'none';\n return `fq:${this.fullQuery}-pt:${pageTarget}-st:${this.host.searchType}-sf:${sortField}-sd:${sortDirection}`;\n }\n\n /**\n * Similar to `pageFetchQueryKey` above, but excludes sort fields since they\n * are not relevant in determining aggregation queries.\n */\n get facetFetchQueryKey(): string {\n const profileKey = `pf;${this.host.withinProfile}--pe;${this.host.profileElement}`;\n const pageTarget = this.host.withinCollection ?? profileKey;\n return `fq:${this.fullQuery}-pt:${pageTarget}-st:${this.host.searchType}`;\n }\n\n /**\n * Constructs a search service FilterMap object from the combination of\n * all the currently-applied filters. This includes any facets, letter\n * filters, and date range.\n */\n get filterMap(): FilterMap {\n const builder = new FilterMapBuilder();\n\n // Add the date range, if applicable\n if (this.host.minSelectedDate) {\n builder.addFilter(\n 'year',\n this.host.minSelectedDate,\n FilterConstraint.GREATER_OR_EQUAL\n );\n }\n if (this.host.maxSelectedDate) {\n builder.addFilter(\n 'year',\n this.host.maxSelectedDate,\n FilterConstraint.LESS_OR_EQUAL\n );\n }\n\n // Add any selected facets\n if (this.host.selectedFacets) {\n for (const [facetName, facetValues] of Object.entries(\n this.host.selectedFacets\n )) {\n const { name, values } = this.prepareFacetForFetch(\n facetName,\n facetValues\n );\n for (const [value, bucket] of Object.entries(values)) {\n let constraint;\n if (bucket.state === 'selected') {\n constraint = FilterConstraint.INCLUDE;\n } else if (bucket.state === 'hidden') {\n constraint = FilterConstraint.EXCLUDE;\n }\n\n if (constraint) {\n builder.addFilter(name, value, constraint);\n }\n }\n }\n }\n\n // Add any letter filters\n if (this.host.selectedTitleFilter) {\n builder.addFilter(\n 'firstTitle',\n this.host.selectedTitleFilter,\n FilterConstraint.INCLUDE\n );\n }\n if (this.host.selectedCreatorFilter) {\n builder.addFilter(\n 'firstCreator',\n this.host.selectedCreatorFilter,\n FilterConstraint.INCLUDE\n );\n }\n\n const filterMap = builder.build();\n return filterMap;\n }\n\n /**\n * Produces a compact unique ID for a search request that can help with debugging\n * on the backend by making related requests easier to trace through different services.\n * (e.g., tying the hits/aggregations requests for the same page back to a single hash).\n *\n * @param params The search service parameters for the request\n * @param kind The kind of request (hits-only, aggregations-only, or both)\n * @returns A Promise resolving to the uid to apply to the request\n */\n async requestUID(params: SearchParams, kind: RequestKind): Promise<string> {\n const paramsToHash = JSON.stringify({\n pageType: params.pageType,\n pageTarget: params.pageTarget,\n query: params.query,\n fields: params.fields,\n filters: params.filters,\n sort: params.sort,\n searchType: this.host.searchType,\n });\n\n const fullQueryHash = (await sha1(paramsToHash)).slice(0, 20); // First 80 bits of SHA-1 are plenty for this\n const sessionId = (await this.host.getSessionId()).slice(0, 20); // Likewise\n const page = params.page ?? 0;\n const kindPrefix = kind.charAt(0); // f = full, h = hits, a = aggregations\n const currentTime = Date.now();\n\n return `R:${fullQueryHash}-S:${sessionId}-P:${page}-K:${kindPrefix}-T:${currentTime}`;\n }\n\n /**\n * @inheritdoc\n */\n get pageSpecifierParams(): PageSpecifierParams | null {\n if (this.host.withinCollection) {\n return {\n pageType: 'collection_details',\n pageTarget: this.host.withinCollection,\n };\n }\n if (this.host.withinProfile) {\n return {\n pageType: 'account_details',\n pageTarget: this.host.withinProfile,\n pageElements: this.host.profileElement\n ? [this.host.profileElement]\n : [],\n };\n }\n return null;\n }\n\n /**\n * The full query, including year facets and date range clauses\n */\n private get fullQuery(): string | undefined {\n let fullQuery = this.host.baseQuery?.trim() ?? '';\n\n const { facetQuery, dateRangeQueryClause, sortFilterQueries } = this;\n\n if (facetQuery) {\n fullQuery += ` AND ${facetQuery}`;\n }\n if (dateRangeQueryClause) {\n fullQuery += ` AND ${dateRangeQueryClause}`;\n }\n if (sortFilterQueries) {\n fullQuery += ` AND ${sortFilterQueries}`;\n }\n return fullQuery.trim();\n }\n\n /**\n * Generates a query string representing the current set of applied facets\n *\n * Example: `mediatype:(\"collection\" OR \"audio\" OR -\"etree\") AND year:(\"2000\" OR \"2001\")`\n */\n private get facetQuery(): string | undefined {\n if (!this.host.selectedFacets) return undefined;\n const facetClauses = [];\n for (const [facetName, facetValues] of Object.entries(\n this.host.selectedFacets\n )) {\n facetClauses.push(this.buildFacetClause(facetName, facetValues));\n }\n return this.joinFacetClauses(facetClauses)?.trim();\n }\n\n private get dateRangeQueryClause(): string | undefined {\n if (!this.host.minSelectedDate || !this.host.maxSelectedDate) {\n return undefined;\n }\n\n return `year:[${this.host.minSelectedDate} TO ${this.host.maxSelectedDate}]`;\n }\n\n private get sortFilterQueries(): string {\n const queries = [this.titleQuery, this.creatorQuery];\n return queries.filter(q => q).join(' AND ');\n }\n\n /**\n * Returns a query clause identifying the currently selected title filter,\n * e.g., `firstTitle:X`.\n */\n private get titleQuery(): string | undefined {\n return this.host.selectedTitleFilter\n ? `firstTitle:${this.host.selectedTitleFilter}`\n : undefined;\n }\n\n /**\n * Returns a query clause identifying the currently selected creator filter,\n * e.g., `firstCreator:X`.\n */\n private get creatorQuery(): string | undefined {\n return this.host.selectedCreatorFilter\n ? `firstCreator:${this.host.selectedCreatorFilter}`\n : undefined;\n }\n\n /**\n * Builds an OR-joined facet clause for the given facet name and values.\n *\n * E.g., for name `subject` and values\n * `{ foo: { state: 'selected' }, bar: { state: 'hidden' } }`\n * this will produce the clause\n * `subject:(\"foo\" OR -\"bar\")`.\n *\n * @param facetName The facet type (e.g., 'collection')\n * @param facetValues The facet buckets, mapped by their keys\n */\n private buildFacetClause(\n facetName: string,\n facetValues: Record<string, FacetBucket>\n ): string {\n const { name: facetQueryName, values } = this.prepareFacetForFetch(\n facetName,\n facetValues\n );\n const facetEntries = Object.entries(values);\n if (facetEntries.length === 0) return '';\n\n const facetValuesArray: string[] = [];\n for (const [key, facetData] of facetEntries) {\n const plusMinusPrefix = facetData.state === 'hidden' ? '-' : '';\n facetValuesArray.push(`${plusMinusPrefix}\"${key}\"`);\n }\n\n const valueQuery = facetValuesArray.join(` OR `);\n return `${facetQueryName}:(${valueQuery})`;\n }\n\n /**\n * Handles some special pre-request normalization steps for certain facet types\n * that require them.\n *\n * @param facetName The name of the facet type (e.g., 'language')\n * @param facetValues An array of values for that facet type\n */\n private prepareFacetForFetch(\n facetName: string,\n facetValues: Record<string, FacetBucket>\n ): { name: string; values: Record<string, FacetBucket> } {\n // eslint-disable-next-line prefer-const\n let [normalizedName, normalizedValues] = [facetName, facetValues];\n\n // The full \"search engine\" name of the lending field is \"lending___status\"\n if (facetName === 'lending') {\n normalizedName = 'lending___status';\n }\n\n return {\n name: normalizedName,\n values: normalizedValues,\n };\n }\n\n /**\n * Takes an array of facet clauses, and combines them into a\n * full AND-joined facet query string. Empty clauses are ignored.\n */\n private joinFacetClauses(facetClauses: string[]): string | undefined {\n const nonEmptyFacetClauses = facetClauses.filter(\n clause => clause.length > 0\n );\n return nonEmptyFacetClauses.length > 0\n ? `(${nonEmptyFacetClauses.join(' AND ')})`\n : undefined;\n }\n\n /**\n * Fires a backend request to fetch a set of aggregations (representing UI facets) for\n * the current search state.\n */\n private async fetchFacets(): Promise<void> {\n const trimmedQuery = this.host.baseQuery?.trim();\n if (!this.canPerformSearch) return;\n\n const { facetFetchQueryKey } = this;\n if (this.fetchesInProgress.has(facetFetchQueryKey)) return;\n this.fetchesInProgress.add(facetFetchQueryKey);\n\n this.setFacetsLoading(true);\n\n const sortParams = this.host.sortParam ? [this.host.sortParam] : [];\n const params: SearchParams = {\n ...this.pageSpecifierParams,\n query: trimmedQuery || '',\n rows: 0,\n filters: this.filterMap,\n // Fetch a few extra buckets beyond the 6 we show, in case some get suppressed\n aggregationsSize: 10,\n // Note: we don't need an aggregations param to fetch the default aggregations from the PPS.\n // The default aggregations for the search_results page type should be what we need here.\n };\n params.uid = await this.requestUID(\n { ...params, sort: sortParams },\n 'aggregations'\n );\n\n const searchResponse = await this.host.searchService?.search(\n params,\n this.host.searchType\n );\n const success = searchResponse?.success;\n\n // This is checking to see if the query has changed since the data was fetched.\n // If so, we just want to discard this set of aggregations because they are\n // likely no longer valid for the newer query.\n const queryChangedSinceFetch =\n !this.fetchesInProgress.has(facetFetchQueryKey);\n this.fetchesInProgress.delete(facetFetchQueryKey);\n if (queryChangedSinceFetch) return;\n\n if (!success) {\n const errorMsg = searchResponse?.error?.message;\n const detailMsg = searchResponse?.error?.details?.message;\n\n if (!errorMsg && !detailMsg) {\n // @ts-ignore: Property 'Sentry' does not exist on type 'Window & typeof globalThis'\n window?.Sentry?.captureMessage?.(\n 'Missing or malformed facet response from backend',\n 'error'\n );\n }\n\n this.setFacetsLoading(false);\n return;\n }\n\n const { aggregations, collectionTitles } = success.response;\n this.aggregations = aggregations;\n\n if (collectionTitles) {\n for (const [id, title] of Object.entries(collectionTitles)) {\n this.collectionTitles.set(id, title);\n }\n }\n\n this.yearHistogramAggregation =\n success?.response?.aggregations?.year_histogram;\n\n this.setFacetsLoading(false);\n this.requestHostUpdate();\n }\n\n /**\n * Performs the initial page fetch(es) for the current search state.\n */\n private async doInitialPageFetch(): Promise<void> {\n this.setSearchResultsLoading(true);\n // Try to batch 2 initial page requests when possible\n await this.fetchPage(this.host.initialPageNumber, 2);\n }\n\n /**\n * Fetches one or more pages of results and updates the data source.\n *\n * @param pageNumber The page number to fetch\n * @param numInitialPages If this is an initial page fetch (`pageNumber = 1`),\n * specifies how many pages to batch together in one request. Ignored\n * if `pageNumber != 1`, defaulting to a single page.\n */\n async fetchPage(pageNumber: number, numInitialPages = 1): Promise<void> {\n const trimmedQuery = this.host.baseQuery?.trim();\n if (!this.canPerformSearch) return;\n\n // if we already have data, don't fetch again\n if (this.hasPage(pageNumber)) return;\n\n if (this.endOfDataReached) return;\n\n // Batch multiple initial page requests together if needed (e.g., can request\n // pages 1 and 2 together in a single request).\n let numPages = pageNumber === 1 ? numInitialPages : 1;\n const numRows = this.pageSize * numPages;\n\n // if a fetch is already in progress for this query and page, don't fetch again\n const { pageFetchQueryKey } = this;\n const currentPageKey = `${pageFetchQueryKey}-p:${pageNumber}`;\n if (this.fetchesInProgress.has(currentPageKey)) return;\n\n for (let i = 0; i < numPages; i += 1) {\n this.fetchesInProgress.add(`${pageFetchQueryKey}-p:${pageNumber + i}`);\n }\n this.previousQueryKey = pageFetchQueryKey;\n\n let sortParams = this.host.sortParam ? [this.host.sortParam] : [];\n // TODO eventually the PPS should handle these defaults natively\n const isDefaultProfileSort =\n this.host.withinProfile && this.host.selectedSort === SortField.default;\n if (isDefaultProfileSort && this.host.defaultSortField) {\n const sortOption = SORT_OPTIONS[this.host.defaultSortField];\n if (sortOption.searchServiceKey) {\n sortParams = [\n {\n field: sortOption.searchServiceKey,\n direction: 'desc',\n },\n ];\n }\n }\n\n const params: SearchParams = {\n ...this.pageSpecifierParams,\n query: trimmedQuery || '',\n page: pageNumber,\n rows: numRows,\n sort: sortParams,\n filters: this.filterMap,\n aggregations: { omit: true },\n };\n params.uid = await this.requestUID(params, 'hits');\n\n // log('=== FIRING PAGE REQUEST ===', params);\n const searchResponse = await this.host.searchService?.search(\n params,\n this.host.searchType\n );\n // log('=== RECEIVED PAGE RESPONSE IN CB ===', searchResponse);\n const success = searchResponse?.success;\n\n // This is checking to see if the fetch has been invalidated since it was fired off.\n // If so, we just want to discard the response since it is for an obsolete query state.\n if (!this.fetchesInProgress.has(currentPageKey)) return;\n for (let i = 0; i < numPages; i += 1) {\n this.fetchesInProgress.delete(`${pageFetchQueryKey}-p:${pageNumber + i}`);\n }\n\n if (!success) {\n const errorMsg = searchResponse?.error?.message;\n const detailMsg = searchResponse?.error?.details?.message;\n\n this.queryErrorMessage = `${errorMsg ?? ''}${\n detailMsg ? `; ${detailMsg}` : ''\n }`;\n\n if (!this.queryErrorMessage) {\n this.queryErrorMessage = 'Missing or malformed response from backend';\n // @ts-ignore: Property 'Sentry' does not exist on type 'Window & typeof globalThis'\n window?.Sentry?.captureMessage?.(this.queryErrorMessage, 'error');\n }\n\n this.setSearchResultsLoading(false);\n this.requestHostUpdate();\n return;\n }\n\n this.setTotalResultCount(success.response.totalResults - this.offset);\n if (this.activeOnHost && this.totalResults === 0) {\n // display event to offshoot when result count is zero.\n this.host.emitEmptyResults();\n }\n\n this.sessionContext = success.sessionContext;\n if (this.host.withinCollection) {\n this.collectionExtraInfo = success.response.collectionExtraInfo;\n\n // For collections, we want the UI to respect the default sort option\n // which can be specified in metadata, or otherwise assumed to be week:desc\n if (this.activeOnHost) {\n this.host.applyDefaultCollectionSort(this.collectionExtraInfo);\n }\n\n if (this.collectionExtraInfo) {\n this.parentCollections = [].concat(\n this.collectionExtraInfo.public_metadata?.collection ?? []\n );\n }\n } else if (this.host.withinProfile) {\n this.accountExtraInfo = success.response.accountExtraInfo;\n this.pageElements = success.response.pageElements;\n }\n\n const { results, collectionTitles } = success.response;\n if (results && results.length > 0) {\n // Load any collection titles present on the response into the cache,\n // or queue up preload fetches for them if none were present.\n if (collectionTitles) {\n for (const [id, title] of Object.entries(collectionTitles)) {\n this.collectionTitles.set(id, title);\n }\n }\n\n // Update the data source for each returned page.\n // For loans and web archives, we must account for receiving more pages than we asked for.\n const isUnpagedElement = ['lending', 'web_archives'].includes(\n this.host.profileElement!\n );\n if (isUnpagedElement) {\n numPages = Math.ceil(results.length / this.pageSize);\n this.endOfDataReached = true;\n if (this.activeOnHost) this.host.setTileCount(this.totalResults);\n }\n\n for (let i = 0; i < numPages; i += 1) {\n const pageStartIndex = this.pageSize * i;\n this.addFetchedResultsToDataSource(\n pageNumber + i,\n results.slice(pageStartIndex, pageStartIndex + this.pageSize),\n !isUnpagedElement || i === numPages - 1\n );\n }\n }\n\n // When we reach the end of the data, we can set the infinite scroller's\n // item count to the real total number of results (rather than the\n // temporary estimates based on pages rendered so far).\n if (this.size >= this.totalResults || results.length === 0) {\n this.endOfDataReached = true;\n if (this.activeOnHost) this.host.setTileCount(this.size);\n }\n\n this.setSearchResultsLoading(false);\n this.requestHostUpdate();\n }\n\n /**\n * Update the datasource from the fetch response\n *\n * @param pageNumber\n * @param results\n */\n private addFetchedResultsToDataSource(\n pageNumber: number,\n results: SearchResult[],\n needsReload = true\n ): void {\n const tiles: TileModel[] = [];\n results?.forEach(result => {\n if (!result.identifier) return;\n tiles.push(new TileModel(result));\n });\n\n this.addPage(pageNumber, tiles);\n\n if (needsReload) {\n this.refreshVisibleResults();\n }\n }\n\n /**\n * Fetches the aggregation buckets for the given prefix filter type.\n */\n private async fetchPrefixFilterBuckets(\n filterType: PrefixFilterType\n ): Promise<Bucket[]> {\n const trimmedQuery = this.host.baseQuery?.trim();\n if (!this.canPerformSearch) return [];\n\n const filterAggregationKey = prefixFilterAggregationKeys[filterType];\n const sortParams = this.host.sortParam ? [this.host.sortParam] : [];\n\n const params: SearchParams = {\n ...this.pageSpecifierParams,\n query: trimmedQuery || '',\n rows: 0,\n filters: this.filterMap,\n // Only fetch the firstTitle or firstCreator aggregation\n aggregations: { simpleParams: [filterAggregationKey] },\n // Fetch all 26 letter buckets\n aggregationsSize: 26,\n };\n params.uid = await this.requestUID(\n { ...params, sort: sortParams },\n 'aggregations'\n );\n\n const searchResponse = await this.host.searchService?.search(\n params,\n this.host.searchType\n );\n\n return (searchResponse?.success?.response?.aggregations?.[\n filterAggregationKey\n ]?.buckets ?? []) as Bucket[];\n }\n\n /**\n * Fetches and caches the prefix filter counts for the given filter type.\n */\n async updatePrefixFilterCounts(filterType: PrefixFilterType): Promise<void> {\n const { facetFetchQueryKey } = this;\n const buckets = await this.fetchPrefixFilterBuckets(filterType);\n\n // Don't update the filter counts for an outdated query (if it has been changed\n // since we sent the request)\n const queryChangedSinceFetch =\n facetFetchQueryKey !== this.facetFetchQueryKey;\n if (queryChangedSinceFetch) return;\n\n // Unpack the aggregation buckets into a simple map like { 'A': 50, 'B': 25, ... }\n this.prefixFilterCountMap = { ...this.prefixFilterCountMap }; // Clone the object to trigger an update\n this.prefixFilterCountMap[filterType] = buckets.reduce(\n (acc: Record<string, number>, bucket: Bucket) => {\n acc[(bucket.key as string).toUpperCase()] = bucket.doc_count;\n return acc;\n },\n {}\n );\n\n this.requestHostUpdate();\n }\n\n /**\n * @inheritdoc\n */\n async updatePrefixFiltersForCurrentSort(): Promise<void> {\n if (['title', 'creator'].includes(this.host.selectedSort as SortField)) {\n const filterType = this.host.selectedSort as PrefixFilterType;\n if (!this.prefixFilterCountMap[filterType]) {\n this.updatePrefixFilterCounts(filterType);\n }\n }\n }\n\n /**\n * @inheritdoc\n */\n refreshLetterCounts(): void {\n if (Object.keys(this.prefixFilterCountMap).length > 0) {\n this.prefixFilterCountMap = {};\n }\n this.updatePrefixFiltersForCurrentSort();\n this.requestHostUpdate();\n }\n}\n"]}
|
|
1
|
+
{"version":3,"file":"collection-browser-data-source.js","sourceRoot":"","sources":["../../../src/data-source/collection-browser-data-source.ts"],"names":[],"mappings":"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","sourcesContent":["import type { ReactiveControllerHost } from 'lit';\nimport {\n AccountExtraInfo,\n Aggregation,\n Bucket,\n CollectionExtraInfo,\n FilterConstraint,\n FilterMap,\n FilterMapBuilder,\n PageElementMap,\n SearchParams,\n SearchResponseSessionContext,\n SearchResult,\n SearchType,\n} from '@internetarchive/search-service';\nimport {\n prefixFilterAggregationKeys,\n type FacetBucket,\n type PrefixFilterType,\n TileModel,\n PrefixFilterCounts,\n RequestKind,\n SortField,\n SORT_OPTIONS,\n} from '../models';\nimport { FACETLESS_PAGE_ELEMENTS, type PageSpecifierParams } from './models';\nimport type { CollectionBrowserDataSourceInterface } from './collection-browser-data-source-interface';\nimport type { CollectionBrowserSearchInterface } from './collection-browser-query-state';\nimport { sha1 } from '../utils/sha1';\nimport { log } from '../utils/log';\n\nexport class CollectionBrowserDataSource\n implements CollectionBrowserDataSourceInterface\n{\n /**\n * All pages of tile models that have been fetched so far, indexed by their page\n * number (with the first being page 1).\n */\n private pages: Record<string, TileModel[]> = {};\n\n /**\n * Tile offset to apply when looking up tiles in the pages, in order to maintain\n * page alignment after tiles are removed.\n */\n private offset = 0;\n\n /**\n * Total number of tile models stored in this data source's pages\n */\n private numTileModels = 0;\n\n /**\n * A set of fetch IDs that are valid for the current query state\n */\n private fetchesInProgress = new Set<string>();\n\n /**\n * A record of the query key used for the last search.\n * If this changes, we need to load new results.\n */\n private previousQueryKey: string = '';\n\n /**\n * Whether the initial page of search results for the current query state\n * is presently being fetched.\n */\n private searchResultsLoading = false;\n\n /**\n * Whether the facets (aggregations) for the current query state are\n * presently being fetched.\n */\n private facetsLoading = false;\n\n /**\n * Whether the facets are actually visible -- if not, then we can delay any facet\n * fetches until they become visible.\n */\n private facetsReadyToLoad = false;\n\n /**\n * Whether further query changes should be ignored and not trigger fetches\n */\n private suppressFetches = false;\n\n /**\n * @inheritdoc\n */\n totalResults = 0;\n\n /**\n * @inheritdoc\n */\n endOfDataReached = false;\n\n /**\n * @inheritdoc\n */\n queryInitialized = false;\n\n /**\n * @inheritdoc\n */\n aggregations?: Record<string, Aggregation>;\n\n /**\n * @inheritdoc\n */\n yearHistogramAggregation?: Aggregation;\n\n /**\n * @inheritdoc\n */\n collectionTitles = new Map<string, string>();\n\n /**\n * @inheritdoc\n */\n collectionExtraInfo?: CollectionExtraInfo;\n\n /**\n * @inheritdoc\n */\n accountExtraInfo?: AccountExtraInfo;\n\n /**\n * @inheritdoc\n */\n sessionContext?: SearchResponseSessionContext;\n\n /**\n * @inheritdoc\n */\n pageElements?: PageElementMap;\n\n /**\n * @inheritdoc\n */\n parentCollections?: string[] = [];\n\n /**\n * @inheritdoc\n */\n prefixFilterCountMap: Partial<Record<PrefixFilterType, PrefixFilterCounts>> =\n {};\n\n /**\n * @inheritdoc\n */\n queryErrorMessage?: string;\n\n /**\n * Internal property to store the `resolve` function for the most recent\n * `initialSearchComplete` promise, allowing us to resolve it at the appropriate time.\n */\n private _initialSearchCompleteResolver!: (val: boolean) => void;\n\n /**\n * Internal property to store the private value backing the `initialSearchComplete` getter.\n */\n private _initialSearchCompletePromise: Promise<boolean> = new Promise(res => {\n this._initialSearchCompleteResolver = res;\n });\n\n /**\n * @inheritdoc\n */\n get initialSearchComplete(): Promise<boolean> {\n return this._initialSearchCompletePromise;\n }\n\n // eslint-disable-next-line no-useless-constructor\n constructor(\n /** The host element to which this controller should attach listeners */\n private readonly host: ReactiveControllerHost &\n CollectionBrowserSearchInterface,\n /** Default size of result pages */\n private pageSize: number = 50\n ) {\n // Just setting some property values\n }\n\n hostConnected(): void {\n this.setSearchResultsLoading(this.searchResultsLoading);\n this.setFacetsLoading(this.facetsLoading);\n }\n\n hostUpdate(): void {\n // This reactive controller hook is run whenever the host component (collection-browser) performs an update.\n // We check whether the host's state has changed in a way which should trigger a reset & new results fetch.\n\n // Only the currently-installed data source should react to the update\n if (!this.activeOnHost) return;\n\n // Copy loading states onto the host\n this.setSearchResultsLoading(this.searchResultsLoading);\n this.setFacetsLoading(this.facetsLoading);\n\n // Can't perform searches without a search service\n if (!this.host.searchService) return;\n\n // We should only reset if part of the full query state has changed\n const queryKeyChanged = this.pageFetchQueryKey !== this.previousQueryKey;\n if (!queryKeyChanged) return;\n\n // We should only reset if either:\n // (a) our state permits a valid search, or\n // (b) we have a blank query that we want to show empty results for\n const shouldShowEmptyQueryResults = !this.host.baseQuery;\n if (!(this.canPerformSearch || shouldShowEmptyQueryResults)) return;\n\n if (this.activeOnHost) this.host.emitQueryStateChanged();\n this.handleQueryChange();\n }\n\n /**\n * Returns whether this data source is the one currently installed on the host component.\n */\n private get activeOnHost(): boolean {\n return this.host.dataSource === this;\n }\n\n /**\n * @inheritdoc\n */\n get size(): number {\n return this.numTileModels;\n }\n\n /**\n * @inheritdoc\n */\n reset(): void {\n log('Resetting CB data source');\n this.pages = {};\n this.aggregations = {};\n this.yearHistogramAggregation = undefined;\n this.pageElements = undefined;\n this.parentCollections = [];\n this.queryErrorMessage = undefined;\n\n this.offset = 0;\n this.numTileModels = 0;\n this.endOfDataReached = false;\n this.queryInitialized = false;\n this.facetsLoading = false;\n\n // Invalidate any fetches in progress\n this.fetchesInProgress.clear();\n\n this.setTotalResultCount(0);\n this.requestHostUpdate();\n }\n\n /**\n * @inheritdoc\n */\n addPage(pageNum: number, pageTiles: TileModel[]): void {\n this.pages[pageNum] = pageTiles;\n this.numTileModels += pageTiles.length;\n this.requestHostUpdate();\n }\n\n /**\n * @inheritdoc\n */\n addMultiplePages(firstPageNum: number, tiles: TileModel[]): void {\n const numPages = Math.ceil(tiles.length / this.pageSize);\n for (let i = 0; i < numPages; i += 1) {\n const pageStartIndex = this.pageSize * i;\n this.addPage(\n firstPageNum + i,\n tiles.slice(pageStartIndex, pageStartIndex + this.pageSize)\n );\n }\n\n const visiblePages = this.host.currentVisiblePageNumbers;\n const needsReload = visiblePages.some(\n page => page >= firstPageNum && page <= firstPageNum + numPages\n );\n if (needsReload) {\n this.refreshVisibleResults();\n }\n }\n\n /**\n * @inheritdoc\n */\n getPage(pageNum: number): TileModel[] {\n return this.pages[pageNum];\n }\n\n /**\n * @inheritdoc\n */\n getAllPages(): Record<string, TileModel[]> {\n return this.pages;\n }\n\n /**\n * @inheritdoc\n */\n hasPage(pageNum: number): boolean {\n return !!this.pages[pageNum];\n }\n\n /**\n * @inheritdoc\n */\n getTileModelAt(index: number): TileModel | undefined {\n const offsetIndex = index + this.offset;\n const expectedPageNum = Math.floor(offsetIndex / this.pageSize) + 1;\n const expectedIndexOnPage = offsetIndex % this.pageSize;\n\n let page = 1;\n let tilesSeen = 0;\n while (tilesSeen <= offsetIndex) {\n if (!this.pages[page]) {\n // If we encounter a missing page, either we're past all the results or the page data is sparse.\n // So just return the tile at the expected position if it exists.\n return this.pages[expectedPageNum]?.[expectedIndexOnPage];\n }\n\n if (tilesSeen + this.pages[page].length > offsetIndex) {\n return this.pages[page][offsetIndex - tilesSeen];\n }\n\n tilesSeen += this.pages[page].length;\n page += 1;\n }\n\n return this.pages[expectedPageNum]?.[expectedIndexOnPage];\n }\n\n /**\n * @inheritdoc\n */\n indexOf(tile: TileModel): number {\n return Object.values(this.pages).flat().indexOf(tile) - this.offset;\n }\n\n /**\n * @inheritdoc\n */\n getPageSize(): number {\n return this.pageSize;\n }\n\n /**\n * @inheritdoc\n */\n setPageSize(pageSize: number): void {\n this.reset();\n this.pageSize = pageSize;\n }\n\n /**\n * @inheritdoc\n */\n setTotalResultCount(count: number): void {\n this.totalResults = count;\n if (this.activeOnHost) {\n this.host.setTotalResultCount(count);\n }\n }\n\n /**\n * @inheritdoc\n */\n setFetchesSuppressed(suppressed: boolean): void {\n this.suppressFetches = suppressed;\n }\n\n /**\n * @inheritdoc\n */\n async handleQueryChange(): Promise<void> {\n // Don't react to the change if fetches are suppressed for this data source\n if (this.suppressFetches) return;\n\n this.reset();\n\n // Reset the `initialSearchComplete` promise with a new value for the imminent search\n this._initialSearchCompletePromise = new Promise(res => {\n this._initialSearchCompleteResolver = res;\n });\n\n // Fire the initial page & facet requests\n this.queryInitialized = true;\n await Promise.all([\n this.doInitialPageFetch(),\n this.canFetchFacets ? this.fetchFacets() : null,\n ]);\n\n // Resolve the `initialSearchComplete` promise for this search\n this._initialSearchCompleteResolver(true);\n }\n\n /**\n * @inheritdoc\n */\n async handleFacetReadinessChange(ready: boolean): Promise<void> {\n const facetsBecameReady = !this.facetsReadyToLoad && ready;\n this.facetsReadyToLoad = ready;\n\n const needsFetch = facetsBecameReady && this.canFetchFacets;\n if (needsFetch) {\n this.fetchFacets();\n }\n }\n\n /**\n * Whether the data source & its host are in a state where a facet request should be fired.\n * (i.e., they aren't suppressed or already loading, etc.)\n */\n private get canFetchFacets(): boolean {\n console.log('in canFetchFacets');\n console.log(`facetLoadStrategy is ${this.host.facetLoadStrategy}`);\n console.log(`profileElement is ${this.host.profileElement}`);\n // Don't fetch facets if they are suppressed entirely or not required for the current profile page element\n if (this.host.facetLoadStrategy === 'off') return false;\n if (FACETLESS_PAGE_ELEMENTS.includes(this.host.profileElement!))\n return false;\n\n console.log(`facetsReadyToLoad: ${this.facetsReadyToLoad}`);\n // If facets are to be lazy-loaded, don't fetch them if they are not going to be visible anyway\n // (wait until they become visible instead)\n if (this.host.facetLoadStrategy !== 'eager' && !this.facetsReadyToLoad)\n return false;\n\n // Don't fetch facets again if they are already fetched or pending\n const facetsAlreadyFetched =\n Object.keys(this.aggregations ?? {}).length > 0;\n console.log(`facetsAlreadyFetched: ${facetsAlreadyFetched}`);\n console.log(`facetsLoading: ${this.facetsLoading}`);\n if (this.facetsLoading || facetsAlreadyFetched) return false;\n\n console.log('canFetchFacets = true');\n return true;\n }\n\n /**\n * @inheritdoc\n */\n map(\n callback: (model: TileModel, index: number, array: TileModel[]) => TileModel\n ): void {\n this.pages = Object.fromEntries(\n Object.entries(this.pages).map(([page, tileModels]) => [\n page,\n tileModels.map((model, index, array) =>\n model ? callback(model, index, array) : model\n ),\n ])\n );\n this.requestHostUpdate();\n this.refreshVisibleResults();\n }\n\n /**\n * @inheritdoc\n */\n checkAllTiles = (): void => {\n this.map(model => {\n const cloned = model.clone();\n cloned.checked = true;\n return cloned;\n });\n };\n\n /**\n * @inheritdoc\n */\n uncheckAllTiles = (): void => {\n this.map(model => {\n const cloned = model.clone();\n cloned.checked = false;\n return cloned;\n });\n };\n\n /**\n * @inheritdoc\n */\n removeCheckedTiles = (): void => {\n // To make sure our data source remains page-aligned, we will offset our data source by\n // the number of removed tiles, so that we can just add the offset when the infinite\n // scroller queries for cell contents.\n // This only matters while we're still viewing the same set of results. If the user changes\n // their query/filters/sort, then the data source is overwritten and the offset cleared.\n const { checkedTileModels, uncheckedTileModels } = this;\n const numChecked = checkedTileModels.length;\n if (numChecked === 0) return;\n this.offset += numChecked;\n const newPages: typeof this.pages = {};\n\n // Which page the remaining tile models start on, post-offset\n let offsetPageNumber = Math.floor(this.offset / this.pageSize) + 1;\n let indexOnPage = this.offset % this.pageSize;\n\n // Fill the pages up to that point with empty models\n for (let page = 1; page <= offsetPageNumber; page += 1) {\n const remainingHidden = this.offset - this.pageSize * (page - 1);\n const offsetCellsOnPage = Math.min(this.pageSize, remainingHidden);\n newPages[page] = Array(offsetCellsOnPage).fill(undefined);\n }\n\n // Shift all the remaining tiles into their new positions in the data source\n for (const model of uncheckedTileModels) {\n if (!newPages[offsetPageNumber]) newPages[offsetPageNumber] = [];\n newPages[offsetPageNumber].push(model);\n indexOnPage += 1;\n if (indexOnPage >= this.pageSize) {\n offsetPageNumber += 1;\n indexOnPage = 0;\n }\n }\n\n // Swap in the new pages\n this.pages = newPages;\n this.numTileModels -= numChecked;\n this.totalResults -= numChecked;\n this.host.setTileCount(this.size);\n this.host.setTotalResultCount(this.totalResults);\n this.requestHostUpdate();\n this.refreshVisibleResults();\n };\n\n /**\n * @inheritdoc\n */\n get checkedTileModels(): TileModel[] {\n return this.getFilteredTileModels(model => model.checked);\n }\n\n /**\n * @inheritdoc\n */\n get uncheckedTileModels(): TileModel[] {\n return this.getFilteredTileModels(model => !model.checked);\n }\n\n /**\n * Returns a flattened, filtered array of all the tile models in the data source\n * for which the given predicate returns a truthy value.\n *\n * @param predicate A callback function to apply on each tile model, as with Array.filter\n * @returns A filtered array of tile models satisfying the predicate\n */\n private getFilteredTileModels(\n predicate: (model: TileModel, index: number, array: TileModel[]) => unknown\n ): TileModel[] {\n return Object.values(this.pages)\n .flat()\n .filter((model, index, array) =>\n model ? predicate(model, index, array) : false\n );\n }\n\n /**\n * @inheritdoc\n */\n get canPerformSearch(): boolean {\n console.log('in canPerformSearch');\n console.log(`has search service: ${!!this.host.searchService}`);\n if (!this.host.searchService) return false;\n\n const trimmedQuery = this.host.baseQuery?.trim();\n const hasNonEmptyQuery = !!trimmedQuery;\n const isCollectionSearch = !!this.host.withinCollection;\n const isProfileSearch = !!this.host.withinProfile;\n const hasProfileElement = !!this.host.profileElement;\n const isMetadataSearch = this.host.searchType === SearchType.METADATA;\n console.log(`hasNonEmptyQuery: ${hasNonEmptyQuery}\n isCollectionSearch: ${isCollectionSearch}\n isProfileSearch: ${isProfileSearch}\n isMetadataSearch: ${isMetadataSearch}\n hasProfileElement: ${hasProfileElement}`\n );\n\n // Metadata searches within a collection/profile are allowed to have no query.\n // Otherwise, a non-empty query must be set.\n return (\n hasNonEmptyQuery ||\n (isCollectionSearch && isMetadataSearch) ||\n (isProfileSearch && hasProfileElement && isMetadataSearch)\n );\n }\n\n /**\n * Sets the state for whether the initial set of search results for the\n * current query is loading\n */\n private setSearchResultsLoading(loading: boolean): void {\n console.log(`setSearchResultsLoading: ${loading} (active on host: ${this.activeOnHost})`);\n this.searchResultsLoading = loading;\n if (this.activeOnHost) {\n this.host.setSearchResultsLoading(loading);\n }\n }\n\n /**\n * Sets the state for whether the facets for a query is loading\n */\n private setFacetsLoading(loading: boolean): void {\n console.log(`setFacetsLoading: ${loading} (active on host: ${this.activeOnHost})`);\n this.facetsLoading = loading;\n if (this.activeOnHost) {\n this.host.setFacetsLoading(loading);\n }\n }\n\n /**\n * Requests that the host perform an update, provided this data\n * source is actively installed on it.\n */\n private requestHostUpdate(): void {\n if (this.activeOnHost) {\n this.host.requestUpdate();\n }\n }\n\n /**\n * Requests that the host refresh its visible tiles, provided this\n * data source is actively installed on it.\n */\n private refreshVisibleResults(): void {\n if (this.activeOnHost) {\n this.host.refreshVisibleResults();\n }\n }\n\n /**\n * The query key is a string that uniquely identifies the current search.\n * It consists of:\n * - The current base query\n * - The current collection/profile target & page element\n * - The current search type\n * - Any currently-applied facets\n * - Any currently-applied date range\n * - Any currently-applied prefix filters\n * - The current sort options\n *\n * This lets us internally keep track of queries so we don't persist data that's\n * no longer relevant. Not meant to be human-readable.\n */\n get pageFetchQueryKey(): string {\n const profileKey = `pf;${this.host.withinProfile}--pe;${this.host.profileElement}`;\n const pageTarget = this.host.withinCollection ?? profileKey;\n const sortField = this.host.selectedSort ?? 'none';\n const sortDirection = this.host.sortDirection ?? 'none';\n return `fq:${this.fullQuery}-pt:${pageTarget}-st:${this.host.searchType}-sf:${sortField}-sd:${sortDirection}`;\n }\n\n /**\n * Similar to `pageFetchQueryKey` above, but excludes sort fields since they\n * are not relevant in determining aggregation queries.\n */\n get facetFetchQueryKey(): string {\n const profileKey = `pf;${this.host.withinProfile}--pe;${this.host.profileElement}`;\n const pageTarget = this.host.withinCollection ?? profileKey;\n return `fq:${this.fullQuery}-pt:${pageTarget}-st:${this.host.searchType}`;\n }\n\n /**\n * Constructs a search service FilterMap object from the combination of\n * all the currently-applied filters. This includes any facets, letter\n * filters, and date range.\n */\n get filterMap(): FilterMap {\n const builder = new FilterMapBuilder();\n\n // Add the date range, if applicable\n if (this.host.minSelectedDate) {\n builder.addFilter(\n 'year',\n this.host.minSelectedDate,\n FilterConstraint.GREATER_OR_EQUAL\n );\n }\n if (this.host.maxSelectedDate) {\n builder.addFilter(\n 'year',\n this.host.maxSelectedDate,\n FilterConstraint.LESS_OR_EQUAL\n );\n }\n\n // Add any selected facets\n if (this.host.selectedFacets) {\n for (const [facetName, facetValues] of Object.entries(\n this.host.selectedFacets\n )) {\n const { name, values } = this.prepareFacetForFetch(\n facetName,\n facetValues\n );\n for (const [value, bucket] of Object.entries(values)) {\n let constraint;\n if (bucket.state === 'selected') {\n constraint = FilterConstraint.INCLUDE;\n } else if (bucket.state === 'hidden') {\n constraint = FilterConstraint.EXCLUDE;\n }\n\n if (constraint) {\n builder.addFilter(name, value, constraint);\n }\n }\n }\n }\n\n // Add any letter filters\n if (this.host.selectedTitleFilter) {\n builder.addFilter(\n 'firstTitle',\n this.host.selectedTitleFilter,\n FilterConstraint.INCLUDE\n );\n }\n if (this.host.selectedCreatorFilter) {\n builder.addFilter(\n 'firstCreator',\n this.host.selectedCreatorFilter,\n FilterConstraint.INCLUDE\n );\n }\n\n const filterMap = builder.build();\n return filterMap;\n }\n\n /**\n * Produces a compact unique ID for a search request that can help with debugging\n * on the backend by making related requests easier to trace through different services.\n * (e.g., tying the hits/aggregations requests for the same page back to a single hash).\n *\n * @param params The search service parameters for the request\n * @param kind The kind of request (hits-only, aggregations-only, or both)\n * @returns A Promise resolving to the uid to apply to the request\n */\n async requestUID(params: SearchParams, kind: RequestKind): Promise<string> {\n const paramsToHash = JSON.stringify({\n pageType: params.pageType,\n pageTarget: params.pageTarget,\n query: params.query,\n fields: params.fields,\n filters: params.filters,\n sort: params.sort,\n searchType: this.host.searchType,\n });\n\n const fullQueryHash = (await sha1(paramsToHash)).slice(0, 20); // First 80 bits of SHA-1 are plenty for this\n const sessionId = (await this.host.getSessionId()).slice(0, 20); // Likewise\n const page = params.page ?? 0;\n const kindPrefix = kind.charAt(0); // f = full, h = hits, a = aggregations\n const currentTime = Date.now();\n\n return `R:${fullQueryHash}-S:${sessionId}-P:${page}-K:${kindPrefix}-T:${currentTime}`;\n }\n\n /**\n * @inheritdoc\n */\n get pageSpecifierParams(): PageSpecifierParams | null {\n if (this.host.withinCollection) {\n return {\n pageType: 'collection_details',\n pageTarget: this.host.withinCollection,\n };\n }\n if (this.host.withinProfile) {\n return {\n pageType: 'account_details',\n pageTarget: this.host.withinProfile,\n pageElements: this.host.profileElement\n ? [this.host.profileElement]\n : [],\n };\n }\n return null;\n }\n\n /**\n * The full query, including year facets and date range clauses\n */\n private get fullQuery(): string | undefined {\n let fullQuery = this.host.baseQuery?.trim() ?? '';\n\n const { facetQuery, dateRangeQueryClause, sortFilterQueries } = this;\n\n if (facetQuery) {\n fullQuery += ` AND ${facetQuery}`;\n }\n if (dateRangeQueryClause) {\n fullQuery += ` AND ${dateRangeQueryClause}`;\n }\n if (sortFilterQueries) {\n fullQuery += ` AND ${sortFilterQueries}`;\n }\n return fullQuery.trim();\n }\n\n /**\n * Generates a query string representing the current set of applied facets\n *\n * Example: `mediatype:(\"collection\" OR \"audio\" OR -\"etree\") AND year:(\"2000\" OR \"2001\")`\n */\n private get facetQuery(): string | undefined {\n if (!this.host.selectedFacets) return undefined;\n const facetClauses = [];\n for (const [facetName, facetValues] of Object.entries(\n this.host.selectedFacets\n )) {\n facetClauses.push(this.buildFacetClause(facetName, facetValues));\n }\n return this.joinFacetClauses(facetClauses)?.trim();\n }\n\n private get dateRangeQueryClause(): string | undefined {\n if (!this.host.minSelectedDate || !this.host.maxSelectedDate) {\n return undefined;\n }\n\n return `year:[${this.host.minSelectedDate} TO ${this.host.maxSelectedDate}]`;\n }\n\n private get sortFilterQueries(): string {\n const queries = [this.titleQuery, this.creatorQuery];\n return queries.filter(q => q).join(' AND ');\n }\n\n /**\n * Returns a query clause identifying the currently selected title filter,\n * e.g., `firstTitle:X`.\n */\n private get titleQuery(): string | undefined {\n return this.host.selectedTitleFilter\n ? `firstTitle:${this.host.selectedTitleFilter}`\n : undefined;\n }\n\n /**\n * Returns a query clause identifying the currently selected creator filter,\n * e.g., `firstCreator:X`.\n */\n private get creatorQuery(): string | undefined {\n return this.host.selectedCreatorFilter\n ? `firstCreator:${this.host.selectedCreatorFilter}`\n : undefined;\n }\n\n /**\n * Builds an OR-joined facet clause for the given facet name and values.\n *\n * E.g., for name `subject` and values\n * `{ foo: { state: 'selected' }, bar: { state: 'hidden' } }`\n * this will produce the clause\n * `subject:(\"foo\" OR -\"bar\")`.\n *\n * @param facetName The facet type (e.g., 'collection')\n * @param facetValues The facet buckets, mapped by their keys\n */\n private buildFacetClause(\n facetName: string,\n facetValues: Record<string, FacetBucket>\n ): string {\n const { name: facetQueryName, values } = this.prepareFacetForFetch(\n facetName,\n facetValues\n );\n const facetEntries = Object.entries(values);\n if (facetEntries.length === 0) return '';\n\n const facetValuesArray: string[] = [];\n for (const [key, facetData] of facetEntries) {\n const plusMinusPrefix = facetData.state === 'hidden' ? '-' : '';\n facetValuesArray.push(`${plusMinusPrefix}\"${key}\"`);\n }\n\n const valueQuery = facetValuesArray.join(` OR `);\n return `${facetQueryName}:(${valueQuery})`;\n }\n\n /**\n * Handles some special pre-request normalization steps for certain facet types\n * that require them.\n *\n * @param facetName The name of the facet type (e.g., 'language')\n * @param facetValues An array of values for that facet type\n */\n private prepareFacetForFetch(\n facetName: string,\n facetValues: Record<string, FacetBucket>\n ): { name: string; values: Record<string, FacetBucket> } {\n // eslint-disable-next-line prefer-const\n let [normalizedName, normalizedValues] = [facetName, facetValues];\n\n // The full \"search engine\" name of the lending field is \"lending___status\"\n if (facetName === 'lending') {\n normalizedName = 'lending___status';\n }\n\n return {\n name: normalizedName,\n values: normalizedValues,\n };\n }\n\n /**\n * Takes an array of facet clauses, and combines them into a\n * full AND-joined facet query string. Empty clauses are ignored.\n */\n private joinFacetClauses(facetClauses: string[]): string | undefined {\n const nonEmptyFacetClauses = facetClauses.filter(\n clause => clause.length > 0\n );\n return nonEmptyFacetClauses.length > 0\n ? `(${nonEmptyFacetClauses.join(' AND ')})`\n : undefined;\n }\n\n /**\n * Fires a backend request to fetch a set of aggregations (representing UI facets) for\n * the current search state.\n */\n private async fetchFacets(): Promise<void> {\n console.log('in fetchFacets');\n console.log(`can perform search? ${this.canPerformSearch}`);\n const trimmedQuery = this.host.baseQuery?.trim();\n if (!this.canPerformSearch) return;\n\n const { facetFetchQueryKey } = this;\n console.log(`facetFetchQueryKey: ${facetFetchQueryKey}`);\n console.log(`fetch in progress? ${this.fetchesInProgress.has(facetFetchQueryKey)}`);\n if (this.fetchesInProgress.has(facetFetchQueryKey)) return;\n this.fetchesInProgress.add(facetFetchQueryKey);\n\n this.setFacetsLoading(true);\n\n const sortParams = this.host.sortParam ? [this.host.sortParam] : [];\n const params: SearchParams = {\n ...this.pageSpecifierParams,\n query: trimmedQuery || '',\n rows: 0,\n filters: this.filterMap,\n // Fetch a few extra buckets beyond the 6 we show, in case some get suppressed\n aggregationsSize: 10,\n // Note: we don't need an aggregations param to fetch the default aggregations from the PPS.\n // The default aggregations for the search_results page type should be what we need here.\n };\n params.uid = await this.requestUID(\n { ...params, sort: sortParams },\n 'aggregations'\n );\n\n console.log('sending facet request');\n const searchResponse = await this.host.searchService?.search(\n params,\n this.host.searchType\n );\n const success = searchResponse?.success;\n\n // This is checking to see if the query has changed since the data was fetched.\n // If so, we just want to discard this set of aggregations because they are\n // likely no longer valid for the newer query.\n const queryChangedSinceFetch =\n !this.fetchesInProgress.has(facetFetchQueryKey);\n this.fetchesInProgress.delete(facetFetchQueryKey);\n if (queryChangedSinceFetch) return;\n\n if (!success) {\n const errorMsg = searchResponse?.error?.message;\n const detailMsg = searchResponse?.error?.details?.message;\n\n if (!errorMsg && !detailMsg) {\n // @ts-ignore: Property 'Sentry' does not exist on type 'Window & typeof globalThis'\n window?.Sentry?.captureMessage?.(\n 'Missing or malformed facet response from backend',\n 'error'\n );\n }\n\n this.setFacetsLoading(false);\n return;\n }\n\n const { aggregations, collectionTitles } = success.response;\n this.aggregations = aggregations;\n\n if (collectionTitles) {\n for (const [id, title] of Object.entries(collectionTitles)) {\n this.collectionTitles.set(id, title);\n }\n }\n\n this.yearHistogramAggregation =\n success?.response?.aggregations?.year_histogram;\n\n this.setFacetsLoading(false);\n this.requestHostUpdate();\n }\n\n /**\n * Performs the initial page fetch(es) for the current search state.\n */\n private async doInitialPageFetch(): Promise<void> {\n this.setSearchResultsLoading(true);\n // Try to batch 2 initial page requests when possible\n await this.fetchPage(this.host.initialPageNumber, 2);\n }\n\n /**\n * Fetches one or more pages of results and updates the data source.\n *\n * @param pageNumber The page number to fetch\n * @param numInitialPages If this is an initial page fetch (`pageNumber = 1`),\n * specifies how many pages to batch together in one request. Ignored\n * if `pageNumber != 1`, defaulting to a single page.\n */\n async fetchPage(pageNumber: number, numInitialPages = 1): Promise<void> {\n const trimmedQuery = this.host.baseQuery?.trim();\n if (!this.canPerformSearch) return;\n\n // if we already have data, don't fetch again\n if (this.hasPage(pageNumber)) return;\n\n if (this.endOfDataReached) return;\n\n // Batch multiple initial page requests together if needed (e.g., can request\n // pages 1 and 2 together in a single request).\n let numPages = pageNumber === 1 ? numInitialPages : 1;\n const numRows = this.pageSize * numPages;\n\n // if a fetch is already in progress for this query and page, don't fetch again\n const { pageFetchQueryKey } = this;\n const currentPageKey = `${pageFetchQueryKey}-p:${pageNumber}`;\n if (this.fetchesInProgress.has(currentPageKey)) return;\n\n for (let i = 0; i < numPages; i += 1) {\n this.fetchesInProgress.add(`${pageFetchQueryKey}-p:${pageNumber + i}`);\n }\n this.previousQueryKey = pageFetchQueryKey;\n\n let sortParams = this.host.sortParam ? [this.host.sortParam] : [];\n // TODO eventually the PPS should handle these defaults natively\n const isDefaultProfileSort =\n this.host.withinProfile && this.host.selectedSort === SortField.default;\n if (isDefaultProfileSort && this.host.defaultSortField) {\n const sortOption = SORT_OPTIONS[this.host.defaultSortField];\n if (sortOption.searchServiceKey) {\n sortParams = [\n {\n field: sortOption.searchServiceKey,\n direction: 'desc',\n },\n ];\n }\n }\n\n const params: SearchParams = {\n ...this.pageSpecifierParams,\n query: trimmedQuery || '',\n page: pageNumber,\n rows: numRows,\n sort: sortParams,\n filters: this.filterMap,\n aggregations: { omit: true },\n };\n params.uid = await this.requestUID(params, 'hits');\n\n // log('=== FIRING PAGE REQUEST ===', params);\n const searchResponse = await this.host.searchService?.search(\n params,\n this.host.searchType\n );\n // log('=== RECEIVED PAGE RESPONSE IN CB ===', searchResponse);\n const success = searchResponse?.success;\n\n // This is checking to see if the fetch has been invalidated since it was fired off.\n // If so, we just want to discard the response since it is for an obsolete query state.\n if (!this.fetchesInProgress.has(currentPageKey)) return;\n for (let i = 0; i < numPages; i += 1) {\n this.fetchesInProgress.delete(`${pageFetchQueryKey}-p:${pageNumber + i}`);\n }\n\n if (!success) {\n const errorMsg = searchResponse?.error?.message;\n const detailMsg = searchResponse?.error?.details?.message;\n\n this.queryErrorMessage = `${errorMsg ?? ''}${\n detailMsg ? `; ${detailMsg}` : ''\n }`;\n\n if (!this.queryErrorMessage) {\n this.queryErrorMessage = 'Missing or malformed response from backend';\n // @ts-ignore: Property 'Sentry' does not exist on type 'Window & typeof globalThis'\n window?.Sentry?.captureMessage?.(this.queryErrorMessage, 'error');\n }\n\n this.setSearchResultsLoading(false);\n this.requestHostUpdate();\n return;\n }\n\n this.setTotalResultCount(success.response.totalResults - this.offset);\n if (this.activeOnHost && this.totalResults === 0) {\n // display event to offshoot when result count is zero.\n this.host.emitEmptyResults();\n }\n\n this.sessionContext = success.sessionContext;\n if (this.host.withinCollection) {\n this.collectionExtraInfo = success.response.collectionExtraInfo;\n\n // For collections, we want the UI to respect the default sort option\n // which can be specified in metadata, or otherwise assumed to be week:desc\n if (this.activeOnHost) {\n this.host.applyDefaultCollectionSort(this.collectionExtraInfo);\n }\n\n if (this.collectionExtraInfo) {\n this.parentCollections = [].concat(\n this.collectionExtraInfo.public_metadata?.collection ?? []\n );\n }\n } else if (this.host.withinProfile) {\n this.accountExtraInfo = success.response.accountExtraInfo;\n this.pageElements = success.response.pageElements;\n }\n\n const { results, collectionTitles } = success.response;\n if (results && results.length > 0) {\n // Load any collection titles present on the response into the cache,\n // or queue up preload fetches for them if none were present.\n if (collectionTitles) {\n for (const [id, title] of Object.entries(collectionTitles)) {\n this.collectionTitles.set(id, title);\n }\n }\n\n // Update the data source for each returned page.\n // For loans and web archives, we must account for receiving more pages than we asked for.\n const isUnpagedElement = ['lending', 'web_archives'].includes(\n this.host.profileElement!\n );\n if (isUnpagedElement) {\n numPages = Math.ceil(results.length / this.pageSize);\n this.endOfDataReached = true;\n if (this.activeOnHost) this.host.setTileCount(this.totalResults);\n }\n\n for (let i = 0; i < numPages; i += 1) {\n const pageStartIndex = this.pageSize * i;\n this.addFetchedResultsToDataSource(\n pageNumber + i,\n results.slice(pageStartIndex, pageStartIndex + this.pageSize),\n !isUnpagedElement || i === numPages - 1\n );\n }\n }\n\n // When we reach the end of the data, we can set the infinite scroller's\n // item count to the real total number of results (rather than the\n // temporary estimates based on pages rendered so far).\n if (this.size >= this.totalResults || results.length === 0) {\n this.endOfDataReached = true;\n if (this.activeOnHost) this.host.setTileCount(this.size);\n }\n\n this.setSearchResultsLoading(false);\n this.requestHostUpdate();\n }\n\n /**\n * Update the datasource from the fetch response\n *\n * @param pageNumber\n * @param results\n */\n private addFetchedResultsToDataSource(\n pageNumber: number,\n results: SearchResult[],\n needsReload = true\n ): void {\n const tiles: TileModel[] = [];\n results?.forEach(result => {\n if (!result.identifier) return;\n tiles.push(new TileModel(result));\n });\n\n this.addPage(pageNumber, tiles);\n\n if (needsReload) {\n this.refreshVisibleResults();\n }\n }\n\n /**\n * Fetches the aggregation buckets for the given prefix filter type.\n */\n private async fetchPrefixFilterBuckets(\n filterType: PrefixFilterType\n ): Promise<Bucket[]> {\n const trimmedQuery = this.host.baseQuery?.trim();\n if (!this.canPerformSearch) return [];\n\n const filterAggregationKey = prefixFilterAggregationKeys[filterType];\n const sortParams = this.host.sortParam ? [this.host.sortParam] : [];\n\n const params: SearchParams = {\n ...this.pageSpecifierParams,\n query: trimmedQuery || '',\n rows: 0,\n filters: this.filterMap,\n // Only fetch the firstTitle or firstCreator aggregation\n aggregations: { simpleParams: [filterAggregationKey] },\n // Fetch all 26 letter buckets\n aggregationsSize: 26,\n };\n params.uid = await this.requestUID(\n { ...params, sort: sortParams },\n 'aggregations'\n );\n\n const searchResponse = await this.host.searchService?.search(\n params,\n this.host.searchType\n );\n\n return (searchResponse?.success?.response?.aggregations?.[\n filterAggregationKey\n ]?.buckets ?? []) as Bucket[];\n }\n\n /**\n * Fetches and caches the prefix filter counts for the given filter type.\n */\n async updatePrefixFilterCounts(filterType: PrefixFilterType): Promise<void> {\n const { facetFetchQueryKey } = this;\n const buckets = await this.fetchPrefixFilterBuckets(filterType);\n\n // Don't update the filter counts for an outdated query (if it has been changed\n // since we sent the request)\n const queryChangedSinceFetch =\n facetFetchQueryKey !== this.facetFetchQueryKey;\n if (queryChangedSinceFetch) return;\n\n // Unpack the aggregation buckets into a simple map like { 'A': 50, 'B': 25, ... }\n this.prefixFilterCountMap = { ...this.prefixFilterCountMap }; // Clone the object to trigger an update\n this.prefixFilterCountMap[filterType] = buckets.reduce(\n (acc: Record<string, number>, bucket: Bucket) => {\n acc[(bucket.key as string).toUpperCase()] = bucket.doc_count;\n return acc;\n },\n {}\n );\n\n this.requestHostUpdate();\n }\n\n /**\n * @inheritdoc\n */\n async updatePrefixFiltersForCurrentSort(): Promise<void> {\n if (['title', 'creator'].includes(this.host.selectedSort as SortField)) {\n const filterType = this.host.selectedSort as PrefixFilterType;\n if (!this.prefixFilterCountMap[filterType]) {\n this.updatePrefixFilterCounts(filterType);\n }\n }\n }\n\n /**\n * @inheritdoc\n */\n refreshLetterCounts(): void {\n if (Object.keys(this.prefixFilterCountMap).length > 0) {\n this.prefixFilterCountMap = {};\n }\n this.updatePrefixFiltersForCurrentSort();\n this.requestHostUpdate();\n }\n}\n"]}
|
package/package.json
CHANGED
|
@@ -206,8 +206,7 @@ export class CollectionBrowserDataSource
|
|
|
206
206
|
// We should only reset if either:
|
|
207
207
|
// (a) our state permits a valid search, or
|
|
208
208
|
// (b) we have a blank query that we want to show empty results for
|
|
209
|
-
const shouldShowEmptyQueryResults =
|
|
210
|
-
this.host.clearResultsOnEmptyQuery && this.host.baseQuery === '';
|
|
209
|
+
const shouldShowEmptyQueryResults = !this.host.baseQuery;
|
|
211
210
|
if (!(this.canPerformSearch || shouldShowEmptyQueryResults)) return;
|
|
212
211
|
|
|
213
212
|
if (this.activeOnHost) this.host.emitQueryStateChanged();
|
|
@@ -415,11 +414,15 @@ export class CollectionBrowserDataSource
|
|
|
415
414
|
* (i.e., they aren't suppressed or already loading, etc.)
|
|
416
415
|
*/
|
|
417
416
|
private get canFetchFacets(): boolean {
|
|
417
|
+
console.log('in canFetchFacets');
|
|
418
|
+
console.log(`facetLoadStrategy is ${this.host.facetLoadStrategy}`);
|
|
419
|
+
console.log(`profileElement is ${this.host.profileElement}`);
|
|
418
420
|
// Don't fetch facets if they are suppressed entirely or not required for the current profile page element
|
|
419
421
|
if (this.host.facetLoadStrategy === 'off') return false;
|
|
420
422
|
if (FACETLESS_PAGE_ELEMENTS.includes(this.host.profileElement!))
|
|
421
423
|
return false;
|
|
422
424
|
|
|
425
|
+
console.log(`facetsReadyToLoad: ${this.facetsReadyToLoad}`);
|
|
423
426
|
// If facets are to be lazy-loaded, don't fetch them if they are not going to be visible anyway
|
|
424
427
|
// (wait until they become visible instead)
|
|
425
428
|
if (this.host.facetLoadStrategy !== 'eager' && !this.facetsReadyToLoad)
|
|
@@ -428,8 +431,11 @@ export class CollectionBrowserDataSource
|
|
|
428
431
|
// Don't fetch facets again if they are already fetched or pending
|
|
429
432
|
const facetsAlreadyFetched =
|
|
430
433
|
Object.keys(this.aggregations ?? {}).length > 0;
|
|
434
|
+
console.log(`facetsAlreadyFetched: ${facetsAlreadyFetched}`);
|
|
435
|
+
console.log(`facetsLoading: ${this.facetsLoading}`);
|
|
431
436
|
if (this.facetsLoading || facetsAlreadyFetched) return false;
|
|
432
437
|
|
|
438
|
+
console.log('canFetchFacets = true');
|
|
433
439
|
return true;
|
|
434
440
|
}
|
|
435
441
|
|
|
@@ -555,6 +561,8 @@ export class CollectionBrowserDataSource
|
|
|
555
561
|
* @inheritdoc
|
|
556
562
|
*/
|
|
557
563
|
get canPerformSearch(): boolean {
|
|
564
|
+
console.log('in canPerformSearch');
|
|
565
|
+
console.log(`has search service: ${!!this.host.searchService}`);
|
|
558
566
|
if (!this.host.searchService) return false;
|
|
559
567
|
|
|
560
568
|
const trimmedQuery = this.host.baseQuery?.trim();
|
|
@@ -563,6 +571,12 @@ export class CollectionBrowserDataSource
|
|
|
563
571
|
const isProfileSearch = !!this.host.withinProfile;
|
|
564
572
|
const hasProfileElement = !!this.host.profileElement;
|
|
565
573
|
const isMetadataSearch = this.host.searchType === SearchType.METADATA;
|
|
574
|
+
console.log(`hasNonEmptyQuery: ${hasNonEmptyQuery}
|
|
575
|
+
isCollectionSearch: ${isCollectionSearch}
|
|
576
|
+
isProfileSearch: ${isProfileSearch}
|
|
577
|
+
isMetadataSearch: ${isMetadataSearch}
|
|
578
|
+
hasProfileElement: ${hasProfileElement}`
|
|
579
|
+
);
|
|
566
580
|
|
|
567
581
|
// Metadata searches within a collection/profile are allowed to have no query.
|
|
568
582
|
// Otherwise, a non-empty query must be set.
|
|
@@ -578,6 +592,7 @@ export class CollectionBrowserDataSource
|
|
|
578
592
|
* current query is loading
|
|
579
593
|
*/
|
|
580
594
|
private setSearchResultsLoading(loading: boolean): void {
|
|
595
|
+
console.log(`setSearchResultsLoading: ${loading} (active on host: ${this.activeOnHost})`);
|
|
581
596
|
this.searchResultsLoading = loading;
|
|
582
597
|
if (this.activeOnHost) {
|
|
583
598
|
this.host.setSearchResultsLoading(loading);
|
|
@@ -588,6 +603,7 @@ export class CollectionBrowserDataSource
|
|
|
588
603
|
* Sets the state for whether the facets for a query is loading
|
|
589
604
|
*/
|
|
590
605
|
private setFacetsLoading(loading: boolean): void {
|
|
606
|
+
console.log(`setFacetsLoading: ${loading} (active on host: ${this.activeOnHost})`);
|
|
591
607
|
this.facetsLoading = loading;
|
|
592
608
|
if (this.activeOnHost) {
|
|
593
609
|
this.host.setFacetsLoading(loading);
|
|
@@ -909,10 +925,14 @@ export class CollectionBrowserDataSource
|
|
|
909
925
|
* the current search state.
|
|
910
926
|
*/
|
|
911
927
|
private async fetchFacets(): Promise<void> {
|
|
928
|
+
console.log('in fetchFacets');
|
|
929
|
+
console.log(`can perform search? ${this.canPerformSearch}`);
|
|
912
930
|
const trimmedQuery = this.host.baseQuery?.trim();
|
|
913
931
|
if (!this.canPerformSearch) return;
|
|
914
932
|
|
|
915
933
|
const { facetFetchQueryKey } = this;
|
|
934
|
+
console.log(`facetFetchQueryKey: ${facetFetchQueryKey}`);
|
|
935
|
+
console.log(`fetch in progress? ${this.fetchesInProgress.has(facetFetchQueryKey)}`);
|
|
916
936
|
if (this.fetchesInProgress.has(facetFetchQueryKey)) return;
|
|
917
937
|
this.fetchesInProgress.add(facetFetchQueryKey);
|
|
918
938
|
|
|
@@ -934,6 +954,7 @@ export class CollectionBrowserDataSource
|
|
|
934
954
|
'aggregations'
|
|
935
955
|
);
|
|
936
956
|
|
|
957
|
+
console.log('sending facet request');
|
|
937
958
|
const searchResponse = await this.host.searchService?.search(
|
|
938
959
|
params,
|
|
939
960
|
this.host.searchType
|