Skip to content

Commit 5b42f44

Browse files
atljsatya164
andauthored
feat(bob): support custom target definitions (#732)
### Summary This adds the `custom` target to bob. Users are able to pass arbitrary scripts via this target and bob will call those scripts. ### Test plan 1. Configure bob in a new project 2. Define the `custom` target 3. Define a script that generates some files 4. Call bob build and make sure the script was called with the right package manager --------- Co-authored-by: Satyajit Sahoo <[email protected]>
1 parent f06514d commit 5b42f44

File tree

4 files changed

+99
-1
lines changed

4 files changed

+99
-1
lines changed

docs/pages/build.md

+12
Original file line numberDiff line numberDiff line change
@@ -188,6 +188,18 @@ Generates the [React Native Codegen](https://reactnative.dev/docs/the-new-archit
188188

189189
You can ensure your Codegen generated scaffold code is stable through different React Native versions by shipping it with your library. You can find more in the [React Native Official Docs](https://reactnative.dev/docs/the-new-architecture/codegen-cli#including-generated-code-into-libraries).
190190

191+
#### `custom`
192+
193+
Define a custom build target. This is useful to call code generators during the build process.
194+
195+
##### `script`
196+
197+
Accepts a script name. `bob` will call the matching script defined under `package.json`'s `scripts` property. The build process **will throw and exit** if the target is defined without this option.
198+
199+
##### `clean`
200+
201+
You can pass a path to this option and `bob` will delete all the files on that path. The path is resolved relatively to where `build` was called from.
202+
191203
#### `commonjs`
192204

193205
Enable compiling source files with Babel and use CommonJS module system.

packages/react-native-builder-bob/src/index.ts

+9
Original file line numberDiff line numberDiff line change
@@ -11,6 +11,7 @@ import buildCommonJS from './targets/commonjs';
1111
import buildModule from './targets/module';
1212
import buildTypescript from './targets/typescript';
1313
import buildCodegen from './targets/codegen';
14+
import customTarget from './targets/custom';
1415
import type { Options, Report, Target } from './types';
1516

1617
type ArgName = 'target';
@@ -584,6 +585,14 @@ async function buildTarget(
584585
report,
585586
});
586587
break;
588+
case 'custom':
589+
await customTarget({
590+
options: targetOptions,
591+
source: path.resolve(root, source),
592+
report,
593+
root,
594+
});
595+
break;
587596
default:
588597
logger.exit(`Invalid target ${kleur.blue(targetName)}.`);
589598
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,72 @@
1+
import kleur from 'kleur';
2+
import path from 'path';
3+
import fs from 'fs-extra';
4+
import type { Input } from '../types';
5+
import { spawn } from '../utils/spawn';
6+
import dedent from 'dedent';
7+
import del from 'del';
8+
9+
type Options = Omit<Input, 'output'> & {
10+
options?: {
11+
script?: string;
12+
clean?: string;
13+
};
14+
};
15+
16+
export default async function customTarget({ options, root, report }: Options) {
17+
if (options?.script == null) {
18+
report.error(
19+
dedent(
20+
`No script was provided with the custom target.
21+
Example: ${kleur.green('{["custom", { "script": "generateTypes" }}')}`
22+
)
23+
);
24+
process.exit(1);
25+
}
26+
27+
const pathToClean = options.clean
28+
? path.relative(root, options.clean)
29+
: undefined;
30+
31+
if (pathToClean) {
32+
report.info(`Cleaning up ${kleur.blue(pathToClean)}`);
33+
34+
await del([path.resolve(root, pathToClean)]);
35+
}
36+
37+
const packageManagerExecutable = process.env.npm_execpath ?? 'npm';
38+
const packageManagerArgs = ['run', options.script];
39+
40+
// usr/bin/yarn -> yarn
41+
const packageManagerName = path.basename(packageManagerExecutable);
42+
report.info(
43+
`Running ${kleur.blue(packageManagerName)} ${kleur.blue(
44+
packageManagerArgs.join(' ')
45+
)}`
46+
);
47+
48+
try {
49+
await spawn(packageManagerExecutable, packageManagerArgs, {
50+
stdio: ['ignore', 'ignore', 'inherit'],
51+
});
52+
} catch (e) {
53+
report.error(
54+
`An error occurred when running ${kleur.blue(options.script)}`
55+
);
56+
process.exit(1);
57+
}
58+
59+
report.success(`Ran the ${kleur.blue(options.script)} script succesfully`);
60+
61+
if (options.clean && pathToClean && !(await fs.pathExists(pathToClean))) {
62+
report.warn(
63+
`Custom target with the ${kleur.blue(
64+
options.script
65+
)} script has ${kleur.blue(options.clean)} as the ${kleur.bold(
66+
'clean'
67+
)} option but this path wasn't created after running the script. Are you sure you've defined the ${kleur.bold(
68+
'clean'
69+
)} path correctly?`
70+
);
71+
}
72+
}

packages/react-native-builder-bob/src/types.ts

+6-1
Original file line numberDiff line numberDiff line change
@@ -13,7 +13,12 @@ export type Input = {
1313
report: Report;
1414
};
1515

16-
export type Target = 'commonjs' | 'module' | 'typescript' | 'codegen';
16+
export type Target =
17+
| 'commonjs'
18+
| 'module'
19+
| 'typescript'
20+
| 'codegen'
21+
| 'custom';
1722

1823
export type Options = {
1924
source?: string;

0 commit comments

Comments
 (0)