UNPKG

react-native

Version:

A framework for building native apps using React

189 lines (164 loc) • 6.25 kB
/** * 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 */ 'use strict'; 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);