budibase/packages/server/src/utilities/schema.ts

261 lines
7.8 KiB
TypeScript
Raw Normal View History

2024-03-20 19:33:39 +01:00
import {
FieldType,
2024-04-26 12:23:11 +02:00
BBReferenceFieldSubType,
2024-03-20 19:33:39 +01:00
TableSchema,
FieldSchema,
Row,
Table,
2024-03-20 19:33:39 +01:00
} from "@budibase/types"
2024-06-03 16:56:12 +02:00
import { ValidColumnNameRegex, helpers, utils } from "@budibase/shared-core"
2023-10-09 18:02:21 +02:00
import { db } from "@budibase/backend-core"
type Rows = Array<Row>
interface SchemaValidation {
[index: string]: boolean
}
interface ValidationResults {
schemaValidation: SchemaValidation
allValid: boolean
invalidColumns: Array<string>
errors: Record<string, string>
}
2024-03-20 19:33:39 +01:00
export function isSchema(schema: any): schema is TableSchema {
return (
typeof schema === "object" &&
2024-03-20 19:33:39 +01:00
Object.values<FieldSchema>(schema).every(column => {
return (
column !== null &&
typeof column === "object" &&
typeof column.type === "string" &&
Object.values(FieldType).includes(column.type as FieldType)
)
})
)
}
export function isRows(rows: any): rows is Rows {
return Array.isArray(rows) && rows.every(row => typeof row === "object")
}
2024-07-31 17:14:02 +02:00
export function validate(
rows: Rows,
schema: TableSchema,
protectedColumnNames: readonly string[]
): ValidationResults {
const results: ValidationResults = {
schemaValidation: {},
allValid: false,
invalidColumns: [],
errors: {},
}
2024-07-31 17:14:02 +02:00
protectedColumnNames = protectedColumnNames.map(x => x.toLowerCase())
rows.forEach(row => {
Object.entries(row).forEach(([columnName, columnData]) => {
2024-03-20 19:33:39 +01:00
const {
type: columnType,
subtype: columnSubtype,
autocolumn: isAutoColumn,
2024-05-02 13:19:19 +02:00
constraints,
2024-04-03 17:01:36 +02:00
} = schema[columnName] || {}
2023-10-09 18:02:21 +02:00
// If the column had an invalid value we don't want to override it
if (results.schemaValidation[columnName] === false) {
return
}
2024-07-31 17:14:02 +02:00
if (protectedColumnNames.includes(columnName.toLowerCase())) {
results.schemaValidation[columnName] = false
2024-07-31 17:40:30 +02:00
results.errors[columnName] = `${columnName} is a protected column name`
2024-07-31 17:14:02 +02:00
return
}
// If the columnType is not a string, then it's not present in the schema, and should be added to the invalid columns array
if (typeof columnType !== "string") {
results.invalidColumns.push(columnName)
} else if (!columnName.match(ValidColumnNameRegex)) {
// Check for special characters in column names
results.schemaValidation[columnName] = false
results.errors[columnName] =
"Column names can't contain special characters"
2023-05-02 13:48:05 +02:00
} else if (
columnData == null &&
!schema[columnName].constraints?.presence
) {
results.schemaValidation[columnName] = true
} else if (
// If there's no data for this field don't bother with further checks
// If the field is already marked as invalid there's no need for further checks
columnData == null ||
isAutoColumn
) {
return
} else if (columnType === FieldType.NUMBER && isNaN(Number(columnData))) {
// If provided must be a valid number
results.schemaValidation[columnName] = false
} else if (
// If provided must be a valid date
columnType === FieldType.DATETIME &&
isNaN(new Date(columnData).getTime())
) {
results.schemaValidation[columnName] = false
2023-10-09 18:02:21 +02:00
} else if (
2024-04-22 11:14:23 +02:00
(columnType === FieldType.BB_REFERENCE ||
columnType === FieldType.BB_REFERENCE_SINGLE) &&
2024-05-27 13:39:43 +02:00
!isValidBBReference(
columnData,
columnType,
columnSubtype,
2024-06-03 16:56:12 +02:00
helpers.schema.isRequired(constraints)
2024-05-27 13:39:43 +02:00
)
2023-10-09 18:02:21 +02:00
) {
results.schemaValidation[columnName] = false
} else {
results.schemaValidation[columnName] = true
}
})
})
2024-07-31 17:40:30 +02:00
for (const schemaField of Object.keys(schema)) {
if (protectedColumnNames.includes(schemaField.toLowerCase())) {
results.schemaValidation[schemaField] = false
results.errors[schemaField] = `${schemaField} is a protected column name`
}
}
results.allValid =
Object.values(results.schemaValidation).length > 0 &&
Object.values(results.schemaValidation).every(column => column)
// Select unique values
results.invalidColumns = [...new Set(results.invalidColumns)]
return results
}
export function parse(rows: Rows, table: Table): Rows {
return rows.map(row => {
const parsedRow: Row = {}
Object.entries(row).forEach(([columnName, columnData]) => {
const schema = table.schema
if (!(columnName in schema)) {
// Objects can be present in the row data but not in the schema, so make sure we don't proceed in such a case
return
}
if (
schema[columnName].autocolumn &&
!table.primary?.includes(columnName)
) {
// Don't want the user specifying values for autocolumns unless they're updating
// a row through its primary key.
return
}
2024-05-20 16:34:22 +02:00
const columnSchema = schema[columnName]
const { type: columnType } = columnSchema
2024-08-02 11:51:19 +02:00
if ([FieldType.NUMBER, FieldType.BIGINT].includes(columnType)) {
// If provided must be a valid number
parsedRow[columnName] = columnData ? Number(columnData) : columnData
2024-05-22 11:58:57 +02:00
} else if (
columnType === FieldType.DATETIME &&
!columnSchema.timeOnly &&
!columnSchema.dateOnly
) {
// If provided must be a valid date
parsedRow[columnName] = columnData
? new Date(columnData).toISOString()
: columnData
2024-08-02 11:51:19 +02:00
} else if (
columnType === FieldType.JSON &&
typeof columnData === "string"
) {
parsedRow[columnName] = parseJsonExport(columnData)
} else if (columnType === FieldType.BB_REFERENCE) {
2024-05-10 11:45:15 +02:00
let parsedValues: { _id: string }[] = columnData || []
2024-08-02 11:51:19 +02:00
if (columnData && typeof columnData === "string") {
parsedValues = parseJsonExport<{ _id: string }[]>(columnData)
2024-05-10 11:45:15 +02:00
}
2024-05-03 17:53:59 +02:00
parsedRow[columnName] = parsedValues?.map(u => u._id)
2024-04-25 16:31:15 +02:00
} else if (columnType === FieldType.BB_REFERENCE_SINGLE) {
2024-08-02 11:51:19 +02:00
let parsedValue = columnData
if (columnData && typeof columnData === "string") {
parsedValue = parseJsonExport<{ _id: string }>(columnData)
}
2024-04-25 16:31:15 +02:00
parsedRow[columnName] = parsedValue?._id
2024-04-03 17:01:36 +02:00
} else if (
(columnType === FieldType.ATTACHMENTS ||
columnType === FieldType.ATTACHMENT_SINGLE ||
columnType === FieldType.SIGNATURE_SINGLE) &&
2024-04-03 17:01:36 +02:00
typeof columnData === "string"
) {
2024-08-02 11:51:19 +02:00
parsedRow[columnName] = parseJsonExport(columnData)
} else {
parsedRow[columnName] = columnData
}
})
return parsedRow
})
}
2023-10-09 18:02:21 +02:00
function isValidBBReference(
2024-04-25 15:50:28 +02:00
data: any,
type: FieldType.BB_REFERENCE | FieldType.BB_REFERENCE_SINGLE,
2024-05-02 13:19:19 +02:00
subtype: BBReferenceFieldSubType,
isRequired: boolean
2023-10-09 18:02:21 +02:00
): boolean {
2024-04-25 15:50:28 +02:00
if (typeof data !== "string") {
return false
}
if (type === FieldType.BB_REFERENCE_SINGLE) {
if (!data) {
2024-05-02 13:19:19 +02:00
return !isRequired
2024-04-25 15:50:28 +02:00
}
2024-08-02 11:51:19 +02:00
const user = parseJsonExport<{ _id: string }>(data)
2024-04-25 15:50:28 +02:00
return db.isGlobalUserID(user._id)
}
switch (subtype) {
2024-04-26 12:23:11 +02:00
case BBReferenceFieldSubType.USER:
case BBReferenceFieldSubType.USERS: {
2024-08-02 11:51:19 +02:00
const userArray = parseJsonExport<{ _id: string }[]>(data)
2023-10-10 15:39:05 +02:00
if (!Array.isArray(userArray)) {
2023-10-09 18:02:21 +02:00
return false
}
2023-10-10 14:20:26 +02:00
const constainsWrongId = userArray.find(
user => !db.isGlobalUserID(user._id)
)
return !constainsWrongId
2024-03-19 16:58:25 +01:00
}
2023-10-09 18:02:21 +02:00
default:
2024-04-25 15:50:28 +02:00
throw utils.unreachable(subtype)
2023-10-09 18:02:21 +02:00
}
}
2024-08-02 11:51:19 +02:00
function parseJsonExport<T>(value: string) {
try {
const parsed = JSON.parse(value)
return parsed as T
} catch (e: any) {
if (
e.message.startsWith("Expected property name or '}' in JSON at position ")
) {
// This was probably converted as CSV and it has single quotes instead of double ones
const parsed = JSON.parse(value.replace(/'/g, '"'))
return parsed as T
}
// It is no a valid JSON
throw e
}
}