diff --git a/README.md b/README.md index 42d9caf4..7053aa56 100644 --- a/README.md +++ b/README.md @@ -146,3 +146,7 @@ Use the `./scripts/run-version.sh` script setting the branch references for the This repo has two main parts: `app/server` (Express server) and `app/static` (Vue frontend). The normal way to spin up the app without hot reloading is build frontend, copy build files into public folder in the server (`npm run build --prefix=app/static` builds the frontend and copies the files to the server) and run the server (`npm run serve --prefix=app/server`). The html file will then look for its javascript src `wodin.js` and css src `wodin.css` in the `app/server/public` directory and load the app from these. The hot reloading setup is different. The `server.js` file (from building the server) takes in a `--hot-reload` boolean option which tells it to instead look for the javascript src at `http://localhost:5173/src/wodin.ts` since `wodin.ts` is our entrypoint (`http://localhost:5173` is the default url for `vite` dev mode and it serves the files, after transpiling to javascript, based on your folder structure) and css src at `http://localhost:5173/src/scss/style.scss`. To run it this way we run `npm run dev --prefix=app/static` to start the vite dev server and then we run `npm run serve-hot --prefix=app/server`. + +## Static build + +For details about the static build please read the README [here](./config-static/README.md) diff --git a/app/server/src/static-site-builder/builder.ts b/app/server/src/static-site-builder/builder.ts new file mode 100644 index 00000000..e60f5ce5 --- /dev/null +++ b/app/server/src/static-site-builder/builder.ts @@ -0,0 +1,43 @@ +import path from "path"; +import fs from "fs"; +import { readFile } from "../configReader"; +import axios from "axios"; + +export const buildWodinStaticSite = async (configPath: string, destPath: string) => { + const storesPath = path.resolve(configPath, "stores") + const stores = fs.readdirSync(storesPath, { recursive: false }) as string[]; + + // bootstrapping and making sure the folder structure is correct + if (!fs.existsSync(destPath)) { + fs.mkdirSync(destPath); + } + const destStoresPath = path.resolve(destPath, "stores"); + if (fs.existsSync(destStoresPath)) { + fs.rmSync(destStoresPath, { recursive: true }); + } + fs.mkdirSync(destStoresPath); + + // copy in their config specific files (assets to come soon) + fs.cpSync(path.resolve(configPath, "index.html"), path.resolve(destPath, "index.html")); + + // get runners + const runnerOdeResponse = await axios.get("http://localhost:8001/support/runner-ode"); + fs.writeFileSync(path.resolve(destStoresPath, "runnerOde.js"), runnerOdeResponse.data.data); + const runnerDiscreteResponse = await axios.get("http://localhost:8001/support/runner-discrete"); + fs.writeFileSync(path.resolve(destStoresPath, "runnerDiscrete.js"), runnerDiscreteResponse.data.data); + + // make folder per store with config.json (their config + default code) and + // model.json (model response from odin.api) + stores.forEach(async store => { + const destStorePath = path.resolve(destStoresPath, store); + fs.mkdirSync(destStorePath); + const model = readFile(path.resolve(storesPath, store, "model.R")).split("\n"); + + const config = JSON.parse(readFile(path.resolve(storesPath, store, "config.json"))) as { appType: string }; + fs.writeFileSync(path.resolve(destStorePath, "config.json"), JSON.stringify({ ...config, defaultCode: model })); + + const timeType = config.appType === "stochastic" ? "discrete" : "continuous"; + const modelResponse = await axios.post("http://localhost:8001/compile", { model, requirements: { timeType } }); + fs.writeFileSync(path.resolve(destStorePath, `model.json`), JSON.stringify(modelResponse.data.data)); + }); +}; diff --git a/app/server/src/static-site-builder/wodinBuilder.ts b/app/server/src/static-site-builder/wodinBuilder.ts index 6cb0ca2e..f8e09f99 100644 --- a/app/server/src/static-site-builder/wodinBuilder.ts +++ b/app/server/src/static-site-builder/wodinBuilder.ts @@ -1,50 +1,6 @@ -import path from "path"; -import fs from "fs"; -import { readFile } from "../configReader"; -import axios from "axios"; import { processArgs } from "./args"; - -const mkdirForce = (path: string) => { - if (!fs.existsSync(path)) { - fs.mkdirSync(path); - } -}; +import { buildWodinStaticSite } from "./builder"; const { configPath, destPath } = processArgs(); -const storesPath = path.resolve(configPath, "stores") -const stores = fs.readdirSync(storesPath, { recursive: false }) as string[]; - -mkdirForce(destPath); -const destStoresPath = path.resolve(destPath, "stores"); -if (fs.existsSync(destStoresPath)) { - fs.rmSync(destStoresPath, { recursive: true }); -} -fs.mkdirSync(destStoresPath); - -fs.cpSync(path.resolve(configPath, "index.html"), path.resolve(destPath, "index.html")); - -const getRunners = async () => { - const runnerOdeResponse = await axios.get("http://localhost:8001/support/runner-ode"); - fs.writeFileSync(path.resolve(destStoresPath, "runnerOde.js"), runnerOdeResponse.data.data); - - const runnerDiscreteResponse = await axios.get("http://localhost:8001/support/runner-discrete"); - fs.writeFileSync(path.resolve(destStoresPath, "runnerDiscrete.js"), runnerDiscreteResponse.data.data); -} -getRunners(); - -stores.forEach(async store => { - const destStorePath = path.resolve(destStoresPath, store); - fs.mkdirSync(destStorePath); - - const model = readFile(path.resolve(storesPath, store, "model.R")).split("\n"); - - const config = JSON.parse(readFile(path.resolve(storesPath, store, "config.json"))) as { appType: string }; - fs.writeFileSync(path.resolve(destStorePath, "config.json"), JSON.stringify({ ...config, defaultCode: model })); - - const modelResponse = await axios.post("http://localhost:8001/compile", { - model, - requirements: { timeType: config.appType === "stochastic" ? "discrete" : "continuous" } - }); - fs.writeFileSync(path.resolve(destStorePath, `model.json`), JSON.stringify(modelResponse.data.data)); -}); +buildWodinStaticSite(configPath, destPath); diff --git a/app/server/tests/static-site-builder/args.test.ts b/app/server/tests/static-site-builder/args.test.ts new file mode 100644 index 00000000..b8ffb645 --- /dev/null +++ b/app/server/tests/static-site-builder/args.test.ts @@ -0,0 +1,29 @@ +import { processArgs } from "../../src/static-site-builder/args"; + +describe("args", () => { + const realArgs = process.argv; + afterEach(() => { + process.argv = realArgs; + }); + + const builderPath = "/builder", + cfgPath = "/configPath", + destPath = "/destPath"; + + it("throws error if no config path or dest path arg", () => { + const argv = ["node", builderPath]; + expect(() => { processArgs(argv); }) + .toThrow("Usage:"); + + const argv1 = ["node", builderPath, cfgPath]; + expect(() => { processArgs(argv1); }) + .toThrow("Usage:"); + }); + + it("returns config path and dest path", () => { + const argv = ["node", builderPath, cfgPath, destPath]; + const args = processArgs(argv); + expect(args.configPath).toBe(cfgPath); + expect(args.destPath).toBe(destPath); + }); +}); diff --git a/app/server/tests/static-site-builder/builder.test.ts b/app/server/tests/static-site-builder/builder.test.ts new file mode 100644 index 00000000..3968da89 --- /dev/null +++ b/app/server/tests/static-site-builder/builder.test.ts @@ -0,0 +1,90 @@ +import path from "path"; +import fs from "fs"; +import { tmpdirTest } from "./tempDirTestHelper" +import { buildWodinStaticSite } from "../../src/static-site-builder/builder" + +const buildSite = async (tmpdir: string) => { + await buildWodinStaticSite("../../config-static", tmpdir); +} + +const { + mockRunnerOde, mockRunnerDiscrete, + mockModel, mockGet, mockPost +} = vi.hoisted(() => { + const mockRunnerOde = "var odinRunnerOde;"; + const mockRunnerDiscrete = "var odinRunnerDiscrete;"; + const mockModel = "class odinModel {};"; + const mockGet = vi.fn().mockImplementation((url: string) => { + if (url === "http://localhost:8001/support/runner-ode") { + return { data: { data: mockRunnerOde } }; + } else if (url === "http://localhost:8001/support/runner-discrete") { + return { data: { data: mockRunnerDiscrete } }; + } + }); + const mockPost = vi.fn().mockImplementation((url: string) => { + if (url === "http://localhost:8001/compile") { + return { data: { data: mockModel } }; + } + }); + return { + mockRunnerOde, mockRunnerDiscrete, + mockModel, mockGet, mockPost + }; +}); + +vi.mock("axios", () => { + return { + default: { + get: mockGet, + post: mockPost + } + } +}); + +const p = path.resolve; + +const expectPath = (...paths: string[]) => { + expect(fs.existsSync(p(...paths))).toBe(true); +}; + +const expectPathContent = (content: string, ...paths: string[]) => { + expect(fs.readFileSync(p(...paths)).toString()).toBe(content); +} + +const expectPathContentContains = (content: string, ...paths: string[]) => { + expect(fs.readFileSync(p(...paths)).toString()).toContain(content); +} + +describe("Wodin builder", () => { + tmpdirTest("creates dest dir if it doesn't exist", async ({ tmpdir }) => { + fs.rmdirSync(tmpdir); + await buildSite(tmpdir); + expectPath(tmpdir); + }); + + tmpdirTest("generates site with correct files and folder structure", async ({ tmpdir }) => { + const storesPath = p(tmpdir, "stores"); + await buildSite(tmpdir); + + expectPath(tmpdir, "index.html"); + + expectPath(storesPath, "runnerOde.js"); + expectPathContent(mockRunnerOde, storesPath, "runnerOde.js"); + expectPath(storesPath, "runnerDiscrete.js"); + expectPathContent(mockRunnerDiscrete, storesPath, "runnerDiscrete.js"); + + expectPath(storesPath, "basic", "model.json"); + expectPathContentContains(mockModel, storesPath, "basic", "model.json"); + expectPath(storesPath, "basic", "config.json"); + }); + + tmpdirTest("overwrites existing stores folder", async ({ tmpdir }) => { + const storesPath = p(tmpdir, "stores"); + fs.mkdirSync(storesPath); + fs.writeFileSync(p(storesPath, "trash.txt"), "whats up"); + + await buildSite(tmpdir); + + expect(fs.existsSync(p(storesPath, "trash.txt"))).toBe(false) + }); +}) \ No newline at end of file diff --git a/app/server/tests/static-site-builder/tempDirTestHelper.ts b/app/server/tests/static-site-builder/tempDirTestHelper.ts new file mode 100644 index 00000000..5f56d5e4 --- /dev/null +++ b/app/server/tests/static-site-builder/tempDirTestHelper.ts @@ -0,0 +1,22 @@ +import { test } from "vitest"; +import os from "node:os"; +import path from "path"; +import fs from "fs"; + +type TmpDirTestFixture = { tmpdir: string; } + +const createTempDir = () => { + const ostmpdir = os.tmpdir(); + const tmpdir = path.resolve(ostmpdir, "static-site-builder-tests-"); + return fs.mkdtempSync(tmpdir); +}; + +export const tmpdirTest = test.extend({ + // eslint-disable-next-line no-empty-pattern + tmpdir: async ({}, use) => { + const dir = createTempDir(); + await use(dir); + + fs.rmSync(dir, { recursive: true }) + } +}); diff --git a/app/server/vitest/vitest.unit.config.mjs b/app/server/vitest/vitest.unit.config.mjs index e4a0aef9..ca1fd2b4 100644 --- a/app/server/vitest/vitest.unit.config.mjs +++ b/app/server/vitest/vitest.unit.config.mjs @@ -9,7 +9,7 @@ export default mergeConfig( coverage: { provider: "istanbul", include: ["src"], - exclude: ["**/tests/**", "src/server/server.ts"] + exclude: ["**/tests/**", "src/server/server.ts", "src/static-site-builder/wodinBuilder.ts"] } } }) diff --git a/app/static/src/externalScriptSrc.ts b/app/static/src/externalScriptSrc.ts index 60cd4ed8..5f440e74 100644 --- a/app/static/src/externalScriptSrc.ts +++ b/app/static/src/externalScriptSrc.ts @@ -1,8 +1,25 @@ -const externalScripts = [ +/* + Originally dynamic wodin returned a mustache template from the express server + which had a + + + + tag in it but with the introduction of the wodin static build, we no longer + control the html that is shipped to the user (the scientists write the html + research papers with the static build). + + We could enforce that they add this script tag manually otherwise mathjax + (latex style rendering in html) won't work but for the sake of the researchers' + experience lets make it easier and less prone to errors by just programmatically + adding the script tag. + + To load more external scripts, just append the src to the externalScripts array. +*/ +export const externalScripts = [ "https://cdn.jsdelivr.net/npm/mathjax@3/es5/tex-chtml.js" ]; -export const mountScriptTags = () => { +export const loadThirdPartyCDNScripts = () => { externalScripts.forEach(src => { const script = document.createElement("script"); script.defer = true; diff --git a/app/static/src/mainUtils.ts b/app/static/src/mainUtils.ts deleted file mode 100644 index c2971265..00000000 --- a/app/static/src/mainUtils.ts +++ /dev/null @@ -1,18 +0,0 @@ -import { AppType } from "./store/appState/state"; -import { storeOptions as basicStoreOptions } from "./store/basic/basic"; -import { storeOptions as fitStoreOptions } from "./store/fit/fit"; -import { storeOptions as stochasticStoreOptions } from "./store/stochastic/stochastic"; - -const { Basic, Fit, Stochastic } = AppType; -export const getStoreOptions = (appType: AppType) => { - switch (appType) { - case Basic: - return basicStoreOptions; - case Fit: - return fitStoreOptions; - case Stochastic: - return stochasticStoreOptions; - default: - throw new Error("Unknown app type"); - } -}; diff --git a/app/static/src/wodin-static.ts b/app/static/src/wodin-static.ts deleted file mode 100644 index 95da8907..00000000 --- a/app/static/src/wodin-static.ts +++ /dev/null @@ -1,106 +0,0 @@ -import { createApp } from "vue"; -import { Store, StoreOptions } from "vuex"; -import axios from "axios"; -import "./scss/style.scss" -import { AppConfig, OdinModelResponse, OdinRunnerDiscrete, OdinRunnerOde } from "./types/responseTypes"; -import { AppState, AppType } from "./store/appState/state"; -import { AppStateMutation } from "./store/appState/mutations"; -import { ModelMutation } from "./store/model/mutations"; -import { RunAction } from "./store/run/actions"; -import { ModelAction } from "./store/model/actions"; -import { getStoreOptions } from "./mainUtils"; -import { mountScriptTags } from "./externalScriptSrc"; -import RunTab from "./components/run/RunTab.vue"; -import SensitivityTab from "./components/sensitivity/SensitivityTab.vue"; - -declare let odinjs: OdinRunnerOde -declare let dust: OdinRunnerDiscrete - -const blockingScripts = ["./stores/runnerOde.js", "./stores/runnerDiscrete.js"]; - -const deepCopyStateIfExists = (obj: T) => { - if (obj.state) obj.state = JSON.parse(JSON.stringify(obj.state)); -} - -const componentsAndSelectors = (s: string) => ([ - { selector: `.w-run-graph[data-w-store="${s}"]`, component: RunTab }, - { selector: `.w-sens-graph[data-w-store="${s}"]`, component: SensitivityTab } -]); - -const boot = async () => { - // inject external scripts - mountScriptTags(); - - // inject internal runner scripts - let numScriptsLoaded = 0; - blockingScripts.forEach(src => { - const script = document.createElement("script"); - script.async = false; - script.src = src; - document.body.append(script); - script.onload = () => numScriptsLoaded++; - }); - - // wait for runner scripts to load (waiting for correct number of - // onload events from script tags above) - await new Promise(res => setInterval(() => { - if (numScriptsLoaded === blockingScripts.length) return res(null) - }, 100)); - - // find all stores the user has put in the page - const stores = document.querySelectorAll("[data-w-store]")!; - const storesInPage: string[] = [] - stores.forEach(el => { - const elStore = el.getAttribute("data-w-store")!; - if (!storesInPage.includes(elStore)) { - storesInPage.push(elStore); - } - }); - - // get relevant store configs and model code - const configPromises = storesInPage.map(s => axios.get(`./stores/${s}/config.json`)); - const modelResponsePromises = storesInPage.map(s => axios.get(`./stores/${s}/model.json`)); - const configs = (await Promise.all(configPromises)).map(res => res.data) as { appType: AppType, defaultCode: string[] }[]; - const modelResponses = (await Promise.all(modelResponsePromises)).map(res => res.data) as OdinModelResponse[]; - - storesInPage.forEach(async (s, i) => { - const { appType, defaultCode } = configs[i]; - const modelResponse = modelResponses[i]; - const originalStore = getStoreOptions(appType) as StoreOptions; - - // recursively deep copy state because without this we would have multiple - // stores writing to the same state objects - const deepCopyStore = { ...originalStore }; - deepCopyStateIfExists(deepCopyStore); - Object.keys(deepCopyStore.modules!).forEach(m => deepCopyStateIfExists(deepCopyStore.modules![m])); - - const store = new Store(deepCopyStore); - - // manually initialise store - const appConfigPayload = { - appType, - basicProp: "", - defaultCode, - endTime: 100, - readOnlyCode: true, - stateUploadIntervalMillis: 2_000_000, - maxReplicatesRun: 100, - maxReplicatesDisplay: 50 - } as AppConfig - store.commit(AppStateMutation.SetConfig, appConfigPayload); - store.commit(`model/${ModelMutation.SetOdinRunnerOde}`, odinjs); - store.commit(`model/${ModelMutation.SetOdinRunnerDiscrete}`, dust); - store.commit(`model/${ModelMutation.SetOdinResponse}`, modelResponse); - await store.dispatch(`model/${ModelAction.CompileModel}`) - await store.dispatch(`run/${RunAction.RunModel}`) - - // mount components to dom elements based on selectors - componentsAndSelectors(s).forEach(({ component, selector }) => { - const applet = createApp(component); - applet.use(store); - document.querySelectorAll(selector)?.forEach(el => applet.mount(el)); - }); - }) -}; - -boot() diff --git a/app/static/src/wodin.ts b/app/static/src/wodin.ts index 35cd0ef7..5adb0666 100644 --- a/app/static/src/wodin.ts +++ b/app/static/src/wodin.ts @@ -10,13 +10,13 @@ import "./assets/fontawesome.css"; import "./scss/style.scss" import help from "./directives/help"; import App from "./components/App.vue"; -import { getStoreOptions } from "./mainUtils"; +import { getStoreOptions } from "./wodinStaticUtils"; import { Store, StoreOptions } from "vuex"; -import { mountScriptTags } from "./externalScriptSrc"; +import { loadThirdPartyCDNScripts } from "./externalScriptSrc"; declare let appType: AppType; -mountScriptTags(); +loadThirdPartyCDNScripts(); const { Basic, Fit, Stochastic } = AppType; export const getComponent = (appType: AppType) => { diff --git a/app/static/src/wodinStatic.ts b/app/static/src/wodinStatic.ts new file mode 100644 index 00000000..c495d6dc --- /dev/null +++ b/app/static/src/wodinStatic.ts @@ -0,0 +1,57 @@ +import { createApp } from "vue"; +import { getStoreOptions } from "./wodinStaticUtils"; +import { Store, StoreOptions } from "vuex"; +import "./scss/style.scss" +import { AppState } from "./store/appState/state"; +import { loadThirdPartyCDNScripts } from "./externalScriptSrc"; +import { + componentsAndSelectors, getConfigAndModelForStores, getDeepCopiedStoreOptions, + getStoresInPage, initialiseStore, waitForBlockingScripts +} from "./wodinStaticUtils"; + +/* + This is the entrypoint to the wodin static build. This boot function just gets called + at the end of the file. Since we are not allowed top level awaits we have to create + an async function and call it at the bottom. + + To load any scripts that you need before the runs just append the src to the + blockingScripts array. +*/ + +const blockingScripts = ["./stores/runnerOde.js", "./stores/runnerDiscrete.js"]; + +const boot = async () => { + // inject external scripts such as mathjax + loadThirdPartyCDNScripts(); + + // wait for scripts we can't load without such as runner code + await waitForBlockingScripts(blockingScripts); + + // find all stores the user has put in the page + const storesInPage = getStoresInPage(); + + // get relevant store configs and model code + const configAndModelObj = await getConfigAndModelForStores(storesInPage); + + storesInPage.forEach(async s => { + const { config, modelResponse } = configAndModelObj[s]; + const originalStoreOptions = getStoreOptions(config.appType) as StoreOptions; + + // recursively deep copy state because without this we would have multiple + // stores writing to the same state objects + const store = new Store(getDeepCopiedStoreOptions(originalStoreOptions)); + + // manually initialise store because this is the responsibility of apps like + // WodinSession which aren't used in the static build + await initialiseStore(store, config, modelResponse); + + // mount components to dom elements based on selectors + componentsAndSelectors(s).forEach(({ component, selector }) => { + const applet = createApp(component); + applet.use(store); + document.querySelectorAll(selector)?.forEach(el => applet.mount(el)); + }); + }) +}; + +boot() diff --git a/app/static/src/wodinStaticUtils.ts b/app/static/src/wodinStaticUtils.ts new file mode 100644 index 00000000..53aeeb5a --- /dev/null +++ b/app/static/src/wodinStaticUtils.ts @@ -0,0 +1,133 @@ +import axios from "axios"; +import RunTab from "./components/run/RunTab.vue"; +import SensitivityTab from "./components/sensitivity/SensitivityTab.vue"; +import { AppState, AppType } from "./store/appState/state"; +import { AppConfig, OdinModelResponse, OdinRunnerDiscrete, OdinRunnerOde } from "./types/responseTypes"; +import { Store, StoreOptions } from "vuex"; +import { AppStateMutation } from "./store/appState/mutations"; +import { ModelMutation } from "./store/model/mutations"; +import { ModelAction } from "./store/model/actions"; +import { RunAction } from "./store/run/actions"; +import { storeOptions as basicStoreOptions } from "./store/basic/basic"; +import { storeOptions as fitStoreOptions } from "./store/fit/fit"; +import { storeOptions as stochasticStoreOptions } from "./store/stochastic/stochastic"; + +const { Basic, Fit, Stochastic } = AppType; +export const getStoreOptions = (appType: AppType) => { + switch (appType) { + case Basic: + return basicStoreOptions; + case Fit: + return fitStoreOptions; + case Stochastic: + return stochasticStoreOptions; + default: + throw new Error("Unknown app type"); + } +}; + +export const componentsAndSelectors = (s: string) => ([ + { selector: `.w-run-graph[data-w-store="${s}"]`, component: RunTab }, + { selector: `.w-sens-graph[data-w-store="${s}"]`, component: SensitivityTab } +]); + +export const waitForBlockingScripts = async (blockingScripts: string[]) => { + // inject internal runner scripts + let numScriptsLoaded = 0; + blockingScripts.forEach(src => { + const script = document.createElement("script"); + script.async = false; + script.src = src; + document.body.append(script); + script.onload = () => numScriptsLoaded++; + }); + + // wait for runner scripts to load (waiting for correct number of + // onload events from script tags above) + await new Promise(res => setInterval(() => { + if (numScriptsLoaded === blockingScripts.length) return res(null) + }, 100)); +}; + +export const getStoresInPage = () => { + const stores = document.querySelectorAll("[data-w-store]")!; + const storesInPage: string[] = [] + stores.forEach(el => { + const elStore = el.getAttribute("data-w-store")!; + if (!storesInPage.includes(elStore)) { + storesInPage.push(elStore); + } + }); + return storesInPage; +}; + +// TODO more tolerant error handling, maybe one config didnt work (for error handling PR) +export type StaticConfig = { appType: AppType, defaultCode: string[] }; +type ConfigAndModel = { config: StaticConfig, modelResponse: OdinModelResponse }; + +export const getConfigAndModelForStores = async (storesInPage: string[]) => { + const configPromises = storesInPage.map(s => axios.get(`./stores/${s}/config.json`)); + const modelResponsePromises = storesInPage.map(s => axios.get(`./stores/${s}/model.json`)); + const configs = (await Promise.all(configPromises)).map(res => res.data) as StaticConfig[]; + const modelResponses = (await Promise.all(modelResponsePromises)).map(res => res.data) as OdinModelResponse[]; + + return Object.fromEntries(storesInPage.map((s, i) => { + const cfgAndModel: ConfigAndModel = { + config: configs[i], + modelResponse: modelResponses[i] + }; + return [ s, cfgAndModel ]; + })); +}; + +export const getDeepCopiedStoreOptions = (storeOptions: StoreOptions) => { + const deepCopy = { ...storeOptions }; + if (deepCopy.state) { + deepCopy.state = JSON.parse(JSON.stringify(deepCopy.state)); + } + deepCopy.modules = { ...deepCopy.modules }; + Object.keys(deepCopy.modules!).forEach(m => { + deepCopy.modules![m] = { ...deepCopy.modules![m] }; + if (deepCopy.modules![m].state) { + deepCopy.modules![m].state = JSON.parse(JSON.stringify(deepCopy.modules![m].state)); + } + }); + return deepCopy; +}; + +declare let odinjs: OdinRunnerOde +declare let dust: OdinRunnerDiscrete + +/* + Traditionally in dynamic wodin, initialising the store is the responsibility + of the components on mount (such as WodinSession initialising the app config) + however, we do not mount these components anymore in static wodin. Static wodin + may or may not mount these components so we need to guarantee that these bits + of state are initialised. + + Note: we have not thoroughly explored mounting components that impact these + specific parts of the store yet as there is no need to in static wodin. In theory + we have disabled the API service so they should not change the state but if you + are getting API related errors on static build then it returning undefined as the + response may be the cause. +*/ +export const initialiseStore = async ( + store: Store, config: StaticConfig, modelResponse: OdinModelResponse +) => { + const appConfigPayload = { + appType: config.appType, + basicProp: "", + defaultCode: config.defaultCode, + endTime: 100, + readOnlyCode: true, + stateUploadIntervalMillis: 2_000_000, + maxReplicatesRun: 100, + maxReplicatesDisplay: 50 + } as AppConfig + store.commit(AppStateMutation.SetConfig, appConfigPayload); + store.commit(`model/${ModelMutation.SetOdinRunnerOde}`, odinjs); + store.commit(`model/${ModelMutation.SetOdinRunnerDiscrete}`, dust); + store.commit(`model/${ModelMutation.SetOdinResponse}`, modelResponse); + await store.dispatch(`model/${ModelAction.CompileModel}`) + await store.dispatch(`run/${RunAction.RunModel}`) +}; diff --git a/app/static/tests/unit/externalScriptSrc.test.ts b/app/static/tests/unit/externalScriptSrc.test.ts new file mode 100644 index 00000000..e87ea544 --- /dev/null +++ b/app/static/tests/unit/externalScriptSrc.test.ts @@ -0,0 +1,12 @@ +import { externalScripts, loadThirdPartyCDNScripts } from "../../src/externalScriptSrc" + +describe("external script src", () => { + test("mounts enternal scripts to document", () => { + loadThirdPartyCDNScripts(); + externalScripts.forEach((src, i) => { + const script = document.body.getElementsByTagName("script").item(i) + expect(script!.src).toBe(src); + expect(script!.defer).toBe(true); + }) + }); +}); diff --git a/app/static/tests/unit/wodinStaticUtils.test.ts b/app/static/tests/unit/wodinStaticUtils.test.ts new file mode 100644 index 00000000..e55162d9 --- /dev/null +++ b/app/static/tests/unit/wodinStaticUtils.test.ts @@ -0,0 +1,136 @@ +import { AppType } from "@/store/appState/state"; +import { + componentsAndSelectors, getStoreOptions, waitForBlockingScripts, + getStoresInPage, getConfigAndModelForStores, getDeepCopiedStoreOptions, + initialiseStore +} from "@/wodinStaticUtils"; +import { storeOptions as basicStoreOptions } from "@/store/basic/basic"; +import { storeOptions as fitStoreOptions } from "@/store/fit/fit"; +import { storeOptions as stochasticStoreOptions } from "@/store/stochastic/stochastic"; +import { AppStateMutation } from "@/store/appState/mutations"; +import { ModelMutation } from "@/store/model/mutations"; +import { ModelAction } from "@/store/model/actions"; +import { RunAction } from "@/store/run/actions"; + +const { mockConfigRes, mockModelRes, mockGet } = vi.hoisted(() => { + const mockConfigRes = (s: string) => `config-${s}`; + const mockModelRes = (s: string) => `model-${s}`; + const mockGet = vi.fn().mockImplementation((url: string) => { + const store = url.split("/").at(-2); + const type = url.split("/").at(-1) === "config.json" ? "config" : "model"; + return { data: `${type}-${store}` }; + }); + return { mockConfigRes, mockModelRes, mockGet }; +}); +vi.mock("axios", () => ({ default: { get: mockGet } })); + +describe("wodin static utils", () => { + test("can get correct store options", () => { + const b = getStoreOptions(AppType.Basic); + expect(b).toStrictEqual(basicStoreOptions); + + const f = getStoreOptions(AppType.Fit); + expect(f).toStrictEqual(fitStoreOptions); + + const s = getStoreOptions(AppType.Stochastic); + expect(s).toStrictEqual(stochasticStoreOptions); + }); + + test("get store options throws error if not correct app type", () => { + expect(() => getStoreOptions("hey" as any)).toThrowError("Unknown app type"); + }); + + test("component and selector interpolates store correctly", () => { + const { selector } = componentsAndSelectors("test-store")[0]; + expect(selector).toContain(`data-w-store="test-store"`); + }); + + test("can wait for blocking scripts", async () => { + let promiseDone = false; + const mockScript = { async: true, src: "", onload: null }; + const createElementSpy = vi.spyOn(document, "createElement") + .mockImplementation(() => mockScript as any); + const documentBodySpy = vi.spyOn(document.body, "append"); + + waitForBlockingScripts(["test-script"]).then(() => promiseDone = true); + (mockScript.onload as any)(); + await vi.waitFor(() => expect(promiseDone).toBe(true)); + + expect(createElementSpy).toBeCalledTimes(1); + expect((documentBodySpy.mock.calls[0][0] as any).async).toBe(false); + expect((documentBodySpy.mock.calls[0][0] as any).src).toBe("test-script"); + }); + + test("get stores in page returns unique list of stores", async () => { + const testStores = ["store1", "store1", "store2"]; + vi.spyOn(document, "querySelectorAll").mockImplementation(() => { + return testStores.map(s => ({ getAttribute: vi.fn().mockImplementation(() => s) })) as any + }); + const storesInPage = getStoresInPage(); + expect(storesInPage).toStrictEqual(["store1", "store2"]); + }); + + test("can get configs and models for stores", async () => { + const testStores = ["store1", "store2"]; + const configAndModelObj = await getConfigAndModelForStores(testStores); + testStores.forEach(s => { + expect(configAndModelObj[s].config).toBe(mockConfigRes(s)); + expect(configAndModelObj[s].modelResponse).toBe(mockModelRes(s)); + }) + }); + + test("can deep copy store options", () => { + const mockStoreOptions = { + state: { key1: "hey1" }, + modules: { + mod1: { state: { key2: "hey2" } }, + mod2: { state: { key3: "hey3" } } + } + }; + const copy = getDeepCopiedStoreOptions(mockStoreOptions as any) as any; + copy.state.key1 = "what1"; + copy.modules.mod1.state.key2 = "what2"; + copy.modules.mod2.state.key3 = "what3"; + expect(mockStoreOptions.state.key1).toBe("hey1"); + expect(mockStoreOptions.modules.mod1.state.key2).toBe("hey2"); + expect(mockStoreOptions.modules.mod2.state.key3).toBe("hey3"); + }); + + test("can initialise store", async () => { + (globalThis as any).odinjs = "test odinjs"; + (globalThis as any).dust = "test dust"; + + const commit = vi.fn(); + const dispatch = vi.fn(); + await initialiseStore( + { commit, dispatch } as any, + { appType: AppType.Basic, defaultCode: ["test", "code"] }, + "test model res" as any + ); + + expect(commit).toBeCalledTimes(4); + + expect(commit.mock.calls[0][0]).toBe(AppStateMutation.SetConfig); + expect(commit.mock.calls[0][1]).toStrictEqual({ + appType: AppType.Basic, + basicProp: "", + defaultCode: ["test", "code"], + endTime: 100, + readOnlyCode: true, + stateUploadIntervalMillis: 2_000_000, + maxReplicatesRun: 100, + maxReplicatesDisplay: 50 + }); + expect(commit.mock.calls[1][0]).toBe(`model/${ModelMutation.SetOdinRunnerOde}`); + expect(commit.mock.calls[1][1]).toBe("test odinjs"); + expect(commit.mock.calls[2][0]).toBe(`model/${ModelMutation.SetOdinRunnerDiscrete}`); + expect(commit.mock.calls[2][1]).toBe("test dust"); + expect(commit.mock.calls[3][0]).toBe(`model/${ModelMutation.SetOdinResponse}`); + expect(commit.mock.calls[3][1]).toBe("test model res"); + + expect(dispatch).toBeCalledTimes(2); + + expect(dispatch.mock.calls[0][0]).toBe(`model/${ModelAction.CompileModel}`); + expect(dispatch.mock.calls[1][0]).toBe(`run/${RunAction.RunModel}`); + }); +}); diff --git a/app/static/vite-static.config.ts b/app/static/vite-static.config.ts index 28fc7343..c9999887 100644 --- a/app/static/vite-static.config.ts +++ b/app/static/vite-static.config.ts @@ -6,7 +6,7 @@ import config from "./vite-common.config" export default mergeConfig(config, { build: { rollupOptions: { - input: resolve(__dirname, './src/wodin-static.ts') + input: resolve(__dirname, './src/wodinStatic.ts') } } }); diff --git a/app/static/vitest.config.ts b/app/static/vitest.config.ts index 92f9e762..7e32c007 100644 --- a/app/static/vitest.config.ts +++ b/app/static/vitest.config.ts @@ -14,7 +14,7 @@ export default mergeConfig( coverage: { provider: "istanbul", include: ["src", "translationPackage"], - exclude: ["**/wodin.ts", "**/App.vue", "**/tests/**"] + exclude: ["**/wodin.ts", "**/App.vue", "**/tests/**", "**/wodinStatic.ts"] } } }) diff --git a/config-static/README.md b/config-static/README.md new file mode 100644 index 00000000..b0ba8266 --- /dev/null +++ b/config-static/README.md @@ -0,0 +1,40 @@ +# Wodin Static + +## Background + +We would like researchers to be able to build a site without a server and publish a static site somewhere like github pages. This will be to build interactive research papers/articles that allow policy makers to better understand the researchers' models. Since the models will be static we can remove both the `express` backend, `redis` and `odin.api`. + +Static wodin sites will allow the user to update parameters and re-run the model, fit data, run sensitivity and download data. They do not support updating model code or saving sessions. + +## Example static site + +An example of the config a user needs to design is in this folder. To convert this example into a site run: +1. `./scripts/run-dev-dependencies.sh` +1. `./scripts/build-and-serve-static-site.sh` +The site should be available at `localhost:3000` (the containers which are run by the dependencies script are only involved in generating the model javascript for this site. Once the site is built, you can stop the containers and still change the HTML document, e.g. adding a new graph, and it'll update everything accordingly when you refresh the page - you don't need to re-run the `build-and-serve` script either). + +## How researchers will build a static site + +The researchers will not be running the `build-and-serve-static-site.sh` manually. There are two stages to how the static site works for them: + +1. Build time: The researchers will be using a github action on a repository that looks like `config-static` to generate model code and static assets of the site. +1. Run time: The built Javascript querying the HTML page and mounting the correct components with the correct stores. + +### Build time + +After the user writes the config, a github action will checkout this repo, run the `odin.api` docker container and run the `wodinBuilder.ts` script which takes in the path of the config and path of the output folder as args. This script expects a `stores` folder which contains folders that label your stores, ``. In each `` folder it expects a `config.json` and `model.R` file. This script does a couple of things: +1. Copies the `index.html` file the user has written to the output folder +1. Gets the Javascript code for the ODE and discrete runners from `odin.api` and saves them into `stores/runnerOde.js` and `stores/runnerDiscrete.js` (the runner are generic and shared across all stores) +1. For each ``, it saves the config into `stores//config.json` and it compiles the model code (also via the odin api) and saves the response into `stores//model.json`. + +We also build `wodin` frontend in static mode (just normal `wodin` frontend with an early return in the api service so we don't do any network requests since there are no `odin.api` and `redis` containers running) and copy the js and css files into the output folder. After this, we just commit the output folder into a github pages branch and github will deploy it. + +### Run time + +The `wodinStatic.ts` script is the entrypoint for run time wodin static. This script does the following: +1. Loads third party CDN scripts (e.g. for mathjax). It does this by creating a script tag with the right source and appends it to the body of the HTML document. +1. Awaits blocking scripts, e.g. loading the `runnerOde.js` and `runnerDiscrete.js`, that the app cannot run without, these runners loaded as global variables `odinjs` and `dust` respectively. +1. Queries the document and finds all tags with `data-w-store` attributes - these tags define the wodin components to render by class name, and the `data-w-store` attribute specifies the name of the store to use. These attribute values define all the stores requires by the site, and should match the `` folder names in the user's config. +1. Gets the `stores//config.json` and `stores//model.json` for only the stores the user has used in the page. +1. For each `` the user has used in the page, it initialises the store with the config, runners (using the global variables), model and finally runs the model with the runners. This initialisation was the responsibility of on mount hook of components like `WodinSession.vue` but we no longer mount those components so we have to manually initialise the store. Note: the api service is disabled in the wodin static build so returns undefined for all responses, this may break things if you want to mount certain components that do api requests. We intend to fix this soon. +1. For each `` we loop over the components we want to mount. We define a selector for each one in `componentsAndSelectors` function and query the DOM for these selectors making sure they have `data-w-store` tag with value ``. We mount the component to the correct selector with the correct store. `index.html` in this folder includes examples of all supported components.