-
Notifications
You must be signed in to change notification settings - Fork 546
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
- Loading branch information
1 parent
ea98900
commit 084090b
Showing
10 changed files
with
734 additions
and
338 deletions.
There are no files selected for viewing
Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.
Oops, something went wrong.
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file was deleted.
Oops, something went wrong.
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -1,129 +1,136 @@ | ||
import _ from 'lodash-es' | ||
import _ from 'lodash-es'; | ||
import PromiseWorker from 'promise-worker'; | ||
import {get} from 'svelte/store' | ||
import {site} from '@primocms/builder' | ||
import {locale} from '@primocms/builder' | ||
import rollupWorker from './workers/rollup.worker.js?worker' | ||
import postCSSWorker from './workers/postcss.worker.js?worker' | ||
import { get } from 'svelte/store'; | ||
import { site } from '@primocms/builder'; | ||
import { locale } from '@primocms/builder'; | ||
import rollupWorker from './workers/rollup.worker.js?worker'; | ||
import postCSSWorker from './workers/postcss.worker.js?worker'; | ||
|
||
const cssPromiseWorker = new PromiseWorker(new postCSSWorker()); | ||
const htmlPromiseWorker = new PromiseWorker(new rollupWorker()); | ||
const postcss_worker = new PromiseWorker(new postCSSWorker()); | ||
const rollup_worker = new PromiseWorker(new rollupWorker()); | ||
|
||
const componentsMap = new Map(); | ||
const COMPILED_COMPONENTS_CACHE = new Map(); | ||
|
||
export async function html({ component, buildStatic = true, format = 'esm'}) { | ||
/** | ||
* Compiles and renders a given component or page, caching the result. | ||
* @async | ||
* @param {Object} options - The options for rendering. | ||
* @param {Object|Object[]} options.component - The component(s) to be rendered. Can be a single component or an array of components for a page. | ||
* @param {boolean} [options.buildStatic=true] - Indicates whether to build the component statically or not. | ||
* @param {string} [options.format='esm'] - The module format to use, such as 'esm' for ES Modules. | ||
* @returns {Promise<Object>} Returns a payload containing the rendered HTML, CSS, JS, and other relevant data. | ||
* @throws {Error} Throws an error if the compilation or rendering fails. | ||
*/ | ||
export async function html({ component, buildStatic = true, format = 'esm' }) { | ||
let cache_key; | ||
if (!buildStatic) { | ||
cache_key = JSON.stringify({ | ||
component, | ||
format | ||
}); | ||
if (COMPILED_COMPONENTS_CACHE.has(cache_key)) { | ||
return COMPILED_COMPONENTS_CACHE.get(cache_key); | ||
} | ||
} | ||
|
||
// return { | ||
// error: 'none' | ||
// } | ||
const compile_page = Array.isArray(component); | ||
|
||
let cacheKey | ||
if (!buildStatic) { | ||
cacheKey = JSON.stringify({ | ||
component, | ||
format | ||
}) | ||
if (componentsMap.has(cacheKey)) { | ||
const cached = componentsMap.get(cacheKey) | ||
return cached | ||
} | ||
} | ||
let res; | ||
try { | ||
const has_js = compile_page ? component.some((s) => s.js) : !!component.js; | ||
res = await rollup_worker.postMessage({ | ||
component, | ||
hydrated: buildStatic && has_js, | ||
buildStatic, | ||
format, | ||
site: get(site), | ||
locale: get(locale) | ||
}); | ||
} catch (e) { | ||
console.log('error', e); | ||
res = { | ||
error: e.toString() | ||
}; | ||
} | ||
|
||
let res | ||
try { | ||
const has_js = Array.isArray(component) ? component.some(s => s.js) : !!component.js | ||
res = await htmlPromiseWorker.postMessage({ | ||
component, | ||
hydrated: buildStatic && has_js, | ||
buildStatic, | ||
format, | ||
site: get(site), | ||
locale: get(locale) | ||
}) | ||
} catch(e) { | ||
console.log('error', e) | ||
res = { | ||
error: e.toString() | ||
} | ||
} | ||
let payload; | ||
|
||
let final | ||
if (!res) { | ||
payload = { | ||
html: '<h1 style="text-align: center">could not render</h1>' | ||
}; | ||
res = {}; | ||
} else if (res.error) { | ||
console.error(res.error); | ||
payload = { | ||
error: escapeHtml(res.error) | ||
}; | ||
function escapeHtml(unsafe) { | ||
return unsafe | ||
.replace(/&/g, '&') | ||
.replace(/</g, '<') | ||
.replace(/>/g, '>') | ||
.replace(/"/g, '"') | ||
.replace(/'/g, '''); | ||
} | ||
} else if (buildStatic) { | ||
const blob = new Blob([res.ssr], { type: 'text/javascript' }); | ||
const url = URL.createObjectURL(blob); | ||
const { default: App } = await import(url /* @vite-ignore */); | ||
URL.revokeObjectURL(url); | ||
|
||
if (!res) { | ||
final = { | ||
html: '<h1 style="text-align: center">could not render</h1>' | ||
} | ||
res = {} | ||
} else if (res.error) { | ||
console.error(res.error) | ||
final = { | ||
error: escapeHtml(res.error) | ||
} | ||
function escapeHtml(unsafe) { | ||
return unsafe | ||
.replace(/&/g, "&") | ||
.replace(/</g, "<") | ||
.replace(/>/g, ">") | ||
.replace(/"/g, """) | ||
.replace(/'/g, "'"); | ||
} | ||
} else if (buildStatic) { | ||
const blob = new Blob([res.ssr], { type: 'text/javascript' }); | ||
const url = URL.createObjectURL(blob); | ||
const {default:App} = await import(url/* @vite-ignore */) | ||
let component_data; | ||
if (compile_page) { | ||
// get the component data for the page | ||
component_data = component.reduce((accumulator, item, i) => { | ||
if (!_.isEmpty(item.data)) { | ||
accumulator[`component_${i}_props`] = item.data; | ||
} | ||
return accumulator; | ||
}, {}); | ||
} else { | ||
component_data = component.data; | ||
} | ||
|
||
let component_data | ||
if (Array.isArray(component)) { | ||
component_data = component.reduce((accumulator, item, i) => { | ||
if (!_.isEmpty(item.data)) { | ||
accumulator[`component_${i}_props`] = item.data | ||
} | ||
return accumulator; | ||
}, {}); | ||
} else { | ||
component_data = component.data | ||
} | ||
const rendered = App.render(component_data); | ||
|
||
const rendered = App.render(component_data) | ||
payload = { | ||
head: rendered.head, | ||
html: rendered.html, | ||
css: rendered.css.code, | ||
js: res.dom | ||
}; | ||
} else { | ||
payload = { | ||
js: res.dom | ||
}; | ||
} | ||
|
||
final = { | ||
head: rendered.head, | ||
html: rendered.html, | ||
css: rendered.css.code, | ||
js: res.dom | ||
} | ||
} else { | ||
final = { | ||
js: res.dom | ||
} | ||
} | ||
if (!buildStatic) { | ||
COMPILED_COMPONENTS_CACHE.set(cache_key, payload); | ||
} | ||
|
||
if (!buildStatic) { | ||
componentsMap.set(cacheKey, final) | ||
} | ||
|
||
return final | ||
return payload; | ||
} | ||
|
||
|
||
const cssMap = new Map() | ||
const cssMap = new Map(); | ||
export async function css(raw) { | ||
// return { | ||
// css: '' | ||
// } | ||
if (cssMap.has(raw)) { | ||
return ({ css: cssMap.get(raw) }) | ||
} | ||
const processed = await cssPromiseWorker.postMessage({ | ||
css: raw | ||
}) | ||
if (processed.message) { | ||
return { | ||
error: processed.message | ||
} | ||
} | ||
cssMap.set(raw, processed) | ||
return { | ||
css: processed | ||
} | ||
} | ||
// return { | ||
// css: '' | ||
// } | ||
if (cssMap.has(raw)) { | ||
return { css: cssMap.get(raw) }; | ||
} | ||
const processed = await postcss_worker.postMessage({ | ||
css: raw | ||
}); | ||
if (processed.message) { | ||
return { | ||
error: processed.message | ||
}; | ||
} | ||
cssMap.set(raw, processed); | ||
return { | ||
css: processed | ||
}; | ||
} |
Oops, something went wrong.