155 lines
4.7 KiB
JavaScript
155 lines
4.7 KiB
JavaScript
const { join } = require("path")
|
|
const { ObjectStoreBuckets } = require("../../constants")
|
|
const fs = require("fs")
|
|
const { upload, retrieveToTmp, streamUpload } = require("./utilities")
|
|
const { resolve } = require("../centralPath")
|
|
const env = require("../../environment")
|
|
const TOP_LEVEL_PATH = join(__dirname, "..", "..", "..")
|
|
|
|
/**
|
|
* Client library paths in the object store:
|
|
* Previously, the entire standard-components package was downloaded from NPM
|
|
* as a tarball and extracted to the object store, even though only the manifest
|
|
* was ever needed. Therefore we need to support old apps which may still have
|
|
* the manifest at this location for the first update.
|
|
*
|
|
* The new paths for the in-use version are:
|
|
* {appId}/manifest.json
|
|
* {appId}/budibase-client.js
|
|
*
|
|
* The paths for the backups are:
|
|
* {appId}/manifest.json.bak
|
|
* {appId}/budibase-client.js.bak
|
|
*
|
|
* We don't rely on NPM at all any more, as when updating to the latest version
|
|
* we pull both the manifest and client bundle from the server's dependencies
|
|
* in the local file system.
|
|
*/
|
|
|
|
/**
|
|
* Backs up the current client library version by copying both the manifest
|
|
* and client bundle to .bak extensions in the object store. Only the one
|
|
* previous version is stored as a backup, which can be reverted to.
|
|
* @param appId The app ID to backup
|
|
* @returns {Promise<void>}
|
|
*/
|
|
exports.backupClientLibrary = async appId => {
|
|
// Copy existing manifest to tmp
|
|
let tmpManifestPath
|
|
try {
|
|
// Try to load the manifest from the new file location
|
|
tmpManifestPath = await retrieveToTmp(
|
|
ObjectStoreBuckets.APPS,
|
|
join(appId, "manifest.json")
|
|
)
|
|
} catch (error) {
|
|
// Fallback to loading it from the old location for old apps
|
|
tmpManifestPath = await retrieveToTmp(
|
|
ObjectStoreBuckets.APPS,
|
|
join(
|
|
appId,
|
|
"node_modules",
|
|
"budibase",
|
|
"standard-components",
|
|
"package",
|
|
"manifest.json"
|
|
)
|
|
)
|
|
}
|
|
|
|
// Copy existing client lib to tmp
|
|
const tmpClientPath = await retrieveToTmp(
|
|
ObjectStoreBuckets.APPS,
|
|
join(appId, "budibase-client.js")
|
|
)
|
|
|
|
// Upload manifest and client library as backups
|
|
const manifestUpload = upload({
|
|
bucket: ObjectStoreBuckets.APPS,
|
|
filename: join(appId, "manifest.json.bak"),
|
|
path: tmpManifestPath,
|
|
type: "application/json",
|
|
})
|
|
const clientUpload = upload({
|
|
bucket: ObjectStoreBuckets.APPS,
|
|
filename: join(appId, "budibase-client.js.bak"),
|
|
path: tmpClientPath,
|
|
type: "application/javascript",
|
|
})
|
|
await Promise.all([manifestUpload, clientUpload])
|
|
}
|
|
|
|
/**
|
|
* Uploads the latest version of the component manifest and the client library
|
|
* to the object store, overwriting the existing version.
|
|
* @param appId The app ID to update
|
|
* @returns {Promise<void>}
|
|
*/
|
|
exports.updateClientLibrary = async appId => {
|
|
let manifest, client
|
|
|
|
if (env.isDev()) {
|
|
// Load the symlinked version in dev which is always the newest
|
|
manifest = require.resolve("@budibase/standard-components/manifest.json")
|
|
client = require.resolve("@budibase/client")
|
|
} else {
|
|
// Load the bundled version in prod
|
|
manifest = resolve(TOP_LEVEL_PATH, "client", "manifest.json")
|
|
client = resolve(TOP_LEVEL_PATH, "client", "budibase-client.js")
|
|
}
|
|
|
|
// Upload latest manifest and client library
|
|
const manifestUpload = streamUpload(
|
|
ObjectStoreBuckets.APPS,
|
|
join(appId, "manifest.json"),
|
|
fs.createReadStream(manifest),
|
|
{
|
|
ContentType: "application/json",
|
|
}
|
|
)
|
|
const clientUpload = streamUpload(
|
|
ObjectStoreBuckets.APPS,
|
|
join(appId, "budibase-client.js"),
|
|
fs.createReadStream(client),
|
|
{
|
|
ContentType: "application/javascript",
|
|
}
|
|
)
|
|
await Promise.all([manifestUpload, clientUpload])
|
|
}
|
|
|
|
/**
|
|
* Reverts the version of the client library and manifest to the previously
|
|
* used version for an app.
|
|
* @param appId The app ID to revert
|
|
* @returns {Promise<void>}
|
|
*/
|
|
exports.revertClientLibrary = async appId => {
|
|
// Copy backups manifest to tmp directory
|
|
const tmpManifestPath = await retrieveToTmp(
|
|
ObjectStoreBuckets.APPS,
|
|
join(appId, "manifest.json.bak")
|
|
)
|
|
|
|
// Copy backup client lib to tmp
|
|
const tmpClientPath = await retrieveToTmp(
|
|
ObjectStoreBuckets.APPS,
|
|
join(appId, "budibase-client.js.bak")
|
|
)
|
|
|
|
// Upload backups as new versions
|
|
const manifestUpload = upload({
|
|
bucket: ObjectStoreBuckets.APPS,
|
|
filename: join(appId, "manifest.json"),
|
|
path: tmpManifestPath,
|
|
type: "application/json",
|
|
})
|
|
const clientUpload = upload({
|
|
bucket: ObjectStoreBuckets.APPS,
|
|
filename: join(appId, "budibase-client.js"),
|
|
path: tmpClientPath,
|
|
type: "application/javascript",
|
|
})
|
|
await Promise.all([manifestUpload, clientUpload])
|
|
}
|