create-expo-module 0.0.0 → 0.2.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
package/.eslintrc.js ADDED
@@ -0,0 +1,2 @@
1
+ // @generated by expo-module-scripts
2
+ module.exports = require('expo-module-scripts/eslintrc.base.js');
package/README.md ADDED
@@ -0,0 +1 @@
1
+ # create-expo-module
@@ -0,0 +1,3 @@
1
+ #!/usr/bin/env node
2
+
3
+ require('../build/create-expo-module.js')
@@ -0,0 +1 @@
1
+ export {};
@@ -0,0 +1,236 @@
1
+ "use strict";
2
+ var __importDefault = (this && this.__importDefault) || function (mod) {
3
+ return (mod && mod.__esModule) ? mod : { "default": mod };
4
+ };
5
+ Object.defineProperty(exports, "__esModule", { value: true });
6
+ const spawn_async_1 = __importDefault(require("@expo/spawn-async"));
7
+ const commander_1 = require("commander");
8
+ const download_tarball_1 = __importDefault(require("download-tarball"));
9
+ const ejs_1 = __importDefault(require("ejs"));
10
+ const fs_extra_1 = __importDefault(require("fs-extra"));
11
+ const path_1 = __importDefault(require("path"));
12
+ const prompts_1 = __importDefault(require("prompts"));
13
+ const packageJson = require('../package.json');
14
+ // `yarn run` may change the current working dir, then we should use `INIT_CWD` env.
15
+ const CWD = process.env.INIT_CWD || process.cwd();
16
+ // Ignore some paths. Especially `package.json` as it is rendered
17
+ // from `$package.json` file instead of the original one.
18
+ const IGNORES_PATHS = ['.DS_Store', 'build', 'node_modules', 'package.json'];
19
+ /**
20
+ * The main function of the command.
21
+ *
22
+ * @param target Path to the directory where to create the module. Defaults to current working dir.
23
+ * @param command An object from `commander`.
24
+ */
25
+ async function main(target, options) {
26
+ const targetDir = target ? path_1.default.join(CWD, target) : CWD;
27
+ options.target = targetDir;
28
+ await fs_extra_1.default.ensureDir(targetDir);
29
+ const data = await askForSubstitutionDataAsync(targetDir, options);
30
+ const packageManager = await selectPackageManagerAsync();
31
+ const packagePath = options.source
32
+ ? path_1.default.join(CWD, options.source)
33
+ : await downloadPackageAsync(targetDir);
34
+ const files = await getFilesAsync(packagePath);
35
+ console.log('🎨 Creating Expo module from the template files...');
36
+ // Iterate through all template files.
37
+ for (const file of files) {
38
+ const renderedRelativePath = ejs_1.default.render(file.replace(/^\$/, ''), data, {
39
+ openDelimiter: '{',
40
+ closeDelimiter: '}',
41
+ escape: (value) => value.replace('.', path_1.default.sep),
42
+ });
43
+ const fromPath = path_1.default.join(packagePath, file);
44
+ const toPath = path_1.default.join(targetDir, renderedRelativePath);
45
+ const template = await fs_extra_1.default.readFile(fromPath, { encoding: 'utf8' });
46
+ const renderedContent = ejs_1.default.render(template, data);
47
+ await fs_extra_1.default.outputFile(toPath, renderedContent, { encoding: 'utf8' });
48
+ }
49
+ if (!options.source) {
50
+ // Files in the downloaded tarball are wrapped in `package` dir.
51
+ // We should remove it after all.
52
+ await fs_extra_1.default.remove(packagePath);
53
+ }
54
+ if (!options.withReadme) {
55
+ await fs_extra_1.default.remove(path_1.default.join(targetDir, 'README.md'));
56
+ }
57
+ if (!options.withChangelog) {
58
+ await fs_extra_1.default.remove(path_1.default.join(targetDir, 'CHANGELOG.md'));
59
+ }
60
+ // Install dependencies and build
61
+ await postActionsAsync(packageManager, targetDir);
62
+ console.log('✅ Successfully created Expo module');
63
+ }
64
+ /**
65
+ * Recursively scans for the files within the directory. Returned paths are relative to the `root` path.
66
+ */
67
+ async function getFilesAsync(root, dir = null) {
68
+ const files = [];
69
+ const baseDir = dir ? path_1.default.join(root, dir) : root;
70
+ for (const file of await fs_extra_1.default.readdir(baseDir)) {
71
+ const relativePath = dir ? path_1.default.join(dir, file) : file;
72
+ if (IGNORES_PATHS.includes(relativePath) || IGNORES_PATHS.includes(file)) {
73
+ continue;
74
+ }
75
+ const fullPath = path_1.default.join(baseDir, file);
76
+ const stat = await fs_extra_1.default.lstat(fullPath);
77
+ if (stat.isDirectory()) {
78
+ files.push(...(await getFilesAsync(root, relativePath)));
79
+ }
80
+ else {
81
+ files.push(relativePath);
82
+ }
83
+ }
84
+ return files;
85
+ }
86
+ /**
87
+ * Asks NPM registry for the url to the tarball.
88
+ */
89
+ async function getNpmTarballUrl(packageName, version = 'latest') {
90
+ const { stdout } = await (0, spawn_async_1.default)('npm', ['view', `${packageName}@${version}`, 'dist.tarball']);
91
+ return stdout.trim();
92
+ }
93
+ /**
94
+ * Gets the username of currently logged in user. Used as a default in the prompt asking for the module author.
95
+ */
96
+ async function npmWhoamiAsync(targetDir) {
97
+ try {
98
+ const { stdout } = await (0, spawn_async_1.default)('npm', ['whoami'], { cwd: targetDir });
99
+ return stdout.trim();
100
+ }
101
+ catch {
102
+ return null;
103
+ }
104
+ }
105
+ /**
106
+ * Downloads the template from NPM registry.
107
+ */
108
+ async function downloadPackageAsync(targetDir) {
109
+ const tarballUrl = await getNpmTarballUrl('expo-module-template');
110
+ console.log('⬇️ Downloading module template from npm...');
111
+ await (0, download_tarball_1.default)({
112
+ url: tarballUrl,
113
+ dir: targetDir,
114
+ });
115
+ return path_1.default.join(targetDir, 'package');
116
+ }
117
+ /**
118
+ * Asks whether to use Yarn or npm as a dependency package manager.
119
+ */
120
+ async function selectPackageManagerAsync() {
121
+ const { packageManager } = await (0, prompts_1.default)({
122
+ type: 'select',
123
+ name: 'packageManager',
124
+ message: 'Which package manager do you want to use to install dependencies?',
125
+ choices: [
126
+ { title: 'yarn', value: 'yarn' },
127
+ { title: 'npm', value: 'npm' },
128
+ ],
129
+ });
130
+ return packageManager;
131
+ }
132
+ /**
133
+ * Installs dependencies and builds TypeScript files.
134
+ */
135
+ async function postActionsAsync(packageManager, targetDir) {
136
+ async function run(...args) {
137
+ await (0, spawn_async_1.default)(packageManager, args, {
138
+ cwd: targetDir,
139
+ stdio: 'ignore',
140
+ });
141
+ }
142
+ console.log('📦 Installing dependencies...');
143
+ await run('install');
144
+ console.log('🛠 Compiling TypeScript files...');
145
+ await run('run', 'build');
146
+ }
147
+ /**
148
+ * Asks the user for some data necessary to render the template.
149
+ * Some values may already be provided by command options, the prompt is skipped in that case.
150
+ */
151
+ async function askForSubstitutionDataAsync(targetDir, options) {
152
+ var _a, _b;
153
+ const defaultPackageSlug = path_1.default.basename(targetDir);
154
+ const defaultProjectName = defaultPackageSlug
155
+ .replace(/^./, (match) => match.toUpperCase())
156
+ .replace(/\W+(\w)/g, (_, p1) => p1.toUpperCase());
157
+ const promptQueries = [
158
+ {
159
+ type: 'text',
160
+ name: 'slug',
161
+ message: 'What is the package slug?',
162
+ initial: defaultPackageSlug,
163
+ resolvedValue: options.target ? defaultPackageSlug : null,
164
+ },
165
+ {
166
+ type: 'text',
167
+ name: 'name',
168
+ message: 'What is the project name?',
169
+ initial: defaultProjectName,
170
+ },
171
+ {
172
+ type: 'text',
173
+ name: 'description',
174
+ message: 'How would you describe the module?',
175
+ },
176
+ {
177
+ type: 'text',
178
+ name: 'package',
179
+ message: 'What is the Android package name?',
180
+ initial: `expo.modules.${defaultPackageSlug.replace(/\W/g, '').toLowerCase()}`,
181
+ },
182
+ {
183
+ type: 'text',
184
+ name: 'author',
185
+ message: 'Who is the author?',
186
+ initial: (_a = (await npmWhoamiAsync(targetDir))) !== null && _a !== void 0 ? _a : '',
187
+ },
188
+ {
189
+ type: 'text',
190
+ name: 'license',
191
+ message: 'What is the license?',
192
+ initial: 'MIT',
193
+ },
194
+ {
195
+ type: 'text',
196
+ name: 'repo',
197
+ message: 'What is the repository URL?',
198
+ },
199
+ ];
200
+ const answers = {};
201
+ for (const query of promptQueries) {
202
+ const { name, resolvedValue } = query;
203
+ answers[name] = (_b = resolvedValue !== null && resolvedValue !== void 0 ? resolvedValue : options[name]) !== null && _b !== void 0 ? _b : (await (0, prompts_1.default)(query))[name];
204
+ }
205
+ const { slug, name, description, package: projectPackage, author, license, repo } = answers;
206
+ return {
207
+ project: {
208
+ slug,
209
+ name,
210
+ version: '0.1.0',
211
+ description,
212
+ package: projectPackage,
213
+ },
214
+ author,
215
+ license,
216
+ repo,
217
+ };
218
+ }
219
+ const program = new commander_1.Command();
220
+ program
221
+ .name(packageJson.name)
222
+ .version(packageJson.version)
223
+ .description(packageJson.description)
224
+ .arguments('[target_dir]')
225
+ .option('-s, --source <source_dir>', 'Local path to the template. By default it downloads `expo-module-template` from NPM.')
226
+ .option('-n, --name <module_name>', 'Name of the native module.')
227
+ .option('-d, --description <description>', 'Description of the module.')
228
+ .option('-p, --package <package>', 'The Android package name.')
229
+ .option('-a, --author <author>', 'The author name.')
230
+ .option('-l, --license <license>', 'The license that the module is distributed with.')
231
+ .option('-r, --repo <repo_url>', 'The URL to the repository.')
232
+ .option('--with-readme', 'Whether to include README.md file.', false)
233
+ .option('--with-changelog', 'Whether to include CHANGELOG.md file.', false)
234
+ .action(main);
235
+ program.parse(process.argv);
236
+ //# sourceMappingURL=create-expo-module.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"create-expo-module.js","sourceRoot":"","sources":["../src/create-expo-module.ts"],"names":[],"mappings":"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","sourcesContent":["import spawnAsync from '@expo/spawn-async';\nimport { Command } from 'commander';\nimport downloadTarball from 'download-tarball';\nimport ejs from 'ejs';\nimport fs from 'fs-extra';\nimport path from 'path';\nimport prompts, { PromptObject } from 'prompts';\n\nconst packageJson = require('../package.json');\n\n// `yarn run` may change the current working dir, then we should use `INIT_CWD` env.\nconst CWD = process.env.INIT_CWD || process.cwd();\n\n// Ignore some paths. Especially `package.json` as it is rendered\n// from `$package.json` file instead of the original one.\nconst IGNORES_PATHS = ['.DS_Store', 'build', 'node_modules', 'package.json'];\n\n/**\n * Possible command options.\n */\ntype CommandOptions = {\n target: string;\n source?: string;\n name?: string;\n description?: string;\n package?: string;\n author?: string;\n license?: string;\n repo?: string;\n withReadme: boolean;\n withChangelog: boolean;\n};\n\n/**\n * Represents an object that is passed to `ejs` when rendering the template.\n */\ntype SubstitutionData = {\n project: {\n slug: string;\n name: string;\n version: string;\n description: string;\n package: string;\n };\n author: string;\n license: string;\n repo: string;\n};\n\ntype CustomPromptObject = PromptObject & {\n name: string;\n resolvedValue?: string | null;\n};\n\ntype PackageManager = 'npm' | 'yarn';\n\n/**\n * The main function of the command.\n *\n * @param target Path to the directory where to create the module. Defaults to current working dir.\n * @param command An object from `commander`.\n */\nasync function main(target: string | undefined, options: CommandOptions) {\n const targetDir = target ? path.join(CWD, target) : CWD;\n\n options.target = targetDir;\n await fs.ensureDir(targetDir);\n\n const data = await askForSubstitutionDataAsync(targetDir, options);\n const packageManager = await selectPackageManagerAsync();\n const packagePath = options.source\n ? path.join(CWD, options.source)\n : await downloadPackageAsync(targetDir);\n const files = await getFilesAsync(packagePath);\n\n console.log('🎨 Creating Expo module from the template files...');\n\n // Iterate through all template files.\n for (const file of files) {\n const renderedRelativePath = ejs.render(file.replace(/^\\$/, ''), data, {\n openDelimiter: '{',\n closeDelimiter: '}',\n escape: (value: string) => value.replace('.', path.sep),\n });\n const fromPath = path.join(packagePath, file);\n const toPath = path.join(targetDir, renderedRelativePath);\n const template = await fs.readFile(fromPath, { encoding: 'utf8' });\n const renderedContent = ejs.render(template, data);\n\n await fs.outputFile(toPath, renderedContent, { encoding: 'utf8' });\n }\n\n if (!options.source) {\n // Files in the downloaded tarball are wrapped in `package` dir.\n // We should remove it after all.\n await fs.remove(packagePath);\n }\n if (!options.withReadme) {\n await fs.remove(path.join(targetDir, 'README.md'));\n }\n if (!options.withChangelog) {\n await fs.remove(path.join(targetDir, 'CHANGELOG.md'));\n }\n\n // Install dependencies and build\n await postActionsAsync(packageManager, targetDir);\n\n console.log('✅ Successfully created Expo module');\n}\n\n/**\n * Recursively scans for the files within the directory. Returned paths are relative to the `root` path.\n */\nasync function getFilesAsync(root: string, dir: string | null = null): Promise<string[]> {\n const files: string[] = [];\n const baseDir = dir ? path.join(root, dir) : root;\n\n for (const file of await fs.readdir(baseDir)) {\n const relativePath = dir ? path.join(dir, file) : file;\n\n if (IGNORES_PATHS.includes(relativePath) || IGNORES_PATHS.includes(file)) {\n continue;\n }\n\n const fullPath = path.join(baseDir, file);\n const stat = await fs.lstat(fullPath);\n\n if (stat.isDirectory()) {\n files.push(...(await getFilesAsync(root, relativePath)));\n } else {\n files.push(relativePath);\n }\n }\n return files;\n}\n\n/**\n * Asks NPM registry for the url to the tarball.\n */\nasync function getNpmTarballUrl(packageName: string, version: string = 'latest'): Promise<string> {\n const { stdout } = await spawnAsync('npm', ['view', `${packageName}@${version}`, 'dist.tarball']);\n return stdout.trim();\n}\n\n/**\n * Gets the username of currently logged in user. Used as a default in the prompt asking for the module author.\n */\nasync function npmWhoamiAsync(targetDir: string): Promise<string | null> {\n try {\n const { stdout } = await spawnAsync('npm', ['whoami'], { cwd: targetDir });\n return stdout.trim();\n } catch {\n return null;\n }\n}\n\n/**\n * Downloads the template from NPM registry.\n */\nasync function downloadPackageAsync(targetDir: string): Promise<string> {\n const tarballUrl = await getNpmTarballUrl('expo-module-template');\n\n console.log('⬇️ Downloading module template from npm...');\n\n await downloadTarball({\n url: tarballUrl,\n dir: targetDir,\n });\n return path.join(targetDir, 'package');\n}\n\n/**\n * Asks whether to use Yarn or npm as a dependency package manager.\n */\nasync function selectPackageManagerAsync(): Promise<PackageManager> {\n const { packageManager } = await prompts({\n type: 'select',\n name: 'packageManager',\n message: 'Which package manager do you want to use to install dependencies?',\n choices: [\n { title: 'yarn', value: 'yarn' },\n { title: 'npm', value: 'npm' },\n ],\n });\n return packageManager;\n}\n\n/**\n * Installs dependencies and builds TypeScript files.\n */\nasync function postActionsAsync(packageManager: PackageManager, targetDir: string) {\n async function run(...args: string[]) {\n await spawnAsync(packageManager, args, {\n cwd: targetDir,\n stdio: 'ignore',\n });\n }\n\n console.log('📦 Installing dependencies...');\n await run('install');\n\n console.log('🛠 Compiling TypeScript files...');\n await run('run', 'build');\n}\n\n/**\n * Asks the user for some data necessary to render the template.\n * Some values may already be provided by command options, the prompt is skipped in that case.\n */\nasync function askForSubstitutionDataAsync(\n targetDir: string,\n options: CommandOptions\n): Promise<SubstitutionData> {\n const defaultPackageSlug = path.basename(targetDir);\n const defaultProjectName = defaultPackageSlug\n .replace(/^./, (match) => match.toUpperCase())\n .replace(/\\W+(\\w)/g, (_, p1) => p1.toUpperCase());\n\n const promptQueries: CustomPromptObject[] = [\n {\n type: 'text',\n name: 'slug',\n message: 'What is the package slug?',\n initial: defaultPackageSlug,\n resolvedValue: options.target ? defaultPackageSlug : null,\n },\n {\n type: 'text',\n name: 'name',\n message: 'What is the project name?',\n initial: defaultProjectName,\n },\n {\n type: 'text',\n name: 'description',\n message: 'How would you describe the module?',\n },\n {\n type: 'text',\n name: 'package',\n message: 'What is the Android package name?',\n initial: `expo.modules.${defaultPackageSlug.replace(/\\W/g, '').toLowerCase()}`,\n },\n {\n type: 'text',\n name: 'author',\n message: 'Who is the author?',\n initial: (await npmWhoamiAsync(targetDir)) ?? '',\n },\n {\n type: 'text',\n name: 'license',\n message: 'What is the license?',\n initial: 'MIT',\n },\n {\n type: 'text',\n name: 'repo',\n message: 'What is the repository URL?',\n },\n ];\n\n const answers: Record<string, string> = {};\n for (const query of promptQueries) {\n const { name, resolvedValue } = query;\n answers[name] = resolvedValue ?? options[name] ?? (await prompts(query))[name];\n }\n\n const { slug, name, description, package: projectPackage, author, license, repo } = answers;\n\n return {\n project: {\n slug,\n name,\n version: '0.1.0',\n description,\n package: projectPackage,\n },\n author,\n license,\n repo,\n };\n}\n\nconst program = new Command();\n\nprogram\n .name(packageJson.name)\n .version(packageJson.version)\n .description(packageJson.description)\n .arguments('[target_dir]')\n .option(\n '-s, --source <source_dir>',\n 'Local path to the template. By default it downloads `expo-module-template` from NPM.'\n )\n .option('-n, --name <module_name>', 'Name of the native module.')\n .option('-d, --description <description>', 'Description of the module.')\n .option('-p, --package <package>', 'The Android package name.')\n .option('-a, --author <author>', 'The author name.')\n .option('-l, --license <license>', 'The license that the module is distributed with.')\n .option('-r, --repo <repo_url>', 'The URL to the repository.')\n .option('--with-readme', 'Whether to include README.md file.', false)\n .option('--with-changelog', 'Whether to include CHANGELOG.md file.', false)\n .action(main);\n\nprogram.parse(process.argv);\n"]}
package/package.json CHANGED
@@ -1,11 +1,49 @@
1
1
  {
2
2
  "name": "create-expo-module",
3
- "version": "0.0.0",
4
- "description": "",
5
- "main": "index.js",
3
+ "version": "0.2.0",
4
+ "description": "The script to create the Expo module",
5
+ "main": "build/create-expo-module.js",
6
+ "types": "build/create-expo-module.d.ts",
6
7
  "scripts": {
7
- "test": "echo \"Error: no test specified\" && exit 1"
8
+ "build": "expo-module build",
9
+ "clean": "expo-module clean",
10
+ "lint": "expo-module lint",
11
+ "expo-module": "expo-module"
8
12
  },
9
- "author": "",
10
- "license": "MIT"
13
+ "bin": {
14
+ "create-expo-module": "./bin/create-expo-module"
15
+ },
16
+ "repository": {
17
+ "type": "git",
18
+ "url": "https://github.com/expo/expo.git",
19
+ "directory": "packages/create-expo-module"
20
+ },
21
+ "keywords": [
22
+ "expo",
23
+ "module",
24
+ "modules",
25
+ "library",
26
+ "react",
27
+ "native"
28
+ ],
29
+ "author": "Expo",
30
+ "license": "MIT",
31
+ "bugs": {
32
+ "url": "https://github.com/expo/expo/issues"
33
+ },
34
+ "homepage": "https://github.com/expo/expo/tree/main/packages/expo",
35
+ "dependencies": {
36
+ "@expo/spawn-async": "^1.5.0",
37
+ "commander": "^8.3.0",
38
+ "download-tarball": "^2.0.0",
39
+ "ejs": "^3.1.6",
40
+ "fs-extra": "^10.0.0",
41
+ "prompts": "^2.4.2"
42
+ },
43
+ "devDependencies": {
44
+ "@types/ejs": "^3.1.0",
45
+ "@types/prompts": "^2.0.14",
46
+ "expo-module-scripts": "^2.0.0"
47
+ },
48
+ "gitHead": "89a27c0ca0ca8becd7546697298e874a15e94faf"
11
49
  }
@@ -0,0 +1,306 @@
1
+ import spawnAsync from '@expo/spawn-async';
2
+ import { Command } from 'commander';
3
+ import downloadTarball from 'download-tarball';
4
+ import ejs from 'ejs';
5
+ import fs from 'fs-extra';
6
+ import path from 'path';
7
+ import prompts, { PromptObject } from 'prompts';
8
+
9
+ const packageJson = require('../package.json');
10
+
11
+ // `yarn run` may change the current working dir, then we should use `INIT_CWD` env.
12
+ const CWD = process.env.INIT_CWD || process.cwd();
13
+
14
+ // Ignore some paths. Especially `package.json` as it is rendered
15
+ // from `$package.json` file instead of the original one.
16
+ const IGNORES_PATHS = ['.DS_Store', 'build', 'node_modules', 'package.json'];
17
+
18
+ /**
19
+ * Possible command options.
20
+ */
21
+ type CommandOptions = {
22
+ target: string;
23
+ source?: string;
24
+ name?: string;
25
+ description?: string;
26
+ package?: string;
27
+ author?: string;
28
+ license?: string;
29
+ repo?: string;
30
+ withReadme: boolean;
31
+ withChangelog: boolean;
32
+ };
33
+
34
+ /**
35
+ * Represents an object that is passed to `ejs` when rendering the template.
36
+ */
37
+ type SubstitutionData = {
38
+ project: {
39
+ slug: string;
40
+ name: string;
41
+ version: string;
42
+ description: string;
43
+ package: string;
44
+ };
45
+ author: string;
46
+ license: string;
47
+ repo: string;
48
+ };
49
+
50
+ type CustomPromptObject = PromptObject & {
51
+ name: string;
52
+ resolvedValue?: string | null;
53
+ };
54
+
55
+ type PackageManager = 'npm' | 'yarn';
56
+
57
+ /**
58
+ * The main function of the command.
59
+ *
60
+ * @param target Path to the directory where to create the module. Defaults to current working dir.
61
+ * @param command An object from `commander`.
62
+ */
63
+ async function main(target: string | undefined, options: CommandOptions) {
64
+ const targetDir = target ? path.join(CWD, target) : CWD;
65
+
66
+ options.target = targetDir;
67
+ await fs.ensureDir(targetDir);
68
+
69
+ const data = await askForSubstitutionDataAsync(targetDir, options);
70
+ const packageManager = await selectPackageManagerAsync();
71
+ const packagePath = options.source
72
+ ? path.join(CWD, options.source)
73
+ : await downloadPackageAsync(targetDir);
74
+ const files = await getFilesAsync(packagePath);
75
+
76
+ console.log('🎨 Creating Expo module from the template files...');
77
+
78
+ // Iterate through all template files.
79
+ for (const file of files) {
80
+ const renderedRelativePath = ejs.render(file.replace(/^\$/, ''), data, {
81
+ openDelimiter: '{',
82
+ closeDelimiter: '}',
83
+ escape: (value: string) => value.replace('.', path.sep),
84
+ });
85
+ const fromPath = path.join(packagePath, file);
86
+ const toPath = path.join(targetDir, renderedRelativePath);
87
+ const template = await fs.readFile(fromPath, { encoding: 'utf8' });
88
+ const renderedContent = ejs.render(template, data);
89
+
90
+ await fs.outputFile(toPath, renderedContent, { encoding: 'utf8' });
91
+ }
92
+
93
+ if (!options.source) {
94
+ // Files in the downloaded tarball are wrapped in `package` dir.
95
+ // We should remove it after all.
96
+ await fs.remove(packagePath);
97
+ }
98
+ if (!options.withReadme) {
99
+ await fs.remove(path.join(targetDir, 'README.md'));
100
+ }
101
+ if (!options.withChangelog) {
102
+ await fs.remove(path.join(targetDir, 'CHANGELOG.md'));
103
+ }
104
+
105
+ // Install dependencies and build
106
+ await postActionsAsync(packageManager, targetDir);
107
+
108
+ console.log('✅ Successfully created Expo module');
109
+ }
110
+
111
+ /**
112
+ * Recursively scans for the files within the directory. Returned paths are relative to the `root` path.
113
+ */
114
+ async function getFilesAsync(root: string, dir: string | null = null): Promise<string[]> {
115
+ const files: string[] = [];
116
+ const baseDir = dir ? path.join(root, dir) : root;
117
+
118
+ for (const file of await fs.readdir(baseDir)) {
119
+ const relativePath = dir ? path.join(dir, file) : file;
120
+
121
+ if (IGNORES_PATHS.includes(relativePath) || IGNORES_PATHS.includes(file)) {
122
+ continue;
123
+ }
124
+
125
+ const fullPath = path.join(baseDir, file);
126
+ const stat = await fs.lstat(fullPath);
127
+
128
+ if (stat.isDirectory()) {
129
+ files.push(...(await getFilesAsync(root, relativePath)));
130
+ } else {
131
+ files.push(relativePath);
132
+ }
133
+ }
134
+ return files;
135
+ }
136
+
137
+ /**
138
+ * Asks NPM registry for the url to the tarball.
139
+ */
140
+ async function getNpmTarballUrl(packageName: string, version: string = 'latest'): Promise<string> {
141
+ const { stdout } = await spawnAsync('npm', ['view', `${packageName}@${version}`, 'dist.tarball']);
142
+ return stdout.trim();
143
+ }
144
+
145
+ /**
146
+ * Gets the username of currently logged in user. Used as a default in the prompt asking for the module author.
147
+ */
148
+ async function npmWhoamiAsync(targetDir: string): Promise<string | null> {
149
+ try {
150
+ const { stdout } = await spawnAsync('npm', ['whoami'], { cwd: targetDir });
151
+ return stdout.trim();
152
+ } catch {
153
+ return null;
154
+ }
155
+ }
156
+
157
+ /**
158
+ * Downloads the template from NPM registry.
159
+ */
160
+ async function downloadPackageAsync(targetDir: string): Promise<string> {
161
+ const tarballUrl = await getNpmTarballUrl('expo-module-template');
162
+
163
+ console.log('⬇️ Downloading module template from npm...');
164
+
165
+ await downloadTarball({
166
+ url: tarballUrl,
167
+ dir: targetDir,
168
+ });
169
+ return path.join(targetDir, 'package');
170
+ }
171
+
172
+ /**
173
+ * Asks whether to use Yarn or npm as a dependency package manager.
174
+ */
175
+ async function selectPackageManagerAsync(): Promise<PackageManager> {
176
+ const { packageManager } = await prompts({
177
+ type: 'select',
178
+ name: 'packageManager',
179
+ message: 'Which package manager do you want to use to install dependencies?',
180
+ choices: [
181
+ { title: 'yarn', value: 'yarn' },
182
+ { title: 'npm', value: 'npm' },
183
+ ],
184
+ });
185
+ return packageManager;
186
+ }
187
+
188
+ /**
189
+ * Installs dependencies and builds TypeScript files.
190
+ */
191
+ async function postActionsAsync(packageManager: PackageManager, targetDir: string) {
192
+ async function run(...args: string[]) {
193
+ await spawnAsync(packageManager, args, {
194
+ cwd: targetDir,
195
+ stdio: 'ignore',
196
+ });
197
+ }
198
+
199
+ console.log('📦 Installing dependencies...');
200
+ await run('install');
201
+
202
+ console.log('🛠 Compiling TypeScript files...');
203
+ await run('run', 'build');
204
+ }
205
+
206
+ /**
207
+ * Asks the user for some data necessary to render the template.
208
+ * Some values may already be provided by command options, the prompt is skipped in that case.
209
+ */
210
+ async function askForSubstitutionDataAsync(
211
+ targetDir: string,
212
+ options: CommandOptions
213
+ ): Promise<SubstitutionData> {
214
+ const defaultPackageSlug = path.basename(targetDir);
215
+ const defaultProjectName = defaultPackageSlug
216
+ .replace(/^./, (match) => match.toUpperCase())
217
+ .replace(/\W+(\w)/g, (_, p1) => p1.toUpperCase());
218
+
219
+ const promptQueries: CustomPromptObject[] = [
220
+ {
221
+ type: 'text',
222
+ name: 'slug',
223
+ message: 'What is the package slug?',
224
+ initial: defaultPackageSlug,
225
+ resolvedValue: options.target ? defaultPackageSlug : null,
226
+ },
227
+ {
228
+ type: 'text',
229
+ name: 'name',
230
+ message: 'What is the project name?',
231
+ initial: defaultProjectName,
232
+ },
233
+ {
234
+ type: 'text',
235
+ name: 'description',
236
+ message: 'How would you describe the module?',
237
+ },
238
+ {
239
+ type: 'text',
240
+ name: 'package',
241
+ message: 'What is the Android package name?',
242
+ initial: `expo.modules.${defaultPackageSlug.replace(/\W/g, '').toLowerCase()}`,
243
+ },
244
+ {
245
+ type: 'text',
246
+ name: 'author',
247
+ message: 'Who is the author?',
248
+ initial: (await npmWhoamiAsync(targetDir)) ?? '',
249
+ },
250
+ {
251
+ type: 'text',
252
+ name: 'license',
253
+ message: 'What is the license?',
254
+ initial: 'MIT',
255
+ },
256
+ {
257
+ type: 'text',
258
+ name: 'repo',
259
+ message: 'What is the repository URL?',
260
+ },
261
+ ];
262
+
263
+ const answers: Record<string, string> = {};
264
+ for (const query of promptQueries) {
265
+ const { name, resolvedValue } = query;
266
+ answers[name] = resolvedValue ?? options[name] ?? (await prompts(query))[name];
267
+ }
268
+
269
+ const { slug, name, description, package: projectPackage, author, license, repo } = answers;
270
+
271
+ return {
272
+ project: {
273
+ slug,
274
+ name,
275
+ version: '0.1.0',
276
+ description,
277
+ package: projectPackage,
278
+ },
279
+ author,
280
+ license,
281
+ repo,
282
+ };
283
+ }
284
+
285
+ const program = new Command();
286
+
287
+ program
288
+ .name(packageJson.name)
289
+ .version(packageJson.version)
290
+ .description(packageJson.description)
291
+ .arguments('[target_dir]')
292
+ .option(
293
+ '-s, --source <source_dir>',
294
+ 'Local path to the template. By default it downloads `expo-module-template` from NPM.'
295
+ )
296
+ .option('-n, --name <module_name>', 'Name of the native module.')
297
+ .option('-d, --description <description>', 'Description of the module.')
298
+ .option('-p, --package <package>', 'The Android package name.')
299
+ .option('-a, --author <author>', 'The author name.')
300
+ .option('-l, --license <license>', 'The license that the module is distributed with.')
301
+ .option('-r, --repo <repo_url>', 'The URL to the repository.')
302
+ .option('--with-readme', 'Whether to include README.md file.', false)
303
+ .option('--with-changelog', 'Whether to include CHANGELOG.md file.', false)
304
+ .action(main);
305
+
306
+ program.parse(process.argv);
package/tsconfig.json ADDED
@@ -0,0 +1,23 @@
1
+ {
2
+ "extends": "@tsconfig/node12/tsconfig.json",
3
+ "include": ["./src"],
4
+ "exclude": ["**/__mocks__/*", "**/__tests__/*"],
5
+ "compilerOptions": {
6
+ "outDir": "./build",
7
+ "module": "commonjs",
8
+ "moduleResolution": "node",
9
+ "types": [],
10
+ "typeRoots": ["./ts-declarations", "node_modules/@types"],
11
+ "sourceMap": true,
12
+ "declaration": true,
13
+ "inlineSources": true,
14
+ "strictNullChecks": true,
15
+ "strictPropertyInitialization": true,
16
+ "strictFunctionTypes": true,
17
+ "skipLibCheck": true,
18
+ "noImplicitAny": false,
19
+ "noImplicitThis": true,
20
+ "noImplicitReturns": true,
21
+ "allowSyntheticDefaultImports": true
22
+ }
23
+ }