Skip to content

Commit

Permalink
Merge branch 'main' of github.com:plopjs/plop into feat/async-run
Browse files Browse the repository at this point in the history
  • Loading branch information
benallfree committed Apr 21, 2024
2 parents c5c458a + 6b1f1b7 commit cd366c8
Show file tree
Hide file tree
Showing 6 changed files with 90 additions and 33 deletions.
4 changes: 4 additions & 0 deletions .prettierrc
Original file line number Diff line number Diff line change
@@ -0,0 +1,4 @@
{
"singleQuote": false,
"semi": true
}
54 changes: 45 additions & 9 deletions README.md
Original file line number Diff line number Diff line change
Expand Up @@ -130,6 +130,47 @@ $ plop component -- --type react
### Running a Generator Forcefully
By default Plop actions keep your files safe by failing when things look fishy. The most obvious example of this is not allowing an [`add`](#add) action to overwrite a file that already exists. Plop actions individually support the `force` property but you can also use the `--force` flag when running Plop from the terminal. Using the `--force` flag will tell every action to run forcefully. With great power...🕷

### Using TypeScript plopfiles

Plop bundles TypeScript declarations and supports TypeScript plopfiles via [tsx loaders](https://github.com/privatenumber/tsx?tab=readme-ov-file#nodejs-loader), a feature of [NodeJS command line imports](https://nodejs.org/api/cli.html#--importmodule).

First, make a TypesScript plopfile using `plop --init-ts` or by hand:

```ts
// plopfile.ts
import {NodePlopAPI} from 'plop';

export default function (plop: NodePlopAPI) {
// plop generator code
};
```

Next, install [tsx](https://github.com/privatenumber/tsx) and optionally [cross-env](https://www.npmjs.com/package/cross-env):

```bash
npm i -D tsx cross-env
```

Finally, use `NODE_OPTIONS` to activate the tsx loader. Now Plop can import your `plopfile.ts`:

**Node.js v20.6 and above**

```json
// package.json
"scripts": {
"cross-env NODE_OPTIONS='--import tsx' plop --plopfile=plopfile.ts"
}
```

**Node.js v20.5.1 and below**

```json
// package.json
"scripts": {
"cross-env NODE_OPTIONS='--loader tsx' plop --plopfile=plopfile.ts"
}
```

## Why Generators?
Because when you create your boilerplate separate from your code, you naturally put more time and thought into it.

Expand All @@ -140,18 +181,13 @@ Because [context switching is expensive](https://www.petrikainulainen.net/softwa
# Plopfile API
The plopfile api is the collection of methods that are exposed by the `plop` object. Most of the work is done by [`setGenerator`](#setgenerator) but this section documents the other methods that you may also find useful in your plopfile.

## TypeScript Declarations
## TypeScript Support

`plop` bundles TypeScript declarations. Whether or not you write your plopfile in TypeScript, many editors will offer code assistance via these declarations.
Plop bundles TypeScript declarations. See [using TypeScript plopfiles](#using-typescript-plopfiles) for more details.

```javascript
// plopfile.ts
import {NodePlopAPI} from 'plop';
## JSDoc Support

export default function (plop: NodePlopAPI) {
// plop generator code
};
```
Whether or not you write your plopfile in TypeScript, many editors will offer code assistance via JSDoc declarations.

```javascript
// plopfile.js
Expand Down
49 changes: 32 additions & 17 deletions packages/plop/src/console-out.js
Original file line number Diff line number Diff line change
Expand Up @@ -62,6 +62,7 @@ function displayHelpScreen() {
" -t, --show-type-names " +
chalk.dim("Show type names instead of abbreviations"),
" -i, --init " + chalk.dim("Generate a basic plopfile.js"),
" --init-ts " + chalk.dim("Generate a basic plopfile.ts"),
" -v, --version " + chalk.dim("Print current version"),
" -f, --force " + chalk.dim("Run the generator forcefully"),
"",
Expand Down Expand Up @@ -89,25 +90,39 @@ function displayHelpScreen() {
);
}

function createInitPlopfile(force = false) {
var initString =
"export default function (plop) {\n\n" +
"\tplop.setGenerator('basics', {\n" +
"\t\tdescription: 'this is a skeleton plopfile',\n" +
"\t\tprompts: [],\n" +
"\t\tactions: []\n" +
"\t});\n\n" +
"};";
function createInitPlopfile(force = false, useTypescript = false) {
var initString = (() => {
if (useTypescript) {
return (
"import type { NodePlopAPI } from 'plop'\n" +
"\n" +
"export default async function (plop: NodePlopAPI) {\n" +
"\n" +
"}\n" +
"\n"
);
} else {
return (
"export default function (plop) {\n\n" +
"\tplop.setGenerator('basics', {\n" +
"\t\tdescription: 'this is a skeleton plopfile',\n" +
"\t\tprompts: [],\n" +
"\t\tactions: []\n" +
"\t});\n\n" +
"};"
);
}
})();

if (fs.existsSync(process.cwd() + "/plopfile.js") && force === false) {
throw Error('"plopfile.js" already exists at this location.');
}

if (fs.existsSync(process.cwd() + "/plopfile.cjs") && force === false) {
throw Error('"plopfile.cjs" already exists at this location.');
}
[`js`, `cjs`, `ts`].forEach((ext) => {
const name = `plopfile.${ext}`;
if (fs.existsSync(process.cwd() + `/${name}`) && force === false) {
throw Error(`"${name}" already exists at this location.`);
}
});

fs.writeFileSync(process.cwd() + "/plopfile.js", initString);
const outExt = useTypescript ? `ts` : `js`;
fs.writeFileSync(process.cwd() + `/plopfile.${outExt}`, initString);
}

const typeDisplay = {
Expand Down
14 changes: 7 additions & 7 deletions packages/plop/src/input-processing.js
Original file line number Diff line number Diff line change
Expand Up @@ -18,11 +18,11 @@ function getBypassAndGenerator(plop, passArgsBeforeDashes) {
const { plopArgV, eoaArg } = passArgsBeforeDashes
? { plopArgV: argv }
: eoaIndex === -1
? { plopArgV: [] }
: {
plopArgV: minimist(args.slice(eoaIndex + 1, args.length)),
eoaArg: args[eoaIndex + 1],
};
? { plopArgV: [] }
: {
plopArgV: minimist(args.slice(eoaIndex + 1, args.length)),
eoaArg: args[eoaIndex + 1],
};

// locate the generator name based on input and take the rest of the
// user's input as prompt bypass data to be passed into the generator
Expand Down Expand Up @@ -70,10 +70,10 @@ function handleArgFlags(env) {
}

// handle request for initializing a new plopfile
if (argv.init || argv.i) {
if (argv.init || argv.i || argv[`init-ts`]) {
const force = argv.force === true || argv.f === true || false;
try {
out.createInitPlopfile(force);
out.createInitPlopfile(force, !!argv[`init-ts`]);
process.exit(0);
} catch (err) {
console.error(chalk.red("[PLOP] ") + err.message);
Expand Down
1 change: 1 addition & 0 deletions packages/plop/src/plop.d.ts
Original file line number Diff line number Diff line change
Expand Up @@ -12,6 +12,7 @@ export {
PlopGenerator,
NodePlopAPI,
PlopGeneratorConfig,
Actions
} from "node-plop";

export const Plop: Liftoff;
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -11,6 +11,7 @@ Options:
-h, --help Show this help display
-t, --show-type-names Show type names instead of abbreviations
-i, --init Generate a basic plopfile.js
--init-ts Generate a basic plopfile.ts
-v, --version Print current version
-f, --force Run the generator forcefully
Expand Down

0 comments on commit cd366c8

Please sign in to comment.