Bundle static imported bindings from modules which will be dynamically imported somewhere, and treeshake those static imported bindings from the dynamic imported modules
Note
This plugin is only used in build time.
pnpm add rollup-plugin-proxy-dynamic-import -D
In a nutshell, this plugin does this (using SvelteKit naming conventions as an example):
Input | Output |
---|---|
// +page.js
// This exported `route_config` will get
// bundled into router.js and treeshake out
// from this module after build.
export const route_config = {
path: '/',
component: () => import('./comp.vue'),
meta: {
service: () => import('./+page.js'),
},
};
export function load() {
// omitted for example
return { a: 1 };
}
export const actions = {
default() {
// omitted for example
throw redirect('/user/123');
},
}; // router.js
import { route_config } from './+page.js';
export const router = createRouter({
routes: [route_config],
}); |
// _page-[hash].js
// Notice that there is no `route_config`
export function load() {
// omitted for example
return { a: 1 };
}
export const actions = {
default() {
// omitted for example
throw redirect('/user/123');
},
}; // router-[hash].js
// Hey, I got bundled into router.js
const route_config = {
path: '/',
component: () => import('./comp-[hash].vue'),
meta: {
service: () => import('./_page-[hash].js'),
},
};
export const router = createRouter({
routes: [route_config],
}); |
In the past, before file system based routing, we configure the routes in the giant route file. Later, there has been many improvements / ideas in the modern web development, namely file based routing, and then Remix Route Module, and now SvelteKit and Next.js directory based routing.
Personally, I prefer config based routing to FS based one, but I am not eager to
define every route in a giant route file. At the same time, I track SvelteKit
development, and saw their directory based routing proposal. It gives you the
structure for co-locating data loading code, view code, and the rest. Since
SvelteKit routing is FS based, they don't need another file for routing. But for
config based routing, what is the best place to define routing? In +routing.js
(taken from Angular naming convention) ?
If possible, I would like to define them in +page.js
and +layout.js
and
import them in router file with Vite's import.meta.glob
. That way, the app
will still work even if you delete one route. However, in JavaScript, when you
import a module, you get a module even without using every exports (which hurt
the purpose of using dynamic import for lazy loading).
What if we could separate the static imported bindings into another module and the rest in the original module at build time? This way, we could still be able to import static bindings and get the benefit of lazy loading. This plugin does that for you. Without further ado, let's use it.
This plugin works in both Rollup and Vite setup.
// rollup.config.js / vite.config.js
import { proxy_dynamic_import } from 'rollup-plugin-proxy-dynamic-import';
export default {
plugins: [proxy_dynamic_import(opts)],
};
export interface ProxyDynamicImportOptions {
/**
* `include` argument of {@link createFilter}.
*/
include: FilterPattern;
/**
* `exclude` argument of {@link createFilter}.
*/
exclude?: FilterPattern;
/**
* This option is required to be defined to treeshake only used exports in
* dynamic imports. This plugin creates facade modules to re-export used bindings
* so that Rollup can treeshake ununsed ones. This is a workaround until Rollup
* can treeshake dynamic imports OOTB.
*
* https://github.com/rollup/rollup/issues/3447
*
* Export names to re-export. Use `*` to re-export all, not that it does not
* re-export `default` export.
*/
reexports?: string[] | '*';
/**
* List of file extensions to resolve for imports that omit extensions.
*
* When used as a Vite plugin, it takes `resolve.extensions` instead of given
* value without `.json`.
*
* When used as a Rollup plugin, its default is
* `['.js'. '.ts', '.jsx', '.tsx', '.mjs']`.
*/
extensions?: string[];
}
Use snake_case
everywhere, the rest is handled by Prettier.
-
Install deps:
pnpm i
-
Format Code
pnpm fmt
-
Lint
pnpm lint
-
Typecheck
pnpm tsc
-
Test
pnpm run test