From b6612f9e5fbe801505e5a2c6300574adb0ab2f8e Mon Sep 17 00:00:00 2001 From: Tom Coleman Date: Mon, 17 Oct 2016 12:23:23 +1100 Subject: [PATCH 1/3] Fix issue with SSR on a mutation wrapping a query We were hoisting `fetchData` onto the mutation HOC, which led to the query executing twice --- Changelog.md | 3 +++ src/graphql.tsx | 5 ++-- test/react-web/server/index.test.tsx | 40 ++++++++++++++++++++++++++++ typings.d.ts | 2 +- 4 files changed, 46 insertions(+), 4 deletions(-) diff --git a/Changelog.md b/Changelog.md index 48b6dc18b3..da064bfce9 100644 --- a/Changelog.md +++ b/Changelog.md @@ -4,6 +4,9 @@ Expect active development and potentially significant breaking changes in the `0 ### vNext +- Bug: Fix issue with SSR queries running twice when a mutation wraps a query [#274](https://github.com/apollostack/react-apollo/issue/274) + + ### v0.5.10 - Bug: Fix issue with changing outer props *and not changing variables*, ultimately caused by https://github.com/apollostack/apollo-client/pull/694 diff --git a/src/graphql.tsx b/src/graphql.tsx index 11a6583944..92a042da10 100644 --- a/src/graphql.tsx +++ b/src/graphql.tsx @@ -109,7 +109,7 @@ export function withApollo(WrappedComponent) { } // Make sure we preserve any custom statics on the original component. - return hoistNonReactStatics(WithApollo, WrappedComponent); + return hoistNonReactStatics(WithApollo, WrappedComponent, { fetchData: true }); }; export interface OperationOption { @@ -498,8 +498,7 @@ export default function graphql( if (operation.type === DocumentType.Query) (GraphQL as any).fetchData = fetchData; // Make sure we preserve any custom statics on the original component. - return hoistNonReactStatics(GraphQL, WrappedComponent); - + return hoistNonReactStatics(GraphQL, WrappedComponent, { fetchData: true }); }; }; diff --git a/test/react-web/server/index.test.tsx b/test/react-web/server/index.test.tsx index 825883418d..13ec93c227 100644 --- a/test/react-web/server/index.test.tsx +++ b/test/react-web/server/index.test.tsx @@ -264,6 +264,46 @@ describe('SSR', () => { ; }); + it('should correctly handle SSR mutations, reverse order', () => { + + const query = gql`{ currentUser { firstName } }`; + const data1 = { currentUser: { firstName: 'James' } }; + + const mutation = gql`mutation { logRoutes { id } }`; + const mutationData= { logRoutes: { id: 'foo' } }; + + const networkInterface = mockNetworkInterface( + { request: { query }, result: { data: data1 }, delay: 5 }, + { request: { query: mutation }, result: { data: mutationData }, delay: 5 } + ); + const apolloClient = new ApolloClient({ networkInterface }); + + const withQuery = graphql(query, { + props: ({ ownProps, data }) => { + expect(ownProps.mutate).toBeTruthy(); + return { + data, + }; + }, + }); + + const withMutation = graphql(mutation); + const Element = (({ data }) => ( +
{data.loading ? 'loading' : data.currentUser.firstName}
+ )); + + const WrappedElement = withMutation(withQuery(Element)); + + const app = (); + + return getDataFromTree(app) + .then(() => { + const markup = ReactDOM.renderToString(app); + expect(markup).toMatch(/James/); + }) + ; + }); + it('should not require `ApolloProvider` to be the root component', () => { const query = gql`{ currentUser { firstName } }`; diff --git a/typings.d.ts b/typings.d.ts index 101765feed..d3ae7c735d 100644 --- a/typings.d.ts +++ b/typings.d.ts @@ -36,7 +36,7 @@ declare module 'hoist-non-react-statics' { * * Returns the target component. */ - function hoistNonReactStatics(targetComponent: any, sourceComponent: any): any; + function hoistNonReactStatics(targetComponent: any, sourceComponent: any, customStatics: {[name: string]: boolean}): any; namespace hoistNonReactStatics {} export = hoistNonReactStatics; } From 39e950e76a8988ecfb50b6135e661b158d284a99 Mon Sep 17 00:00:00 2001 From: Tom Coleman Date: Mon, 17 Oct 2016 12:52:42 +1100 Subject: [PATCH 2/3] Added an explicit nested queries test (trying to replicate #250) --- test/react-web/server/index.test.tsx | 31 ++++++++++++++++++++++++++++ 1 file changed, 31 insertions(+) diff --git a/test/react-web/server/index.test.tsx b/test/react-web/server/index.test.tsx index 13ec93c227..a65c1bc249 100644 --- a/test/react-web/server/index.test.tsx +++ b/test/react-web/server/index.test.tsx @@ -54,6 +54,37 @@ describe('SSR', () => { }); }); + it('should handle nested queries that depend on each other', () => { + const idQuery = gql`{ currentUser { id } }`; + const idData = { currentUser: { id: '1234' } }; + const userQuery = gql`query getUser($id: String) { user(id: $id) { firstName } }`; + const variables = { id: '1234' }; + const userData = { user: { firstName: 'James' } }; + const networkInterface = mockNetworkInterface( + { request: { query: idQuery }, result: { data: idData }, delay: 50 }, + { request: { query: userQuery, variables }, result: { data: userData }, delay: 50 }, + ); + const apolloClient = new ApolloClient({ networkInterface }); + + const withId = graphql(idQuery); + const withUser = graphql(userQuery, { + skip: ({ data: { loading } }) => loading, + options: ({ data }) => ({ variables: { id: data.currentUser.id } }), + }); + const Component = ({ data }) => ( +
{data.loading ? 'loading' : data.user.firstName}
+ ); + const WrappedComponent = withId(withUser(Component)); + + const app = (); + + return getDataFromTree(app) + .then(() => { + const markup = ReactDOM.renderToString(app); + expect(markup).toMatch(/James/); + }); + }); + it('should correctly skip queries (deprecated)', () => { const query = gql`{ currentUser { firstName } }`; From 65127351a7f90a316ed38c71f42569568021eeeb Mon Sep 17 00:00:00 2001 From: Tom Coleman Date: Mon, 17 Oct 2016 13:04:57 +1100 Subject: [PATCH 3/3] Added a query with a nested container. Again for #250 --- test/react-web/server/index.test.tsx | 24 ++++++++++++++++++++++++ 1 file changed, 24 insertions(+) diff --git a/test/react-web/server/index.test.tsx b/test/react-web/server/index.test.tsx index a65c1bc249..41eddac900 100644 --- a/test/react-web/server/index.test.tsx +++ b/test/react-web/server/index.test.tsx @@ -54,6 +54,30 @@ describe('SSR', () => { }); }); + it('should pick up queries deep in the render tree', () => { + + const query = gql`{ currentUser { firstName } }`; + const data = { currentUser: { firstName: 'James' } }; + const networkInterface = mockNetworkInterface( + { request: { query }, result: { data }, delay: 50 } + ); + const apolloClient = new ApolloClient({ networkInterface }); + + const WrappedElement = graphql(query)(({ data }) => ( +
{data.loading ? 'loading' : data.currentUser.firstName}
+ )); + + const Page = () => (
Hi
); + + const app = (); + + return getDataFromTree(app) + .then(() => { + const markup = ReactDOM.renderToString(app); + expect(markup).toMatch(/James/); + }); + }); + it('should handle nested queries that depend on each other', () => { const idQuery = gql`{ currentUser { id } }`; const idData = { currentUser: { id: '1234' } };