Easy, transparent ”watch mode” for filesystem access (powered by Chokidar)
The problem: You're writing a script that uses the filesystem as one of its inputs. You want to watch files for changes, so you can rerun your script when they change.
The solution: Use jumpgen.
Now, your script can use filesystem APIs without worrying about how to watch files for changes, leaving you to focus on the logic of your generator.
- Your script will rerun automatically when files it relies on get added, changed, or deleted.
- Glob patterns are also watched for you.
- Jumpgen determines if a path needs to be watched recursively, only its children, or just the path itself. This means your generator will only rerun when it needs to, reducing unnecessary work.
- Incremental updates are easier than ever, thanks to Jumpgen's taskAPI. (coming soon)
- If you have a long-running script, it can be automatically aborted if a file changes during its execution.
- If your script relies on third-party code that accesses the filesystem, you can tell Jumpgen to watch those files too, optionally blaming other files when it detects a change.
- If your script reads from a configuration file, you can tell Jumpgen to “hard reset” the generator when that file changes. This is useful for far-reaching changes that might invalidate your entire script's output.
- It uses the chokidar@4,picomatch,tinyglobby, andfdirnpm packages under the hood for file watching and globbing.
See the API Reference for the full documentation.
- Jumpgen isn't focused on ”project scaffolding”, like ploporyeoman. But you can absolutely use it to build your own scaffolding tools.
- Jumpgen isn't focused on basic file-watching tasks. If you're looking for a simple way to watch files for changes, use chokidardirectly.
- Jumpgen isn't focused on basic script re-running. If you're looking to rerun a script when files change, use something like watchlistinstead.
Jumpgen saves the day when your script has complex, configurable logic and/or could benefit from incremental updates. For example, I'm using it to build a type-safe RPC library, alien-rpc, which is highly configurable and greatly benefits from knowing when certain files change.
pnpm add jumpgenDefine your generator with a name and a function that receives a Context object with helper functions for reading, scanning, and writing files. Your generator should avoid using node:fs APIs directly, or else file-watching will break.
import { jumpgen } from 'jumpgen'
export default jumpgen('my-generator', async ({ fs, dedent }) => {
  // Find files to use as source modules. If a file matching your globs
  // is later added or removed, your generator will be rerun (if watch
  // mode is enabled).
  const sourceModulePaths = fs.scan(['src/**/*.ts', '!**/*.test.ts'], {
    absolute: true,
  })
  // When you read a file, and you later change or delete it, your generator
  // will be rerun (if watch mode is enabled).
  const contents = sourceModulePaths.map(p => fs.read(p))
  // When you write a file, your generator emits a "write" event. This
  // is useful for logging, which helps you understand what's happening.
  contents.forEach((content, i) => {
    const outPath = sourceModulePaths[i].replace(/\.ts$/, '.js')
    fs.write(outPath, transform(content))
  })
  // Use the "dedent" function to remove excess indentation from your
  // template literals.
  fs.write(
    'foo.ts',
    dedent`
      export const foo = true
    `
  )
})To run your generator, simply import and call it.
import myGenerator from './my-generator.js'
// This example uses the default options.
const runner = myGenerator({
  // All file operations are relative to this path.
  root: process.cwd(),
  // Watch mode must be explicitly enabled.
  watch: false,
  // You may provide your own EventEmitter, which is mainly useful for
  // consolidating events across multiple generators. Whether or not you
  // provide one, you can listen for events on the `runner.events` property.
  events: undefined,
})
// The generator runs immediately. To wait for it to finish, you can
// await it or call its "then" method.
await runner
// or
runner.then(() => {
  console.log('done')
})
// End the current run early and stop watching for changes.
await runner.destroy()
// Listen to events from the runner.
runner.events.on('start', generatorName => {
  console.log(generatorName, 'started')
})
runner.events.on('write', (file, generatorName) => {
  console.log(generatorName, 'wrote', file)
})
runner.events.on('finish', (result, generatorName) => {
  console.log(generatorName, 'finished with', result)
})
runner.events.on('error', (error, generatorName) => {
  console.error(generatorName, 'errored with', error)
})The compose function lets you combine multiple generators into a single generator that runs them all in parallel.
import { compose } from 'jumpgen'
// The returned generator has the same API as the generators you pass to it,
// except it resolves with an array containing the results of all the generators.
const myGenerator = compose(generatorA, generatorB)See the testing guide for information on how to test your generators.
MIT