Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

perf: Optimize Browse page queries #1236

Merged
merged 48 commits into from
Oct 31, 2023
Merged
Show file tree
Hide file tree
Changes from 1 commit
Commits
Show all changes
48 commits
Select commit Hold shift + click to select a range
6143c45
fix: Allow dataset tags to wrap
bprusinowski Oct 25, 2023
c82f421
perf: Do not query dataset counts
bprusinowski Oct 25, 2023
5621193
perf: Do not use two queries when searching for cubes
bprusinowski Oct 25, 2023
4ed5fdf
fix: Point to a correct graph to retrieve creator labels
bprusinowski Oct 25, 2023
3d23a44
perf: Use scalar for nested objects (GQL SearchCubes)
bprusinowski Oct 25, 2023
a6220e5
style: Prevent layout shift when draft tag is added
bprusinowski Oct 26, 2023
ea5501e
fix: Types
bprusinowski Oct 26, 2023
6e39852
refactor: Use scalar for SearchCube
bprusinowski Oct 26, 2023
8a11fb2
fix: Do not measure timing twice
bprusinowski Oct 26, 2023
f8fbe2f
perf: Optimize cube search query
bprusinowski Oct 26, 2023
fbba91b
fix: Order cubes by version history and by desc(iri) to select newest…
bprusinowski Oct 26, 2023
467a17a
fix: Types
bprusinowski Oct 26, 2023
15d0b76
fix: Do not filter by subthemes on the server side
bprusinowski Oct 26, 2023
e4c0278
fix: Tests
bprusinowski Oct 26, 2023
a3b4a8f
refactor: Clean up
bprusinowski Oct 26, 2023
ac5e6ed
fix: Missing text input value after refresh
bprusinowski Oct 26, 2023
3143679
fix: Type
bprusinowski Oct 26, 2023
91a298c
fix: Removing browse theme and organization filters
bprusinowski Oct 26, 2023
826ebeb
feat: Allow subthemes to be displayed when organization is second sec…
bprusinowski Oct 26, 2023
a2b4aef
fix: Selecting subthemes when both sections are filtered
bprusinowski Oct 26, 2023
bde860c
fix: Do not clear the cubes when new query is being executed
bprusinowski Oct 26, 2023
d1eff45
fix: Only update browse state if router is ready
bprusinowski Oct 26, 2023
4f65f10
docs: Update CHANGELOG
bprusinowski Oct 26, 2023
a726792
fix: Include subthemes when calculating cube search scores
bprusinowski Oct 27, 2023
95341b1
perf: Specify graph when querying themes
bprusinowski Oct 27, 2023
986e842
perf: Specify graph for subthemes
bprusinowski Oct 27, 2023
7b9c508
chore: Comment out published date datatype condition for now
bprusinowski Oct 27, 2023
4b9e50e
fix: Keep multi-lang filtering
bprusinowski Oct 27, 2023
d16b2b9
fix: Tests
bprusinowski Oct 27, 2023
6b1cedc
test: Add more E2E search tests
bprusinowski Oct 27, 2023
41353f6
fix: creativeWorkStatus has to be set
bprusinowski Oct 27, 2023
c02a7d2
style: Animate filters title
bprusinowski Oct 27, 2023
7631f7d
perf: Do not fire cubes, themes and organizations queries on dataset …
bprusinowski Oct 27, 2023
e2de8ca
fix: Do not filter publishers by language
bprusinowski Oct 27, 2023
3ed4efc
perf: Further optimize cube search query
bprusinowski Oct 30, 2023
a01b3f7
feat: Improve GQLDebugPanel
bprusinowski Oct 30, 2023
e90b2a2
fix: Search query keywords
bprusinowski Oct 30, 2023
b4e87bc
fix: Copy button placement
bprusinowski Oct 30, 2023
d8c50c2
fix: Show published and draft versions of cubes with the same version…
bprusinowski Oct 30, 2023
8d87a82
perf: Optimize search query
bprusinowski Oct 30, 2023
4652385
perf: Remove version history-related logic from SearchCubes query
bprusinowski Oct 30, 2023
e708814
perf: Optimize search cube queries by concatenating themes and sub-th…
bprusinowski Oct 30, 2023
d4117a6
fix: No results (search cubes)
bprusinowski Oct 31, 2023
67c6bdc
chore: Typo
bprusinowski Oct 31, 2023
056c640
fix: Theme filtering needs to happen in HAVING part of the query
bprusinowski Oct 31, 2023
7815ab2
perf: Do not query themes and organizations separately
bprusinowski Oct 31, 2023
0497a97
fix: Exclude topic when constructing remove URL (search filters nav i…
bprusinowski Oct 31, 2023
edb4e91
perf: Drop HAVING in favor of direct filtering
bprusinowski Oct 31, 2023
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
6 changes: 3 additions & 3 deletions app/browser/dataset-browse.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -69,7 +69,7 @@ export const SearchDatasetInput = ({

const searchLabel = t({
id: "dataset.search.label",
message: `Search datasets`,
message: "Search datasets",
});

const placeholderLabel = t({
Expand All @@ -95,13 +95,13 @@ export const SearchDatasetInput = ({
inputRef={inputRef}
id="datasetSearch"
label={searchLabel}
defaultValue={search || ""}
defaultValue={search ?? ""}
InputProps={{
inputProps: {
"data-testid": "datasetSearch",
},
onKeyPress: handleKeyPress,
onReset: onReset,
onReset,
onFocus: () => setShowDraftCheckbox(true),
}}
placeholder={placeholderLabel}
Expand Down
1 change: 1 addition & 0 deletions app/components/form.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -591,6 +591,7 @@ export const SearchField = ({
},
[inputRef, onReset]
);

return (
<Box
sx={{ color: "grey.700", fontSize: "1rem", position: "relative", ...sx }}
Expand Down
21 changes: 14 additions & 7 deletions app/domain/data.ts
Original file line number Diff line number Diff line change
Expand Up @@ -12,6 +12,7 @@ import {
TemporalDimension,
TemporalOrdinalDimension,
} from "@/graphql/query-hooks";
import { DataCubePublicationStatus } from "@/graphql/resolver-types";
import { ResolvedDimension } from "@/graphql/shared-types";

export type RawObservationValue = Literal | NamedNode;
Expand Down Expand Up @@ -71,16 +72,22 @@ export type GeoData = {
};

// Extracted for performance reasons.
ptbrowne marked this conversation as resolved.
Show resolved Hide resolved
export type SearchCubeCreator = {
export type SearchCube = {
iri: string;
label: string;
title: string;
description: string | null;
publicationStatus: DataCubePublicationStatus;
datePublished: string | null;
creator: {
iri: string;
label: string;
} | null;
themes: {
iri: string;
label: string;
}[];
};

export type SearchCubeThemes = {
iri: string;
label: string;
}[];

const xmlSchema = "http://www.w3.org/2001/XMLSchema#";
export const parseRDFLiteral = <T = ObservationValue>(value: Literal): T => {
const v = value.value;
Expand Down
10 changes: 1 addition & 9 deletions app/graphql/queries/data-cubes.graphql
Original file line number Diff line number Diff line change
Expand Up @@ -99,15 +99,7 @@ query SearchCubes(
) {
highlightedTitle
highlightedDescription
cube {
iri
title
description
publicationStatus
datePublished
creator
themes
}
cube
}
}

Expand Down
32 changes: 5 additions & 27 deletions app/graphql/query-hooks.ts
Original file line number Diff line number Diff line change
Expand Up @@ -2,8 +2,7 @@ import { DimensionValue } from '../domain/data';
import { QueryFilters } from '../configurator';
import { Observation } from '../domain/data';
import { RawObservation } from '../domain/data';
import { SearchCubeCreator } from '../domain/data';
import { SearchCubeThemes } from '../domain/data';
import { SearchCube } from '../domain/data';
import gql from 'graphql-tag';
import * as Urql from 'urql';
export type Maybe<T> = T | null;
Expand All @@ -24,8 +23,7 @@ export type Scalars = {
GeoShapes: any;
Observation: Observation;
RawObservation: RawObservation;
SearchCubeCreator: SearchCubeCreator;
SearchCubeThemes: SearchCubeThemes;
SearchCube: SearchCube;
ValueIdentifier: any;
ValuePosition: any;
};
Expand Down Expand Up @@ -442,17 +440,6 @@ export enum ScaleType {
Ratio = 'Ratio'
}

export type SearchCube = {
__typename: 'SearchCube';
iri: Scalars['String'];
title: Scalars['String'];
description?: Maybe<Scalars['String']>;
creator?: Maybe<Scalars['SearchCubeCreator']>;
publicationStatus: DataCubePublicationStatus;
datePublished?: Maybe<Scalars['String']>;
themes: Scalars['SearchCubeThemes'];
};


export type SearchCubeFilter = {
type: Scalars['String'];
Expand All @@ -462,7 +449,7 @@ export type SearchCubeFilter = {
export type SearchCubeResult = {
__typename: 'SearchCubeResult';
score?: Maybe<Scalars['Float']>;
cube: SearchCube;
cube: Scalars['SearchCube'];
highlightedTitle?: Maybe<Scalars['String']>;
highlightedDescription?: Maybe<Scalars['String']>;
};
Expand All @@ -473,7 +460,6 @@ export enum SearchCubeResultOrder {
CreatedDesc = 'CREATED_DESC'
}


export type StandardErrorDimension = Dimension & {
__typename: 'StandardErrorDimension';
iri: Scalars['String'];
Expand Down Expand Up @@ -838,7 +824,7 @@ export type SearchCubesQueryVariables = Exact<{
}>;


export type SearchCubesQuery = { __typename: 'Query', searchCubes: Array<{ __typename: 'SearchCubeResult', highlightedTitle?: Maybe<string>, highlightedDescription?: Maybe<string>, cube: { __typename: 'SearchCube', iri: string, title: string, description?: Maybe<string>, publicationStatus: DataCubePublicationStatus, datePublished?: Maybe<string>, creator?: Maybe<SearchCubeCreator>, themes: SearchCubeThemes } }> };
export type SearchCubesQuery = { __typename: 'Query', searchCubes: Array<{ __typename: 'SearchCubeResult', highlightedTitle?: Maybe<string>, highlightedDescription?: Maybe<string>, cube: SearchCube }> };

export type DataCubePreviewQueryVariables = Exact<{
iri: Scalars['String'];
Expand Down Expand Up @@ -1255,15 +1241,7 @@ export const SearchCubesDocument = gql`
) {
highlightedTitle
highlightedDescription
cube {
iri
title
description
publicationStatus
datePublished
creator
themes
}
cube
}
}
`;
Expand Down
51 changes: 8 additions & 43 deletions app/graphql/resolver-types.ts
Original file line number Diff line number Diff line change
Expand Up @@ -2,8 +2,7 @@ import { DimensionValue } from '../domain/data';
import { Filters } from '../configurator';
import { Observation } from '../domain/data';
import { RawObservation } from '../domain/data';
import { SearchCubeCreator } from '../domain/data';
import { SearchCubeThemes } from '../domain/data';
import { SearchCube } from '../domain/data';
import { GraphQLResolveInfo, GraphQLScalarType, GraphQLScalarTypeConfig } from 'graphql';
import { ResolvedDataCube, ResolvedObservationsQuery, ResolvedMeasure, ResolvedDimension } from './shared-types';
import { VisualizeGraphQLContext } from './context';
Expand All @@ -25,8 +24,7 @@ export type Scalars = {
GeoShapes: any;
Observation: Observation;
RawObservation: RawObservation;
SearchCubeCreator: SearchCubeCreator;
SearchCubeThemes: SearchCubeThemes;
SearchCube: SearchCube;
ValueIdentifier: any;
ValuePosition: any;
};
Expand Down Expand Up @@ -443,17 +441,6 @@ export enum ScaleType {
Ratio = 'Ratio'
}

export type SearchCube = {
__typename?: 'SearchCube';
iri: Scalars['String'];
title: Scalars['String'];
description?: Maybe<Scalars['String']>;
creator?: Maybe<Scalars['SearchCubeCreator']>;
publicationStatus: DataCubePublicationStatus;
datePublished?: Maybe<Scalars['String']>;
themes: Scalars['SearchCubeThemes'];
};


export type SearchCubeFilter = {
type: Scalars['String'];
Expand All @@ -463,7 +450,7 @@ export type SearchCubeFilter = {
export type SearchCubeResult = {
__typename?: 'SearchCubeResult';
score?: Maybe<Scalars['Float']>;
cube: SearchCube;
cube: Scalars['SearchCube'];
highlightedTitle?: Maybe<Scalars['String']>;
highlightedDescription?: Maybe<Scalars['String']>;
};
Expand All @@ -474,7 +461,6 @@ export enum SearchCubeResultOrder {
CreatedDesc = 'CREATED_DESC'
}


export type StandardErrorDimension = Dimension & {
__typename?: 'StandardErrorDimension';
iri: Scalars['String'];
Expand Down Expand Up @@ -674,12 +660,10 @@ export type ResolversTypes = ResolversObject<{
RawObservation: ResolverTypeWrapper<Scalars['RawObservation']>;
RelatedDimension: ResolverTypeWrapper<RelatedDimension>;
ScaleType: ScaleType;
SearchCube: ResolverTypeWrapper<SearchCube>;
SearchCubeCreator: ResolverTypeWrapper<Scalars['SearchCubeCreator']>;
SearchCube: ResolverTypeWrapper<Scalars['SearchCube']>;
SearchCubeFilter: SearchCubeFilter;
SearchCubeResult: ResolverTypeWrapper<SearchCubeResult>;
SearchCubeResultOrder: SearchCubeResultOrder;
SearchCubeThemes: ResolverTypeWrapper<Scalars['SearchCubeThemes']>;
StandardErrorDimension: ResolverTypeWrapper<ResolvedDimension>;
TemporalDimension: ResolverTypeWrapper<ResolvedDimension>;
TemporalOrdinalDimension: ResolverTypeWrapper<ResolvedDimension>;
Expand Down Expand Up @@ -717,11 +701,9 @@ export type ResolversParentTypes = ResolversObject<{
Query: {};
RawObservation: Scalars['RawObservation'];
RelatedDimension: RelatedDimension;
SearchCube: SearchCube;
SearchCubeCreator: Scalars['SearchCubeCreator'];
SearchCube: Scalars['SearchCube'];
SearchCubeFilter: SearchCubeFilter;
SearchCubeResult: SearchCubeResult;
SearchCubeThemes: Scalars['SearchCubeThemes'];
StandardErrorDimension: ResolvedDimension;
TemporalDimension: ResolvedDimension;
TemporalOrdinalDimension: ResolvedDimension;
Expand Down Expand Up @@ -962,19 +944,8 @@ export type RelatedDimensionResolvers<ContextType = VisualizeGraphQLContext, Par
__isTypeOf?: IsTypeOfResolverFn<ParentType, ContextType>;
}>;

export type SearchCubeResolvers<ContextType = VisualizeGraphQLContext, ParentType extends ResolversParentTypes['SearchCube'] = ResolversParentTypes['SearchCube']> = ResolversObject<{
iri?: Resolver<ResolversTypes['String'], ParentType, ContextType>;
title?: Resolver<ResolversTypes['String'], ParentType, ContextType>;
description?: Resolver<Maybe<ResolversTypes['String']>, ParentType, ContextType>;
creator?: Resolver<Maybe<ResolversTypes['SearchCubeCreator']>, ParentType, ContextType>;
publicationStatus?: Resolver<ResolversTypes['DataCubePublicationStatus'], ParentType, ContextType>;
datePublished?: Resolver<Maybe<ResolversTypes['String']>, ParentType, ContextType>;
themes?: Resolver<ResolversTypes['SearchCubeThemes'], ParentType, ContextType>;
__isTypeOf?: IsTypeOfResolverFn<ParentType, ContextType>;
}>;

export interface SearchCubeCreatorScalarConfig extends GraphQLScalarTypeConfig<ResolversTypes['SearchCubeCreator'], any> {
name: 'SearchCubeCreator';
export interface SearchCubeScalarConfig extends GraphQLScalarTypeConfig<ResolversTypes['SearchCube'], any> {
name: 'SearchCube';
}

export type SearchCubeResultResolvers<ContextType = VisualizeGraphQLContext, ParentType extends ResolversParentTypes['SearchCubeResult'] = ResolversParentTypes['SearchCubeResult']> = ResolversObject<{
Expand All @@ -985,10 +956,6 @@ export type SearchCubeResultResolvers<ContextType = VisualizeGraphQLContext, Par
__isTypeOf?: IsTypeOfResolverFn<ParentType, ContextType>;
}>;

export interface SearchCubeThemesScalarConfig extends GraphQLScalarTypeConfig<ResolversTypes['SearchCubeThemes'], any> {
name: 'SearchCubeThemes';
}

export type StandardErrorDimensionResolvers<ContextType = VisualizeGraphQLContext, ParentType extends ResolversParentTypes['StandardErrorDimension'] = ResolversParentTypes['StandardErrorDimension']> = ResolversObject<{
iri?: Resolver<ResolversTypes['String'], ParentType, ContextType>;
label?: Resolver<ResolversTypes['String'], ParentType, ContextType>;
Expand Down Expand Up @@ -1071,10 +1038,8 @@ export type Resolvers<ContextType = VisualizeGraphQLContext> = ResolversObject<{
Query?: QueryResolvers<ContextType>;
RawObservation?: GraphQLScalarType;
RelatedDimension?: RelatedDimensionResolvers<ContextType>;
SearchCube?: SearchCubeResolvers<ContextType>;
SearchCubeCreator?: GraphQLScalarType;
SearchCube?: GraphQLScalarType;
SearchCubeResult?: SearchCubeResultResolvers<ContextType>;
SearchCubeThemes?: GraphQLScalarType;
StandardErrorDimension?: StandardErrorDimensionResolvers<ContextType>;
TemporalDimension?: TemporalDimensionResolvers<ContextType>;
TemporalOrdinalDimension?: TemporalOrdinalDimensionResolvers<ContextType>;
Expand Down
13 changes: 1 addition & 12 deletions app/graphql/schema.graphql
Original file line number Diff line number Diff line change
Expand Up @@ -323,18 +323,7 @@ type OrdinalMeasure implements Dimension {
hierarchy(sourceType: String!, sourceUrl: String!): [HierarchyValue!]
}

scalar SearchCubeCreator
scalar SearchCubeThemes

type SearchCube {
iri: String!
title: String!
description: String
creator: SearchCubeCreator
publicationStatus: DataCubePublicationStatus!
datePublished: String
themes: SearchCubeThemes!
}
scalar SearchCube
ptbrowne marked this conversation as resolved.
Show resolved Hide resolved

type SearchCubeResult {
score: Float
Expand Down
33 changes: 15 additions & 18 deletions app/rdf/query-search.ts
Original file line number Diff line number Diff line change
Expand Up @@ -5,7 +5,7 @@ import { Literal, NamedNode } from "rdf-js";
import StreamClient from "sparql-http-client";
import ParsingClient from "sparql-http-client/ParsingClient";

import { SearchCubeCreator, SearchCubeThemes } from "@/domain/data";
import { SearchCube } from "@/domain/data";
import { truthy } from "@/domain/types";
import { RequestQueryMeta } from "@/graphql/query-meta";
import {
Expand Down Expand Up @@ -280,16 +280,17 @@ export const searchCubes = async ({

const localizedCubes = getCubesByLocale(rawCubeByLang, locale);

if (!localizedCubes) {
if (!localizedCubes?.length) {
return null;
}

const parsedCube: any = {
iri: null,
title: null,
const parsedCube: SearchCube = {
iri: localizedCubes[0].iri,
title: localizedCubes[0].title,
description: null,
creator: null,
publicationStatus: null,
publicationStatus: localizedCubes[0]
.publicationStatus as DataCubePublicationStatus,
datePublished: null,
themes: [],
};
Expand All @@ -308,28 +309,23 @@ export const searchCubes = async ({
}

if (!parsedCube.creator && cube.creator) {
const creator: SearchCubeCreator = {
parsedCube.creator = {
iri: cube.creator,
label: cube.creatorLabel,
};

parsedCube.creator = creator;
}

if (!parsedCube.datePublished) {
parsedCube.datePublished = cube.datePublished;
}

if (!parsedCube.publisher) {
parsedCube.publisher = cube.publisher;
}

if (!parsedCube.publicationStatus) {
parsedCube.publicationStatus = cube.publicationStatus;
parsedCube.publicationStatus =
cube.publicationStatus as DataCubePublicationStatus;
}

if (cube.theme || cube.themeName) {
(parsedCube.themes as SearchCubeThemes).push({
parsedCube.themes.push({
iri: cube.theme,
label: cube.themeName,
});
Expand All @@ -347,9 +343,10 @@ export const searchCubes = async ({
.map((cube) => ({
cube,
highlightedTitle: query ? highlight(cube.title, query) : cube.title,
highlightedDescription: query
? highlight(cube.description, query)
: cube.description,
highlightedDescription:
query && cube.description
? highlight(cube.description, query)
: cube.description,
}));

return {
Expand Down
Loading