react-native
Version:
A framework for building native apps using React
189 lines (164 loc) • 6.25 kB
JavaScript
/**
* Copyright (c) Meta Platforms, Inc. and affiliates.
*
* This source code is licensed under the MIT license found in the
* LICENSE file in the root directory of this source tree.
*
* @flow strict-local
* @format
*/
;
const {spawnSync} = require('child_process');
const fs = require('fs');
const os = require('os');
const path = require('path');
const yargs = require('yargs');
const LAST_BUILD_FILENAME = 'React-Core-prebuilt/.last_build_configuration';
function validateBuildConfiguration(configuration /*: string */) {
if (!['Debug', 'Release'].includes(configuration)) {
throw new Error(`Invalid configuration ${configuration}`);
}
}
function validateVersion(version /*: ?string */) {
if (version == null || version === '') {
throw new Error('Version cannot be empty');
}
}
function shouldReplaceRnCoreConfiguration(configuration /*: string */) {
const fileExists = fs.existsSync(LAST_BUILD_FILENAME);
if (fileExists) {
console.log(`Found ${LAST_BUILD_FILENAME} file`);
const oldConfiguration = fs.readFileSync(LAST_BUILD_FILENAME).toString();
if (oldConfiguration === configuration) {
console.log(
'Same config of the previous build. No need to replace React-Core-prebuilt',
);
return false;
}
}
// Assumption: if there is no stored last build, we assume that it was build for debug.
if (!fileExists && configuration === 'Debug') {
console.log(
'No previous build detected, but Debug Configuration. No need to replace React-Core-prebuilt',
);
return false;
}
return true;
}
function replaceRNCoreConfiguration(
configuration /*: string */,
version /*: string */,
podsRoot /*: string */,
) {
// Filename comes from rncore.rb
const tarballURLPath = `${podsRoot}/ReactNativeCore-artifacts/reactnative-core-${version.toLowerCase()}-${configuration.toLowerCase()}.tar.gz`;
const finalLocation = 'React-Core-prebuilt';
// Extract to a temporary directory on a regular filesystem first, then move
// into the final location. This avoids issues with partial tar extraction on
// certain filesystems (e.g. EdenFS) where extracting directly can silently
// produce incomplete results.
const tmpDir = fs.mkdtempSync(path.join(os.tmpdir(), 'rncore-'));
const tmpExtractDir = path.join(tmpDir, 'React-Core-prebuilt');
fs.mkdirSync(tmpExtractDir, {recursive: true});
try {
console.log('Extracting the tarball to temp dir', tarballURLPath);
const result = spawnSync(
'tar',
['-xf', tarballURLPath, '-C', tmpExtractDir],
{
stdio: 'inherit',
},
);
if (result.status !== 0) {
throw new Error(`tar extraction failed with exit code ${result.status}`);
}
// Verify extraction produced the expected xcframework structure
const xcfwPath = path.join(tmpExtractDir, 'React.xcframework');
const modulemapPath = path.join(xcfwPath, 'Modules', 'module.modulemap');
if (!fs.existsSync(modulemapPath)) {
throw new Error(
`Extraction verification failed: ${modulemapPath} not found`,
);
}
// Delete all directories in finalLocation - not files, since we want to
// keep the React-VFS.yaml file
const dirs = fs
.readdirSync(finalLocation, {withFileTypes: true})
.filter(dirent => dirent.isDirectory());
for (const dirent of dirs) {
const direntName =
typeof dirent.name === 'string' ? dirent.name : dirent.name.toString();
const dirPath = `${finalLocation}/${direntName}`;
console.log('Removing directory', dirPath);
fs.rmSync(dirPath, {force: true, recursive: true});
}
// Move extracted directories from temp to final location
const extractedEntries = fs
.readdirSync(tmpExtractDir, {withFileTypes: true})
.filter(dirent => dirent.isDirectory());
for (const dirent of extractedEntries) {
const direntName =
typeof dirent.name === 'string' ? dirent.name : dirent.name.toString();
const src = path.join(tmpExtractDir, direntName);
const dst = path.join(finalLocation, direntName);
const mvResult = spawnSync('mv', [src, dst], {stdio: 'inherit'});
if (mvResult.status !== 0) {
// Fallback: copy recursively then remove source
console.log(`mv failed for ${direntName}, falling back to cp -R`);
const cpResult = spawnSync('cp', ['-R', src, dst], {
stdio: 'inherit',
});
if (cpResult.status !== 0) {
throw new Error(
`cp fallback failed with exit code ${cpResult.status}`,
);
}
}
}
} finally {
// Clean up temp directory
fs.rmSync(tmpDir, {force: true, recursive: true});
}
}
function updateLastBuildConfiguration(configuration /*: string */) {
console.log(`Updating ${LAST_BUILD_FILENAME} with ${configuration}`);
fs.writeFileSync(LAST_BUILD_FILENAME, configuration);
}
function main(
configuration /*: string */,
version /*: string */,
podsRoot /*: string */,
) {
validateBuildConfiguration(configuration);
validateVersion(version);
if (!shouldReplaceRnCoreConfiguration(configuration)) {
return;
}
replaceRNCoreConfiguration(configuration, version, podsRoot);
updateLastBuildConfiguration(configuration);
console.log('Done replacing React Native prebuilt');
}
// This script is executed in the Pods folder, which is usually not synched to Github, so it should be ok
const argv = yargs
.option('c', {
alias: 'configuration',
description:
'Configuration to use to download the right React-Core prebuilt version. Allowed values are "Debug" and "Release".',
})
.option('r', {
alias: 'reactNativeVersion',
description:
'The Version of React Native associated with the React-Core prebuilt tarball.',
})
.option('p', {
alias: 'podsRoot',
description: 'The path to the Pods root folder',
})
.usage('Usage: $0 -c Debug -r <version> -p <path/to/react-native>').argv;
// $FlowFixMe[prop-missing]
const configuration = argv.configuration;
// $FlowFixMe[prop-missing]
const version = argv.reactNativeVersion;
// $FlowFixMe[prop-missing]
const podsRoot = argv.podsRoot;
main(configuration, version, podsRoot);