From 87c261857fbb62a6fd40757ff75c56d053ea22dd Mon Sep 17 00:00:00 2001 From: Ian MacLeod Date: Sun, 10 Jul 2016 19:44:05 -0700 Subject: [PATCH 1/2] Pass a context object around with store data This makes it quite a bit easier to pass additional configuration to the data/* functions --- src/QueryManager.ts | 30 ++++++++++++++--------- src/data/diffAgainstStore.ts | 46 ++++++++++++++++-------------------- src/data/readFromStore.ts | 18 +++++++------- test/diffAgainstStore.ts | 31 +++++++++++++++++------- 4 files changed, 70 insertions(+), 55 deletions(-) diff --git a/src/QueryManager.ts b/src/QueryManager.ts index f45e0fbb53f..9b4263f3c68 100644 --- a/src/QueryManager.ts +++ b/src/QueryManager.ts @@ -325,12 +325,14 @@ export class QueryManager { try { const resultFromStore = { data: readSelectionSetFromStore({ - store: this.getDataWithOptimisticResults(), + context: { + store: this.getDataWithOptimisticResults(), + fragmentMap: queryStoreValue.fragmentMap, + }, rootId: queryStoreValue.query.id, selectionSet: queryStoreValue.query.selectionSet, variables: queryStoreValue.variables, returnPartialData: options.returnPartialData || options.noFetch, - fragmentMap: queryStoreValue.fragmentMap, }), loading: queryStoreValue.loading, }; @@ -558,15 +560,17 @@ export class QueryManager { } const previousResult = readSelectionSetFromStore({ - // In case of an optimistic change, apply reducer on top of the - // results including previous optimistic updates. Otherwise, apply it - // on top of the real data only. - store: isOptimistic ? this.getDataWithOptimisticResults() : this.getApolloState().data, + context: { + // In case of an optimistic change, apply reducer on top of the + // results including previous optimistic updates. Otherwise, apply it + // on top of the real data only. + store: isOptimistic ? this.getDataWithOptimisticResults() : this.getApolloState().data, + fragmentMap: createFragmentMap(fragments || []), + }, rootId: 'ROOT_QUERY', selectionSet: queryDefinition.selectionSet, variables: queryOptions.variables, returnPartialData: queryOptions.returnPartialData || queryOptions.noFetch, - fragmentMap: createFragmentMap(fragments || []), }); return { @@ -674,12 +678,14 @@ export class QueryManager { initialResult: Object, } { const { missingSelectionSets, result } = diffSelectionSetAgainstStore({ + context: { + store: this.store.getState()[this.reduxRootKey].data, + fragmentMap, + }, selectionSet: queryDef.selectionSet, - store: this.store.getState()[this.reduxRootKey].data, throwOnMissingField: false, rootId, variables, - fragmentMap, }); const initialResult = result; @@ -761,12 +767,14 @@ export class QueryManager { // this will throw an error if there are missing fields in // the results if returnPartialData is false. resultFromStore = readSelectionSetFromStore({ - store: this.getApolloState().data, + context: { + store: this.getApolloState().data, + fragmentMap, + }, rootId: querySS.id, selectionSet: querySS.selectionSet, variables, returnPartialData: returnPartialData || noFetch, - fragmentMap, }); // ensure multiple errors don't get thrown /* tslint:disable */ diff --git a/src/data/diffAgainstStore.ts b/src/data/diffAgainstStore.ts index c1710364ec0..6e8544b6269 100644 --- a/src/data/diffAgainstStore.ts +++ b/src/data/diffAgainstStore.ts @@ -52,6 +52,13 @@ export interface DiffResult { missingSelectionSets?: SelectionSetWithRoot[]; } +// Contexual state and configuration that is used throught a request from the +// store. +export interface StoreContext { + store: NormalizedCache; + fragmentMap: FragmentMap; +} + export function diffQueryAgainstStore({ store, query, @@ -64,7 +71,7 @@ export function diffQueryAgainstStore({ const queryDef = getQueryDefinition(query); return diffSelectionSetAgainstStore({ - store, + context: { store, fragmentMap: {} }, rootId: 'ROOT_QUERY', selectionSet: queryDef.selectionSet, throwOnMissingField: false, @@ -86,7 +93,7 @@ export function diffFragmentAgainstStore({ const fragmentDef = getFragmentDefinition(fragment); return diffSelectionSetAgainstStore({ - store, + context: { store, fragmentMap: {} }, rootId, selectionSet: fragmentDef.selectionSet, throwOnMissingField: false, @@ -126,28 +133,22 @@ export function handleFragmentErrors(fragmentErrors: { [typename: string]: Error * @return {result: Object, missingSelectionSets: [SelectionSet]} */ export function diffSelectionSetAgainstStore({ + context, selectionSet, - store, rootId, throwOnMissingField = false, variables, - fragmentMap, }: { + context: StoreContext, selectionSet: SelectionSet, - store: NormalizedCache, rootId: string, throwOnMissingField: boolean, variables: Object, - fragmentMap?: FragmentMap, }): DiffResult { if (selectionSet.kind !== 'SelectionSet') { throw new Error('Must be a selection set.'); } - if (!fragmentMap) { - fragmentMap = {}; - } - const result = {}; const missingFields: Selection[] = []; @@ -177,12 +178,11 @@ export function diffSelectionSetAgainstStore({ if (isField(selection)) { const diffResult = diffFieldAgainstStore({ + context, field: selection, throwOnMissingField, variables, rootId, - store, - fragmentMap, included, }); fieldIsMissing = diffResult.isMissing; @@ -204,12 +204,11 @@ export function diffSelectionSetAgainstStore({ if (included) { try { const diffResult = diffSelectionSetAgainstStore({ + context, selectionSet: selection.selectionSet, throwOnMissingField, variables, rootId, - store, - fragmentMap, }); fieldIsMissing = diffResult.isMissing; fieldResult = diffResult.result; @@ -232,7 +231,7 @@ export function diffSelectionSetAgainstStore({ } } } else { - const fragment = fragmentMap[selection.name.value]; + const fragment = context.fragmentMap[selection.name.value]; if (!fragment) { throw new Error(`No fragment named ${selection.name.value}`); @@ -243,12 +242,11 @@ export function diffSelectionSetAgainstStore({ if (included) { try { const diffResult = diffSelectionSetAgainstStore({ + context, selectionSet: fragment.selectionSet, throwOnMissingField, variables, rootId, - store, - fragmentMap, }); fieldIsMissing = diffResult.isMissing; fieldResult = diffResult.result; @@ -311,23 +309,21 @@ export function diffSelectionSetAgainstStore({ } function diffFieldAgainstStore({ + context, field, throwOnMissingField, variables, rootId, - store, - fragmentMap, included = true, }: { + context: StoreContext, field: Field, throwOnMissingField: boolean, variables: Object, rootId: string, - store: NormalizedCache, - fragmentMap?: FragmentMap, included?: Boolean, }): FieldDiffResult { - const storeObj = store[rootId] || {}; + const storeObj = context.store[rootId] || {}; const storeFieldKey = storeKeyNameFromField(field, variables); if (! has(storeObj, storeFieldKey)) { @@ -382,12 +378,11 @@ Perhaps you want to use the \`returnPartialData\` option?`, } const itemDiffResult = diffSelectionSetAgainstStore({ - store, + context, throwOnMissingField, rootId: id, selectionSet: field.selectionSet, variables, - fragmentMap, }); if (itemDiffResult.isMissing) { @@ -409,12 +404,11 @@ Perhaps you want to use the \`returnPartialData\` option?`, if (isIdValue(storeValue)) { const unescapedId = storeValue.id; return diffSelectionSetAgainstStore({ - store, + context, throwOnMissingField, rootId: unescapedId, selectionSet: field.selectionSet, variables, - fragmentMap, }); } diff --git a/src/data/readFromStore.ts b/src/data/readFromStore.ts index 510b7ea7e95..a7164cdfbbd 100644 --- a/src/data/readFromStore.ts +++ b/src/data/readFromStore.ts @@ -1,5 +1,6 @@ import { diffSelectionSetAgainstStore, + StoreContext, } from './diffAgainstStore'; import { @@ -37,12 +38,14 @@ export function readQueryFromStore({ const queryDef = getQueryDefinition(query); return readSelectionSetFromStore({ - store, + context: { + store, + fragmentMap: fragmentMap || {}, + }, rootId: 'ROOT_QUERY', selectionSet: queryDef.selectionSet, variables, returnPartialData, - fragmentMap, }); } @@ -62,7 +65,7 @@ export function readFragmentFromStore({ const fragmentDef = getFragmentDefinition(fragment); return readSelectionSetFromStore({ - store, + context: { store, fragmentMap: {} }, rootId, selectionSet: fragmentDef.selectionSet, variables, @@ -71,29 +74,26 @@ export function readFragmentFromStore({ } export function readSelectionSetFromStore({ - store, + context, rootId, selectionSet, variables, returnPartialData = false, - fragmentMap, }: { - store: NormalizedCache, + context: StoreContext, rootId: string, selectionSet: SelectionSet, variables: Object, returnPartialData?: boolean, - fragmentMap?: FragmentMap, }): Object { const { result, } = diffSelectionSetAgainstStore({ + context, selectionSet, rootId, - store, throwOnMissingField: !returnPartialData, variables, - fragmentMap, }); return result; diff --git a/test/diffAgainstStore.ts b/test/diffAgainstStore.ts index 7b2cc3830d3..b218e9c6f0b 100644 --- a/test/diffAgainstStore.ts +++ b/test/diffAgainstStore.ts @@ -311,7 +311,10 @@ describe('diffing queries against the store', () => { }`; assert.throws(() => { diffSelectionSetAgainstStore({ - store, + context: { + store, + fragmentMap: {}, + }, rootId: 'ROOT_QUERY', selectionSet: getQueryDefinition(unionQuery).selectionSet, variables: null, @@ -353,7 +356,10 @@ describe('diffing queries against the store', () => { }`; assert.doesNotThrow(() => { diffSelectionSetAgainstStore({ - store, + context: { + store, + fragmentMap: {}, + }, rootId: 'ROOT_QUERY', selectionSet: getQueryDefinition(unionQuery).selectionSet, variables: null, @@ -395,12 +401,14 @@ describe('diffing queries against the store', () => { }`; assert.doesNotThrow(() => { diffSelectionSetAgainstStore({ - store, + context: { + store, + fragmentMap: createFragmentMap(getFragmentDefinitions(unionQuery)), + }, rootId: 'ROOT_QUERY', selectionSet: getQueryDefinition(unionQuery).selectionSet, variables: null, throwOnMissingField: true, - fragmentMap: createFragmentMap(getFragmentDefinitions(unionQuery)), }); }); }); @@ -439,12 +447,14 @@ describe('diffing queries against the store', () => { }`; assert.throw(() => { diffSelectionSetAgainstStore({ - store, + context: { + store, + fragmentMap: createFragmentMap(getFragmentDefinitions(unionQuery)), + }, rootId: 'ROOT_QUERY', selectionSet: getQueryDefinition(unionQuery).selectionSet, variables: null, throwOnMissingField: true, - fragmentMap: createFragmentMap(getFragmentDefinitions(unionQuery)), }); }); }); @@ -484,7 +494,10 @@ describe('diffing queries against the store', () => { assert.throw(() => { diffSelectionSetAgainstStore({ - store, + context: { + store, + fragmentMap: {}, + }, rootId: 'ROOT_QUERY', selectionSet: getQueryDefinition(unionQuery).selectionSet, variables: null, @@ -527,7 +540,7 @@ describe('diffing queries against the store', () => { `; const { result } = diffSelectionSetAgainstStore({ - store, + context: { store, fragmentMap: {} }, rootId: 'ROOT_QUERY', selectionSet: getQueryDefinition(queryWithMissingField).selectionSet, variables: null, @@ -541,7 +554,7 @@ describe('diffing queries against the store', () => { }); assert.throws(function() { diffSelectionSetAgainstStore({ - store, + context: { store, fragmentMap: {} }, rootId: 'ROOT_QUERY', selectionSet: getQueryDefinition(queryWithMissingField).selectionSet, variables: null, From d1ecb7e87b8c3c160f53797665c52975ab6ce6f2 Mon Sep 17 00:00:00 2001 From: Ian MacLeod Date: Sun, 10 Jul 2016 12:16:51 -0700 Subject: [PATCH 2/2] storeFetchMiddleware By allowing users to rewrite data lookups from the store, they can satisfy behavior like #332 --- CHANGELOG.md | 1 + ambient.d.ts | 5 + package.json | 1 + src/QueryManager.ts | 11 ++ src/data/diffAgainstStore.ts | 20 ++- src/data/fetchMiddleware.ts | 70 ++++++++++ src/index.ts | 11 ++ test/client.ts | 128 ++++++++++++++++++ .../browser/globals/apollo-client/index.d.ts | 7 +- typings/main/globals/apollo-client/index.d.ts | 7 +- 10 files changed, 256 insertions(+), 5 deletions(-) create mode 100644 src/data/fetchMiddleware.ts diff --git a/CHANGELOG.md b/CHANGELOG.md index 634c77d519e..0d4bd91bcb9 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -5,6 +5,7 @@ Expect active development and potentially significant breaking changes in the `0 ### vNEXT - Fixed an issue with named fragments in batched queries. [PR #509](https://github.com/apollostack/apollo-client/pull/509) and [Issue #501](https://github.com/apollostack/apollo-client/issues/501). - Fixed an issue with unused variables in queries after diffing queries against information available in the store. [PR #518](https://github.com/apollostack/apollo-client/pull/518) and [Issue #496](https://github.com/apollostack/apollo-client/issues/496). +- Added a `storeFetchMiddleware` option to `ApolloClient` that allows transformation of values returned from the store. Also exposes a `cachedFetchById` middleware to handle the common case of fetching cached resources by id. [PR #376](https://github.com/apollostack/apollo-client/pull/376) ### v0.4.11 diff --git a/ambient.d.ts b/ambient.d.ts index 47ef18ba9e5..0a64672fba3 100644 --- a/ambient.d.ts +++ b/ambient.d.ts @@ -103,6 +103,11 @@ declare module 'lodash.pick' { export = main.pick; } +declare module 'lodash.every' { + import main = require('~lodash/index'); + export = main.every; +} + /* GRAPHQL diff --git a/package.json b/package.json index ad874694e3b..da4e5b1275e 100644 --- a/package.json +++ b/package.json @@ -42,6 +42,7 @@ "lodash.assign": "^4.0.8", "lodash.clonedeep": "^4.3.2", "lodash.countby": "^4.4.0", + "lodash.every": "^4.4.0", "lodash.flatten": "^4.2.0", "lodash.forown": "^4.1.0", "lodash.has": "^4.3.1", diff --git a/src/QueryManager.ts b/src/QueryManager.ts index 9b4263f3c68..8e88647240c 100644 --- a/src/QueryManager.ts +++ b/src/QueryManager.ts @@ -60,6 +60,10 @@ import { removeUnusedVariablesFromQuery, } from './data/diffAgainstStore'; +import { + StoreFetchMiddleware, +} from './data/fetchMiddleware'; + import { MutationBehavior, MutationQueryReducersMap, @@ -103,6 +107,7 @@ export class QueryManager { private networkInterface: NetworkInterface; private reduxRootKey: string; private queryTransformer: QueryTransformer; + private storeFetchMiddleware: StoreFetchMiddleware; private queryListeners: { [queryId: string]: QueryListener }; // A map going from queryId to the last result/state that the queryListener was told about. @@ -140,6 +145,7 @@ export class QueryManager { store, reduxRootKey, queryTransformer, + storeFetchMiddleware, shouldBatch = false, batchInterval = 10, }: { @@ -147,6 +153,7 @@ export class QueryManager { store: ApolloStore, reduxRootKey: string, queryTransformer?: QueryTransformer, + storeFetchMiddleware?: StoreFetchMiddleware, shouldBatch?: Boolean, batchInterval?: number, }) { @@ -156,6 +163,7 @@ export class QueryManager { this.store = store; this.reduxRootKey = reduxRootKey; this.queryTransformer = queryTransformer; + this.storeFetchMiddleware = storeFetchMiddleware; this.pollingTimers = {}; this.batchInterval = batchInterval; this.queryListeners = {}; @@ -328,6 +336,7 @@ export class QueryManager { context: { store: this.getDataWithOptimisticResults(), fragmentMap: queryStoreValue.fragmentMap, + fetchMiddleware: this.storeFetchMiddleware, }, rootId: queryStoreValue.query.id, selectionSet: queryStoreValue.query.selectionSet, @@ -681,6 +690,7 @@ export class QueryManager { context: { store: this.store.getState()[this.reduxRootKey].data, fragmentMap, + fetchMiddleware: this.storeFetchMiddleware, }, selectionSet: queryDef.selectionSet, throwOnMissingField: false, @@ -770,6 +780,7 @@ export class QueryManager { context: { store: this.getApolloState().data, fragmentMap, + fetchMiddleware: this.storeFetchMiddleware, }, rootId: querySS.id, selectionSet: querySS.selectionSet, diff --git a/src/data/diffAgainstStore.ts b/src/data/diffAgainstStore.ts index 6e8544b6269..89bf808c6a4 100644 --- a/src/data/diffAgainstStore.ts +++ b/src/data/diffAgainstStore.ts @@ -1,5 +1,6 @@ import isArray = require('lodash.isarray'); import isNull = require('lodash.isnull'); +import isUndefined = require('lodash.isundefined'); import has = require('lodash.has'); import assign = require('lodash.assign'); @@ -16,6 +17,10 @@ import { isIdValue, } from './store'; +import { + StoreFetchMiddleware, +} from './fetchMiddleware'; + import { SelectionSetWithRoot, } from '../queries/store'; @@ -57,6 +62,7 @@ export interface DiffResult { export interface StoreContext { store: NormalizedCache; fragmentMap: FragmentMap; + fetchMiddleware?: StoreFetchMiddleware; } export function diffQueryAgainstStore({ @@ -326,7 +332,17 @@ function diffFieldAgainstStore({ const storeObj = context.store[rootId] || {}; const storeFieldKey = storeKeyNameFromField(field, variables); - if (! has(storeObj, storeFieldKey)) { + let storeValue, fieldMissing; + // Give the transformer a chance to yield a rewritten result. + if (context.fetchMiddleware) { + storeValue = context.fetchMiddleware(field, variables, context.store, () => storeObj[storeFieldKey]); + fieldMissing = isUndefined(storeValue); + } else { + storeValue = storeObj[storeFieldKey]; + fieldMissing = !has(storeObj, storeFieldKey); + } + + if (fieldMissing) { if (throwOnMissingField && included) { throw new ApolloError({ errorMessage: `Can't find field ${storeFieldKey} on object (${rootId}) ${JSON.stringify(storeObj, null, 2)}. @@ -342,8 +358,6 @@ Perhaps you want to use the \`returnPartialData\` option?`, }; } - const storeValue = storeObj[storeFieldKey]; - // Handle all scalar types here if (! field.selectionSet) { if (isJsonValue(storeValue)) { diff --git a/src/data/fetchMiddleware.ts b/src/data/fetchMiddleware.ts new file mode 100644 index 00000000000..08442d62149 --- /dev/null +++ b/src/data/fetchMiddleware.ts @@ -0,0 +1,70 @@ +import every = require('lodash.every'); +import has = require('lodash.has'); + +import { + Field, + Variable, +} from 'graphql'; + +import { + NormalizedCache, + StoreValue, + IdValue, +} from './store'; + +// Middleware that is given an opportunity to rewrite results from the store. +// It should call `next()` to look up the default value. +export type StoreFetchMiddleware = ( + field: Field, + variables: {}, + store: NormalizedCache, + next: () => StoreValue +) => StoreValue; + +// StoreFetchMiddleware that special cases all parameterized queries containing +// either `id` or `ids` to retrieve nodes by those ids directly from the store. +// +// This allows the client to avoid an extra round trip when it is fetching a +// node by id that was previously fetched by a different query. +// +// NOTE: This middleware assumes that you are mapping data ids to the id of +// your nodes. E.g. `dataIdFromObject: value => value.id`. +export function cachedFetchById( + field: Field, + variables: {}, + store: NormalizedCache, + next: () => StoreValue +): StoreValue { + // Note that we are careful to _not_ return an id if it doesn't exist in the + // store! apollo-client assumes that if an id exists in the store, the node + // referenced must also exist. + if (field.arguments && field.arguments.length === 1) { + const onlyArg = field.arguments[0]; + // Only supports variables, for now. + if (onlyArg.value.kind === 'Variable') { + const variable = onlyArg.value; + if (onlyArg.name.value === 'id') { + const id = variables[variable.name.value]; + if (has(store, id)) { + return toIdValue(id); + } + } else if (onlyArg.name.value === 'ids') { + const ids = variables[variable.name.value]; + if (every(ids, id => has(store, id))) { + return ids; + } + } + } + } + + // Otherwise, fall back to the regular behavior. + return next(); +} + +function toIdValue(id): IdValue { + return { + type: 'id', + id, + generated: false, + }; +} diff --git a/src/index.ts b/src/index.ts index 405fcbe9fbe..d93d5149135 100644 --- a/src/index.ts +++ b/src/index.ts @@ -59,6 +59,11 @@ import { addTypenameToSelectionSet, } from './queries/queryTransform'; +import { + cachedFetchById, + StoreFetchMiddleware, +} from './data/fetchMiddleware'; + import { MutationBehavior, MutationBehaviorReducerMap, @@ -87,6 +92,7 @@ export { readQueryFromStore, readFragmentFromStore, addTypenameToSelectionSet as addTypename, + cachedFetchById, writeQueryToStore, writeFragmentToStore, print as printAST, @@ -167,6 +173,7 @@ export default class ApolloClient { public queryManager: QueryManager; public reducerConfig: ApolloReducerConfig; public queryTransformer: QueryTransformer; + public storeFetchMiddleware: StoreFetchMiddleware; public shouldBatch: boolean; public shouldForceFetch: boolean; public dataId: IdGetter; @@ -179,6 +186,7 @@ export default class ApolloClient { initialState, dataIdFromObject, queryTransformer, + storeFetchMiddleware, shouldBatch = false, ssrMode = false, ssrForceFetchDelay = 0, @@ -190,6 +198,7 @@ export default class ApolloClient { initialState?: any, dataIdFromObject?: IdGetter, queryTransformer?: QueryTransformer, + storeFetchMiddleware?: StoreFetchMiddleware, shouldBatch?: boolean, ssrMode?: boolean, ssrForceFetchDelay?: number @@ -201,6 +210,7 @@ export default class ApolloClient { this.networkInterface = networkInterface ? networkInterface : createNetworkInterface('/graphql'); this.queryTransformer = queryTransformer; + this.storeFetchMiddleware = storeFetchMiddleware; this.shouldBatch = shouldBatch; this.shouldForceFetch = !(ssrMode || ssrForceFetchDelay > 0); this.dataId = dataIdFromObject; @@ -307,6 +317,7 @@ export default class ApolloClient { reduxRootKey: this.reduxRootKey, store, queryTransformer: this.queryTransformer, + storeFetchMiddleware: this.storeFetchMiddleware, shouldBatch: this.shouldBatch, batchInterval: this.batchInterval, }); diff --git a/test/client.ts b/test/client.ts index 7264aa0ee49..c0e9e78cb87 100644 --- a/test/client.ts +++ b/test/client.ts @@ -56,6 +56,8 @@ import { import { addTypenameToSelectionSet } from '../src/queries/queryTransform'; +import { cachedFetchById } from '../src/data/fetchMiddleware'; + import mockNetworkInterface from './mocks/mockNetworkInterface'; import { getFragmentDefinitions } from '../src/queries/getFromAST'; @@ -802,6 +804,132 @@ describe('client', () => { }); }); + describe('store fetch middleware (with cachedFetchById)', () => { + + let fetchAll, fetchOne, fetchMany, tasks, flatTasks, client, requests; + beforeEach(() => { + fetchAll = gql` + query fetchAll { + tasks { + id + name + } + } + `; + fetchOne = gql` + query fetchOne($taskId: ID!) { + task(id: $taskId) { + id + name + } + } + `; + fetchMany = gql` + query fetchMany($taskIds: [ID]!) { + tasks(ids: $taskIds) { + id + name + } + } + `; + tasks = { + abc123: {id: 'abc123', name: 'Do stuff'}, + def456: {id: 'def456', name: 'Do things'}, + }; + flatTasks = Object.keys(tasks).map(k => tasks[k]); + requests = []; + const networkInterface: NetworkInterface = { + query(request: Request): Promise { + return new Promise((resolve) => { + requests.push(request); + if (request.operationName === 'fetchAll') { + resolve({ data: { tasks: flatTasks } }); + } else if (request.operationName === 'fetchMany') { + const ids = request.variables['taskIds']; + resolve({ data: { tasks: ids.map(i => tasks[i] || null) } }); + } else if (request.operationName === 'fetchOne') { + resolve({ data: { task: tasks[request.variables['taskId']] || null } }); + } + }); + }, + }; + client = new ApolloClient({ + networkInterface, + dataIdFromObject: (value) => (value).id, + storeFetchMiddleware: cachedFetchById, + }); + }); + + it('should support directly querying with an empty cache', () => { + return client.query({ query: fetchOne, variables: { taskId: 'abc123' } }) + .then((actualResult) => { + assert.deepEqual(actualResult.data, { task: tasks['abc123'] }); + assert.deepEqual(requests.map(r => r.operationName), ['fetchOne']); + }); + }); + + it('should support directly querying with cache lookups', () => { + return client.query({ query: fetchOne, variables: { taskId: 'abc123' } }) + .then((actualResult) => { + assert.deepEqual(actualResult.data, { task: tasks['abc123'] }); + return client.query({ query: fetchOne, variables: { taskId: 'abc123' } }); + }) + .then((actualResult) => { + assert.deepEqual(actualResult.data, { task: tasks['abc123'] }); + assert.deepEqual(requests.map(r => r.operationName), ['fetchOne']); + }); + }); + + it('should support rewrites from other queries', () => { + return client.query({ query: fetchAll }) + .then((actualResult) => { + assert.deepEqual(actualResult.data, { tasks: flatTasks }); + return client.query({ query: fetchOne, variables: { taskId: 'abc123' } }); + }) + .then((actualResult) => { + assert.deepEqual(actualResult.data, { task: tasks['abc123'] }); + assert.deepEqual(requests.map(r => r.operationName), ['fetchAll']); + }); + }); + + it('should handle cache misses when rewriting', () => { + return client.query({ query: fetchAll }) + .then((actualResult) => { + assert.deepEqual(actualResult.data, { tasks: flatTasks }); + return client.query({ query: fetchOne, variables: { taskId: 'badid' } }); + }) + .then((actualResult) => { + assert.deepEqual(actualResult.data, { task: null }); + assert.deepEqual(requests.map(r => r.operationName), ['fetchAll', 'fetchOne']); + }); + }); + + it('should handle bulk fetching from cache', () => { + return client.query({ query: fetchAll }) + .then((actualResult) => { + assert.deepEqual(actualResult.data, { tasks: flatTasks }); + return client.query({ query: fetchMany, variables: { taskIds: ['def456', 'abc123'] } }); + }) + .then((actualResult) => { + assert.deepEqual(actualResult.data, { tasks: [tasks['def456'], tasks['abc123']] }); + assert.deepEqual(requests.map(r => r.operationName), ['fetchAll']); + }); + }); + + it('should handle cache misses when bulk fetching', () => { + return client.query({ query: fetchAll }) + .then((actualResult) => { + assert.deepEqual(actualResult.data, { tasks: flatTasks }); + return client.query({ query: fetchMany, variables: { taskIds: ['def456', 'badid'] } }); + }) + .then((actualResult) => { + assert.deepEqual(actualResult.data, { tasks: [tasks['def456'], null] }); + assert.deepEqual(requests.map(r => r.operationName), ['fetchAll', 'fetchMany']); + }); + }); + + }); + it('should send operationName along with the mutation to the server', (done) => { const mutation = gql` mutation myMutationName { diff --git a/typings/browser/globals/apollo-client/index.d.ts b/typings/browser/globals/apollo-client/index.d.ts index 398410c95fd..238f3c2f2b8 100644 --- a/typings/browser/globals/apollo-client/index.d.ts +++ b/typings/browser/globals/apollo-client/index.d.ts @@ -100,6 +100,11 @@ declare module 'lodash.pick' { export = main.pick; } +declare module 'lodash.every' { + import main = require('~lodash/index'); + export = main.every; +} + /* GRAPHQL @@ -116,4 +121,4 @@ declare module 'graphql-tag/parser' { declare module 'graphql-tag/printer' { function print(ast: any): string; -} \ No newline at end of file +} diff --git a/typings/main/globals/apollo-client/index.d.ts b/typings/main/globals/apollo-client/index.d.ts index 398410c95fd..238f3c2f2b8 100644 --- a/typings/main/globals/apollo-client/index.d.ts +++ b/typings/main/globals/apollo-client/index.d.ts @@ -100,6 +100,11 @@ declare module 'lodash.pick' { export = main.pick; } +declare module 'lodash.every' { + import main = require('~lodash/index'); + export = main.every; +} + /* GRAPHQL @@ -116,4 +121,4 @@ declare module 'graphql-tag/parser' { declare module 'graphql-tag/printer' { function print(ast: any): string; -} \ No newline at end of file +}