const env = require("../environment") const { OBJ_STORE_DIRECTORY } = require("../constants") const { sanitizeKey } = require("@budibase/backend-core/objectStore") const { generateMetadataID } = require("../db/utils") const Readable = require("stream").Readable const { getAppDB } = require("@budibase/backend-core/context") const BB_CDN = "https://cdn.budi.live" exports.wait = ms => new Promise(resolve => setTimeout(resolve, ms)) exports.isDev = env.isDev exports.removeFromArray = (array, element) => { const index = array.indexOf(element) if (index !== -1) { array.splice(index, 1) } return array } /** * Makes sure that a URL has the correct number of slashes, while maintaining the * http(s):// double slashes. * @param {string} url The URL to test and remove any extra double slashes. * @return {string} The updated url. */ exports.checkSlashesInUrl = url => { return url.replace(/(https?:\/\/)|(\/)+/g, "$1$2") } /** * Gets the address of the object store, depending on whether self hosted or in cloud. * @return {string} The base URL of the object store (MinIO or S3). */ exports.objectStoreUrl = () => { if (env.SELF_HOSTED || env.MINIO_URL) { // can use a relative url for this as all goes through the proxy (this is hosted in minio) return OBJ_STORE_DIRECTORY } else { return BB_CDN } } /** * In production the client library is stored in the object store, however in development * we use the symlinked version produced by lerna, located in node modules. We link to this * via a specific endpoint (under /api/assets/client). * @param {string} appId In production we need the appId to look up the correct bucket, as the * version of the client lib may differ between apps. * @param {string} version The version to retrieve. * @return {string} The URL to be inserted into appPackage response or server rendered * app index file. */ exports.clientLibraryPath = (appId, version) => { if (env.isProd()) { let url = `${exports.objectStoreUrl()}/${sanitizeKey( appId )}/budibase-client.js` // append app version to bust the cache if (version) { url += `?v=${version}` } return url } else { return `/api/assets/client` } } exports.attachmentsRelativeURL = attachmentKey => { return exports.checkSlashesInUrl( `${exports.objectStoreUrl()}/${attachmentKey}` ) } exports.updateEntityMetadata = async (type, entityId, updateFn) => { const db = getAppDB() const id = generateMetadataID(type, entityId) // read it to see if it exists, we'll overwrite it no matter what let rev, metadata = {} try { const oldMetadata = await db.get(id) rev = oldMetadata._rev metadata = updateFn(oldMetadata) } catch (err) { rev = null metadata = updateFn({}) } metadata._id = id if (rev) { metadata._rev = rev } const response = await db.put(metadata) return { ...metadata, _id: id, _rev: response.rev, } } exports.saveEntityMetadata = async (type, entityId, metadata) => { return exports.updateEntityMetadata(type, entityId, () => { return metadata }) } exports.deleteEntityMetadata = async (type, entityId) => { const db = getAppDB() const id = generateMetadataID(type, entityId) let rev try { const metadata = await db.get(id) if (metadata) { rev = metadata._rev } } catch (err) { // don't need to error if it doesn't exist } if (id && rev) { await db.remove(id, rev) } } exports.escapeDangerousCharacters = string => { return string .replace(/[\\]/g, "\\\\") .replace(/[\b]/g, "\\b") .replace(/[\f]/g, "\\f") .replace(/[\n]/g, "\\n") .replace(/[\r]/g, "\\r") .replace(/[\t]/g, "\\t") } exports.stringToReadStream = string => { return new Readable({ read() { this.push(string) this.push(null) }, }) } exports.formatBytes = bytes => { const units = ["B", "KB", "MB", "GB", "TB", "PB", "EB", "ZB", "YB"] const byteIncrements = 1024 let unit = 0 let size = parseInt(bytes, 10) || 0 while (size >= byteIncrements && ++unit) { size /= byteIncrements } return `${size.toFixed(size < 10 && unit > 0 ? 1 : 0)}${units[unit]}` }