From 86298c3519edf5fb25f94c261a805f65a7f62c69 Mon Sep 17 00:00:00 2001 From: Matt Travi Date: Mon, 17 Dec 2018 11:26:20 -0600 Subject: [PATCH] feat(custom-render): called a custom renderer when provided --- src/router-wrapper.js | 16 +++++---------- test/unit/router-wrapper-test.js | 35 ++++++++++++++++++++++---------- 2 files changed, 29 insertions(+), 22 deletions(-) diff --git a/src/router-wrapper.js b/src/router-wrapper.js index 9762f8fe..7a48446f 100644 --- a/src/router-wrapper.js +++ b/src/router-wrapper.js @@ -1,12 +1,10 @@ -import React from 'react'; -import {renderToString} from 'react-dom/server'; -import {RouterContext} from 'react-router'; import Boom from 'boom'; import {MOVED_PERMANENTLY, MOVED_TEMPORARILY} from 'http-status-codes'; import matchRoute from './route-matcher'; import fetchData from './data-fetcher'; +import defaultRenderFactory from './default-render-factory'; -export default async function renderThroughReactRouter(request, h, {routes, respond, Root, store}) { +export default async function renderThroughReactRouter(request, h, {render, routes, respond, Root, store}) { try { const {renderProps, status, redirectLocation} = await matchRoute(request.raw.req.url, routes); @@ -24,16 +22,12 @@ export default async function renderThroughReactRouter(request, h, {routes, resp } else { await fetchData({renderProps, store}); + const defaultRender = defaultRenderFactory(request, store, renderProps, Root); + return respond(h, { store, status, - renderedContent: { - html: renderToString(( - - - - )) - } + renderedContent: render ? render(defaultRender) : {html: defaultRender()} }); } } catch (e) { diff --git a/test/unit/router-wrapper-test.js b/test/unit/router-wrapper-test.js index 08836f90..d71c15b8 100644 --- a/test/unit/router-wrapper-test.js +++ b/test/unit/router-wrapper-test.js @@ -1,12 +1,10 @@ -import React from 'react'; -import {RouterContext} from 'react-router'; -import domServer from 'react-dom/server'; -import {MOVED_TEMPORARILY, MOVED_PERMANENTLY} from 'http-status-codes'; +import {MOVED_PERMANENTLY, MOVED_TEMPORARILY} from 'http-status-codes'; import sinon from 'sinon'; import {assert} from 'chai'; import any from '@travi/any'; import Boom from 'boom'; import renderThroughReactRouter from '../../src/router-wrapper'; +import * as defaultRenderFactory from '../../src/default-render-factory'; import * as routeMatcher from '../../src/route-matcher'; import * as dataFetcher from '../../src/data-fetcher'; @@ -25,8 +23,7 @@ suite('router-wrapper', () => { sandbox.stub(routeMatcher, 'default'); sandbox.stub(dataFetcher, 'default'); sandbox.stub(Boom, 'wrap'); - sandbox.stub(React, 'createElement'); - sandbox.stub(domServer, 'renderToString'); + sandbox.stub(defaultRenderFactory, 'default'); }); teardown(() => sandbox.restore()); @@ -36,20 +33,36 @@ suite('router-wrapper', () => { const reply = sinon.spy(); const renderProps = any.simpleObject(); const status = any.integer(); - const context = any.simpleObject(); - const rootComponent = any.simpleObject(); const html = any.string(); const response = any.string(); + const defaultRender = sinon.stub(); routeMatcher.default.withArgs(url, routes).resolves({renderProps, status}); dataFetcher.default.withArgs({renderProps, store, status}).resolves({renderProps, status}); - React.createElement.withArgs(RouterContext, sinon.match(renderProps)).returns(context); - React.createElement.withArgs(Root, {request, store}).returns(rootComponent); - domServer.renderToString.withArgs(rootComponent).returns(html); + defaultRender.returns(html); + defaultRenderFactory.default.withArgs(request, store, renderProps, Root).returns(defaultRender); respond.withArgs(reply, {renderedContent: {html}, store, status}).returns(response); return assert.becomes(renderThroughReactRouter(request, reply, {routes, respond, Root, store}), response); }); + test('that response contains the custom-rendered content when a custom renderer is provided', async () => { + const respond = sinon.stub(); + const reply = sinon.spy(); + const renderProps = any.simpleObject(); + const status = any.integer(); + const response = any.string(); + const render = sinon.stub(); + const renderedContent = any.simpleObject(); + const defaultRender = () => undefined; + routeMatcher.default.withArgs(url, routes).resolves({renderProps, status}); + dataFetcher.default.withArgs({renderProps, store, status}).resolves({renderProps, status}); + respond.withArgs(reply, {renderedContent, store, status}).returns(response); + defaultRenderFactory.default.withArgs(request, store, renderProps, Root).returns(defaultRender); + render.withArgs(defaultRender).returns(renderedContent); + + assert.equal(await renderThroughReactRouter(request, reply, {render, routes, respond, Root, store}), response); + }); + test('that a temporary redirect results when a redirectLocation is defined with a 302 status', () => { const respond = sinon.stub(); const redirect = sinon.stub();