2021-06-24 19:17:26 +02:00
|
|
|
import {
|
2023-05-15 18:36:16 +02:00
|
|
|
ConnectionInfo,
|
2023-12-19 11:11:51 +01:00
|
|
|
Database,
|
2023-05-05 16:47:55 +02:00
|
|
|
DatasourceFeature,
|
2022-08-11 14:50:05 +02:00
|
|
|
DatasourceFieldType,
|
2023-01-31 20:49:31 +01:00
|
|
|
Document,
|
|
|
|
Integration,
|
2022-08-11 12:48:58 +02:00
|
|
|
IntegrationBase,
|
2023-01-31 20:49:31 +01:00
|
|
|
QueryType,
|
2022-08-11 12:48:58 +02:00
|
|
|
} from "@budibase/types"
|
2023-01-31 20:49:31 +01:00
|
|
|
import { db as dbCore } from "@budibase/backend-core"
|
2024-03-28 13:14:56 +01:00
|
|
|
import { HOST_ADDRESS } from "./utils"
|
2021-06-24 19:16:48 +02:00
|
|
|
|
2022-08-12 18:03:06 +02:00
|
|
|
interface CouchDBConfig {
|
|
|
|
url: string
|
|
|
|
database: string
|
|
|
|
}
|
2021-06-24 19:16:48 +02:00
|
|
|
|
2022-08-12 18:03:06 +02:00
|
|
|
const SCHEMA: Integration = {
|
|
|
|
docs: "https://docs.couchdb.org/en/stable/",
|
|
|
|
friendlyName: "CouchDB",
|
|
|
|
type: "Non-relational",
|
|
|
|
description:
|
|
|
|
"Apache CouchDB is an open-source document-oriented NoSQL database, implemented in Erlang.",
|
2023-05-24 10:50:51 +02:00
|
|
|
features: {
|
|
|
|
[DatasourceFeature.CONNECTION_CHECKING]: true,
|
|
|
|
},
|
2022-08-12 18:03:06 +02:00
|
|
|
datasource: {
|
|
|
|
url: {
|
|
|
|
type: DatasourceFieldType.STRING,
|
|
|
|
required: true,
|
2024-03-28 13:14:56 +01:00
|
|
|
default: `http://${HOST_ADDRESS}:5984`,
|
2021-06-24 19:16:48 +02:00
|
|
|
},
|
2022-08-12 18:03:06 +02:00
|
|
|
database: {
|
|
|
|
type: DatasourceFieldType.STRING,
|
|
|
|
required: true,
|
|
|
|
},
|
|
|
|
},
|
|
|
|
query: {
|
|
|
|
create: {
|
|
|
|
type: QueryType.JSON,
|
|
|
|
},
|
|
|
|
read: {
|
|
|
|
type: QueryType.JSON,
|
|
|
|
},
|
|
|
|
update: {
|
|
|
|
type: QueryType.JSON,
|
|
|
|
},
|
2023-01-31 20:49:31 +01:00
|
|
|
get: {
|
|
|
|
type: QueryType.FIELDS,
|
|
|
|
fields: {
|
|
|
|
id: {
|
|
|
|
type: DatasourceFieldType.STRING,
|
|
|
|
required: true,
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
2022-08-12 18:03:06 +02:00
|
|
|
delete: {
|
|
|
|
type: QueryType.FIELDS,
|
|
|
|
fields: {
|
|
|
|
id: {
|
|
|
|
type: DatasourceFieldType.STRING,
|
|
|
|
required: true,
|
2021-06-24 19:16:48 +02:00
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
2022-08-12 18:03:06 +02:00
|
|
|
},
|
|
|
|
}
|
2021-06-24 19:16:48 +02:00
|
|
|
|
2022-08-12 18:03:06 +02:00
|
|
|
class CouchDBIntegration implements IntegrationBase {
|
2023-12-19 11:11:51 +01:00
|
|
|
private readonly client: Database
|
2021-06-24 19:16:48 +02:00
|
|
|
|
2022-08-12 18:03:06 +02:00
|
|
|
constructor(config: CouchDBConfig) {
|
2024-07-03 13:50:33 +02:00
|
|
|
if (!config.url || !config.database) {
|
|
|
|
throw new Error("Unable to connect without URL or database")
|
|
|
|
}
|
2023-12-19 11:11:51 +01:00
|
|
|
this.client = dbCore.DatabaseWithConnection(config.database, config.url)
|
2022-08-12 18:03:06 +02:00
|
|
|
}
|
2021-06-24 19:16:48 +02:00
|
|
|
|
2023-05-12 12:35:07 +02:00
|
|
|
async testConnection() {
|
2023-05-15 18:36:16 +02:00
|
|
|
const response: ConnectionInfo = {
|
|
|
|
connected: false,
|
|
|
|
}
|
2023-05-12 12:35:07 +02:00
|
|
|
try {
|
2024-07-03 17:59:31 +02:00
|
|
|
response.connected = await this.client.exists()
|
2023-05-12 12:35:07 +02:00
|
|
|
} catch (e: any) {
|
2023-05-15 18:36:16 +02:00
|
|
|
response.error = e.message as string
|
2023-05-12 12:35:07 +02:00
|
|
|
}
|
2023-05-15 18:36:16 +02:00
|
|
|
return response
|
2023-05-12 12:35:07 +02:00
|
|
|
}
|
|
|
|
|
2023-01-31 20:49:31 +01:00
|
|
|
private parse(query: { json: string | object }) {
|
|
|
|
return typeof query.json === "string" ? JSON.parse(query.json) : query.json
|
|
|
|
}
|
|
|
|
|
|
|
|
async create(query: { json: string | object }) {
|
|
|
|
const parsed = this.parse(query)
|
2024-07-03 17:59:31 +02:00
|
|
|
return await this.client.put(parsed)
|
2022-08-12 18:03:06 +02:00
|
|
|
}
|
2021-06-24 19:16:48 +02:00
|
|
|
|
2023-01-31 20:49:31 +01:00
|
|
|
async read(query: { json: string | object }) {
|
|
|
|
const parsed = this.parse(query)
|
2024-07-03 13:50:33 +02:00
|
|
|
const params = {
|
|
|
|
include_docs: true,
|
|
|
|
...parsed,
|
|
|
|
}
|
2024-07-03 17:59:31 +02:00
|
|
|
const result = await this.client.allDocs(params)
|
2024-07-03 13:50:33 +02:00
|
|
|
return result.rows.map(row => row.doc)
|
2022-08-12 18:03:06 +02:00
|
|
|
}
|
2021-06-24 19:16:48 +02:00
|
|
|
|
2023-01-31 20:49:31 +01:00
|
|
|
async update(query: { json: string | object }) {
|
|
|
|
const parsed: Document = this.parse(query)
|
|
|
|
if (!parsed?._rev && parsed?._id) {
|
|
|
|
const oldDoc = await this.get({ id: parsed._id })
|
|
|
|
parsed._rev = oldDoc._rev
|
|
|
|
}
|
2024-07-03 17:59:31 +02:00
|
|
|
return await this.client.put(parsed)
|
2023-01-31 20:49:31 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
async get(query: { id: string }) {
|
2024-07-03 17:59:31 +02:00
|
|
|
return await this.client.get(query.id)
|
2021-06-24 19:16:48 +02:00
|
|
|
}
|
|
|
|
|
2022-08-12 18:03:06 +02:00
|
|
|
async delete(query: { id: string }) {
|
2024-07-03 17:59:31 +02:00
|
|
|
return await this.client.remove(query.id)
|
2021-06-24 19:16:48 +02:00
|
|
|
}
|
|
|
|
}
|
2022-08-12 18:03:06 +02:00
|
|
|
|
|
|
|
export default {
|
|
|
|
schema: SCHEMA,
|
|
|
|
integration: CouchDBIntegration,
|
|
|
|
}
|