Summary: 1) moved "sonar/desktop/src" to "sonar/desktop/app/src", so "app" is now a separate package containing the core Flipper app code 2) Configured yarn workspaces with the root in "sonar/desktop": app, static, pkg, doctor, headless-tests. Plugins are not included for now, I plan to do this later. Reviewed By: jknoxville Differential Revision: D20535782 fbshipit-source-id: 600b2301960f37c7d72166e0d04eba462bec9fc1
217 lines
5.9 KiB
TypeScript
217 lines
5.9 KiB
TypeScript
/**
|
|
* Copyright (c) Facebook, Inc. and its affiliates.
|
|
*
|
|
* This source code is licensed under the MIT license found in the
|
|
* LICENSE file in the root directory of this source tree.
|
|
*
|
|
* @format
|
|
*/
|
|
|
|
import Metro from 'metro';
|
|
import compilePlugins from '../static/compilePlugins';
|
|
import util from 'util';
|
|
import tmp from 'tmp';
|
|
import path from 'path';
|
|
import fs from 'fs-extra';
|
|
import {spawn} from 'promisify-child-process';
|
|
import recursiveReaddir from 'recursive-readdir';
|
|
import {default as getWatchFolders} from '../static/get-watch-folders';
|
|
import {appDir, staticDir, pluginsDir, headlessDir} from './paths';
|
|
|
|
async function mostRecentlyChanged(
|
|
dir: string,
|
|
ignores: string[],
|
|
): Promise<Date> {
|
|
const files = await util.promisify<string, string[], string[]>(
|
|
recursiveReaddir,
|
|
)(dir, ignores);
|
|
return files
|
|
.map(f => fs.lstatSync(f).ctime)
|
|
.reduce((a, b) => (a > b ? a : b), new Date(0));
|
|
}
|
|
|
|
export function die(err: Error) {
|
|
console.error(err.stack);
|
|
process.exit(1);
|
|
}
|
|
|
|
export function compileDefaultPlugins(
|
|
defaultPluginDir: string,
|
|
skipAll: boolean = false,
|
|
) {
|
|
return compilePlugins(
|
|
null,
|
|
skipAll ? [] : [pluginsDir, path.join(pluginsDir, 'fb')],
|
|
defaultPluginDir,
|
|
{force: true, failSilently: false, recompileOnChanges: false},
|
|
)
|
|
.then(defaultPlugins =>
|
|
fs.writeFileSync(
|
|
path.join(defaultPluginDir, 'index.json'),
|
|
JSON.stringify(
|
|
defaultPlugins.map(({entry, rootDir, out, ...plugin}) => ({
|
|
...plugin,
|
|
out: path.parse(out).base,
|
|
})),
|
|
),
|
|
),
|
|
)
|
|
.catch(die);
|
|
}
|
|
|
|
async function compile(
|
|
buildFolder: string,
|
|
projectRoot: string,
|
|
watchFolders: string[],
|
|
entry: string,
|
|
) {
|
|
await Metro.runBuild(
|
|
{
|
|
reporter: {update: () => {}},
|
|
projectRoot,
|
|
watchFolders,
|
|
serializer: {},
|
|
transformer: {
|
|
babelTransformerPath: path.join(staticDir, 'transforms', 'index.js'),
|
|
},
|
|
resolver: {
|
|
resolverMainFields: ['flipper:source', 'module', 'main'],
|
|
blacklistRE: /\.native\.js$/,
|
|
},
|
|
},
|
|
{
|
|
dev: false,
|
|
minify: false,
|
|
resetCache: true,
|
|
sourceMap: true,
|
|
entry,
|
|
out: path.join(buildFolder, 'bundle.js'),
|
|
},
|
|
);
|
|
}
|
|
|
|
export async function compileHeadless(buildFolder: string) {
|
|
console.log(`⚙️ Compiling headless bundle...`);
|
|
const watchFolders = [
|
|
headlessDir,
|
|
...(await getWatchFolders(staticDir)),
|
|
...(await getWatchFolders(appDir)),
|
|
path.join(pluginsDir, 'navigation'),
|
|
path.join(pluginsDir, 'fb', 'layout', 'sidebar_extensions'),
|
|
path.join(pluginsDir, 'fb', 'mobileconfig'),
|
|
path.join(pluginsDir, 'fb', 'watch'),
|
|
].filter(fs.pathExists);
|
|
try {
|
|
await compile(
|
|
buildFolder,
|
|
headlessDir,
|
|
watchFolders,
|
|
path.join(headlessDir, 'index.tsx'),
|
|
);
|
|
console.log('✅ Compiled headless bundle.');
|
|
} catch (err) {
|
|
die(err);
|
|
}
|
|
}
|
|
|
|
export async function compileRenderer(buildFolder: string) {
|
|
console.log(`⚙️ Compiling renderer bundle...`);
|
|
const watchFolders = [
|
|
...(await getWatchFolders(appDir)),
|
|
path.join(pluginsDir, 'navigation'),
|
|
path.join(pluginsDir, 'fb', 'layout', 'sidebar_extensions'),
|
|
path.join(pluginsDir, 'fb', 'mobileconfig'),
|
|
path.join(pluginsDir, 'fb', 'watch'),
|
|
].filter(fs.pathExists);
|
|
try {
|
|
await compile(
|
|
buildFolder,
|
|
appDir,
|
|
watchFolders,
|
|
path.join(appDir, 'src', 'init.tsx'),
|
|
);
|
|
console.log('✅ Compiled renderer bundle.');
|
|
} catch (err) {
|
|
die(err);
|
|
}
|
|
}
|
|
|
|
export async function compileMain({dev}: {dev: boolean}) {
|
|
const out = path.join(staticDir, 'main.bundle.js');
|
|
// check if main needs to be compiled
|
|
if (await fs.pathExists(out)) {
|
|
const staticDirCtime = await mostRecentlyChanged(staticDir, ['*.bundle.*']);
|
|
const bundleCtime = (await fs.lstat(out)).ctime;
|
|
if (staticDirCtime < bundleCtime) {
|
|
console.log(`🥫 Using cached version of main bundle...`);
|
|
return;
|
|
}
|
|
}
|
|
console.log(`⚙️ Compiling main bundle... ${staticDir}`);
|
|
try {
|
|
const config = Object.assign({}, await Metro.loadConfig(), {
|
|
reporter: {update: () => {}},
|
|
projectRoot: staticDir,
|
|
watchFolders: await getWatchFolders(staticDir),
|
|
transformer: {
|
|
babelTransformerPath: path.join(staticDir, 'transforms', 'index.js'),
|
|
},
|
|
resolver: {
|
|
sourceExts: ['tsx', 'ts', 'js'],
|
|
resolverMainFields: ['flipper:source', 'module', 'main'],
|
|
blacklistRE: /\.native\.js$/,
|
|
},
|
|
});
|
|
await Metro.runBuild(config, {
|
|
platform: 'web',
|
|
entry: path.join(staticDir, 'main.ts'),
|
|
out,
|
|
dev,
|
|
minify: false,
|
|
sourceMap: true,
|
|
resetCache: true,
|
|
});
|
|
console.log('✅ Compiled main bundle.');
|
|
} catch (err) {
|
|
die(err);
|
|
}
|
|
}
|
|
export function buildFolder(): Promise<string> {
|
|
// eslint-disable-next-line no-console
|
|
console.log('Creating build directory');
|
|
return new Promise<string>((resolve, reject) => {
|
|
tmp.dir({prefix: 'flipper-build-'}, (err, buildFolder) => {
|
|
if (err) {
|
|
reject(err);
|
|
} else {
|
|
resolve(buildFolder);
|
|
}
|
|
});
|
|
}).catch(e => {
|
|
die(e);
|
|
return '';
|
|
});
|
|
}
|
|
export function getVersionNumber() {
|
|
let {version} = require('../package.json');
|
|
const buildNumber = process.argv.join(' ').match(/--version=(\d+)/);
|
|
if (buildNumber && buildNumber.length > 0) {
|
|
version = [...version.split('.').slice(0, 2), buildNumber[1]].join('.');
|
|
}
|
|
return version;
|
|
}
|
|
|
|
// Asynchronously determine current mercurial revision as string or `null` in case of any error.
|
|
export function genMercurialRevision(): Promise<string | null> {
|
|
return spawn('hg', ['log', '-r', '.', '-T', '{node}'], {encoding: 'utf8'})
|
|
.then(
|
|
res =>
|
|
(res &&
|
|
(typeof res.stdout === 'string'
|
|
? res.stdout
|
|
: res.stdout?.toString())) ||
|
|
null,
|
|
)
|
|
.catch(() => null);
|
|
}
|