2020-02-03 10:24:25 +01:00
|
|
|
const { appPackageFolder, appsFolder } = require("../createAppPackage")
|
|
|
|
const {
|
|
|
|
readJSON,
|
|
|
|
writeJSON,
|
|
|
|
readdir,
|
|
|
|
ensureDir,
|
|
|
|
rename,
|
|
|
|
unlink,
|
|
|
|
rmdir,
|
|
|
|
} = require("fs-extra")
|
|
|
|
const { join, dirname } = require("path")
|
|
|
|
const { $ } = require("@budibase/core").common
|
2020-02-10 22:35:51 +01:00
|
|
|
const { intersection, map, values, flatten } = require("lodash/fp")
|
2020-02-03 10:24:25 +01:00
|
|
|
const { merge } = require("lodash")
|
|
|
|
|
|
|
|
const { componentLibraryInfo } = require("./componentLibraryInfo")
|
2020-02-10 16:51:09 +01:00
|
|
|
const buildPage = require("./buildPage")
|
2020-02-10 22:35:51 +01:00
|
|
|
const getPages = require("./getPages")
|
|
|
|
const listScreens = require("./listScreens")
|
|
|
|
const saveBackend = require("./saveBackend")
|
2020-02-03 10:24:25 +01:00
|
|
|
|
2020-02-10 22:35:51 +01:00
|
|
|
module.exports.buildPage = buildPage
|
|
|
|
module.exports.listScreens = listScreens
|
|
|
|
module.exports.saveBackend = saveBackend
|
2020-02-03 10:24:25 +01:00
|
|
|
|
|
|
|
const getAppDefinition = async appPath =>
|
|
|
|
await readJSON(`${appPath}/appDefinition.json`)
|
2019-09-29 07:40:06 +02:00
|
|
|
|
2019-07-13 11:35:57 +02:00
|
|
|
module.exports.getPackageForBuilder = async (config, appname) => {
|
2020-02-03 10:24:25 +01:00
|
|
|
const appPath = appPackageFolder(config, appname)
|
2019-07-26 16:13:15 +02:00
|
|
|
|
2020-02-03 10:24:25 +01:00
|
|
|
const pages = await getPages(appPath)
|
2019-07-26 16:13:15 +02:00
|
|
|
|
2020-02-03 10:24:25 +01:00
|
|
|
return {
|
|
|
|
appDefinition: await getAppDefinition(appPath),
|
2019-07-13 11:35:57 +02:00
|
|
|
|
2020-02-03 10:24:25 +01:00
|
|
|
accessLevels: await readJSON(`${appPath}/access_levels.json`),
|
2019-07-25 08:31:54 +02:00
|
|
|
|
2020-02-03 10:24:25 +01:00
|
|
|
pages,
|
2019-07-26 16:13:15 +02:00
|
|
|
|
2020-02-03 10:24:25 +01:00
|
|
|
components: await getComponents(appPath, pages),
|
|
|
|
}
|
2019-07-13 11:35:57 +02:00
|
|
|
}
|
|
|
|
|
2019-10-11 18:14:23 +02:00
|
|
|
module.exports.getApps = async (config, master) => {
|
2020-02-03 10:24:25 +01:00
|
|
|
const dirs = await readdir(appsFolder(config))
|
2019-10-11 18:14:23 +02:00
|
|
|
|
2020-02-03 10:24:25 +01:00
|
|
|
return $(master.listApplications(), [map(a => a.name), intersection(dirs)])
|
2019-10-11 18:14:23 +02:00
|
|
|
}
|
2019-07-14 08:46:36 +02:00
|
|
|
|
2020-02-10 16:51:09 +01:00
|
|
|
const screenPath = (appPath, pageName, name) =>
|
|
|
|
join(appPath, "pages", pageName, "screens", name + ".json")
|
2019-07-26 18:08:59 +02:00
|
|
|
|
2020-02-10 16:51:09 +01:00
|
|
|
module.exports.saveScreen = async (config, appname, pagename, screen) => {
|
|
|
|
const appPath = appPackageFolder(config, appname)
|
|
|
|
const compPath = screenPath(appPath, pagename, screen.name)
|
2020-02-03 10:24:25 +01:00
|
|
|
await ensureDir(dirname(compPath))
|
2020-02-10 16:51:09 +01:00
|
|
|
if (screen._css) {
|
|
|
|
delete screen._css
|
|
|
|
}
|
|
|
|
await writeJSON(compPath, screen, {
|
2020-02-03 10:24:25 +01:00
|
|
|
encoding: "utf8",
|
|
|
|
flag: "w",
|
|
|
|
spaces: 2,
|
|
|
|
})
|
2020-02-11 12:19:17 +01:00
|
|
|
component.stateOrigins = buildStateOrigins(component);
|
|
|
|
return component;
|
2019-07-26 18:08:59 +02:00
|
|
|
}
|
|
|
|
|
2020-02-10 16:51:09 +01:00
|
|
|
module.exports.renameScreen = async (
|
|
|
|
config,
|
|
|
|
appname,
|
|
|
|
pagename,
|
|
|
|
oldName,
|
|
|
|
newName
|
|
|
|
) => {
|
2020-02-03 10:24:25 +01:00
|
|
|
const appPath = appPackageFolder(config, appname)
|
2019-07-26 18:08:59 +02:00
|
|
|
|
2020-02-10 16:51:09 +01:00
|
|
|
const oldComponentPath = screenPath(appPath, pagename, oldName)
|
2019-07-26 18:08:59 +02:00
|
|
|
|
2020-02-10 16:51:09 +01:00
|
|
|
const newComponentPath = screenPath(appPath, pagename, newName)
|
2019-07-26 18:08:59 +02:00
|
|
|
|
2020-02-03 10:24:25 +01:00
|
|
|
await ensureDir(dirname(newComponentPath))
|
|
|
|
await rename(oldComponentPath, newComponentPath)
|
2019-07-26 18:08:59 +02:00
|
|
|
}
|
|
|
|
|
2020-02-10 16:51:09 +01:00
|
|
|
module.exports.deleteScreen = async (config, appname, pagename, name) => {
|
2020-02-03 10:24:25 +01:00
|
|
|
const appPath = appPackageFolder(config, appname)
|
2020-02-10 16:51:09 +01:00
|
|
|
const componentFile = screenPath(appPath, pagename, name)
|
2020-02-03 10:24:25 +01:00
|
|
|
await unlink(componentFile)
|
|
|
|
|
|
|
|
const dir = dirname(componentFile)
|
|
|
|
if ((await readdir(dir)).length === 0) {
|
|
|
|
await rmdir(dir)
|
|
|
|
}
|
2019-07-26 18:08:59 +02:00
|
|
|
}
|
|
|
|
|
2020-02-10 16:51:09 +01:00
|
|
|
module.exports.savePage = async (config, appname, pagename, page) => {
|
|
|
|
const appPath = appPackageFolder(config, appname)
|
|
|
|
const pageDir = join(appPath, "pages", pagename)
|
|
|
|
|
|
|
|
await ensureDir(pageDir)
|
|
|
|
await writeJSON(join(pageDir, "page.json"), page, {
|
|
|
|
encoding: "utf8",
|
|
|
|
flag: "w",
|
|
|
|
space: 2,
|
|
|
|
})
|
|
|
|
const appDefinition = await getAppDefinition(appPath)
|
|
|
|
await buildPage(config, appname, appDefinition, pagename, page)
|
|
|
|
}
|
|
|
|
|
2019-09-09 06:24:14 +02:00
|
|
|
module.exports.componentLibraryInfo = async (config, appname, lib) => {
|
2020-02-03 10:24:25 +01:00
|
|
|
const appPath = appPackageFolder(config, appname)
|
|
|
|
return await componentLibraryInfo(appPath, lib)
|
|
|
|
}
|
2019-08-19 22:18:23 +02:00
|
|
|
|
2020-02-03 10:24:25 +01:00
|
|
|
const getComponents = async (appPath, pages, lib) => {
|
|
|
|
let libs
|
|
|
|
if (!lib) {
|
2020-02-10 16:51:09 +01:00
|
|
|
pages = pages || (await getPages(appPath))
|
2019-08-19 22:18:23 +02:00
|
|
|
|
2020-02-10 16:51:09 +01:00
|
|
|
if (!pages) return []
|
2019-07-25 08:31:54 +02:00
|
|
|
|
2020-02-10 16:51:09 +01:00
|
|
|
libs = $(pages, [values, map(p => p.componentLibraries), flatten])
|
2020-02-03 10:24:25 +01:00
|
|
|
} else {
|
|
|
|
libs = [lib]
|
|
|
|
}
|
2019-07-25 08:31:54 +02:00
|
|
|
|
2020-02-03 10:24:25 +01:00
|
|
|
const components = {}
|
|
|
|
const generators = {}
|
2019-07-25 08:31:54 +02:00
|
|
|
|
2020-02-03 10:24:25 +01:00
|
|
|
for (let l of libs) {
|
|
|
|
const info = await componentLibraryInfo(appPath, l)
|
|
|
|
merge(components, info.components)
|
|
|
|
merge(generators, info.generators)
|
|
|
|
}
|
2019-10-07 07:03:41 +02:00
|
|
|
|
2020-02-03 10:24:25 +01:00
|
|
|
if (components._lib) delete components._lib
|
|
|
|
if (components._generators) delete components._generators
|
2019-07-26 16:13:15 +02:00
|
|
|
|
2020-02-03 10:24:25 +01:00
|
|
|
return { components, generators }
|
2019-07-26 16:13:15 +02:00
|
|
|
}
|
|
|
|
|
2020-02-10 17:58:20 +01:00
|
|
|
/**
|
|
|
|
* buildStateOrigins
|
|
|
|
*
|
|
|
|
* Builds an object that details all the bound state in the application, and what updates it.
|
|
|
|
*
|
|
|
|
* @param screenDefinition - the screen definition metadata.
|
|
|
|
* @returns {Object} an object with the client state values and how they are managed.
|
|
|
|
*/
|
|
|
|
const buildStateOrigins = screenDefinition => {
|
|
|
|
const origins = {};
|
|
|
|
|
|
|
|
function traverse(propValue) {
|
|
|
|
for (let key in propValue) {
|
|
|
|
if (!Array.isArray(propValue[key])) continue;
|
|
|
|
|
|
|
|
if (key === "_children") propValue[key].forEach(traverse);
|
|
|
|
|
|
|
|
for (let element of propValue[key]) {
|
|
|
|
if (element["##eventHandlerType"] === "Set State") origins[element.parameters.path] = element;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
traverse(screenDefinition.props);
|
|
|
|
|
|
|
|
return origins;
|
|
|
|
};
|
|
|
|
|
2020-02-11 12:19:17 +01:00
|
|
|
module.exports.buildStateOrigins = buildStateOrigins;
|
|
|
|
|
2020-02-10 17:58:20 +01:00
|
|
|
const fetchscreens = async (appPath, relativePath = "") => {
|
|
|
|
const currentDir = join(appPath, "components", relativePath)
|
|
|
|
|
|
|
|
const contents = await readdir(currentDir)
|
|
|
|
|
|
|
|
const components = []
|
|
|
|
|
|
|
|
for (let item of contents) {
|
|
|
|
const itemRelativePath = join(relativePath, item)
|
|
|
|
const itemFullPath = join(currentDir, item)
|
|
|
|
const stats = await stat(itemFullPath)
|
|
|
|
|
|
|
|
if (stats.isFile()) {
|
|
|
|
if (!item.endsWith(".json")) continue
|
|
|
|
|
|
|
|
const component = await readJSON(itemFullPath)
|
|
|
|
|
|
|
|
component.name = itemRelativePath
|
|
|
|
.substring(0, itemRelativePath.length - 5)
|
|
|
|
.replace(/\\/g, "/")
|
|
|
|
|
|
|
|
component.props = component.props || {}
|
|
|
|
|
|
|
|
component.stateOrigins = buildStateOrigins(component);
|
|
|
|
|
|
|
|
components.push(component)
|
|
|
|
} else {
|
|
|
|
const childComponents = await fetchscreens(
|
|
|
|
appPath,
|
|
|
|
join(relativePath, item)
|
|
|
|
)
|
|
|
|
|
|
|
|
for (let c of childComponents) {
|
|
|
|
components.push(c)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return components
|
|
|
|
}
|
|
|
|
|
2020-02-03 10:24:25 +01:00
|
|
|
module.exports.getComponents = getComponents
|