-
Notifications
You must be signed in to change notification settings - Fork 0
/
server.js
112 lines (94 loc) · 4.05 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
var http = require('http'),
browserify = require('browserify'),
literalify = require('literalify'),
React = require('react'),
ReactDOMServer = require('react-dom/server'),
DOM = React.DOM, body = DOM.body, div = DOM.div, script = DOM.script,
// This is our React component, shared by server and browser thanks to browserify
App = React.createFactory(require('./App'))
// Just create a plain old HTTP server that responds to two endpoints ('/' and
// '/bundle.js') This would obviously work similarly with any higher level
// library (Express, etc)
http.createServer(function(req, res) {
// If we hit the homepage, then we want to serve up some HTML - including the
// server-side rendered React component(s), as well as the script tags
// pointing to the client-side code
if (req.url == '/') {
res.setHeader('Content-Type', 'text/html; charset=utf-8')
// `props` represents the data to be passed in to the React component for
// rendering - just as you would pass data, or expose variables in
// templates such as Jade or Handlebars. We just use some dummy data
// here (with some potentially dangerous values for testing), but you could
// imagine this would be objects typically fetched async from a DB,
// filesystem or API, depending on the logged-in user, etc.
var props = {
items: [
'Item 0',
'Item 1',
'Item </scRIpt>\u2028',
'Item <!--inject!-->\u2029',
]
}
// Here we're using React to render the outer body, so we just use the
// simpler renderToStaticMarkup function, but you could use any templating
// language (or just a string) for the outer page template
var html = ReactDOMServer.renderToStaticMarkup(body(null,
// The actual server-side rendering of our component occurs here, and we
// pass our data in as `props`. This div is the same one that the client
// will "render" into on the browser from browser.js
div({id: 'content', dangerouslySetInnerHTML: {__html:
ReactDOMServer.renderToString(App(props))
}}),
// The props should match on the client and server, so we stringify them
// on the page to be available for access by the code run in browser.js
// You could use any var name here as long as it's unique
script({dangerouslySetInnerHTML: {__html:
'var APP_PROPS = ' + safeStringify(props) + ';'
}}),
// We'll load React from a CDN - you don't have to do this,
// you can bundle it up or serve it locally if you like
script({src: '//cdnjs.cloudflare.com/ajax/libs/react/15.4.2/react.min.js'}),
script({src: '//cdnjs.cloudflare.com/ajax/libs/react/15.4.2/react-dom.min.js'}),
// Then the browser will fetch and run the browserified bundle consisting
// of browser.js and all its dependencies.
// We serve this from the endpoint a few lines down.
script({src: '/bundle.js'})
))
// Return the page to the browser
res.end(html)
// This endpoint is hit when the browser is requesting bundle.js from the page above
} else if (req.url == '/bundle.js') {
res.setHeader('Content-Type', 'text/javascript')
// Here we invoke browserify to package up browser.js and everything it requires.
// DON'T do it on the fly like this in production - it's very costly -
// either compile the bundle ahead of time, or use some smarter middleware
// (eg browserify-middleware).
// We also use literalify to transform our `require` statements for React
// so that it uses the global variable (from the CDN JS file) instead of
// bundling it up with everything else
browserify()
.add('./browser.js')
.transform(literalify.configure({
'react': 'window.React',
'react-dom': 'window.ReactDOM',
}))
.bundle()
.pipe(res)
// Return 404 for all other requests
} else {
res.statusCode = 404
res.end()
}
// The http server listens on port 3000
}).listen(3000, function(err) {
if (err) throw err
console.log('Listening on 3000...')
})
// A utility function to safely escape JSON for embedding in a <script> tag
function safeStringify(obj) {
return JSON.stringify(obj)
.replace(/<\/(script)/ig, '<\\/$1')
.replace(/<!--/g, '<\\!--')
.replace(/\u2028/g, '\\u2028') // Only necessary if interpreting as JS, which we do
.replace(/\u2029/g, '\\u2029') // Ditto
}