Skip to content

Interface: FunctionPluginHooks

Defined in: plugin/index.ts:187

Build Hooks

buildEnd()

  • Type: (this, err?) => void
Defined in: plugin/index.ts:362

Called when Rolldown has finished bundling, but before Output Generation Hooks. If an error occurred during the build, it is passed on to this hook.

Parameters

this

PluginContext

err?

Error

The error occurred during the build if applicable.

Returns

void


buildStart()

  • Type: (this, options) => void
Defined in: plugin/index.ts:254

Called on each rolldown() build.

This is the recommended hook to use when you need access to the options passed to rolldown() as it takes the transformations by all options hooks into account and also contains the right default values for unset options.

Parameters

this

PluginContext

options

NormalizedInputOptions

Returns

void


closeWatcher()

  • Type: (this) => void
Defined in: plugin/index.ts:523

Notifies a plugin when the watcher process will close so that all open resources can be closed too.

This hook cannot be used by output plugins.

Parameters

this

PluginContext

Returns

void


load()

  • Type: (this, id) => MaybePromise<undefined | null | string | void | SourceDescription>
Defined in: plugin/index.ts:320

Defines a custom loader.

Returning null defers to other load hooks or the built-in loading mechanism.

You can use this.getModuleInfo() to find out the previous values of meta, moduleSideEffects inside this hook.

Parameters

this

PluginContext

id

string

Returns

MaybePromise<undefined | null | string | void | SourceDescription>


moduleParsed()

  • Type: (this, moduleInfo) => void
Defined in: plugin/index.ts:354

This hook is called each time a module has been fully parsed by Rolldown.

This hook will wait until all imports are resolved so that the information in moduleInfo.importedIds, moduleInfo.dynamicallyImportedIds are complete and accurate. Note however that information about importing modules may be incomplete as additional importers could be discovered later. If you need this information, use the buildEnd hook.

Parameters

this

PluginContext

moduleInfo

ModuleInfo

Returns

void


onLog()

  • Type: (this, level, log) => boolean | undefined | null | void
Defined in: plugin/index.ts:206

A function that receives and filters logs and warnings generated by Rolldown and plugins before they are passed to the onLog option or printed to the console.

If false is returned, the log will be filtered out. Otherwise, the log will be handed to the onLog hook of the next plugin, the onLog option, or printed to the console. Plugins can also change the log level of a log or turn a log into an error by passing the log object to this.error, this.warn, this.info or this.debug and returning false.

Note that unlike other plugin hooks that add e.g. the plugin name to the log, those functions will not add or change properties of the log. Additionally, logs generated by an onLog hook will not be passed back to the onLog hook of the same plugin. If another plugin generates a log in response to such a log in its own onLog hook, this log will not be passed to the original onLog hook, either.

Example

js
function plugin1() {
  return {
    name: 'plugin1',
    buildStart() {
      this.info({ message: 'Hey', pluginCode: 'SPECIAL_CODE' });
    },
    onLog(level, log) {
      if (log.plugin === 'plugin1' && log.pluginCode === 'SPECIAL_CODE') {
        // We turn logs into warnings based on their code. This warnings
        // will not be passed back to the same plugin to avoid an
        // infinite loop, but other plugins will still receive it.
        this.warn(log);
        return false;
      }
    },
  };
}

function plugin2() {
  return {
    name: 'plugin2',
    onLog(level, log) {
      if (log.plugin === 'plugin1' && log.pluginCode === 'SPECIAL_CODE') {
        // You can modify logs in this hooks as well
        log.meta = 'processed by plugin 2';
        // This turns the log back to "info". If this happens in
        // response to the first plugin, it will not be passed back to
        // either plugin to avoid an infinite loop. If both plugins are
        // active, the log will be an info log if the second plugin is
        // placed after the first one
        this.info(log);
        return false;
      }
    },
  };
}

Parameters

this

MinimalPluginContext

level

"info" | "debug" | "warn"

log

RolldownLog

Returns

boolean | undefined | null | void


options()

  • Type: (this, options) => undefined | null | void | InputOptions
Defined in: plugin/index.ts:223

Replaces or manipulates the options object passed to rolldown().

Returning null does not replace anything.

If you just need to read the options, it is recommended to use the buildStart hook as that hook has access to the options after the transformations from all options hooks have been taken into account.

Parameters

this

MinimalPluginContext

options

InputOptions

Returns

undefined | null | void | InputOptions


outputOptions()

Defined in: plugin/index.ts:242

Replaces or manipulates the output options object passed to bundle.generate() or bundle.write().

Returning null does not replace anything.

If you just need to read the output options, it is recommended to use the renderStart hook as this hook has access to the output options after the transformations from all outputOptions hooks have been taken into account.

Parameters

this

MinimalPluginContext

options

OutputOptions

Returns

OutputOptions | undefined | null | void


resolveDynamicImport()

  • Type: (this, source, importer) => undefined | null | string | false | void | PartialResolvedId
Defined in: plugin/index.ts:305

Defines a custom resolver for dynamic imports.

Parameters

this

PluginContext

source

string

importer

string | undefined

Returns

undefined | null | string | false | void | PartialResolvedId

Deprecated

This hook exists only for Rollup compatibility. Please use resolveId instead.


resolveId()

  • Type: (this, source, importer, extraOptions) => undefined | null | string | false | void | PartialResolvedId
Defined in: plugin/index.ts:273

Defines a custom resolver.

A resolver can be useful for e.g. locating third-party dependencies.

Returning null defers to other resolveId hooks and eventually the default resolution behavior. Returning false signals that source should be treated as an external module and not included in the bundle. If this happens for a relative import, the id will be renormalized the same way as when the InputOptions.external option is used. If you return an object, then it is possible to resolve an import to a different id while excluding it from the bundle at the same time.

Note that while resolveId will be called for each import of a module and can therefore resolve to the same id many times, values for external, meta or moduleSideEffects can only be set once before the module is loaded. The reason is that after this call, Rolldown will continue with the load and transform hooks for that module that may override these values and should take precedence if they do so.

Parameters

this

PluginContext

source

string

The importee exactly as it is written in the import statement.

For example, given import foo from './foo.js', the source will be "./foo.js".

importer

The fully resolved id of the importing module.

When resolving entry points, importer will usually be undefined. An exception here is entry points generated via this.emitFile as here, you can provide an importer argument. For those cases, the isEntry option will tell you if we are resolving a user defined entry point, an emitted chunk, or if the isEntry parameter was provided for the this.resolve function.

string | undefined

extraOptions
custom?

CustomPluginOptions

Plugin-specific options.

See Custom resolver options section for more details.

isEntry

boolean

Whether this is resolution for an entry point.

Define custom proxy modules for entry points

This can be used for instance as a mechanism to define custom proxy modules for entry points. The following plugin will proxy all entry points to inject a polyfill import.

js
import { exactRegex } from '@rolldown/pluginutils';
// We prefix the polyfill id with \0 to tell other plugins not to try to load or
// transform it
const POLYFILL_ID = '\0polyfill';
const PROXY_SUFFIX = '?inject-polyfill-proxy';

function injectPolyfillPlugin() {
  return {
    name: 'inject-polyfill',
    async resolveId(source, importer, options) {
      if (source === POLYFILL_ID) {
        // It is important that side effects are always respected for polyfills,
        // otherwise using `treeshake.moduleSideEffects: false` may prevent the
        // polyfill from being included.
        return { id: POLYFILL_ID, moduleSideEffects: true };
      }
      if (options.isEntry) {
        // Determine what the actual entry would have been.
        const resolution = await this.resolve(source, importer, options);
        // If it cannot be resolved or is external, just return it so that Rolldown
        // can display an error
        if (!resolution || resolution.external) return resolution;
        // In the load hook of the proxy, we need to know if the entry has a
        // default export. There, however, we no longer have the full "resolution"
        // object that may contain meta-data from other plugins that is only added
        // on first load. Therefore we trigger loading here.
        const moduleInfo = await this.load(resolution);
        // We need to make sure side effects in the original entry point are
        // respected even for `treeshake.moduleSideEffects: false`. "moduleSideEffects"
        // is a writable property on ModuleInfo.
        moduleInfo.moduleSideEffects = true;
        // It is important that the new entry does not start with `\0` and has the same
        // directory as the original one to not mess up relative external import generation.
        // Also keeping the name and just adding a "?query" to the end ensures that
        // `preserveModules` will generate the original entry name for this entry.
        return `${resolution.id}${PROXY_SUFFIX}`;
      }
      return null;
    },
    load: {
      filter: { id: [exactRegex(POLYFILL_ID), /\?proxy$/] },
      handler(id) {
        if (id === POLYFILL_ID) {
          // Replace with actual polyfill
          return "console.log('polyfill');";
        }
        if (id.endsWith(PROXY_SUFFIX)) {
          const entryId = id.slice(0, -PROXY_SUFFIX.length);
          // We know ModuleInfo.exports is reliable because we awaited this.load in resolveId
          const { exports } = this.getModuleInfo(entryId);
          let code =
            `import ${JSON.stringify(POLYFILL_ID)};` + `export * from ${JSON.stringify(entryId)};`;
          // Namespace reexports do not reexport default, so we need special handling here
          if (exports.includes('default')) {
            code += `export { default } from ${JSON.stringify(entryId)};`;
          }
          return code;
        }
        return null;
      },
    },
  };
}
kind

"import-statement" | "dynamic-import" | "require-call" | "import-rule" | "url-token" | "new-url" | "hot-accept"

The kind of import being resolved.

  • import-statement: import { foo } from './lib.js';
  • dynamic-import: import('./lib.js')
  • require-call: require('./lib.js')
  • import-rule: @import 'bg-color.css' (experimental)
  • url-token: url('./icon.png') (experimental)
  • new-url: new URL('./worker.js', import.meta.url) (experimental)
  • hot-accept: import.meta.hot.accept('./lib.js', () => {}) (experimental)

Returns

undefined | null | string | false | void | PartialResolvedId


transform()

Defined in: plugin/index.ts:331

Can be used to transform individual modules.

Note that it's possible to return only properties and no code transformations.

You can use this.getModuleInfo() to find out the previous values of meta, moduleSideEffects inside this hook.

Parameters

this

TransformPluginContext

code

string

id

string

meta

BindingTransformHookExtraArgs & { ast?: Program; magicString?: BindingMagicString; moduleType: ModuleType; }

Returns

undefined | null | string | void | Omit<SourceDescription, "code"> & { code?: string | BindingMagicString; }


watchChange()

  • Type: (this, id, event) => void
Defined in: plugin/index.ts:510

Notifies a plugin whenever Rolldown has detected a change to a monitored file in watch mode.

If a build is currently running, this hook is called once the build finished. It will be called once for every file that changed.

This hook cannot be used by output plugins.

If you need to be notified immediately when a file changed, you can use the watch.onInvalidate option.

Parameters

this

PluginContext

id

string

event
event

ChangeEvent

Returns

void

Output Generation Hooks

augmentChunkHash()

  • Type: (this, chunk) => string | void
Defined in: plugin/index.ts:428

Can be used to augment the hash of individual chunks. Called for each Rolldown output chunk.

Returning a falsy value will not modify the hash. Truthy values will be used as an additional source for hash calculation.

Example

The following plugin will invalidate the hash of chunk foo with the current timestamp:

js
function augmentWithDatePlugin() {
  return {
    name: 'augment-with-date',
    augmentChunkHash(chunkInfo) {
      if (chunkInfo.name === 'foo') {
        return Date.now().toString();
      }
    },
  };
}

Parameters

this

PluginContext

chunk

RenderedChunk

Returns

string | void


closeBundle()

  • Type: (this, error?) => void
Defined in: plugin/index.ts:492

Can be used to clean up any external service that may be running.

Rolldown's CLI will make sure this hook is called after each run, but it is the responsibility of users of the JavaScript API to manually call bundle.close() once they are done generating bundles. For that reason, any plugin relying on this feature should carefully mention this in its documentation.

If a plugin wants to retain resources across builds in watch mode, they can check for this.meta.watchMode in this hook and perform the necessary cleanup for watch mode in closeWatcher.

Parameters

this

PluginContext

error?

Error

An error that occurred during build or the buildEnd hook, if any.

Returns

void


generateBundle()

  • Type: (this, outputOptions, bundle, isWrite) => void
Defined in: plugin/index.ts:456

Called at the end of bundle.generate() or immediately before the files are written in bundle.write().

To modify the files after they have been written, use the writeBundle hook.

You can prevent files from being emitted by deleting them from the bundle object in this hook. To emit additional files, use the this.emitFile function.

DANGER

Do not directly add assets to the bundle. This will not work as expected as Rolldown will ignore those assets. This is not recommended in Rollup as well.

Instead, always use this.emitFile.

Parameters

this

PluginContext

outputOptions

NormalizedOutputOptions

bundle

OutputBundle

Provides the full list of files being written or generated along with their details.

isWrite

boolean

Returns

void


renderChunk()

Defined in: plugin/index.ts:403

Can be used to transform individual chunks. Called for each Rolldown output chunk file.

Returning null will apply no transformations. If you change code in this hook and want to support source maps, you need to return a map describing your changes, see Source Code Transformations section.

chunk is mutable and changes applied in this hook will propagate to other plugins and to the generated bundle. That means if you add or remove imports or exports in this hook, you should update imports, RenderedChunk.importedBindings | importedBindings and/or exports accordingly.

Parameters

this

PluginContext

code

string

chunk

RenderedChunk

outputOptions

NormalizedOutputOptions

meta

RenderedChunkMeta

Returns

string | undefined | null | void | BindingMagicString | { code: string | BindingMagicString; map?: SourceMapInput | undefined; }


renderError()

  • Type: (this, error) => void
Defined in: plugin/index.ts:443

Called when Rolldown encounters an error during bundle.generate() or bundle.write().

To get notified when generation completes successfully, use the generateBundle hook.

Parameters

this

PluginContext

error

Error

Returns

void


renderStart()

  • Type: (this, outputOptions, inputOptions) => void
Defined in: plugin/index.ts:385

Called initially each time bundle.generate() or bundle.write() is called.

To get notified when generation has completed, use the generateBundle and renderError hooks.

This is the recommended hook to use when you need access to the output options passed to bundle.generate() or bundle.write() as it takes the transformations by all outputOptions hooks into account and also contains the right default values for unset options.

It also receives the input options passed to rolldown() so that plugins that can be used as output plugins, i.e. plugins that only use generate phase hooks, can get access to them.

Parameters

this

PluginContext

outputOptions

NormalizedOutputOptions

inputOptions

NormalizedInputOptions

Returns

void


writeBundle()

  • Type: (this, outputOptions, bundle) => void
Defined in: plugin/index.ts:470

Called only at the end of bundle.write() once all files have been written.

Parameters

this

PluginContext

outputOptions

NormalizedOutputOptions

bundle

OutputBundle

Provides the full list of files being written or generated along with their details.

Returns

void