Skip to content

Commit 160ec72

Browse files
lforstantfu
andauthoredOct 22, 2022
feat: add writeBundle hook (#179)
Co-authored-by: Anthony Fu <anthonyfu117@hotmail.com>
1 parent ab8fe74 commit 160ec72

File tree

9 files changed

+178
-7
lines changed

9 files changed

+178
-7
lines changed
 

‎README.md

+7-5
Original file line numberDiff line numberDiff line change
@@ -27,10 +27,12 @@ Currently supports:
2727
| [`transform`](https://rollupjs.org/guide/en/#transformers) ||||| ✅ <sup>3</sup> |
2828
| [`watchChange`](https://rollupjs.org/guide/en/#watchchange) ||||||
2929
| [`buildEnd`](https://rollupjs.org/guide/en/#buildend) ||||||
30+
| [`writeBundle`](https://rollupjs.org/guide/en/#writebundle)<sup>4</sup> ||||||
3031

3132
1. Rollup and esbuild do not support using `enforce` to control the order of plugins. Users need to maintain the order manually.
3233
2. Webpack's id filter is outside of loader logic; an additional hook is needed for better perf on Webpack. In Rollup and Vite, this hook has been polyfilled to match the behaviors. See for following usage examples.
3334
3. Although esbuild can handle both JavaScript and CSS and many other file formats, you can only return JavaScript in `load` and `transform` results.
35+
4. Currently, `writeBundle` is only serves as a hook for the timing. It doesn't pass any arguments.
3436

3537
### Hook Context
3638

@@ -40,13 +42,13 @@ Currently supports:
4042
| ---- | :----: | :--: | :-------: | :-------: | :-----: |
4143
| [`this.parse`](https://rollupjs.org/guide/en/#thisparse) ||||||
4244
| [`this.addWatchFile`](https://rollupjs.org/guide/en/#thisaddwatchfile) ||||||
43-
| [`this.emitFile`](https://rollupjs.org/guide/en/#thisemitfile)<sup>4</sup> ||||||
45+
| [`this.emitFile`](https://rollupjs.org/guide/en/#thisemitfile)<sup>5</sup> ||||||
4446
| [`this.getWatchFiles`](https://rollupjs.org/guide/en/#thisgetwatchfiles) ||||||
4547
| [`this.warn`](https://rollupjs.org/guide/en/#thiswarn) ||||||
4648
| [`this.error`](https://rollupjs.org/guide/en/#thiserror) ||||||
4749

4850

49-
4. Currently, [`this.emitFile`](https://rollupjs.org/guide/en/#thisemitfile) only supports the `EmittedAsset` variant.
51+
5. Currently, [`this.emitFile`](https://rollupjs.org/guide/en/#thisemitfile) only supports the `EmittedAsset` variant.
5052

5153
## Usage
5254

@@ -83,10 +85,10 @@ Since `v0.10.0`, unplugin supports constructing multiple nested plugins to behav
8385

8486
| Rollup | Vite | Webpack 4 | Webpack 5 | esbuild |
8587
| :----: | :--: | :-------: | :-------: | :-----: |
86-
|`>=3.1`<sup>5</sup> |||| ⚠️<sup>6</sup> |
88+
|`>=3.1`<sup>6</sup> |||| ⚠️<sup>7</sup> |
8789

88-
5. Rollup supports nested plugins since [v3.1.0](https://github.com/rollup/rollup/releases/tag/v3.1.0). Plugin aurthor should ask users to a have a Rollup version of `>=3.1.0` when using nested plugins. For singe plugin format, unplugin works for any versions of Rollup.
89-
6. Since esbuild does not have a built-in transform phase, the `transform` hook of nested plugin will not work on esbuild yet. Other hooks like `load` or `resolveId` work fine. We will try to find a way to support it in the future.
90+
6. Rollup supports nested plugins since [v3.1.0](https://github.com/rollup/rollup/releases/tag/v3.1.0). Plugin aurthor should ask users to a have a Rollup version of `>=3.1.0` when using nested plugins. For singe plugin format, unplugin works for any versions of Rollup.
91+
7. Since esbuild does not have a built-in transform phase, the `transform` hook of nested plugin will not work on esbuild yet. Other hooks like `load` or `resolveId` work fine. We will try to find a way to support it in the future.
9092

9193
###### Usage
9294

‎src/esbuild/index.ts

+9-2
Original file line numberDiff line numberDiff line change
@@ -60,14 +60,21 @@ export function getEsbuildPlugin <UserOptions = {}> (
6060
onStart(() => plugin.buildStart!.call(context))
6161
}
6262

63-
if (plugin.buildEnd || initialOptions.watch) {
63+
if (plugin.buildEnd || plugin.writeBundle || initialOptions.watch) {
6464
const rebuild = () => build({
6565
...initialOptions,
6666
watch: false
6767
})
6868

6969
onEnd(async () => {
70-
await plugin.buildEnd!.call(context)
70+
if (plugin.buildEnd) {
71+
await plugin.buildEnd.call(context)
72+
}
73+
74+
if (plugin.writeBundle) {
75+
await plugin.writeBundle()
76+
}
77+
7178
if (initialOptions.watch) {
7279
Object.keys(watchListRecord).forEach((id) => {
7380
if (!watchList.has(id)) {

‎src/types.ts

+4
Original file line numberDiff line numberDiff line change
@@ -29,13 +29,17 @@ export interface UnpluginOptions {
2929
name: string;
3030
enforce?: 'post' | 'pre' | undefined;
3131

32+
// Build Hooks
3233
buildStart?: (this: UnpluginBuildContext) => Promise<void> | void;
3334
buildEnd?: (this: UnpluginBuildContext) => Promise<void> | void;
3435
transform?: (this: UnpluginBuildContext & UnpluginContext, code: string, id: string) => Thenable<TransformResult>;
3536
load?: (this: UnpluginBuildContext & UnpluginContext, id: string) => Thenable<TransformResult>
3637
resolveId?: (id: string, importer: string | undefined, options: { isEntry: boolean }) => Thenable<string | ExternalIdResult | null | undefined>
3738
watchChange?: (this: UnpluginBuildContext, id: string, change: { event: 'create' | 'update' | 'delete' }) => void
3839

40+
// Output Generation Hooks
41+
writeBundle?: (this: void) => Promise<void> | void
42+
3943
/**
4044
* Custom predicate function to filter modules to be loaded.
4145
* When omitted, all modules will be included (might have potential perf impact on Webpack).

‎src/webpack/index.ts

+6
Original file line numberDiff line numberDiff line change
@@ -218,6 +218,12 @@ export function getWebpackPlugin<UserOptions = {}> (
218218
await plugin.buildEnd!.call(createContext(compilation))
219219
})
220220
}
221+
222+
if (plugin.writeBundle) {
223+
compiler.hooks.afterEmit.tap(plugin.name, () => {
224+
plugin.writeBundle!()
225+
})
226+
}
221227
}
222228
}
223229
}

‎test/unit-tests/utils.ts

+2
Original file line numberDiff line numberDiff line change
@@ -10,6 +10,8 @@ export const rollupBuild = rollup.rollup
1010
export const esbuildBuild = esbuild.build
1111
export const webpackBuild = (webpack.webpack || (webpack as any).default || webpack) as typeof webpack.webpack
1212

13+
export const webpackVersion = ((webpack as any).default || webpack).version
14+
1315
export const build = {
1416
webpack: webpackBuild,
1517
rollup: rollupBuild,
+1
Original file line numberDiff line numberDiff line change
@@ -0,0 +1 @@
1+
test-out
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,3 @@
1+
import someString, { someOtherString } from './import'
2+
3+
process.stdout.write(JSON.stringify({ someString, someOtherString }))
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,3 @@
1+
export default 'some string'
2+
3+
export const someOtherString = 'some other string'
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,143 @@
1+
import * as path from 'path'
2+
import * as fs from 'fs'
3+
import { it, describe, expect, vi, afterEach, Mock, beforeAll } from 'vitest'
4+
import { build, toArray, webpackVersion } from '../utils'
5+
import { createUnplugin, UnpluginOptions, VitePlugin } from 'unplugin'
6+
7+
function createUnpluginWithCallback (writeBundleCallback: UnpluginOptions['writeBundle']) {
8+
return createUnplugin(() => ({
9+
name: 'test-plugin',
10+
writeBundle: writeBundleCallback
11+
}))
12+
}
13+
14+
function generateMockWriteBundleHook (outputPath: string) {
15+
return () => {
16+
// We want to check that at the time the `writeBundle` hook is called, all
17+
// build-artifacts have already been written to disk.
18+
19+
const bundleExists = fs.existsSync(path.join(outputPath, 'output.js'))
20+
const sourceMapExists = fs.existsSync(path.join(outputPath, 'output.js.map'))
21+
22+
expect(bundleExists).toBe(true)
23+
expect(sourceMapExists).toBe(true)
24+
25+
return undefined
26+
}
27+
}
28+
29+
// We extract this check because all bundlers should behave the same
30+
function checkWriteBundleHook (writeBundleCallback: Mock): void {
31+
expect(writeBundleCallback).toHaveBeenCalledOnce()
32+
}
33+
34+
describe('writeBundle hook', () => {
35+
beforeAll(() => {
36+
fs.rmSync(path.resolve(__dirname, 'test-out'), { recursive: true, force: true })
37+
})
38+
39+
afterEach(() => {
40+
vi.restoreAllMocks()
41+
})
42+
43+
it('vite', async () => {
44+
expect.assertions(3)
45+
const mockWriteBundleHook = vi.fn(generateMockWriteBundleHook(path.resolve(__dirname, 'test-out/vite')))
46+
const plugin = createUnpluginWithCallback(mockWriteBundleHook).vite
47+
// we need to define `enforce` here for the plugin to be run
48+
const plugins = toArray(plugin()).map((plugin): VitePlugin => ({ ...plugin, enforce: 'pre' }))
49+
50+
await build.vite({
51+
clearScreen: false,
52+
plugins: [plugins],
53+
build: {
54+
lib: {
55+
entry: path.resolve(__dirname, 'test-src/entry.js'),
56+
name: 'TestLib',
57+
fileName: 'output',
58+
formats: ['cjs']
59+
},
60+
outDir: path.resolve(__dirname, 'test-out/vite'),
61+
sourcemap: true
62+
}
63+
})
64+
65+
checkWriteBundleHook(mockWriteBundleHook)
66+
})
67+
68+
it('rollup', async () => {
69+
expect.assertions(3)
70+
const mockResolveIdHook = vi.fn(generateMockWriteBundleHook(path.resolve(__dirname, 'test-out/rollup')))
71+
const plugin = createUnpluginWithCallback(mockResolveIdHook).rollup
72+
73+
const rollupBuild = await build.rollup({
74+
input: path.resolve(__dirname, 'test-src/entry.js')
75+
})
76+
77+
await rollupBuild.write({
78+
plugins: [plugin()],
79+
file: path.resolve(__dirname, 'test-out/rollup/output.js'),
80+
format: 'cjs',
81+
exports: 'named',
82+
sourcemap: true
83+
})
84+
85+
checkWriteBundleHook(mockResolveIdHook)
86+
})
87+
88+
it('webpack', async () => {
89+
expect.assertions(3)
90+
const mockResolveIdHook = vi.fn(generateMockWriteBundleHook(path.resolve(__dirname, 'test-out/webpack')))
91+
const plugin = createUnpluginWithCallback(mockResolveIdHook).webpack
92+
93+
const webpack4Options = {
94+
entry: path.resolve(__dirname, 'test-src/entry.js'),
95+
cache: false,
96+
output: {
97+
path: path.resolve(__dirname, 'test-out/webpack'),
98+
filename: 'output.js',
99+
libraryTarget: 'commonjs'
100+
},
101+
plugins: [plugin()],
102+
devtool: 'source-map'
103+
}
104+
105+
const webpack5Options = {
106+
entry: path.resolve(__dirname, 'test-src/entry.js'),
107+
plugins: [plugin()],
108+
devtool: 'source-map',
109+
output: {
110+
path: path.resolve(__dirname, 'test-out/webpack'),
111+
filename: 'output.js',
112+
library: {
113+
type: 'commonjs'
114+
}
115+
}
116+
}
117+
118+
await new Promise((resolve) => {
119+
build.webpack(webpackVersion!.startsWith('4') ? webpack4Options : webpack5Options,
120+
resolve
121+
)
122+
})
123+
124+
checkWriteBundleHook(mockResolveIdHook)
125+
})
126+
127+
it('esbuild', async () => {
128+
expect.assertions(3)
129+
const mockResolveIdHook = vi.fn(generateMockWriteBundleHook(path.resolve(__dirname, 'test-out/esbuild')))
130+
const plugin = createUnpluginWithCallback(mockResolveIdHook).esbuild
131+
132+
await build.esbuild({
133+
entryPoints: [path.resolve(__dirname, 'test-src/entry.js')],
134+
plugins: [plugin()],
135+
bundle: true, // actually traverse imports
136+
outfile: path.resolve(__dirname, 'test-out/esbuild/output.js'),
137+
format: 'cjs',
138+
sourcemap: true
139+
})
140+
141+
checkWriteBundleHook(mockResolveIdHook)
142+
})
143+
})

0 commit comments

Comments
 (0)
Please sign in to comment.