self.window = self; // egregious hack to get magic-string to work in a worker let version; let fulfil; let ready = new Promise(f => fulfil = f); self.addEventListener('message', async event => { switch (event.data.type) { case 'init': version = event.data.version; importScripts( version === 'local' ? '/repl/local?file=compiler.js' : `https://unpkg.com/svelte@${version}/compiler.js`, `https://unpkg.com/rollup@0.68/dist/rollup.browser.js` ); fulfil(); break; case 'bundle': if (event.data.components.length === 0) return; await ready; const result = await bundle(event.data.components); if (result) { postMessage(result); } break; } }); const commonCompilerOptions = { dev: true, }; let cached = { dom: null, ssr: null }; let currentToken; const is_svelte_module = id => id === 'svelte' || id.startsWith('svelte/'); const cache = new Map(); function fetch_if_uncached(url) { if (!cache.has(url)) { cache.set(url, fetch(url.startsWith('https://unpkg.com/svelte@local/') ? '/repl/local?file=' + url.slice(31) : url) .then(r => r.text()) .catch(err => { console.error(err); cache.delete(url); })); } return cache.get(url); } async function getBundle(mode, cache, lookup) { let bundle; let error; let warningCount = 0; const info = {}; try { bundle = await rollup.rollup({ input: './App.html', external: id => { if (id[0] === '.') return false; if (is_svelte_module(id)) return false; if (id.startsWith('https://')) return false; return true; }, plugins: [{ resolveId(importee, importer) { // v3 hack if (importee === `svelte`) return `https://unpkg.com/svelte@${version}/index.mjs`; if (importee.startsWith(`svelte/`)) return `https://unpkg.com/svelte@${version}/${importee.slice(7)}.mjs`; if (importer && importer.startsWith(`https://`)) { return new URL(`${importee}.mjs`, importer).href; } if (importee in lookup) return importee; }, load(id) { if (id.startsWith(`https://`)) return fetch_if_uncached(id); if (id in lookup) return lookup[id].source; }, transform(code, id) { if (!/\.html$/.test(id)) return null; const name = id.replace(/^\.\//, '').replace(/\.html$/, ''); const { js, css, stats } = svelte.compile(code, Object.assign({ generate: mode, format: 'esm', name: name, filename: name + '.html', onwarn: warning => { console.warn(warning.message); console.log(warning.frame); warningCount += 1; }, }, commonCompilerOptions)); return js; } }], onwarn(warning) { console.warn(warning); warningCount += 1; }, cache }); } catch (error) { return { error, bundle: null, info: null, warningCount: null } } return { bundle, info, error: null, warningCount }; } async function bundle(components) { // console.clear(); console.log(`running Svelte compiler version %c${svelte.VERSION}`, 'font-weight: bold'); const token = currentToken = {}; const lookup = {}; components.forEach(component => { const path = `./${component.name}.${component.type}`; lookup[path] = component; }); let dom; let error; try { dom = await getBundle('dom', cached.dom, lookup); if (dom.error) { throw dom.error; } if (token !== currentToken) { console.error(`aborted`); return; } cached.dom = dom.bundle; let uid = 1; const importMap = new Map(); const domResult = await dom.bundle.generate({ format: 'iife', name: 'SvelteComponent', globals: id => { const name = `import_${uid++}`; importMap.set(id, name); return name; }, sourcemap: true }); if (token !== currentToken) return; const ssr = dom.info.usesHooks ? await getBundle('ssr', cached.ssr, lookup) : null; if (ssr) { cached.ssr = ssr.bundle; if (ssr.error) { throw ssr.error; } } if (token !== currentToken) return; const ssrResult = ssr ? await ssr.bundle.generate({ format: 'iife', name: 'SvelteComponent', globals: id => importMap.get(id), sourcemap: true }) : null; return { bundle: { imports: dom.bundle.imports, importMap }, dom: domResult, ssr: ssrResult, warningCount: dom.warningCount, error: null }; } catch (err) { const e = error || err; delete e.toString; return { bundle: null, dom: null, ssr: null, warningCount: dom.warningCount, error: Object.assign({}, e, { message: e.message, stack: e.stack }) }; } }