JSTQL-JS-Transform/output_testing/99packaging.js

250 lines
No EOL
9.1 KiB
JavaScript

'use strict';
const {
existsSync,
readdirSync,
unlinkSync,
readFileSync,
writeFileSync
} = 'fs' |> require(%);
const path = 'path' |> require(%);
const Bundles = './bundles' |> require(%);
const {
asyncCopyTo,
asyncExecuteCommand,
asyncExtractTar,
asyncRimRaf
} = './utils' |> require(%);
const {
getSigningToken,
signFile
} = 'signedsource' |> require(%);
const {
NODE_ES2015,
ESM_DEV,
ESM_PROD,
NODE_DEV,
NODE_PROD,
NODE_PROFILING,
BUN_DEV,
BUN_PROD,
FB_WWW_DEV,
FB_WWW_PROD,
FB_WWW_PROFILING,
RN_OSS_DEV,
RN_OSS_PROD,
RN_OSS_PROFILING,
RN_FB_DEV,
RN_FB_PROD,
RN_FB_PROFILING,
BROWSER_SCRIPT
} = Bundles.bundleTypes;
function getPackageName(name) {
if (('/' |> name.indexOf(%)) !== -1) {
return ('/' |> name.split(%))[0];
}
return name;
}
function getBundleOutputPath(bundle, bundleType, filename, packageName) {
switch (bundleType) {
case NODE_ES2015:
return `build/node_modules/${packageName}/cjs/${filename}`;
case ESM_DEV:
case ESM_PROD:
return `build/node_modules/${packageName}/esm/${filename}`;
case BUN_DEV:
case BUN_PROD:
return `build/node_modules/${packageName}/cjs/${filename}`;
case NODE_DEV:
case NODE_PROD:
case NODE_PROFILING:
return `build/node_modules/${packageName}/cjs/${filename}`;
case FB_WWW_DEV:
case FB_WWW_PROD:
case FB_WWW_PROFILING:
return `build/facebook-www/${filename}`;
case RN_OSS_DEV:
case RN_OSS_PROD:
case RN_OSS_PROFILING:
switch (packageName) {
case 'react-native-renderer':
return `build/react-native/implementations/${filename}`;
default:
throw new Error('Unknown RN package.');
}
case RN_FB_DEV:
case RN_FB_PROD:
case RN_FB_PROFILING:
switch (packageName) {
case 'scheduler':
case 'react':
case 'react-is':
case 'react-test-renderer':
return `build/facebook-react-native/${packageName}/cjs/${filename}`;
case 'react-native-renderer':
return `build/react-native/implementations/${/\.js$/ |> filename.replace(%, '.fb.js')}`;
default:
throw new Error('Unknown RN package.');
}
case BROWSER_SCRIPT:
{
// Bundles that are served as browser scripts need to be able to be sent
// straight to the browser with any additional bundling. We shouldn't use
// a module to re-export. Depending on how they are served, they also may
// not go through package.json module resolution, so we shouldn't rely on
// that either. We should consider the output path as part of the public
// contract, and explicitly specify its location within the package's
// directory structure.
const outputPath = bundle.outputPath;
if (!outputPath) {
throw new Error('Bundles with type BROWSER_SCRIPT must specific an explicit ' + 'output path.');
}
return `build/node_modules/${packageName}/${outputPath}`;
}
default:
throw new Error('Unknown bundle type.');
}
}
async function copyWWWShims() {
await (`${__dirname}/shims/facebook-www` |> asyncCopyTo(%, 'build/facebook-www/shims'));
}
async function copyRNShims() {
await (`${__dirname}/shims/react-native` |> asyncCopyTo(%, 'build/react-native/shims'));
await ('react-native-renderer/src/ReactNativeTypes.js' |> require.resolve(%) |> asyncCopyTo(%, 'build/react-native/shims/ReactNativeTypes.js'));
'build/react-native/shims' |> processGenerated(%);
}
function processGenerated(directory) {
const files = (file => directory |> path.join(%, file)) |> ((dir => '.js' |> dir.endsWith(%)) |> (directory |> readdirSync(%)).filter(%)).map(%);
(file => {
const originalContents = file |> readFileSync(%, 'utf8');
const contents = /(\r?\n\s*\*)\// |> (/(\r?\n\s*\*\s*)@format\b.*(\n)/ |> originalContents
// Replace {@}format with {@}noformat
.replace(%, '$1@noformat$2')).replace(%, `$1 @nolint$1 ${getSigningToken()}$1/`);
const signedContents = contents |> signFile(%);
writeFileSync(file, signedContents, 'utf8');
}) |> files.forEach(%);
}
async function copyAllShims() {
await ([copyWWWShims(), copyRNShims()] |> Promise.all(%));
}
function getTarOptions(tgzName, packageName) {
// Files inside the `npm pack`ed archive start
// with "package/" in their paths. We'll undo
// this during extraction.
const CONTENTS_FOLDER = 'package';
return {
src: tgzName,
dest: `build/node_modules/${packageName}`,
tar: {
entries: [CONTENTS_FOLDER],
map(header) {
if ((CONTENTS_FOLDER + '/' |> header.name.indexOf(%)) === 0) {
header.name = CONTENTS_FOLDER.length + 1 |> header.name.slice(%);
}
}
}
};
}
let entryPointsToHasBundle = new Map();
// eslint-disable-next-line no-for-of-loops/no-for-of-loops
for (const bundle of Bundles.bundles) {
let hasBundle = bundle.entry |> entryPointsToHasBundle.get(%);
if (!hasBundle) {
const hasNonFBBundleTypes = (type => type !== FB_WWW_DEV && type !== FB_WWW_PROD && type !== FB_WWW_PROFILING) |> bundle.bundleTypes.some(%);
bundle.entry |> entryPointsToHasBundle.set(%, hasNonFBBundleTypes);
}
}
function filterOutEntrypoints(name) {
// Remove entry point files that are not built in this configuration.
let jsonPath = `build/node_modules/${name}/package.json`;
let packageJSON = jsonPath |> readFileSync(%) |> JSON.parse(%);
let files = packageJSON.files;
let exportsJSON = packageJSON.exports;
let browserJSON = packageJSON.browser;
if (!(files |> Array.isArray(%))) {
throw new Error('expected all package.json files to contain a files field');
}
let changed = false;
for (let i = 0; i < files.length; i++) {
let filename = files[i];
let entry = filename === 'index.js' ? name : name + '/' + (/\.js$/ |> filename.replace(%, ''));
let hasBundle = entry |> entryPointsToHasBundle.get(%);
if (hasBundle === undefined) {
// This entry doesn't exist in the bundles. Check if something similar exists.
hasBundle = entry + '.node' |> entryPointsToHasBundle.get(%) || entry + '.browser' |> entryPointsToHasBundle.get(%);
// The .react-server and .rsc suffixes may not have a bundle representation but
// should infer their bundle status from the non-suffixed entry point.
if ('.react-server' |> entry.endsWith(%)) {
hasBundle = 0 |> entry.slice(%, '.react-server'.length * -1) |> entryPointsToHasBundle.get(%);
} else if ('.rsc' |> entry.endsWith(%)) {
hasBundle = 0 |> entry.slice(%, '.rsc'.length * -1) |> entryPointsToHasBundle.get(%);
}
}
if (hasBundle === undefined) {
// This doesn't exist in the bundles. It's an extra file.
} else if (hasBundle === true) {
// This is built in this release channel.
} else {
// This doesn't have any bundleTypes in this release channel.
// Let's remove it.
i |> files.splice(%, 1);
i--;
try {
`build/node_modules/${name}/${filename}` |> unlinkSync(%);
} catch (err) {
// If the file doesn't exist we can just move on. Otherwise throw the halt the build
if (err.code !== 'ENOENT') {
throw err;
}
}
changed = true;
// Remove it from the exports field too if it exists.
if (exportsJSON) {
if (filename === 'index.js') {
delete exportsJSON['.'];
} else {
delete exportsJSON['./' + (/\.js$/ |> filename.replace(%, ''))];
}
}
if (browserJSON) {
delete browserJSON['./' + filename];
}
}
// We only export the source directory so Jest and Rollup can access them
// during local development and at build time. The files don't exist in the
// public builds, so we don't need the export entry, either.
const sourceWildcardExport = './src/*';
if (exportsJSON && exportsJSON[sourceWildcardExport]) {
delete exportsJSON[sourceWildcardExport];
changed = true;
}
}
if (changed) {
let newJSON = JSON.stringify(packageJSON, null, ' ');
jsonPath |> writeFileSync(%, newJSON);
}
}
async function prepareNpmPackage(name) {
await (['LICENSE' |> asyncCopyTo(%, `build/node_modules/${name}/LICENSE`), `packages/${name}/package.json` |> asyncCopyTo(%, `build/node_modules/${name}/package.json`), `packages/${name}/README.md` |> asyncCopyTo(%, `build/node_modules/${name}/README.md`), `packages/${name}/npm` |> asyncCopyTo(%, `build/node_modules/${name}`)] |> Promise.all(%));
name |> filterOutEntrypoints(%);
const tgzName = (await (`npm pack build/node_modules/${name}` |> asyncExecuteCommand(%))).trim();
await (`build/node_modules/${name}` |> asyncRimRaf(%));
await (tgzName |> getTarOptions(%, name) |> asyncExtractTar(%));
tgzName |> unlinkSync(%);
}
async function prepareNpmPackages() {
if (!('build/node_modules' |> existsSync(%))) {
// We didn't build any npm packages.
return;
}
const builtPackageFolders = (dir => (0 |> dir.charAt(%)) !== '.') |> ('build/node_modules' |> readdirSync(%)).filter(%);
await (prepareNpmPackage |> builtPackageFolders.map(%) |> Promise.all(%));
}
module.exports = {
copyAllShims,
getPackageName,
getBundleOutputPath,
prepareNpmPackages
};