This repository was archived by the owner on Jun 28, 2021. It is now read-only.
-
Notifications
You must be signed in to change notification settings - Fork 365
/
Copy pathserver.js
154 lines (129 loc) · 4.79 KB
/
server.js
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
/* global webpack_isomorphic_tools */
import express from 'express';
import PrettyError from 'pretty-error';
import React from 'react';
import ReactDOM from 'react-dom/server';
import { match } from 'react-router';
import { ReduxAsyncConnect, loadOnServer } from 'redux-connect';
import createHistory from 'react-router/lib/createMemoryHistory';
import { Provider } from 'react-redux';
import { IntlProvider } from 'react-intl';
import cookie from 'react-cookie';
import Raven from 'raven';
import errorhandler from 'errorhandler';
import pdf from 'html-pdf';
import config from 'config';
import expressConfig from './server/config/express';
import routes from './routes';
import ApiClient from './helpers/ApiClient';
import createStore from './redux/create';
import debug from './helpers/debug';
import Html from './helpers/Html';
import PdfHtml from './helpers/PdfHtml';
import { setOption, setUserAgent } from './redux/actions/options.js';
import getLocalMessages from './helpers/setLocal';
const pretty = new PrettyError();
const server = express();
Raven.config(config.sentryServer, {
captureUnhandledRejections: true,
autoBreadcrumbs: true
}).install();
expressConfig(server);
server.use(Raven.requestHandler());
server.use((req, res, next) => {
cookie.plugToRequest(req, res);
const client = new ApiClient(req);
const history = createHistory(req.originalUrl);
const store = createStore(history, client);
const localMessages = getLocalMessages(req);
if (process.env.NODE_ENV === 'development') {
webpack_isomorphic_tools.refresh();
}
if (req.query.DISABLE_SSR) {
return res.status(200).send(
`<!doctype html>\n${ReactDOM.renderToString(<IntlProvider locale="en" messages={localMessages}>
<Html store={store} assets={webpack_isomorphic_tools.assets()} />
</IntlProvider>)}`
);
}
store.dispatch(setUserAgent(req.useragent));
store.dispatch(setOption(cookie.load('options') || {}));
debug('Server', 'Executing navigate action');
match(
{ history, routes: routes(store), location: req.originalUrl },
(error, redirectLocation, renderProps) => {
debug('Server', 'Route match callback');
if (redirectLocation) {
res.redirect(redirectLocation.pathname + redirectLocation.search);
} else if (error) {
console.error('ROUTER ERROR:', pretty.render(error));
res.status(500).send(error);
} else if (renderProps) {
const status = renderProps.location.pathname.indexOf('/error') > -1
? 404
: 200;
loadOnServer({ ...renderProps, store, helpers: { client } })
.then(() => {
const component = (
<IntlProvider messages={localMessages} locale="en">
<Provider store={store}>
<ReduxAsyncConnect {...renderProps} />
</Provider>
</IntlProvider>
);
res.type('html');
res.setHeader('Cache-Control', 'public, max-age=31557600');
res.status(status);
debug('Server', 'Sending markup');
if (req.originalUrl.includes('.pdf')) {
const body = ReactDOM.renderToString(
<PdfHtml
url={`${req.protocol}://${req.get('host')}`}
component={component}
assets={webpack_isomorphic_tools.assets()}
/>
);
const html = `<!doctype html>\n${body}`;
return pdf.create(html).toStream((err, stream) => {
if (err) {
res.status(422).send(err);
}
res.set('Content-type', 'application/pdf');
// NOTE: If you want to export a file.
// res.set('Content-disposition', 'attachment; filename=pdf.pdf');
stream.pipe(res);
});
}
const html = `<!doctype html>\n${ReactDOM.renderToString(<Html component={component} store={store} assets={webpack_isomorphic_tools.assets()} />)}`;
return res.send(html);
})
.catch(next);
}
}
);
return false;
});
server.use(Raven.errorHandler());
if (process.env.NODE_ENV === 'development') {
// only use in development
server.use(errorhandler());
} else {
server.use((req, res) => {
res.status(500);
res.send('OOPS');
});
}
const port = process.env.PORT || 8000;
export default function serve(cb) {
return server.listen(port, () => {
console.info(`==> 🌎 ENV=${process.env.NODE_ENV}`);
console.info(`==> ✅ Server is listening at http://localhost:${port}`);
console.info(`==> 🎯 API at ${process.env.API_URL}`);
Object.keys(config).forEach(
key =>
config[key].constructor.name !== 'Object' &&
console.info(`==> ${key}`, config[key])
);
return cb && cb(this);
});
}