From ef8f131aa61b4cc19d89665461d0e1161efae4b9 Mon Sep 17 00:00:00 2001 From: Patrick Browne Date: Fri, 18 Feb 2022 09:55:03 +0100 Subject: [PATCH 1/2] feat: Only load relevant values in right filter Support dimensions parameter in observations graphql query (measures was not used), and use it to only load columns that belong to the column --- app/configurator/components/datatable.tsx | 2 +- app/graphql/queries/data-cubes.graphql | 8 ++++---- app/graphql/query-hooks.ts | 14 +++++++------- app/graphql/resolver-types.ts | 2 +- app/graphql/resolvers.ts | 4 +++- app/graphql/schema.graphql | 2 +- app/rdf/queries.ts | 10 ++++++++-- 7 files changed, 25 insertions(+), 17 deletions(-) diff --git a/app/configurator/components/datatable.tsx b/app/configurator/components/datatable.tsx index a13c9d014..a7637f9a7 100644 --- a/app/configurator/components/datatable.tsx +++ b/app/configurator/components/datatable.tsx @@ -123,7 +123,7 @@ export const DataSetPreviewTable = ({ variables: { iri: dataSetIri, locale, - measures: measures.map((m) => m.iri), + dimensions: null, }, }); diff --git a/app/graphql/queries/data-cubes.graphql b/app/graphql/queries/data-cubes.graphql index 79d38e2f4..432e46927 100644 --- a/app/graphql/queries/data-cubes.graphql +++ b/app/graphql/queries/data-cubes.graphql @@ -71,11 +71,11 @@ query DataCubePreview( query DataCubePreviewObservations( $iri: String! $locale: String! - $measures: [String!]! + $dimensions: [String!] $latest: Boolean ) { dataCubeByIri(iri: $iri, locale: $locale, latest: $latest) { - observations(limit: 10, measures: $measures) { + observations(limit: 10, dimensions: $dimensions) { data sparql } @@ -195,7 +195,7 @@ query TemporalDimensionValues( query DataCubeObservations( $iri: String! $locale: String! - $measures: [String!]! + $dimensions: [String!] $filters: Filters $latest: Boolean ) { @@ -209,7 +209,7 @@ query DataCubeObservations( measures { ...dimensionMetaData } - observations(measures: $measures, filters: $filters) { + observations(dimensions: $dimensions, filters: $filters) { data sparqlEditorUrl } diff --git a/app/graphql/query-hooks.ts b/app/graphql/query-hooks.ts index 931215cc1..970da3993 100644 --- a/app/graphql/query-hooks.ts +++ b/app/graphql/query-hooks.ts @@ -44,7 +44,7 @@ export type DataCube = { export type DataCubeObservationsArgs = { limit?: Maybe; - measures?: Maybe>; + dimensions?: Maybe>; filters?: Maybe; }; @@ -375,7 +375,7 @@ export type DataCubePreviewQuery = { __typename: 'Query', dataCubeByIri?: Maybe< export type DataCubePreviewObservationsQueryVariables = Exact<{ iri: Scalars['String']; locale: Scalars['String']; - measures: Array | Scalars['String']; + dimensions?: Maybe | Scalars['String']>; latest?: Maybe; }>; @@ -488,7 +488,7 @@ export type TemporalDimensionValuesQuery = { __typename: 'Query', dataCubeByIri? export type DataCubeObservationsQueryVariables = Exact<{ iri: Scalars['String']; locale: Scalars['String']; - measures: Array | Scalars['String']; + dimensions?: Maybe | Scalars['String']>; filters?: Maybe; latest?: Maybe; }>; @@ -627,9 +627,9 @@ export function useDataCubePreviewQuery(options: Omit({ query: DataCubePreviewDocument, ...options }); }; export const DataCubePreviewObservationsDocument = gql` - query DataCubePreviewObservations($iri: String!, $locale: String!, $measures: [String!]!, $latest: Boolean) { + query DataCubePreviewObservations($iri: String!, $locale: String!, $dimensions: [String!], $latest: Boolean) { dataCubeByIri(iri: $iri, locale: $locale, latest: $latest) { - observations(limit: 10, measures: $measures) { + observations(limit: 10, dimensions: $dimensions) { data sparql } @@ -754,7 +754,7 @@ export function useTemporalDimensionValuesQuery(options: Omit({ query: TemporalDimensionValuesDocument, ...options }); }; export const DataCubeObservationsDocument = gql` - query DataCubeObservations($iri: String!, $locale: String!, $measures: [String!]!, $filters: Filters, $latest: Boolean) { + query DataCubeObservations($iri: String!, $locale: String!, $dimensions: [String!], $filters: Filters, $latest: Boolean) { dataCubeByIri(iri: $iri, locale: $locale, latest: $latest) { iri title @@ -765,7 +765,7 @@ export const DataCubeObservationsDocument = gql` measures { ...dimensionMetaData } - observations(measures: $measures, filters: $filters) { + observations(dimensions: $dimensions, filters: $filters) { data sparqlEditorUrl } diff --git a/app/graphql/resolver-types.ts b/app/graphql/resolver-types.ts index 20837c2fc..cb423cb16 100644 --- a/app/graphql/resolver-types.ts +++ b/app/graphql/resolver-types.ts @@ -49,7 +49,7 @@ export type DataCube = { export type DataCubeObservationsArgs = { limit?: Maybe; - measures?: Maybe>; + dimensions?: Maybe>; filters?: Maybe; }; diff --git a/app/graphql/resolvers.ts b/app/graphql/resolvers.ts index 003f0943f..6a682ee1b 100644 --- a/app/graphql/resolvers.ts +++ b/app/graphql/resolvers.ts @@ -62,6 +62,7 @@ const Query: QueryResolvers = { filters: queryFilters, limit: 1, raw: true, + dimensions: null, }); if (obs.length === 0) { continue; @@ -240,12 +241,13 @@ const DataCube: DataCubeResolvers = { return dimension ?? null; }, - observations: async ({ cube, locale }, { limit, filters, measures }) => { + observations: async ({ cube, locale }, { limit, filters, dimensions }) => { const { query, observations, observationsRaw } = await getCubeObservations({ cube, locale, filters: filters ?? undefined, limit: limit ?? undefined, + dimensions, }); // const constructedFilters = filters diff --git a/app/graphql/schema.graphql b/app/graphql/schema.graphql index 714de0113..dd36f6f93 100644 --- a/app/graphql/schema.graphql +++ b/app/graphql/schema.graphql @@ -36,7 +36,7 @@ type DataCube { publicationStatus: DataCubePublicationStatus! observations( limit: Int - measures: [String!] + dimensions: [String!] filters: Filters ): ObservationsQuery! dimensions: [Dimension!]! diff --git a/app/rdf/queries.ts b/app/rdf/queries.ts index 5f1e68d31..3330bde21 100644 --- a/app/rdf/queries.ts +++ b/app/rdf/queries.ts @@ -394,6 +394,7 @@ export const getCubeObservations = async ({ filters, limit, raw, + dimensions, }: { cube: Cube; locale: string; @@ -403,6 +404,7 @@ export const getCubeObservations = async ({ limit?: number; /** Returns IRIs instead of labels for NamedNodes */ raw?: boolean; + dimensions: Maybe | undefined; }): Promise<{ query: string; observations: Observation[]; @@ -417,7 +419,8 @@ export const getCubeObservations = async ({ cd.path && ![ns.rdf.type.value, ns.cube.observedBy.value].includes( cd.path.value ?? "" - ) + ) && + (dimensions ? dimensions.includes(cd.path.value) : true) ) ); @@ -428,7 +431,10 @@ export const getCubeObservations = async ({ /** * Add labels to named dimensions */ - const cubeDimensions = await getCubeDimensions({ cube, locale }); + const allCubeDimensions = await getCubeDimensions({ cube, locale }); + const cubeDimensions = allCubeDimensions.filter((d) => + dimensions ? dimensions.includes(d.data.iri) : true + ); // Find dimensions which are NOT literal const namedDimensions = cubeDimensions.filter( From a68810bb5047de79d03e6e0264cac0ea95eab819 Mon Sep 17 00:00:00 2001 From: Patrick Browne Date: Fri, 18 Feb 2022 09:55:50 +0100 Subject: [PATCH 2/2] feat: Only load values for non measure dimension and non standard error dimension --- app/charts/area/chart-area.tsx | 2 +- app/charts/bar/chart-bar.tsx | 2 +- app/charts/column/chart-column.tsx | 2 +- app/charts/line/chart-lines.tsx | 2 +- app/charts/map/chart-map.tsx | 5 +--- app/charts/pie/chart-pie.tsx | 2 +- app/charts/scatterplot/chart-scatterplot.tsx | 5 +--- app/charts/table/chart-table.tsx | 8 +----- app/components/data-download.tsx | 28 ++----------------- .../components/chart-options-selector.tsx | 2 +- app/domain/data.ts | 13 +++++++++ app/rdf/queries.ts | 9 +++++- app/utils/dimension-hierarchy.ts | 2 +- 13 files changed, 34 insertions(+), 48 deletions(-) diff --git a/app/charts/area/chart-area.tsx b/app/charts/area/chart-area.tsx index f473e577f..a804cc8c7 100644 --- a/app/charts/area/chart-area.tsx +++ b/app/charts/area/chart-area.tsx @@ -45,7 +45,7 @@ export const ChartAreasVisualization = ({ variables: { locale, iri: dataSetIri, - measures: [chartConfig.fields.y.componentIri], // FIXME: Other fields may also be measures + dimensions: null, // FIXME: Try to load less dimensions filters: queryFilters, }, }); diff --git a/app/charts/bar/chart-bar.tsx b/app/charts/bar/chart-bar.tsx index 6a58ba05e..4b6204fe9 100644 --- a/app/charts/bar/chart-bar.tsx +++ b/app/charts/bar/chart-bar.tsx @@ -43,7 +43,7 @@ export const ChartBarsVisualization = ({ variables: { locale, iri: dataSetIri, - measures: [chartConfig.fields.x.componentIri], // FIXME: Other fields may also be measures + dimensions: null, // FIXME: Try to load less dimensions filters: queryFilters, }, }); diff --git a/app/charts/column/chart-column.tsx b/app/charts/column/chart-column.tsx index 992ad0d4c..b6cfe8097 100644 --- a/app/charts/column/chart-column.tsx +++ b/app/charts/column/chart-column.tsx @@ -52,7 +52,7 @@ export const ChartColumnsVisualization = ({ variables: { locale, iri: dataSetIri, - measures: [chartConfig.fields.y.componentIri], // FIXME: Other fields may also be measures + dimensions: null, filters: queryFilters, }, }); diff --git a/app/charts/line/chart-lines.tsx b/app/charts/line/chart-lines.tsx index 5e16a7113..fb939ab48 100644 --- a/app/charts/line/chart-lines.tsx +++ b/app/charts/line/chart-lines.tsx @@ -48,7 +48,7 @@ export const ChartLinesVisualization = ({ variables: { locale, iri: dataSetIri, - measures: [chartConfig.fields.y.componentIri], // FIXME: Other fields may also be measures + dimensions: null, // FIXME: Try to load less dimensions filters: queryFilters, }, }); diff --git a/app/charts/map/chart-map.tsx b/app/charts/map/chart-map.tsx index 870d39aed..477331269 100644 --- a/app/charts/map/chart-map.tsx +++ b/app/charts/map/chart-map.tsx @@ -50,10 +50,7 @@ export const ChartMapVisualization = ({ variables: { locale, iri: dataSetIri, - measures: [ - chartConfig.fields.areaLayer.measureIri, - chartConfig.fields.symbolLayer.measureIri, - ], + dimensions: null, // FIXME: Try to load less dimensions filters: queryFilters, }, }); diff --git a/app/charts/pie/chart-pie.tsx b/app/charts/pie/chart-pie.tsx index d86c6a965..676a90992 100644 --- a/app/charts/pie/chart-pie.tsx +++ b/app/charts/pie/chart-pie.tsx @@ -42,7 +42,7 @@ export const ChartPieVisualization = ({ variables: { locale, iri: dataSetIri, - measures: [chartConfig.fields.y.componentIri], // FIXME: Other fields may also be measures + dimensions: null, filters: queryFilters, }, }); diff --git a/app/charts/scatterplot/chart-scatterplot.tsx b/app/charts/scatterplot/chart-scatterplot.tsx index 68e732442..c81ae3b33 100644 --- a/app/charts/scatterplot/chart-scatterplot.tsx +++ b/app/charts/scatterplot/chart-scatterplot.tsx @@ -50,10 +50,7 @@ export const ChartScatterplotVisualization = ({ variables: { locale, iri: dataSetIri, - measures: [ - chartConfig.fields.x.componentIri, - chartConfig.fields.y.componentIri, - ], // FIXME: Other fields may also be measures + dimensions: null, // FIXME: Other fields may also be measures filters: queryFilters, }, }); diff --git a/app/charts/table/chart-table.tsx b/app/charts/table/chart-table.tsx index f969795b0..de74b604d 100644 --- a/app/charts/table/chart-table.tsx +++ b/app/charts/table/chart-table.tsx @@ -26,17 +26,11 @@ export const ChartTableVisualization = ({ }) => { const locale = useLocale(); - const measures = Object.keys(chartConfig.fields).filter( - (key) => - chartConfig.fields[key].componentType === "Measure" && - !chartConfig.fields[key].isHidden - ); - const [{ data, fetching, error }] = useDataCubeObservationsQuery({ variables: { locale, iri: dataSetIri, - measures, + dimensions: null, filters: chartConfig.filters, }, }); diff --git a/app/components/data-download.tsx b/app/components/data-download.tsx index 245f967d9..e82a876cb 100644 --- a/app/components/data-download.tsx +++ b/app/components/data-download.tsx @@ -4,12 +4,7 @@ import { saveAs } from "file-saver"; import { memo, ReactNode, useMemo } from "react"; import { Box, Button, Link } from "theme-ui"; import { useQueryFilters } from "../charts/shared/chart-helpers"; -import { - ChartConfig, - ChartFields, - isMapConfig, - isTableConfig, -} from "../configurator"; +import { ChartConfig, ChartFields } from "../configurator"; import { Observation } from "../domain/data"; import { DimensionMetaDataFragment, @@ -30,24 +25,7 @@ export const DataDownload = memo( chartConfig: ChartConfig; }) => { const locale = useLocale(); - const measures = useMemo( - () => - "y" in chartConfig.fields - ? [chartConfig.fields.y.componentIri] - : isTableConfig(chartConfig) - ? Object.values(chartConfig.fields).flatMap((f) => - f.componentType === "Measure" && !f.isHidden - ? [f.componentIri] - : [] - ) - : isMapConfig(chartConfig) - ? [ - chartConfig.fields.areaLayer.measureIri, - chartConfig.fields.symbolLayer.measureIri, - ] - : [], - [chartConfig] - ); + const filters = useQueryFilters({ chartConfig, }); @@ -56,7 +34,7 @@ export const DataDownload = memo( variables: { locale, iri: dataSetIri, - measures, // FIXME: Other fields may also be measures + dimensions: null, // FIXME: Other fields may also be measures filters, }, }); diff --git a/app/configurator/components/chart-options-selector.tsx b/app/configurator/components/chart-options-selector.tsx index c2ceae557..467dc9ffa 100644 --- a/app/configurator/components/chart-options-selector.tsx +++ b/app/configurator/components/chart-options-selector.tsx @@ -75,7 +75,7 @@ export const ChartOptionsSelector = ({ variables: { locale, iri: state.dataSet, - measures, + dimensions: null, filters: state.chartConfig.filters, }, }); diff --git a/app/domain/data.ts b/app/domain/data.ts index 0c625552f..57faa4538 100644 --- a/app/domain/data.ts +++ b/app/domain/data.ts @@ -7,6 +7,7 @@ import { NominalDimension, OrdinalDimension, } from "../graphql/query-hooks"; +import { ResolvedDimension } from "../graphql/shared-types"; export type RawObservationValue = Literal | NamedNode; @@ -202,3 +203,15 @@ export const isGeoShapesDimension = ( ): dimension is GeoShapesDimension => { return dimension?.__typename === "GeoShapesDimension"; }; + +export const isStandardErrorResolvedDimension = (dim: ResolvedDimension) => { + return dim.data?.related.some((x) => x.type === "StandardError"); +}; + +export const shouldValuesBeLoadedForResolvedDimension = ( + dim: ResolvedDimension +) => { + return !( + dim.data.isMeasureDimension || isStandardErrorResolvedDimension(dim) + ); +}; diff --git a/app/rdf/queries.ts b/app/rdf/queries.ts index 3330bde21..d2e943caa 100644 --- a/app/rdf/queries.ts +++ b/app/rdf/queries.ts @@ -1,4 +1,5 @@ import { descending, group, index } from "d3"; +import { Maybe } from "graphql-tools"; import { Cube, CubeDimension, @@ -14,6 +15,7 @@ import { DimensionValue, Observation, parseObservationValue, + shouldValuesBeLoadedForResolvedDimension, } from "../domain/data"; import { SPARQL_EDITOR, SPARQL_ENDPOINT } from "../domain/env"; import { DataCubeSearchFilter, DataCubeTheme } from "../graphql/query-hooks"; @@ -250,9 +252,10 @@ export const createCubeDimensionValuesLoader = }; export const getCubeDimensionValues = async ( - { dimension, cube, locale, data }: ResolvedDimension, + rdimension: ResolvedDimension, filters?: Filters ): Promise => { + const { dimension, cube, locale, data } = rdimension; if ( typeof dimension.minInclusive !== "undefined" && typeof dimension.maxInclusive !== "undefined" && @@ -267,6 +270,10 @@ export const getCubeDimensionValues = async ( ]; } + if (!shouldValuesBeLoadedForResolvedDimension(rdimension)) { + return []; + } + return await getCubeDimensionValuesWithLabels({ dimension, cube, diff --git a/app/utils/dimension-hierarchy.ts b/app/utils/dimension-hierarchy.ts index 9511ccca1..79975965f 100644 --- a/app/utils/dimension-hierarchy.ts +++ b/app/utils/dimension-hierarchy.ts @@ -277,7 +277,7 @@ export const useHierarchicalDimensionValuesQuery = ({ variables: { iri: dataSetIri, locale, - measures: path, + dimensions: path, }, });