From 3aed49778f4f1c4b41b8be11bb69cc6a96d4bd1d Mon Sep 17 00:00:00 2001 From: mike12345567 Date: Sat, 26 Nov 2022 15:42:53 +0000 Subject: [PATCH] One file left in the server, some fixes after converting the internal row controller. --- .../server/src/api/controllers/row/index.ts | 4 +- .../row/{internal.js => internal.ts} | 135 ++++++++++-------- .../src/api/controllers/row/staticFormula.ts | 2 +- .../server/src/api/controllers/row/utils.ts | 13 +- .../view/{viewBuilder.js => viewBuilder.ts} | 36 +++-- packages/server/src/db/inMemoryView.ts | 4 +- .../src/utilities/rowProcessor/index.ts | 8 +- packages/types/src/documents/app/view.ts | 2 +- packages/types/src/sdk/db.ts | 1 + 9 files changed, 119 insertions(+), 86 deletions(-) rename packages/server/src/api/controllers/row/{internal.js => internal.ts} (78%) rename packages/server/src/api/controllers/view/{viewBuilder.js => viewBuilder.ts} (85%) diff --git a/packages/server/src/api/controllers/row/index.ts b/packages/server/src/api/controllers/row/index.ts index 09af24a9b1..b59f245098 100644 --- a/packages/server/src/api/controllers/row/index.ts +++ b/packages/server/src/api/controllers/row/index.ts @@ -1,6 +1,6 @@ import { quotas } from "@budibase/pro" -import internal from "./internal" -import external from "./external" +import * as internal from "./internal" +import * as external from "./external" import { isExternalTable } from "../../../integrations/utils" function pickApi(tableId: any) { diff --git a/packages/server/src/api/controllers/row/internal.js b/packages/server/src/api/controllers/row/internal.ts similarity index 78% rename from packages/server/src/api/controllers/row/internal.js rename to packages/server/src/api/controllers/row/internal.ts index 9b105fd3ae..98a89a5038 100644 --- a/packages/server/src/api/controllers/row/internal.js +++ b/packages/server/src/api/controllers/row/internal.ts @@ -1,34 +1,41 @@ -const linkRows = require("../../../db/linkedRows") -const { +import * as linkRows from "../../../db/linkedRows" +import { generateRowID, getRowParams, getTableIDFromRowID, DocumentType, InternalTables, -} = require("../../../db/utils") -const userController = require("../user") -const { +} from "../../../db/utils" +import * as userController from "../user" +import { inputProcessing, outputProcessing, cleanupAttachments, -} = require("../../../utilities/rowProcessor") -const { FieldTypes } = require("../../../constants") -const { validate, findRow } = require("./utils") -const { fullSearch, paginatedSearch } = require("./internalSearch") -const { getGlobalUsersFromMetadata } = require("../../../utilities/global") -const inMemoryViews = require("../../../db/inMemoryView") -const env = require("../../../environment") -const { +} from "../../../utilities/rowProcessor" +import { FieldTypes } from "../../../constants" +import { validate as rowValidate, findRow } from "./utils" +import { fullSearch, paginatedSearch } from "./internalSearch" +import { getGlobalUsersFromMetadata } from "../../../utilities/global" +import * as inMemoryViews from "../../../db/inMemoryView" +import env from "../../../environment" +import { migrateToInMemoryView, migrateToDesignView, getFromDesignDoc, getFromMemoryDoc, -} = require("../view/utils") -const { cloneDeep } = require("lodash/fp") -const { context, db: dbCore } = require("@budibase/backend-core") -const { finaliseRow, updateRelatedFormula } = require("./staticFormula") -const exporters = require("../view/exporters") -const { apiFileReturn } = require("../../../utilities/fileSystem") +} from "../view/utils" +import { cloneDeep } from "lodash/fp" +import { context, db as dbCore } from "@budibase/backend-core" +import { finaliseRow, updateRelatedFormula } from "./staticFormula" +import * as exporters from "../view/exporters" +import { apiFileReturn } from "../../../utilities/fileSystem" +import { + BBContext, + Database, + LinkDocumentValue, + Row, + Table, +} from "@budibase/types" const CALCULATION_TYPES = { SUM: "sum", @@ -36,7 +43,7 @@ const CALCULATION_TYPES = { STATS: "stats", } -async function getView(db, viewName) { +async function getView(db: Database, viewName: string) { let mainGetter = env.SELF_HOSTED ? getFromDesignDoc : getFromMemoryDoc let secondaryGetter = env.SELF_HOSTED ? getFromMemoryDoc : getFromDesignDoc let migration = env.SELF_HOSTED ? migrateToDesignView : migrateToInMemoryView @@ -44,7 +51,7 @@ async function getView(db, viewName) { migrate = false try { viewInfo = await mainGetter(db, viewName) - } catch (err) { + } catch (err: any) { // check if it can be retrieved from design doc (needs migrated) if (err.status !== 404) { viewInfo = null @@ -62,7 +69,7 @@ async function getView(db, viewName) { return viewInfo } -async function getRawTableData(ctx, db, tableId) { +async function getRawTableData(ctx: BBContext, db: Database, tableId: string) { let rows if (tableId === InternalTables.USER_METADATA) { await userController.fetchMetadata(ctx) @@ -75,10 +82,10 @@ async function getRawTableData(ctx, db, tableId) { ) rows = response.rows.map(row => row.doc) } - return rows + return rows as Row[] } -exports.patch = async ctx => { +export async function patch(ctx: BBContext) { const db = context.getAppDB() const inputs = ctx.request.body const tableId = inputs.tableId @@ -103,15 +110,15 @@ exports.patch = async ctx => { } let dbTable = await db.get(tableId) // need to build up full patch fields before coerce - let combinedRow = cloneDeep(oldRow) + let combinedRow: any = cloneDeep(oldRow) for (let key of Object.keys(inputs)) { if (!dbTable.schema[key]) continue combinedRow[key] = inputs[key] } // this returns the table and row incase they have been updated - let { table, row } = inputProcessing(ctx.user, dbTable, combinedRow) - const validateResult = await validate({ + let { table, row } = inputProcessing(ctx.user!, dbTable, combinedRow) + const validateResult = await rowValidate({ row, table, }) @@ -121,12 +128,12 @@ exports.patch = async ctx => { } // returned row is cleaned and prepared for writing to DB - row = await linkRows.updateLinks({ + row = (await linkRows.updateLinks({ eventType: linkRows.EventType.ROW_UPDATE, row, tableId: row.tableId, table, - }) + })) as Row // check if any attachments removed await cleanupAttachments(table, { oldRow, row }) @@ -143,7 +150,7 @@ exports.patch = async ctx => { }) } -exports.save = async function (ctx) { +export async function save(ctx: BBContext) { const db = context.getAppDB() let inputs = ctx.request.body inputs.tableId = ctx.params.tableId @@ -154,8 +161,8 @@ exports.save = async function (ctx) { // this returns the table and row incase they have been updated const dbTable = await db.get(inputs.tableId) - let { table, row } = inputProcessing(ctx.user, dbTable, inputs) - const validateResult = await validate({ + let { table, row } = inputProcessing(ctx.user!, dbTable, inputs) + const validateResult = await rowValidate({ row, table, }) @@ -165,12 +172,12 @@ exports.save = async function (ctx) { } // make sure link rows are up to date - row = await linkRows.updateLinks({ + row = (await linkRows.updateLinks({ eventType: linkRows.EventType.ROW_SAVE, row, tableId: row.tableId, table, - }) + })) as Row return finaliseRow(table, row, { oldTable: dbTable, @@ -178,7 +185,7 @@ exports.save = async function (ctx) { }) } -exports.fetchView = async ctx => { +export async function fetchView(ctx: BBContext) { const viewName = ctx.params.viewName // if this is a table view being looked for just transfer to that @@ -199,7 +206,12 @@ exports.fetchView = async ctx => { } else { const tableId = viewInfo.meta.tableId const data = await getRawTableData(ctx, db, tableId) - response = await inMemoryViews.runView(viewInfo, calculation, group, data) + response = await inMemoryViews.runView( + viewInfo, + calculation as string, + !!group, + data + ) } let rows @@ -240,7 +252,7 @@ exports.fetchView = async ctx => { return rows } -exports.fetch = async ctx => { +export async function fetch(ctx: BBContext) { const db = context.getAppDB() const tableId = ctx.params.tableId @@ -249,7 +261,7 @@ exports.fetch = async ctx => { return outputProcessing(table, rows) } -exports.find = async ctx => { +export async function find(ctx: BBContext) { const db = dbCore.getDB(ctx.appId) const table = await db.get(ctx.params.tableId) let row = await findRow(ctx, ctx.params.tableId, ctx.params.rowId) @@ -257,7 +269,7 @@ exports.find = async ctx => { return row } -exports.destroy = async function (ctx) { +export async function destroy(ctx: BBContext) { const db = context.getAppDB() const { _id } = ctx.request.body let row = await db.get(_id) @@ -293,7 +305,7 @@ exports.destroy = async function (ctx) { return { response, row } } -exports.bulkDestroy = async ctx => { +export async function bulkDestroy(ctx: BBContext) { const db = context.getAppDB() const tableId = ctx.params.tableId const table = await db.get(tableId) @@ -301,10 +313,12 @@ exports.bulkDestroy = async ctx => { // before carrying out any updates, make sure the rows are ready to be returned // they need to be the full rows (including previous relationships) for automations - rows = await outputProcessing(table, rows, { squash: false }) + const processedRows = (await outputProcessing(table, rows, { + squash: false, + })) as Row[] // remove the relationships first - let updates = rows.map(row => + let updates: Promise[] = processedRows.map(row => linkRows.updateLinks({ eventType: linkRows.EventType.ROW_DELETE, row, @@ -313,7 +327,7 @@ exports.bulkDestroy = async ctx => { ) if (tableId === InternalTables.USER_METADATA) { updates = updates.concat( - rows.map(row => { + processedRows.map(row => { ctx.params = { id: row._id, } @@ -321,16 +335,16 @@ exports.bulkDestroy = async ctx => { }) ) } else { - await db.bulkDocs(rows.map(row => ({ ...row, _deleted: true }))) + await db.bulkDocs(processedRows.map(row => ({ ...row, _deleted: true }))) } // remove any attachments that were on the rows from object storage - await cleanupAttachments(table, { rows }) - await updateRelatedFormula(table, rows) + await cleanupAttachments(table, { rows: processedRows }) + await updateRelatedFormula(table, processedRows) await Promise.all(updates) - return { response: { ok: true }, rows } + return { response: { ok: true }, rows: processedRows } } -exports.search = async ctx => { +export async function search(ctx: BBContext) { // Fetch the whole table when running in cypress, as search doesn't work if (!env.COUCH_DB_URL && env.isCypress()) { return { rows: await exports.fetch(ctx) } @@ -362,14 +376,14 @@ exports.search = async ctx => { return response } -exports.validate = async ctx => { - return validate({ +export async function validate(ctx: BBContext) { + return rowValidate({ tableId: ctx.params.tableId, row: ctx.request.body, }) } -exports.exportRows = async ctx => { +export async function exportRows(ctx: BBContext) { const db = context.getAppDB() const table = await db.get(ctx.params.tableId) const rowIds = ctx.request.body.rows @@ -382,8 +396,8 @@ exports.exportRows = async ctx => { }) ).rows.map(row => row.doc) - let result = await outputProcessing(table, response) - let rows = [] + let result = (await outputProcessing(table, response)) as Row[] + let rows: Row[] = [] // Filter data to only specified columns if required if (columns && columns.length) { @@ -398,6 +412,7 @@ exports.exportRows = async ctx => { } let headers = Object.keys(rows[0]) + // @ts-ignore const exporter = exporters[format] const filename = `export.${format}` @@ -406,7 +421,7 @@ exports.exportRows = async ctx => { return apiFileReturn(exporter(headers, rows)) } -exports.fetchEnrichedRow = async ctx => { +export async function fetchEnrichedRow(ctx: BBContext) { const db = context.getAppDB() const tableId = ctx.params.tableId const rowId = ctx.params.rowId @@ -416,10 +431,10 @@ exports.fetchEnrichedRow = async ctx => { findRow(ctx, tableId, rowId), ]) // get the link docs - const linkVals = await linkRows.getLinkDocuments({ + const linkVals = (await linkRows.getLinkDocuments({ tableId, rowId, - }) + })) as LinkDocumentValue[] // look up the actual rows based on the ids let response = ( await db.allDocs({ @@ -428,8 +443,8 @@ exports.fetchEnrichedRow = async ctx => { }) ).rows.map(row => row.doc) // group responses by table - let groups = {}, - tables = {} + let groups: any = {}, + tables: Record = {} for (let row of response) { if (!row.tableId) { row.tableId = getTableIDFromRowID(row._id) @@ -442,11 +457,11 @@ exports.fetchEnrichedRow = async ctx => { groups[linkedTableId].push(row) } } - let linkedRows = [] + let linkedRows: Row[] = [] for (let [tableId, rows] of Object.entries(groups)) { // need to include the IDs in these rows for any links they may have linkedRows = linkedRows.concat( - await outputProcessing(tables[tableId], rows) + await outputProcessing(tables[tableId], rows as Row[]) ) } diff --git a/packages/server/src/api/controllers/row/staticFormula.ts b/packages/server/src/api/controllers/row/staticFormula.ts index 7d97d7e466..6e43f5822f 100644 --- a/packages/server/src/api/controllers/row/staticFormula.ts +++ b/packages/server/src/api/controllers/row/staticFormula.ts @@ -16,7 +16,7 @@ const { cloneDeep } = require("lodash/fp") * updated. * NOTE: this will only for affect static formulas. */ -exports.updateRelatedFormula = async (table: Table, enrichedRows: Row[]) => { +export async function updateRelatedFormula(table: Table, enrichedRows: Row[]) { const db = context.getAppDB() // no formula to update, we're done if (!table.relatedFormula) { diff --git a/packages/server/src/api/controllers/row/utils.ts b/packages/server/src/api/controllers/row/utils.ts index 3719c206e2..c003a16266 100644 --- a/packages/server/src/api/controllers/row/utils.ts +++ b/packages/server/src/api/controllers/row/utils.ts @@ -51,16 +51,19 @@ export async function validate({ }: { tableId?: string row: Row - table: Table + table?: Table }) { + let fetchedTable: Table if (!table) { const db = context.getAppDB() - table = await db.get(tableId) + fetchedTable = await db.get(tableId) + } else { + fetchedTable = table } const errors: any = {} - for (let fieldName of Object.keys(table.schema)) { - const constraints = cloneDeep(table.schema[fieldName].constraints) - const type = table.schema[fieldName].type + for (let fieldName of Object.keys(fetchedTable.schema)) { + const constraints = cloneDeep(fetchedTable.schema[fieldName].constraints) + const type = fetchedTable.schema[fieldName].type // formulas shouldn't validated, data will be deleted anyway if (type === FieldTypes.FORMULA) { continue diff --git a/packages/server/src/api/controllers/view/viewBuilder.js b/packages/server/src/api/controllers/view/viewBuilder.ts similarity index 85% rename from packages/server/src/api/controllers/view/viewBuilder.js rename to packages/server/src/api/controllers/view/viewBuilder.ts index 6596e0d9e7..3076e311b1 100644 --- a/packages/server/src/api/controllers/view/viewBuilder.js +++ b/packages/server/src/api/controllers/view/viewBuilder.ts @@ -1,4 +1,14 @@ -const TOKEN_MAP = { +import { ViewFilter } from "@budibase/types" + +type ViewTemplateOpts = { + field: string + tableId: string + groupBy: string + filters: ViewFilter[] + calculation: string +} + +const TOKEN_MAP: Record = { EQUALS: "===", NOT_EQUALS: "!==", LT: "<", @@ -10,13 +20,13 @@ const TOKEN_MAP = { OR: "||", } -const CONDITIONS = { +const CONDITIONS: Record = { EMPTY: "EMPTY", NOT_EMPTY: "NOT_EMPTY", CONTAINS: "CONTAINS", } -const isEmptyExpression = key => { +function isEmptyExpression(key: string) { return `( doc["${key}"] === undefined || doc["${key}"] === null || @@ -25,19 +35,19 @@ const isEmptyExpression = key => { )` } -const GROUP_PROPERTY = { +const GROUP_PROPERTY: Record = { group: { type: "string", }, } -const FIELD_PROPERTY = { +const FIELD_PROPERTY: Record = { field: { type: "string", }, } -const SCHEMA_MAP = { +const SCHEMA_MAP: Record = { sum: { field: "string", value: "number", @@ -74,7 +84,7 @@ const SCHEMA_MAP = { * @param {Array} filters - an array of filter objects * @returns {String} JS Expression */ -function parseFilterExpression(filters) { +function parseFilterExpression(filters: ViewFilter[]) { const expression = [] let first = true @@ -111,7 +121,7 @@ function parseFilterExpression(filters) { * @param {String?} field - field to use for calculations, if any * @param {String?} groupBy - field to group calculation results on, if any */ -function parseEmitExpression(field, groupBy) { +function parseEmitExpression(field: string, groupBy: string) { return `emit(doc["${groupBy || "_id"}"], doc["${field}"]);` } @@ -126,7 +136,13 @@ function parseEmitExpression(field, groupBy) { * filters: Array of filter objects containing predicates that are parsed into a JS expression * calculation: an optional calculation to be performed over the view data. */ -function viewTemplate({ field, tableId, groupBy, filters = [], calculation }) { +export = function ({ + field, + tableId, + groupBy, + filters = [], + calculation, +}: ViewTemplateOpts) { // first filter can't have a conjunction if (filters && filters.length > 0 && filters[0].conjunction) { delete filters[0].conjunction @@ -179,5 +195,3 @@ function viewTemplate({ field, tableId, groupBy, filters = [], calculation }) { ...reduction, } } - -module.exports = viewTemplate diff --git a/packages/server/src/db/inMemoryView.ts b/packages/server/src/db/inMemoryView.ts index c3659ef628..3c96aa26bd 100644 --- a/packages/server/src/db/inMemoryView.ts +++ b/packages/server/src/db/inMemoryView.ts @@ -8,8 +8,8 @@ const Pouch = dbCore.getPouch({ inMemory: true }) export async function runView( view: View, - calculation: boolean, - group: string, + calculation: string, + group: boolean, data: Row[] ) { // use a different ID each time for the DB, make sure they diff --git a/packages/server/src/utilities/rowProcessor/index.ts b/packages/server/src/utilities/rowProcessor/index.ts index d1aa93b4fe..e519c261c5 100644 --- a/packages/server/src/utilities/rowProcessor/index.ts +++ b/packages/server/src/utilities/rowProcessor/index.ts @@ -6,7 +6,7 @@ import { ObjectStoreBuckets } from "../../constants" import { context, db as dbCore, objectStore } from "@budibase/backend-core" import { InternalTables } from "../../db/utils" import { TYPE_TRANSFORM_MAP } from "./map" -import { Row, User, Table } from "@budibase/types" +import { Row, Table, ContextUser } from "@budibase/types" const { cloneDeep } = require("lodash/fp") export * from "./utils" @@ -49,7 +49,7 @@ function getRemovedAttachmentKeys( * for automatic ID purposes. */ export function processAutoColumn( - user: User | null, + user: ContextUser | null, table: Table, row: Row, opts?: AutoColumnProcessingOpts @@ -132,10 +132,10 @@ export function coerce(row: any, type: any) { * @returns {object} the row which has been prepared to be written to the DB. */ export function inputProcessing( - user: User, + user: ContextUser, table: Table, row: Row, - opts: AutoColumnProcessingOpts + opts?: AutoColumnProcessingOpts ) { let clonedRow = cloneDeep(row) // need to copy the table so it can be differenced on way out diff --git a/packages/types/src/documents/app/view.ts b/packages/types/src/documents/app/view.ts index b3dbe53129..de0dfea7f5 100644 --- a/packages/types/src/documents/app/view.ts +++ b/packages/types/src/documents/app/view.ts @@ -33,7 +33,7 @@ export interface ViewStatisticsSchema { } export interface ViewFilter { - value: any + value?: any condition: string key: string conjunction?: string diff --git a/packages/types/src/sdk/db.ts b/packages/types/src/sdk/db.ts index 200b25c501..84daed0f90 100644 --- a/packages/types/src/sdk/db.ts +++ b/packages/types/src/sdk/db.ts @@ -57,6 +57,7 @@ export type DatabaseQueryOpts = { descending?: boolean key?: string keys?: string[] + group?: boolean } export const isDocument = (doc: any): doc is Document => {