react-saasify-chrisvxd
Version:
React components for Saasify web clients.
116 lines (98 loc) • 2.79 kB
JavaScript
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;