const env = require("../../environment") const { getAllApps } = require("@budibase/auth/db") const CouchDB = require("../../db") const { exportDB, sendTempFile, readFileSync, } = require("../../utilities/fileSystem") const { stringToReadStream } = require("../../utilities") const { getGlobalDBName, getGlobalDB } = require("@budibase/auth/tenancy") const { create } = require("./application") const { getDocParams, DocumentTypes, isDevAppID } = require("../../db/utils") async function createApp(appName, appImport) { const ctx = { request: { body: { templateString: appImport, name: appName, }, }, } return create(ctx) } exports.exportApps = async ctx => { if (env.SELF_HOSTED || !env.MULTI_TENANCY) { ctx.throw(400, "Exporting only allowed in multi-tenant cloud environments.") } const apps = await getAllApps(CouchDB, { all: true }) const globalDBString = await exportDB(getGlobalDBName()) let allDBs = { global: globalDBString, } for (let app of apps) { // only export the dev apps as they will be the latest, the user can republish the apps // in their self hosted environment if (isDevAppID(app._id)) { allDBs[app.name] = await exportDB(app._id) } } const filename = `cloud-export-${new Date().getTime()}.txt` ctx.attachment(filename) ctx.body = sendTempFile(JSON.stringify(allDBs)) } async function getAllDocType(db, docType) { const response = await db.allDocs( getDocParams(docType, null, { include_docs: true, }) ) return response.rows.map(row => row.doc) } exports.importApps = async ctx => { if (!env.SELF_HOSTED || env.MULTI_TENANCY) { ctx.throw(400, "Importing only allowed in self hosted environments.") } const apps = await getAllApps(CouchDB, { all: true }) if ( apps.length !== 0 || !ctx.request.files || !ctx.request.files.importFile ) { ctx.throw( 400, "Import file is required and environment must be fresh to import apps." ) } const importFile = ctx.request.files.importFile const importString = readFileSync(importFile.path) const dbs = JSON.parse(importString) const globalDbImport = dbs.global // remove from the list of apps delete dbs.global const globalDb = getGlobalDB() // load the global db first await globalDb.load(stringToReadStream(globalDbImport)) for (let [appName, appImport] of Object.entries(dbs)) { await createApp(appName, appImport) } // once apps are created clean up the global db let users = await getAllDocType(globalDb, DocumentTypes.USER) for (let user of users) { delete user.tenantId } await globalDb.bulkDocs(users) ctx.body = { message: "Apps successfully imported.", } }