mirror of
https://github.com/status-im/react-native.git
synced 2025-01-16 20:44:10 +00:00
Refactor attachHMRServer to use async/await
Reviewed By: davidaurelio Differential Revision: D5603552 fbshipit-source-id: 8fc5f9bcd35a6adf3e3cdf5de47384d8420601cd
This commit is contained in:
parent
2512a34bea
commit
f32d0eed17
@ -88,48 +88,53 @@ function attachHMRServer<TModule: Moduleish>(
|
|||||||
// - The full list of dependencies.
|
// - The full list of dependencies.
|
||||||
// - The shallow dependencies each file on the dependency list has
|
// - The shallow dependencies each file on the dependency list has
|
||||||
// - Inverse shallow dependencies map
|
// - Inverse shallow dependencies map
|
||||||
function getDependencies(platform: string, bundleEntry: string): Promise<{
|
async function getDependencies(platform: string, bundleEntry: string): Promise<{
|
||||||
dependenciesCache: Array<string>,
|
dependenciesCache: Array<string>,
|
||||||
dependenciesModulesCache: {[mixed]: TModule},
|
dependenciesModulesCache: {[mixed]: TModule},
|
||||||
shallowDependencies: {[string]: Array<TModule>},
|
shallowDependencies: {[string]: Array<TModule>},
|
||||||
inverseDependenciesCache: mixed,
|
inverseDependenciesCache: mixed,
|
||||||
resolutionResponse: ResolutionResponse<TModule>,
|
resolutionResponse: ResolutionResponse<TModule>,
|
||||||
}> {
|
}> {
|
||||||
return packagerServer.getDependencies({
|
const response = await packagerServer.getDependencies({
|
||||||
dev: true,
|
dev: true,
|
||||||
entryFile: bundleEntry,
|
entryFile: bundleEntry,
|
||||||
hot: true,
|
hot: true,
|
||||||
minify: false,
|
minify: false,
|
||||||
platform: platform,
|
platform: platform,
|
||||||
recursive: true,
|
recursive: true,
|
||||||
}).then(response => {
|
});
|
||||||
|
|
||||||
/* $FlowFixMe: getModuleId might be null */
|
/* $FlowFixMe: getModuleId might be null */
|
||||||
const {getModuleId}: {getModuleId: () => number} = response;
|
const {getModuleId}: {getModuleId: () => number} = response;
|
||||||
|
|
||||||
// for each dependency builds the object:
|
// for each dependency builds the object:
|
||||||
// `{path: '/a/b/c.js', deps: ['modA', 'modB', ...]}`
|
// `{path: '/a/b/c.js', deps: ['modA', 'modB', ...]}`
|
||||||
return Promise.all(response.dependencies.map((dep: TModule) => {
|
const deps: Array<{
|
||||||
return dep.getName().then(depName => {
|
path: string,
|
||||||
|
name?: string,
|
||||||
|
deps: Array<TModule>,
|
||||||
|
}> = await Promise.all(response.dependencies.map(async (dep: TModule) => {
|
||||||
|
const depName = await dep.getName();
|
||||||
|
|
||||||
if (dep.isAsset() || dep.isJSON()) {
|
if (dep.isAsset() || dep.isJSON()) {
|
||||||
return Promise.resolve({path: dep.path, deps: []});
|
return {path: dep.path, deps: []};
|
||||||
}
|
}
|
||||||
return packagerServer.getShallowDependencies({
|
const dependencies = await packagerServer.getShallowDependencies({
|
||||||
dev: true,
|
dev: true,
|
||||||
entryFile: dep.path,
|
entryFile: dep.path,
|
||||||
hot: true,
|
hot: true,
|
||||||
minify: false,
|
minify: false,
|
||||||
platform: platform,
|
platform: platform,
|
||||||
recursive: true,
|
recursive: true,
|
||||||
}).then(deps => {
|
});
|
||||||
|
|
||||||
return {
|
return {
|
||||||
path: dep.path,
|
path: dep.path,
|
||||||
name: depName,
|
name: depName,
|
||||||
deps,
|
deps: dependencies,
|
||||||
};
|
};
|
||||||
});
|
}));
|
||||||
});
|
|
||||||
}))
|
|
||||||
.then((deps: Array<{path: string, name?: string, deps: Array<TModule>}>) => {
|
|
||||||
// list with all the dependencies' filenames the bundle entry has
|
// list with all the dependencies' filenames the bundle entry has
|
||||||
const dependenciesCache = response.dependencies.map(dep => dep.path);
|
const dependenciesCache = response.dependencies.map(dep => dep.path);
|
||||||
|
|
||||||
@ -154,7 +159,6 @@ function attachHMRServer<TModule: Moduleish>(
|
|||||||
dependenciesModulesCache[getModuleId(dep)] = dep;
|
dependenciesModulesCache[getModuleId(dep)] = dep;
|
||||||
});
|
});
|
||||||
|
|
||||||
|
|
||||||
const inverseDependenciesCache = Object.create(null);
|
const inverseDependenciesCache = Object.create(null);
|
||||||
const inverseDependencies = getInverseDependencies(response);
|
const inverseDependencies = getInverseDependencies(response);
|
||||||
for (const [module, dependents] of inverseDependencies) {
|
for (const [module, dependents] of inverseDependencies) {
|
||||||
@ -169,167 +173,17 @@ function attachHMRServer<TModule: Moduleish>(
|
|||||||
inverseDependenciesCache,
|
inverseDependenciesCache,
|
||||||
resolutionResponse: response,
|
resolutionResponse: response,
|
||||||
};
|
};
|
||||||
});
|
|
||||||
});
|
|
||||||
}
|
}
|
||||||
|
|
||||||
const WebSocketServer = require('ws').Server;
|
async function prepareResponse(filename): Object {
|
||||||
const wss = new WebSocketServer({
|
try {
|
||||||
server: httpServer,
|
const bundle = await generateBundle(filename);
|
||||||
path: path,
|
|
||||||
});
|
|
||||||
|
|
||||||
wss.on('connection', ws => {
|
|
||||||
/* $FlowFixMe: url might be null */
|
|
||||||
const params = querystring.parse(url.parse(ws.upgradeReq.url).query);
|
|
||||||
|
|
||||||
getDependencies(params.platform, params.bundleEntry)
|
|
||||||
.then(({
|
|
||||||
dependenciesCache,
|
|
||||||
dependenciesModulesCache,
|
|
||||||
shallowDependencies,
|
|
||||||
inverseDependenciesCache,
|
|
||||||
}) => {
|
|
||||||
client = {
|
|
||||||
ws,
|
|
||||||
platform: params.platform,
|
|
||||||
bundleEntry: params.bundleEntry,
|
|
||||||
dependenciesCache,
|
|
||||||
dependenciesModulesCache,
|
|
||||||
shallowDependencies,
|
|
||||||
inverseDependenciesCache,
|
|
||||||
};
|
|
||||||
|
|
||||||
packagerServer.setHMRFileChangeListener((type, filename) => {
|
|
||||||
if (!client) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
const blacklisted = blacklist.find(blacklistedPath =>
|
|
||||||
filename.indexOf(blacklistedPath) !== -1
|
|
||||||
);
|
|
||||||
|
|
||||||
if (blacklisted) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
client.ws.send(JSON.stringify({type: 'update-start'}));
|
|
||||||
const promise = type === 'delete'
|
|
||||||
? Promise.resolve()
|
|
||||||
: packagerServer.getShallowDependencies({
|
|
||||||
dev: true,
|
|
||||||
minify: false,
|
|
||||||
entryFile: filename,
|
|
||||||
hot: true,
|
|
||||||
platform: client.platform,
|
|
||||||
recursive: true,
|
|
||||||
}).then(deps => {
|
|
||||||
if (!client) {
|
|
||||||
return [];
|
|
||||||
}
|
|
||||||
|
|
||||||
// if the file dependencies have change we need to invalidate the
|
|
||||||
// dependencies caches because the list of files we need to send
|
|
||||||
// to the client may have changed
|
|
||||||
const oldDependencies = client.shallowDependencies[filename];
|
|
||||||
if (arrayEquals(deps, oldDependencies)) {
|
|
||||||
// Need to create a resolution response to pass to the bundler
|
|
||||||
// to process requires after transform. By providing a
|
|
||||||
// specific response we can compute a non recursive one which
|
|
||||||
// is the least we need and improve performance.
|
|
||||||
return packagerServer.getDependencies({
|
|
||||||
dev: true,
|
|
||||||
entryFile: filename,
|
|
||||||
hot: true,
|
|
||||||
minify: false,
|
|
||||||
platform: client.platform,
|
|
||||||
recursive: true,
|
|
||||||
}).then(response => {
|
|
||||||
return packagerServer.getModuleForPath(filename).then(module => {
|
|
||||||
return response.copy({dependencies: [module]});
|
|
||||||
});
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
// if there're new dependencies compare the full list of
|
|
||||||
// dependencies we used to have with the one we now have
|
|
||||||
return getDependencies(client.platform, client.bundleEntry)
|
|
||||||
.then(({
|
|
||||||
dependenciesCache: depsCache,
|
|
||||||
dependenciesModulesCache: depsModulesCache,
|
|
||||||
shallowDependencies: shallowDeps,
|
|
||||||
inverseDependenciesCache: inverseDepsCache,
|
|
||||||
resolutionResponse,
|
|
||||||
}) => {
|
|
||||||
if (!client) {
|
|
||||||
return {};
|
|
||||||
}
|
|
||||||
|
|
||||||
const nonNullClient = client;
|
|
||||||
|
|
||||||
return packagerServer.getModuleForPath(filename).then(moduleToUpdate => {
|
|
||||||
// build list of modules for which we'll send HMR updates
|
|
||||||
const modulesToUpdate = [moduleToUpdate];
|
|
||||||
Object.keys(depsModulesCache).forEach(module => {
|
|
||||||
if (!nonNullClient.dependenciesModulesCache[module]) {
|
|
||||||
modulesToUpdate.push(depsModulesCache[module]);
|
|
||||||
}
|
|
||||||
});
|
|
||||||
|
|
||||||
// Need to send modules to the client in an order it can
|
|
||||||
// process them: if a new dependency graph was uncovered
|
|
||||||
// because a new dependency was added, the file that was
|
|
||||||
// changed, which is the root of the dependency tree that
|
|
||||||
// will be sent, needs to be the last module that gets
|
|
||||||
// processed. Reversing the new modules makes sense
|
|
||||||
// because we get them through the resolver which returns
|
|
||||||
// a BFS ordered list.
|
|
||||||
modulesToUpdate.reverse();
|
|
||||||
|
|
||||||
// invalidate caches
|
|
||||||
nonNullClient.dependenciesCache = depsCache;
|
|
||||||
nonNullClient.dependenciesModulesCache = depsModulesCache;
|
|
||||||
nonNullClient.shallowDependencies = shallowDeps;
|
|
||||||
nonNullClient.inverseDependenciesCache = inverseDepsCache;
|
|
||||||
|
|
||||||
return resolutionResponse.copy({
|
|
||||||
dependencies: modulesToUpdate
|
|
||||||
});
|
|
||||||
});
|
|
||||||
});
|
|
||||||
})
|
|
||||||
.then((resolutionResponse) => {
|
|
||||||
if (!client) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
// make sure the file was modified is part of the bundle
|
|
||||||
if (!client.shallowDependencies[filename]) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
const httpServerAddress = httpServer.address();
|
|
||||||
|
|
||||||
// Sanitize the value from the HTTP server
|
|
||||||
let packagerHost = 'localhost';
|
|
||||||
if (httpServer.address().address &&
|
|
||||||
httpServer.address().address !== '::' &&
|
|
||||||
httpServer.address().address !== '') {
|
|
||||||
packagerHost = httpServerAddress.address;
|
|
||||||
}
|
|
||||||
|
|
||||||
return packagerServer.buildBundleForHMR({
|
|
||||||
entryFile: client.bundleEntry,
|
|
||||||
platform: client.platform,
|
|
||||||
resolutionResponse,
|
|
||||||
}, packagerHost, httpServerAddress.port);
|
|
||||||
})
|
|
||||||
.then((bundle: HMRBundle) => {
|
|
||||||
if (!client || !bundle || bundle.isEmpty()) {
|
if (!client || !bundle || bundle.isEmpty()) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
return JSON.stringify({
|
return {
|
||||||
type: 'update',
|
type: 'update',
|
||||||
body: {
|
body: {
|
||||||
modules: bundle.getModulesIdsAndCode(),
|
modules: bundle.getModulesIdsAndCode(),
|
||||||
@ -337,9 +191,8 @@ function attachHMRServer<TModule: Moduleish>(
|
|||||||
sourceURLs: bundle.getSourceURLs(),
|
sourceURLs: bundle.getSourceURLs(),
|
||||||
sourceMappingURLs: bundle.getSourceMappingURLs(),
|
sourceMappingURLs: bundle.getSourceMappingURLs(),
|
||||||
},
|
},
|
||||||
});
|
};
|
||||||
})
|
} catch (error) {
|
||||||
.catch(error => {
|
|
||||||
// send errors to the client instead of killing packager server
|
// send errors to the client instead of killing packager server
|
||||||
let body;
|
let body;
|
||||||
if (error.type === 'TransformError' ||
|
if (error.type === 'TransformError' ||
|
||||||
@ -360,20 +213,181 @@ function attachHMRServer<TModule: Moduleish>(
|
|||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
return JSON.stringify({type: 'error', body});
|
return {type: 'error', body};
|
||||||
})
|
}
|
||||||
.then(update => {
|
}
|
||||||
if (!client || !update) {
|
|
||||||
|
async function generateBundle(filename) {
|
||||||
|
if (client === null) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
client.ws.send(update);
|
const deps = await packagerServer.getShallowDependencies({
|
||||||
|
dev: true,
|
||||||
|
minify: false,
|
||||||
|
entryFile: filename,
|
||||||
|
hot: true,
|
||||||
|
platform: client.platform,
|
||||||
|
recursive: true,
|
||||||
});
|
});
|
||||||
|
|
||||||
promise.then(() => {
|
if (client === null) {
|
||||||
/* $FlowFixMe: assume `client` non-null */
|
return;
|
||||||
client.ws.send(JSON.stringify({type: 'update-done'}));
|
}
|
||||||
|
|
||||||
|
// if the file dependencies have change we need to invalidate the
|
||||||
|
// dependencies caches because the list of files we need to send
|
||||||
|
// to the client may have changed
|
||||||
|
const oldDependencies = client.shallowDependencies[filename];
|
||||||
|
|
||||||
|
let resolutionResponse;
|
||||||
|
|
||||||
|
if (arrayEquals(deps, oldDependencies)) {
|
||||||
|
// Need to create a resolution response to pass to the bundler
|
||||||
|
// to process requires after transform. By providing a
|
||||||
|
// specific response we can compute a non recursive one which
|
||||||
|
// is the least we need and improve performance.
|
||||||
|
const response = await packagerServer.getDependencies({
|
||||||
|
dev: true,
|
||||||
|
entryFile: filename,
|
||||||
|
hot: true,
|
||||||
|
minify: false,
|
||||||
|
platform: client.platform,
|
||||||
|
recursive: true,
|
||||||
});
|
});
|
||||||
|
|
||||||
|
const module = await packagerServer.getModuleForPath(filename);
|
||||||
|
|
||||||
|
resolutionResponse = await response.copy({dependencies: [module]});
|
||||||
|
} else {
|
||||||
|
// if there're new dependencies compare the full list of
|
||||||
|
// dependencies we used to have with the one we now have
|
||||||
|
const {
|
||||||
|
dependenciesCache: depsCache,
|
||||||
|
dependenciesModulesCache: depsModulesCache,
|
||||||
|
shallowDependencies: shallowDeps,
|
||||||
|
inverseDependenciesCache: inverseDepsCache,
|
||||||
|
resolutionResponse: myResolutionReponse,
|
||||||
|
} = await getDependencies(client.platform, client.bundleEntry);
|
||||||
|
|
||||||
|
if (client === null) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
const moduleToUpdate = await packagerServer.getModuleForPath(filename);
|
||||||
|
|
||||||
|
if (client === null) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
// build list of modules for which we'll send HMR updates
|
||||||
|
const modulesToUpdate = [moduleToUpdate];
|
||||||
|
Object.keys(depsModulesCache).forEach(module => {
|
||||||
|
if (!client || !client.dependenciesModulesCache[module]) {
|
||||||
|
modulesToUpdate.push(depsModulesCache[module]);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
// Need to send modules to the client in an order it can
|
||||||
|
// process them: if a new dependency graph was uncovered
|
||||||
|
// because a new dependency was added, the file that was
|
||||||
|
// changed, which is the root of the dependency tree that
|
||||||
|
// will be sent, needs to be the last module that gets
|
||||||
|
// processed. Reversing the new modules makes sense
|
||||||
|
// because we get them through the resolver which returns
|
||||||
|
// a BFS ordered list.
|
||||||
|
modulesToUpdate.reverse();
|
||||||
|
|
||||||
|
// invalidate caches
|
||||||
|
client.dependenciesCache = depsCache;
|
||||||
|
client.dependenciesModulesCache = depsModulesCache;
|
||||||
|
client.shallowDependencies = shallowDeps;
|
||||||
|
client.inverseDependenciesCache = inverseDepsCache;
|
||||||
|
|
||||||
|
resolutionResponse = await myResolutionReponse.copy({
|
||||||
|
dependencies: modulesToUpdate,
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
// make sure the file was modified is part of the bundle
|
||||||
|
if (!client || !client.shallowDependencies[filename]) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
const httpServerAddress = httpServer.address();
|
||||||
|
|
||||||
|
// Sanitize the value from the HTTP server
|
||||||
|
let packagerHost = 'localhost';
|
||||||
|
if (httpServer.address().address &&
|
||||||
|
httpServer.address().address !== '::' &&
|
||||||
|
httpServer.address().address !== '') {
|
||||||
|
packagerHost = httpServerAddress.address;
|
||||||
|
}
|
||||||
|
|
||||||
|
const bundle: HMRBundle = await packagerServer.buildBundleForHMR(
|
||||||
|
{
|
||||||
|
entryFile: client.bundleEntry,
|
||||||
|
platform: client.platform,
|
||||||
|
resolutionResponse,
|
||||||
|
},
|
||||||
|
packagerHost,
|
||||||
|
httpServerAddress.port,
|
||||||
|
);
|
||||||
|
|
||||||
|
return bundle;
|
||||||
|
}
|
||||||
|
|
||||||
|
const WebSocketServer = require('ws').Server;
|
||||||
|
const wss = new WebSocketServer({
|
||||||
|
server: httpServer,
|
||||||
|
path: path,
|
||||||
|
});
|
||||||
|
|
||||||
|
wss.on('connection', async ws => {
|
||||||
|
/* $FlowFixMe: url might be null */
|
||||||
|
const params = querystring.parse(url.parse(ws.upgradeReq.url).query);
|
||||||
|
|
||||||
|
try {
|
||||||
|
const {
|
||||||
|
dependenciesCache,
|
||||||
|
dependenciesModulesCache,
|
||||||
|
shallowDependencies,
|
||||||
|
inverseDependenciesCache,
|
||||||
|
} = await getDependencies(params.platform, params.bundleEntry);
|
||||||
|
|
||||||
|
client = {
|
||||||
|
ws,
|
||||||
|
platform: params.platform,
|
||||||
|
bundleEntry: params.bundleEntry,
|
||||||
|
dependenciesCache,
|
||||||
|
dependenciesModulesCache,
|
||||||
|
shallowDependencies,
|
||||||
|
inverseDependenciesCache,
|
||||||
|
};
|
||||||
|
|
||||||
|
packagerServer.setHMRFileChangeListener(async (type, filename) => {
|
||||||
|
if (client === null) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
const blacklisted = blacklist.find(
|
||||||
|
blacklistedPath => filename.indexOf(blacklistedPath) !== -1,
|
||||||
|
);
|
||||||
|
if (blacklisted) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
client.ws.send(JSON.stringify({type: 'update-start'}));
|
||||||
|
|
||||||
|
if (type !== 'delete') {
|
||||||
|
const response = await prepareResponse(filename);
|
||||||
|
|
||||||
|
if (client && response) {
|
||||||
|
client.ws.send(JSON.stringify(response));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
client.ws.send(JSON.stringify({type: 'update-done'}));
|
||||||
});
|
});
|
||||||
|
|
||||||
client.ws.on('error', e => {
|
client.ws.on('error', e => {
|
||||||
@ -382,10 +396,9 @@ function attachHMRServer<TModule: Moduleish>(
|
|||||||
});
|
});
|
||||||
|
|
||||||
client.ws.on('close', () => disconnect());
|
client.ws.on('close', () => disconnect());
|
||||||
})
|
} catch (err) {
|
||||||
.catch(err => {
|
|
||||||
throw err;
|
throw err;
|
||||||
});
|
}
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
|
Loading…
x
Reference in New Issue
Block a user