An esbuild plugin that resolves bare module imports to fully qualified URLs, perfect for creating CDN-compatible builds.
- Transforms bare module imports to CDN URLs
- Smart entrypoint detection using jsDelivr's API
- Persistent caching for better build performance
- Handles scoped packages and subpath imports correctly
- Supports direct URL mapping for specific packages
- Customizable resolver function for complete control
npm install --save-dev @semantic-ui/esbuild-resolve-bare-imports
import * as esbuild from 'esbuild';
import { resolveBareImports } from '@semantic-ui/esbuild-resolve-bare-imports';
import fs from 'fs/promises';
// Read package.json
const packageJson = JSON.parse(await fs.readFile('./package.json', 'utf-8'));
// Build with CDN imports
await esbuild.build({
entryPoints: ['src/index.js'],
bundle: true,
format: 'esm',
outfile: 'dist/cdn/index.js',
plugins: [
resolveBareImports({ packageJson })
],
});
resolveBareImports({
// Package.json content (for dependency resolution)
packageJson: {},
// Optional: Only transform these dependencies (default: all dependencies)
onlyDependencies: null,
// Directory to store cache in (default: '.cache')
cacheDir: '.cache',
// Base CDN URL (default: 'https://cdn.jsdelivr.net/npm')
cdnRoot: 'https://cdn.jsdelivr.net/npm',
// Logging verbosity: 'silent', 'minimal', 'normal', or 'verbose'
logging: 'normal',
// Custom resolver function (packageName, version, entrypoint) => string
resolver: null,
// Direct replacements for packages - bypasses resolution
directReplacements: {
'lodash': 'https://cdn.example.com/lodash.js'
}
})
You can completely customize how URLs are generated:
resolveBareImports({
packageJson,
resolver: (packageName, version, entrypoint) => {
// Special handling for specific packages
if (packageName === 'react') {
return `https://unpkg.com/${packageName}@${version}/umd/react.production.min.js`;
}
// Default CDN URL pattern for other packages
return `https://cdn.jsdelivr.net/npm/${packageName}@${version}/${entrypoint}`;
}
})
You can explicitly map certain packages to specific URLs:
resolveBareImports({
packageJson,
directReplacements: {
'jquery': 'https://code.jquery.com/jquery-3.6.0.min.js',
'@semantic-ui/utils': 'https://custom-cdn.com/semantic-ui-utils.js'
}
})
The plugin correctly handles subpath imports:
// Input:
import map from 'lodash/map';
import { helpers } from '@semantic-ui/utils/helpers';
// Output (after transformation):
import map from 'https://cdn.jsdelivr.net/npm/lodash@4.17.21/map.js';
import { helpers } from 'https://cdn.jsdelivr.net/npm/@semantic-ui/utils@0.10.0/helpers.js';
MIT