Popular
123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328import log from "../lib/log.ts";import util from "../lib/util.ts";import type { TransformCSSOptions } from "./deps.ts";import { bold, dirname, ensureDir, esbuild, extname, fromFileUrl, join, stopEsbuild, transformCSS } from "./deps.ts";import depGraph from "./graph.ts";import { builtinModuleExts, existsDir, existsFile, fetchCode, getAlephPkgUri, isNpmPkg, restoreUrl, toLocalPath,} from "./helpers.ts";import { parseHtmlLinks } from "./html.ts";import { initRouter } from "./routing.ts";import type { AlephConfig, ConnInfo } from "./types.ts";
export async function optimize( serverHandler: (req: Request, connInfo: ConnInfo) => Promise<Response>, config: AlephConfig, appDir?: string,) { const start = performance.now(); const alephPkgUri = getAlephPkgUri(); const options = config?.optimization ?? {}; const target = options.buildTarget ?? "es2018"; const outputDir = join(appDir ?? Deno.cwd(), options.outputDir ?? "./output");
// clean previous build if (await existsDir(outputDir)) { for await (const entry of Deno.readDir(outputDir)) { await Deno.remove(join(outputDir, entry.name), { recursive: entry.isDirectory }); } } else { await Deno.mkdir(outputDir, { recursive: true }); }
// find route files by the `routes` config const routeFiles: string[] = []; if (config?.router) { const { routes } = await initRouter(config.router, appDir); routes.forEach(([_, { filename }]) => { routeFiles.push(filename); }); }
// look up client modules let queue = [...routeFiles]; const indexHtml = join(appDir ?? Deno.cwd(), "index.html"); if (await existsFile(indexHtml)) { const html = await Deno.readFile(indexHtml); const links = await parseHtmlLinks(html); for (const src of links) { const url = new URL(src, "http://localhost/"); const ext = extname(url.pathname).slice(1); if (ext === "css" || builtinModuleExts.includes(ext)) { const specifier = util.isLikelyHttpURL(src) ? src : "." + util.cleanPath(src); queue.push(specifier); } } } queue.push(`${alephPkgUri}/runtime/core/nomodule.ts`);
const entryModules = new Map(queue.map((task) => [task, 0])); const allClientModules = new Set<string>(); const memFS = new Map<string, string>();
// transform client modules while (queue.length > 0) { const deps = new Set<string>(); await Promise.all(queue.map(async (specifier) => { const url = new URL(util.isLikelyHttpURL(specifier) ? toLocalPath(specifier) : specifier, "http://localhost"); const isCSS = url.pathname.endsWith(".css"); const req = new Request(url.toString()); let savePath = join(outputDir, url.pathname); if (isNpmPkg(specifier)) { savePath += ".js"; } else if (isCSS && url.searchParams.has("module")) { savePath += ".js"; } const addr: Deno.Addr = { transport: "tcp", hostname: "localhost", port: 80 }; const res = await serverHandler(req, { localAddr: addr, remoteAddr: addr }); if ( res.status !== 200 || res.headers.get("Content-Type")?.startsWith("text/html") || res.headers.has("X-Transform-Error") ) { throw new Error("Transform Error"); } memFS.set(savePath, await res.text()); if (!isCSS) { depGraph?.get(specifier)?.deps?.forEach(({ specifier, dynamic }) => { if (dynamic) { entryModules.set(specifier, 1); } if (specifier.endsWith(".css")) { deps.add(specifier + "?module"); } else { deps.add(specifier); } }); } else if (url.searchParams.has("module")) { deps.add(`${alephPkgUri}/runtime/core/style.ts`); } else { await ensureDir(dirname(savePath)); await Deno.writeTextFile(savePath, memFS.get(savePath)!); } allClientModules.add(specifier); })); queue = Array.from(deps).filter((specifier) => !allClientModules.has(specifier)); }
const clientModules = new Map(entryModules); const refs = new Map<string, Set<string>>();
// count client module refs for (const [name] of entryModules) { depGraph?.walk(name, ({ specifier }, importer) => { if (importer) { let set = refs.get(specifier); if (!set) { set = new Set<string>(); refs.set(specifier, set); } set.add(importer.specifier); } }); }
// find shared modules for (const [specifier, counter] of refs) { if (counter.size > 1) { clientModules.set(specifier, 2); } }
// hygiene checks, make sure all shared modules are not only referenced by other shared modules for (let i = 0; i < 100; i++) { const toHygiene = new Set<string>(); for (const [specifier, type] of clientModules) { if (type === 2) { const sharedBy = new Set<string>(); depGraph?.lookup(specifier, (specifier) => { if (clientModules.has(specifier)) { sharedBy.add(specifier); return false; } }); if (sharedBy.size === 1) { toHygiene.add(specifier); } } } // break the loop when there are no more modules to hygiene if (toHygiene.size === 0) { break; } toHygiene.forEach((specifier) => clientModules.delete(specifier)); log.debug(`hygiene#${i + 1}`, toHygiene); }
// bundle client modules await Promise.all( Array.from(clientModules.keys()).map(async (entryPoint) => { const url = new URL(util.isLikelyHttpURL(entryPoint) ? toLocalPath(entryPoint) : entryPoint, "http://localhost"); if (url.pathname.endsWith(".css")) { return; } let jsFile = join(outputDir, url.pathname); if (isNpmPkg(entryPoint)) { jsFile += ".js"; } await esbuild({ entryPoints: [jsFile], outfile: jsFile, allowOverwrite: true, platform: "browser", format: "esm", target: [target], bundle: true, minify: true, treeShaking: true, sourcemap: false, plugins: [{ name: "bundle-client-modules", setup(build) { build.onResolve({ filter: /.*/ }, (args) => { let argsPath = args.path; if (argsPath.startsWith("./") || argsPath.startsWith("../")) { argsPath = join(args.resolveDir, argsPath); } const [fp, q] = util.splitBy(argsPath, "?"); const path = util.trimPrefix(fp, outputDir); let specifier = "." + path; if (args.path.startsWith("/-/")) { specifier = restoreUrl(path); } if (clientModules.has(specifier) && specifier !== entryPoint) { return { path: [path, q].filter(Boolean).join("?"), external: true, }; } let jsFile = join(outputDir, path); if (isNpmPkg(specifier)) { jsFile += ".js"; } else if (specifier.endsWith(".css") && new URLSearchParams(q).has("module")) { jsFile += ".js"; } return { path: jsFile }; }); build.onLoad({ filter: /.*/ }, (args) => { return { contents: memFS.get(args.path), loader: "js", }; }); }, }], }); }), );
stopEsbuild(); memFS.clear();
log.info(`${bold(routeFiles.length.toString())} routes found`); log.info(`${bold(clientModules.size.toString())} client modules built`); log.info(`Done in ${(performance.now() - start).toFixed(2)}ms`); Deno.exit(0);}
/** Bundle the css using `parcel-css` with `nesting` and `customMedia` draft support. */export async function bundleCSS( specifier: string, sourceCode: string, options: { asJsModule?: boolean; hmr?: boolean; } & TransformCSSOptions, _tracing = new Set<string>(),): Promise<{ code: string; cssModulesExports?: Record<string, string>; deps?: string[];}> { let { code: css, dependencies, exports } = await transformCSS( specifier, sourceCode, { ...options, analyzeDependencies: true, drafts: { nesting: true, customMedia: true, }, }, ); const deps = dependencies?.filter((dep) => dep.type === "import" && !dep.media).map((dep) => { let url = dep.url; if (util.isLikelyHttpURL(specifier)) { if (!util.isLikelyHttpURL(url)) { url = new URL(url, specifier).toString(); } } else { url = "." + fromFileUrl(new URL(url, `file://${specifier.slice(1)}`)); } return url; }); dependencies?.forEach((dep) => { if (dep.type === "url") { // todo: use magic-string css = css.replace(`url("${dep.placeholder}")`, `url("${dep.url}")`); } }); const eof = options.minify ? "" : "\n"; if (deps) { const imports = await Promise.all(deps.map(async (url) => { if (_tracing.has(url)) { return ""; } _tracing.add(url); const [css] = await fetchCode(url); const { code, deps: subDeps } = await bundleCSS( url, css, { targets: options.targets, minify: options.minify, }, _tracing, ); if (subDeps) { deps.push(...subDeps); } return code; })); css = imports.join(eof) + eof + css; } const cssModulesExports: Record<string, string> = {}; if (exports) { for (const [key, value] of Object.entries(exports)) { cssModulesExports[key] = value.name; } } if (options.asJsModule) { const alephPkgPath = toLocalPath(getAlephPkgUri()); return { code: [ options.hmr && `import createHotContext from "${alephPkgPath}/runtime/core/hmr.ts";`, options.hmr && `import.meta.hot = createHotContext(${JSON.stringify(specifier)});`, `import { applyCSS } from "${alephPkgPath}/runtime/core/style.ts";`, `export const css = ${JSON.stringify(css)};`, `export default ${JSON.stringify(cssModulesExports)};`, `applyCSS(${JSON.stringify(specifier)}, css);`, options.hmr && `import.meta.hot.accept();`, ].filter(Boolean).join(eof), deps, cssModulesExports, }; } return { code: css, cssModulesExports, deps };}