UNPKG

react-saasify-chrisvxd

Version:

React components for Saasify web clients.

116 lines (98 loc) 2.79 kB
const http = require('http'); const https = require('https'); const WebSocket = require('ws'); const generateCertificate = require('./utils/generateCertificate'); const getCertificate = require('./utils/getCertificate'); const logger = require('@parcel/logger'); class HMRServer { async start(options = {}) { await new Promise(async resolve => { if (!options.https) { this.server = http.createServer(); } else if (typeof options.https === 'boolean') { this.server = https.createServer(generateCertificate(options)); } else { this.server = https.createServer(await getCertificate(options.https)); } let websocketOptions = { server: this.server }; if (options.hmrHostname) { websocketOptions.origin = `${options.https ? 'https' : 'http'}://${ options.hmrHostname }`; } this.wss = new WebSocket.Server(websocketOptions); this.server.listen(options.hmrPort, resolve); }); this.wss.on('connection', ws => { ws.onerror = this.handleSocketError; if (this.unresolvedError) { ws.send(JSON.stringify(this.unresolvedError)); } }); this.wss.on('error', this.handleSocketError); return this.wss._server.address().port; } stop() { this.wss.close(); this.server.close(); } emitError(err) { let {message, stack} = logger.formatError(err); // store the most recent error so we can notify new connections // and so we can broadcast when the error is resolved this.unresolvedError = { type: 'error', error: { message, stack } }; this.broadcast(this.unresolvedError); } emitUpdate(assets, reload = false) { if (this.unresolvedError) { this.unresolvedError = null; this.broadcast({ type: 'error-resolved' }); } const shouldReload = reload || assets.some(asset => asset.hmrPageReload); if (shouldReload) { this.broadcast({ type: 'reload' }); } else { this.broadcast({ type: 'update', assets: assets.map(asset => { let deps = {}; for (let [dep, depAsset] of asset.depAssets) { deps[dep.name] = depAsset.id; } return { id: asset.id, type: asset.type, generated: asset.generated, deps: deps }; }) }); } } handleSocketError(err) { if (err.error.code === 'ECONNRESET') { // This gets triggered on page refresh, ignore this return; } logger.warn(err); } broadcast(msg) { const json = JSON.stringify(msg); for (let ws of this.wss.clients) { ws.send(json); } } } module.exports = HMRServer;