From 3b614b693f3f84c6d86612fa8eefefb78196cc50 Mon Sep 17 00:00:00 2001 From: James Wainwright Date: Thu, 6 Sep 2018 14:07:21 +1000 Subject: [PATCH 1/4] Add noThrow option --- README.md | 15 +++++++++++++++ src/definition.js | 7 ++++++- src/relayMutation.js | 2 +- src/relayNetworkLayer.js | 8 +++++--- src/relayQueries.js | 2 +- 5 files changed, 28 insertions(+), 6 deletions(-) diff --git a/README.md b/README.md index 34388e9..9af3adc 100644 --- a/README.md +++ b/README.md @@ -88,6 +88,21 @@ Middlewares - `prefix` - prefix message (default: `[RELAY-NETWORK] GRAPHQL SERVER ERROR:`) - **deferMiddleware** - _experimental_ Right now `deferMiddleware()` just set `defer` as supported option for Relay. So this middleware allow to community play with `defer()` in cases, which was [described by @wincent](https://github.com/facebook/relay/issues/288#issuecomment-199510058). +Advanced options (2nd argument after middlewares) +=========== + +RelayNetworkLayer may accept additional options: + +```js +const middlewares = []; // array of middlewares +const options = {}; // optional advanced options +const network = new RelayNetworkLayer(middlewares, options); +``` + +Available options: + +- **noThrow** - EXPERIMENTAL (May be deprecated in the future) set true to not throw when an error response is given by the server, and to instead handle errors in your app code. + ### Example of injecting NetworkLayer with middlewares on the **client side**. ```js import Relay from 'react-relay'; diff --git a/src/definition.js b/src/definition.js index c796471..7488e61 100644 --- a/src/definition.js +++ b/src/definition.js @@ -59,7 +59,8 @@ export type RRNLResponseObject = { statusText: string, headers: { [name: string]: string }, url: string, - payload: ?GraphQLResponse, + data?: any, + errors?: GraphQLResponseErrors, }; export type RelayClassicRequest = { @@ -71,3 +72,7 @@ export type RelayClassicRequest = { getVariables: () => Object, getDebugName: () => string, }; + +export type RRNLOptions = { + noThrow: boolean, +}; diff --git a/src/relayMutation.js b/src/relayMutation.js index 39ec5ef..5577126 100644 --- a/src/relayMutation.js +++ b/src/relayMutation.js @@ -26,7 +26,7 @@ export default function mutation( } return fetchWithMiddleware(req) - .then(data => relayRequest.resolve({ response: data })) + .then(({ data }) => relayRequest.resolve({ response: data })) .catch(err => { relayRequest.reject(err); throw err; diff --git a/src/relayNetworkLayer.js b/src/relayNetworkLayer.js index 8753376..9f95f2f 100644 --- a/src/relayNetworkLayer.js +++ b/src/relayNetworkLayer.js @@ -5,7 +5,9 @@ import mutation from './relayMutation'; import fetchWithMiddleware from './fetchWithMiddleware'; import type { Middleware, RelayClassicRequest } from './definition'; -export type RRNLOptions = {}; +export type RRNLOptions = { + noThrow: boolean, +}; export default class RelayNetworkLayer { _options: RRNLOptions; @@ -40,10 +42,10 @@ export default class RelayNetworkLayer { } sendQueries(requests: RelayClassicRequest[]): Promise { - return queries(requests, req => fetchWithMiddleware(req, this._middlewares)); + return queries(requests, req => fetchWithMiddleware(req, this._middlewares, this._options)); } sendMutation(request: RelayClassicRequest): Promise { - return mutation(request, req => fetchWithMiddleware(req, this._middlewares)); + return mutation(request, req => fetchWithMiddleware(req, this._middlewares, this._options)); } } diff --git a/src/relayQueries.js b/src/relayQueries.js index 4235d77..40763f3 100644 --- a/src/relayQueries.js +++ b/src/relayQueries.js @@ -26,7 +26,7 @@ export default function queries( }; return fetchWithMiddleware(req) - .then(data => relayRequest.resolve({ response: data })) + .then(({ data }) => relayRequest.resolve({ response: data })) .catch(err => { relayRequest.reject(err); throw err; From d700fa10850f0ab4fc1439354664656ddc2312f8 Mon Sep 17 00:00:00 2001 From: James Wainwright Date: Thu, 6 Sep 2018 14:33:34 +1000 Subject: [PATCH 2/4] Fix tests + commit fixes --- src/__tests__/fetchWithMiddleware.test.js | 14 +++++++++----- src/fetchWithMiddleware.js | 18 +++++++++++++++--- src/middleware/__tests__/auth.test.js | 3 +-- 3 files changed, 25 insertions(+), 10 deletions(-) diff --git a/src/__tests__/fetchWithMiddleware.test.js b/src/__tests__/fetchWithMiddleware.test.js index 022fc53..00b0fd5 100644 --- a/src/__tests__/fetchWithMiddleware.test.js +++ b/src/__tests__/fetchWithMiddleware.test.js @@ -35,7 +35,7 @@ describe('fetchWithMiddleware', () => { it('should make a successfull request without middlewares', async () => { fetchMock.post('/graphql', { id: 1, data: { user: 123 } }); - const data = await fetchWithMiddleware(createMockReq(1), []); + const { data } = await fetchWithMiddleware(createMockReq(1), [], {}); expect(data).toEqual({ user: 123 }); }); @@ -53,10 +53,14 @@ describe('fetchWithMiddleware', () => { fetchMock.post('/graphql', { id: 1, data: { num: 1 } }); - const data = await fetchWithMiddleware(createMockReq(1), [ - numPlus5, - numMultiply10, // should be first, when changing response - ]); + const { data } = await fetchWithMiddleware( + createMockReq(1), + [ + numPlus5, + numMultiply10, // should be first, when changing response + ], + {} + ); expect(data).toEqual({ num: 15 }); }); }); diff --git a/src/fetchWithMiddleware.js b/src/fetchWithMiddleware.js index 67bf41d..7427fcf 100644 --- a/src/fetchWithMiddleware.js +++ b/src/fetchWithMiddleware.js @@ -7,6 +7,7 @@ import type { RRNLRequestObject, RRNLResponseObject, MiddlewareNextFn, + RRNLOptions, } from './definition'; function runFetch(req: RRNLRequestObject): Promise { @@ -40,16 +41,27 @@ function runFetch(req: RRNLRequestObject): Promise { export default function fetchWithMiddleware( req: RRNLRequestObject, - middlewares: Middleware[] + middlewares: Middleware[], + options: RRNLOptions ): Promise { const wrappedFetch: MiddlewareNextFn = compose(...middlewares)(runFetch); return wrappedFetch(req).then(res => { const { payload } = res; - if (!payload || payload.hasOwnProperty('errors') || !payload.hasOwnProperty('data')) { + const { noThrow = false } = options; + const hasErrors = + !payload || payload.hasOwnProperty('errors') || !payload.hasOwnProperty('data'); + + /** Only throw the Error if noThrow === false */ + if (!noThrow && hasErrors) { throw createRequestError(req, res); } - return payload.data; + + /** Return payload.data as well as the errors (if they exist) */ + return { + data: (payload && payload.data) || null, + errors: hasErrors ? createRequestError(req, res) : null, + }; }); } diff --git a/src/middleware/__tests__/auth.test.js b/src/middleware/__tests__/auth.test.js index fa7f591..d129262 100644 --- a/src/middleware/__tests__/auth.test.js +++ b/src/middleware/__tests__/auth.test.js @@ -38,10 +38,9 @@ describe('Middleware / auth', () => { expect(reqs[0][1].headers.Authorization).toBe('Bearer 123'); }); - it('should work with mutation', async () => { + it.only('should work with mutation', async () => { const req1 = mockReq(); await rnl.sendMutation(req1); - expect(req1.payload).toEqual({ response: 'PAYLOAD' }); const reqs = fetchMock.calls('/graphql'); expect(reqs).toHaveLength(1); From edbc7d8b5af8eca8f0330d5bf5ee1f9cbb6c27e4 Mon Sep 17 00:00:00 2001 From: James Wainwright Date: Thu, 6 Sep 2018 14:49:54 +1000 Subject: [PATCH 3/4] Make noThrow option optional in Flow --- src/definition.js | 5 ++--- src/relayNetworkLayer.js | 8 ++------ 2 files changed, 4 insertions(+), 9 deletions(-) diff --git a/src/definition.js b/src/definition.js index 7488e61..8889cd1 100644 --- a/src/definition.js +++ b/src/definition.js @@ -59,8 +59,7 @@ export type RRNLResponseObject = { statusText: string, headers: { [name: string]: string }, url: string, - data?: any, - errors?: GraphQLResponseErrors, + payload?: GraphQLResponse, }; export type RelayClassicRequest = { @@ -74,5 +73,5 @@ export type RelayClassicRequest = { }; export type RRNLOptions = { - noThrow: boolean, + noThrow?: boolean, }; diff --git a/src/relayNetworkLayer.js b/src/relayNetworkLayer.js index 9f95f2f..1ebdef6 100644 --- a/src/relayNetworkLayer.js +++ b/src/relayNetworkLayer.js @@ -3,11 +3,7 @@ import queries from './relayQueries'; import mutation from './relayMutation'; import fetchWithMiddleware from './fetchWithMiddleware'; -import type { Middleware, RelayClassicRequest } from './definition'; - -export type RRNLOptions = { - noThrow: boolean, -}; +import type { Middleware, RelayClassicRequest, RRNLOptions } from './definition'; export default class RelayNetworkLayer { _options: RRNLOptions; @@ -18,7 +14,7 @@ export default class RelayNetworkLayer { sendMutation: Function; constructor(middlewares: Middleware[] | Middleware, options?: RRNLOptions) { - this._options = options || {}; + this._options = typeof options === 'object' ? options : {}; this._middlewares = Array.isArray(middlewares) ? middlewares : [middlewares]; this._supportedOptions = []; From ec73b12fb74f40e43e1f66af32b4f0038d0e59bc Mon Sep 17 00:00:00 2001 From: James Wainwright Date: Thu, 6 Sep 2018 14:55:18 +1000 Subject: [PATCH 4/4] Rm it.only --- src/middleware/__tests__/auth.test.js | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/src/middleware/__tests__/auth.test.js b/src/middleware/__tests__/auth.test.js index d129262..82b7cd7 100644 --- a/src/middleware/__tests__/auth.test.js +++ b/src/middleware/__tests__/auth.test.js @@ -38,7 +38,7 @@ describe('Middleware / auth', () => { expect(reqs[0][1].headers.Authorization).toBe('Bearer 123'); }); - it.only('should work with mutation', async () => { + it('should work with mutation', async () => { const req1 = mockReq(); await rnl.sendMutation(req1); expect(req1.payload).toEqual({ response: 'PAYLOAD' });