2022-01-25 20:22:43 +01:00
|
|
|
import { API } from "api"
|
2022-01-20 12:50:18 +01:00
|
|
|
import { JSONUtils } from "@budibase/frontend-core"
|
|
|
|
import TableFetch from "@budibase/frontend-core/src/fetch/TableFetch.js"
|
|
|
|
import ViewFetch from "@budibase/frontend-core/src/fetch/ViewFetch.js"
|
|
|
|
import QueryFetch from "@budibase/frontend-core/src/fetch/QueryFetch.js"
|
|
|
|
import RelationshipFetch from "@budibase/frontend-core/src/fetch/RelationshipFetch.js"
|
|
|
|
import NestedProviderFetch from "@budibase/frontend-core/src/fetch/NestedProviderFetch.js"
|
|
|
|
import FieldFetch from "@budibase/frontend-core/src/fetch/FieldFetch.js"
|
|
|
|
import JSONArrayFetch from "@budibase/frontend-core/src/fetch/JSONArrayFetch.js"
|
2021-12-17 09:22:04 +01:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Fetches the schema of any kind of datasource.
|
2022-01-04 16:02:43 +01:00
|
|
|
* All datasource fetch classes implement their own functionality to get the
|
|
|
|
* schema of a datasource of their respective types.
|
2022-01-17 15:03:47 +01:00
|
|
|
* @param datasource the datasource to fetch the schema for
|
|
|
|
* @param options options for enriching the schema
|
2021-12-17 09:22:04 +01:00
|
|
|
*/
|
2022-01-17 15:03:47 +01:00
|
|
|
export const fetchDatasourceSchema = async (
|
|
|
|
datasource,
|
|
|
|
options = { enrichRelationships: false }
|
|
|
|
) => {
|
2021-12-17 19:39:48 +01:00
|
|
|
const handler = {
|
|
|
|
table: TableFetch,
|
|
|
|
view: ViewFetch,
|
|
|
|
query: QueryFetch,
|
2022-01-04 16:02:43 +01:00
|
|
|
link: RelationshipFetch,
|
|
|
|
provider: NestedProviderFetch,
|
|
|
|
field: FieldFetch,
|
|
|
|
jsonarray: JSONArrayFetch,
|
|
|
|
}[datasource?.type]
|
|
|
|
if (!handler) {
|
|
|
|
return null
|
2022-01-04 15:34:09 +01:00
|
|
|
}
|
2022-01-20 12:50:18 +01:00
|
|
|
const instance = new handler({ API })
|
2022-01-04 15:34:09 +01:00
|
|
|
|
2022-01-04 16:02:43 +01:00
|
|
|
// Get the datasource definition and then schema
|
2022-01-20 12:50:18 +01:00
|
|
|
const definition = await instance.getDefinition(datasource)
|
|
|
|
let schema = instance.getSchema(datasource, definition)
|
2022-01-04 16:02:43 +01:00
|
|
|
if (!schema) {
|
|
|
|
return null
|
2021-12-17 09:22:04 +01:00
|
|
|
}
|
|
|
|
|
2022-01-04 16:02:43 +01:00
|
|
|
// Check for any JSON fields so we can add any top level properties
|
|
|
|
let jsonAdditions = {}
|
|
|
|
Object.keys(schema).forEach(fieldKey => {
|
|
|
|
const fieldSchema = schema[fieldKey]
|
|
|
|
if (fieldSchema?.type === "json") {
|
2022-01-20 12:50:18 +01:00
|
|
|
const jsonSchema = JSONUtils.convertJSONSchemaToTableSchema(fieldSchema, {
|
2022-01-04 16:02:43 +01:00
|
|
|
squashObjects: true,
|
|
|
|
})
|
|
|
|
Object.keys(jsonSchema).forEach(jsonKey => {
|
|
|
|
jsonAdditions[`${fieldKey}.${jsonKey}`] = {
|
|
|
|
type: jsonSchema[jsonKey].type,
|
|
|
|
nestedJSON: true,
|
|
|
|
}
|
|
|
|
})
|
|
|
|
}
|
|
|
|
})
|
2022-01-17 15:03:47 +01:00
|
|
|
schema = { ...schema, ...jsonAdditions }
|
|
|
|
|
|
|
|
// Check for any relationship fields if required
|
2022-01-17 17:48:49 +01:00
|
|
|
if (options?.enrichRelationships && definition.sql) {
|
2022-01-17 15:03:47 +01:00
|
|
|
let relationshipAdditions = {}
|
|
|
|
for (let fieldKey of Object.keys(schema)) {
|
|
|
|
const fieldSchema = schema[fieldKey]
|
|
|
|
if (fieldSchema?.type === "link") {
|
|
|
|
const linkSchema = await fetchDatasourceSchema({
|
|
|
|
type: "table",
|
|
|
|
tableId: fieldSchema?.tableId,
|
|
|
|
})
|
|
|
|
Object.keys(linkSchema || {}).forEach(linkKey => {
|
|
|
|
relationshipAdditions[`${fieldKey}.${linkKey}`] = {
|
|
|
|
type: linkSchema[linkKey].type,
|
|
|
|
}
|
|
|
|
})
|
|
|
|
}
|
|
|
|
}
|
|
|
|
schema = { ...schema, ...relationshipAdditions }
|
|
|
|
}
|
|
|
|
|
|
|
|
// Ensure schema structure is correct
|
2022-01-20 12:50:18 +01:00
|
|
|
return instance.enrichSchema(schema)
|
2021-12-17 09:22:04 +01:00
|
|
|
}
|