Updating datasource information endpoint to POST which allows sending up an unfinished/unsaved datasource for fetching information with. Also changing how verification and information endpoints work so that enrichment is used and therefore env vars can also be used.

This commit is contained in:
mike12345567 2023-05-31 17:04:29 +01:00
parent bfb3ae66a9
commit 651d50a064
8 changed files with 41 additions and 31 deletions

View File

@ -21,6 +21,7 @@ import {
CreateDatasourceRequest, CreateDatasourceRequest,
VerifyDatasourceRequest, VerifyDatasourceRequest,
VerifyDatasourceResponse, VerifyDatasourceResponse,
FetchDatasourceInfoRequest,
FetchDatasourceInfoResponse, FetchDatasourceInfoResponse,
IntegrationBase, IntegrationBase,
DatasourcePlus, DatasourcePlus,
@ -57,6 +58,21 @@ async function getConnector(
return new Connector(datasource.config) return new Connector(datasource.config)
} }
async function getAndMergeDatasource(datasource: Datasource) {
let existingDatasource: undefined | Datasource
if (datasource._id) {
existingDatasource = await sdk.datasources.get(datasource._id)
}
let enrichedDatasource = datasource
if (existingDatasource) {
enrichedDatasource = sdk.datasources.mergeConfigs(
datasource,
existingDatasource
)
}
return await sdk.datasources.enrich(enrichedDatasource)
}
async function buildSchemaHelper(datasource: Datasource) { async function buildSchemaHelper(datasource: Datasource) {
const connector = (await getConnector(datasource)) as DatasourcePlus const connector = (await getConnector(datasource)) as DatasourcePlus
await connector.buildSchema(datasource._id!, datasource.entities!) await connector.buildSchema(datasource._id!, datasource.entities!)
@ -132,17 +148,7 @@ export async function verify(
ctx: UserCtx<VerifyDatasourceRequest, VerifyDatasourceResponse> ctx: UserCtx<VerifyDatasourceRequest, VerifyDatasourceResponse>
) { ) {
const { datasource } = ctx.request.body const { datasource } = ctx.request.body
let existingDatasource: undefined | Datasource const enrichedDatasource = await getAndMergeDatasource(datasource)
if (datasource._id) {
existingDatasource = await sdk.datasources.get(datasource._id)
}
let enrichedDatasource = datasource
if (existingDatasource) {
enrichedDatasource = sdk.datasources.mergeConfigs(
datasource,
existingDatasource
)
}
const connector = await getConnector(enrichedDatasource) const connector = await getConnector(enrichedDatasource)
if (!connector.testConnection) { if (!connector.testConnection) {
ctx.throw(400, "Connection information verification not supported") ctx.throw(400, "Connection information verification not supported")
@ -156,11 +162,11 @@ export async function verify(
} }
export async function information( export async function information(
ctx: UserCtx<void, FetchDatasourceInfoResponse> ctx: UserCtx<FetchDatasourceInfoRequest, FetchDatasourceInfoResponse>
) { ) {
const datasourceId = ctx.params.datasourceId const { datasource } = ctx.request.body
const datasource = await sdk.datasources.get(datasourceId, { enriched: true }) const enrichedDatasource = await getAndMergeDatasource(datasource)
const connector = (await getConnector(datasource)) as DatasourcePlus const connector = (await getConnector(enrichedDatasource)) as DatasourcePlus
if (!connector.getTableNames) { if (!connector.getTableNames) {
ctx.throw(400, "Table name fetching not supported by datasource") ctx.throw(400, "Table name fetching not supported by datasource")
} }
@ -297,7 +303,7 @@ export async function update(ctx: UserCtx<any, UpdateDatasourceResponse>) {
ctx.body = { ctx.body = {
datasource: await sdk.datasources.removeSecretSingle(datasource), datasource: await sdk.datasources.removeSecretSingle(datasource),
} }
builderSocket.emitDatasourceUpdate(ctx, datasource) builderSocket?.emitDatasourceUpdate(ctx, datasource)
} }
export async function save( export async function save(
@ -340,7 +346,7 @@ export async function save(
response.error = schemaError response.error = schemaError
} }
ctx.body = response ctx.body = response
builderSocket.emitDatasourceUpdate(ctx, datasource) builderSocket?.emitDatasourceUpdate(ctx, datasource)
} }
async function destroyInternalTablesBySourceId(datasourceId: string) { async function destroyInternalTablesBySourceId(datasourceId: string) {
@ -400,7 +406,7 @@ export async function destroy(ctx: UserCtx) {
ctx.message = `Datasource deleted.` ctx.message = `Datasource deleted.`
ctx.status = 200 ctx.status = 200
builderSocket.emitDatasourceDeletion(ctx, datasourceId) builderSocket?.emitDatasourceDeletion(ctx, datasourceId)
} }
export async function find(ctx: UserCtx) { export async function find(ctx: UserCtx) {

View File

@ -71,7 +71,7 @@ export async function create(ctx: any) {
const doc = await pro.plugins.storePlugin(metadata, directory, source) const doc = await pro.plugins.storePlugin(metadata, directory, source)
clientAppSocket.emit("plugins-update", { name, hash: doc.hash }) clientAppSocket?.emit("plugins-update", { name, hash: doc.hash })
ctx.body = { ctx.body = {
message: "Plugin uploaded successfully", message: "Plugin uploaded successfully",
plugins: [doc], plugins: [doc],

View File

@ -78,7 +78,7 @@ export async function save(ctx: UserCtx) {
ctx.eventEmitter && ctx.eventEmitter &&
ctx.eventEmitter.emitTable(`table:save`, appId, savedTable) ctx.eventEmitter.emitTable(`table:save`, appId, savedTable)
ctx.body = savedTable ctx.body = savedTable
builderSocket.emitTableUpdate(ctx, savedTable) builderSocket?.emitTableUpdate(ctx, savedTable)
} }
export async function destroy(ctx: UserCtx) { export async function destroy(ctx: UserCtx) {
@ -91,7 +91,7 @@ export async function destroy(ctx: UserCtx) {
ctx.status = 200 ctx.status = 200
ctx.table = deletedTable ctx.table = deletedTable
ctx.body = { message: `Table ${tableId} deleted.` } ctx.body = { message: `Table ${tableId} deleted.` }
builderSocket.emitTableDeletion(ctx, tableId) builderSocket?.emitTableDeletion(ctx, tableId)
} }
export async function bulkImport(ctx: UserCtx) { export async function bulkImport(ctx: UserCtx) {

View File

@ -58,7 +58,7 @@ export async function save(ctx: Ctx) {
await handleViewEvents(existingTable.views[viewName], table.views[viewName]) await handleViewEvents(existingTable.views[viewName], table.views[viewName])
ctx.body = table.views[viewName] ctx.body = table.views[viewName]
builderSocket.emitTableUpdate(ctx, table) builderSocket?.emitTableUpdate(ctx, table)
} }
export async function calculationEvents(existingView: View, newView: View) { export async function calculationEvents(existingView: View, newView: View) {
@ -127,7 +127,7 @@ export async function destroy(ctx: Ctx) {
await events.view.deleted(view) await events.view.deleted(view)
ctx.body = view ctx.body = view
builderSocket.emitTableUpdate(ctx, table) builderSocket?.emitTableUpdate(ctx, table)
} }
export async function exportView(ctx: Ctx) { export async function exportView(ctx: Ctx) {

View File

@ -20,8 +20,8 @@ router
authorized(permissions.BUILDER), authorized(permissions.BUILDER),
datasourceController.verify datasourceController.verify
) )
.get( .post(
"/api/datasources/:datasourceId/info", "/api/datasources/info",
authorized(permissions.BUILDER), authorized(permissions.BUILDER),
datasourceController.information datasourceController.information
) )

View File

@ -25,6 +25,7 @@ const config = setup.getConfig()!
jest.setTimeout(30000) jest.setTimeout(30000)
jest.unmock("pg") jest.unmock("pg")
jest.mock("../websockets")
describe("postgres integrations", () => { describe("postgres integrations", () => {
let makeRequest: MakeRequestResponse, let makeRequest: MakeRequestResponse,
@ -1055,13 +1056,12 @@ describe("postgres integrations", () => {
}) })
}) })
describe("GET /api/datasources/:datasourceId/info", () => { describe("POST /api/datasources/info", () => {
it("should fetch information about postgres datasource", async () => { it("should fetch information about postgres datasource", async () => {
const primaryName = primaryPostgresTable.name const primaryName = primaryPostgresTable.name
const response = await makeRequest( const response = await makeRequest("post", "/api/datasources/info", {
"get", datasource: postgresDatasource,
`/api/datasources/${postgresDatasource._id}/info` })
)
expect(response.status).toBe(200) expect(response.status).toBe(200)
expect(response.body.tableNames).toBeDefined() expect(response.body.tableNames).toBeDefined()
expect(response.body.tableNames.indexOf(primaryName)).not.toBe(-1) expect(response.body.tableNames.indexOf(primaryName)).not.toBe(-1)

View File

@ -36,6 +36,6 @@ export async function processUploaded(plugin: FileType, source?: PluginSource) {
} }
const doc = await pro.plugins.storePlugin(metadata, directory, source) const doc = await pro.plugins.storePlugin(metadata, directory, source)
clientAppSocket.emit("plugin-update", { name: doc.name, hash: doc.hash }) clientAppSocket?.emit("plugin-update", { name: doc.name, hash: doc.hash })
return doc return doc
} }

View File

@ -23,6 +23,10 @@ export interface VerifyDatasourceResponse {
error?: string error?: string
} }
export interface FetchDatasourceInfoRequest {
datasource: Datasource
}
export interface FetchDatasourceInfoResponse { export interface FetchDatasourceInfoResponse {
tableNames: string[] tableNames: string[]
} }