2024-01-02 12:36:32 +01:00
|
|
|
import {
|
|
|
|
DocumentDestroyResponse,
|
|
|
|
DocumentInsertResponse,
|
|
|
|
DocumentBulkResponse,
|
|
|
|
OkResponse,
|
|
|
|
} from "@budibase/nano"
|
|
|
|
import {
|
|
|
|
AllDocsResponse,
|
|
|
|
AnyDocument,
|
|
|
|
Database,
|
|
|
|
DatabaseDumpOpts,
|
|
|
|
DatabasePutOpts,
|
|
|
|
DatabaseQueryOpts,
|
|
|
|
Document,
|
2024-02-29 17:28:00 +01:00
|
|
|
RowValue,
|
2024-04-10 18:36:57 +02:00
|
|
|
SqlQueryBinding,
|
2024-01-02 12:36:32 +01:00
|
|
|
} from "@budibase/types"
|
|
|
|
import tracer from "dd-trace"
|
|
|
|
import { Writable } from "stream"
|
|
|
|
|
|
|
|
export class DDInstrumentedDatabase implements Database {
|
|
|
|
constructor(private readonly db: Database) {}
|
|
|
|
|
|
|
|
get name(): string {
|
|
|
|
return this.db.name
|
|
|
|
}
|
|
|
|
|
2024-03-01 13:59:51 +01:00
|
|
|
exists(docId?: string): Promise<boolean> {
|
2024-01-02 12:36:32 +01:00
|
|
|
return tracer.trace("db.exists", span => {
|
2024-11-21 17:36:31 +01:00
|
|
|
span.addTags({ db_name: this.name, doc_id: docId })
|
2024-03-01 13:59:51 +01:00
|
|
|
if (docId) {
|
|
|
|
return this.db.exists(docId)
|
|
|
|
}
|
2024-01-02 12:36:32 +01:00
|
|
|
return this.db.exists()
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
get<T extends Document>(id?: string | undefined): Promise<T> {
|
|
|
|
return tracer.trace("db.get", span => {
|
2024-11-21 17:36:31 +01:00
|
|
|
span.addTags({ db_name: this.name, doc_id: id })
|
2024-01-02 12:36:32 +01:00
|
|
|
return this.db.get(id)
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
2024-10-10 16:49:00 +02:00
|
|
|
tryGet<T extends Document>(id?: string | undefined): Promise<T | undefined> {
|
2024-11-21 17:36:31 +01:00
|
|
|
return tracer.trace("db.tryGet", async span => {
|
|
|
|
span.addTags({ db_name: this.name, doc_id: id })
|
|
|
|
const doc = await this.db.tryGet<T>(id)
|
|
|
|
span.addTags({ doc_found: doc !== undefined })
|
|
|
|
return doc
|
2024-10-10 16:49:00 +02:00
|
|
|
})
|
|
|
|
}
|
|
|
|
|
2024-01-02 12:36:32 +01:00
|
|
|
getMultiple<T extends Document>(
|
2025-03-10 16:50:08 +01:00
|
|
|
ids?: string[],
|
2024-01-02 12:36:32 +01:00
|
|
|
opts?: { allowMissing?: boolean | undefined } | undefined
|
|
|
|
): Promise<T[]> {
|
2024-11-21 17:36:31 +01:00
|
|
|
return tracer.trace("db.getMultiple", async span => {
|
|
|
|
span.addTags({
|
2024-01-02 12:36:32 +01:00
|
|
|
db_name: this.name,
|
2025-03-10 16:50:08 +01:00
|
|
|
num_docs: ids?.length || 0,
|
2024-01-02 12:36:32 +01:00
|
|
|
allow_missing: opts?.allowMissing,
|
|
|
|
})
|
2024-11-21 17:36:31 +01:00
|
|
|
const docs = await this.db.getMultiple<T>(ids, opts)
|
|
|
|
span.addTags({ num_docs_found: docs.length })
|
|
|
|
return docs
|
2024-01-02 12:36:32 +01:00
|
|
|
})
|
|
|
|
}
|
|
|
|
|
2024-05-16 14:27:54 +02:00
|
|
|
remove(idOrDoc: Document): Promise<DocumentDestroyResponse>
|
|
|
|
remove(idOrDoc: string, rev?: string): Promise<DocumentDestroyResponse>
|
2024-01-02 12:36:32 +01:00
|
|
|
remove(
|
2024-05-16 14:27:54 +02:00
|
|
|
idOrDoc: string | Document,
|
|
|
|
rev?: string
|
2024-01-02 12:36:32 +01:00
|
|
|
): Promise<DocumentDestroyResponse> {
|
2024-11-21 17:36:31 +01:00
|
|
|
return tracer.trace("db.remove", async span => {
|
|
|
|
span.addTags({ db_name: this.name, doc_id: idOrDoc, rev })
|
2024-05-16 14:52:38 +02:00
|
|
|
const isDocument = typeof idOrDoc === "object"
|
|
|
|
const id = isDocument ? idOrDoc._id! : idOrDoc
|
|
|
|
rev = isDocument ? idOrDoc._rev : rev
|
2024-11-21 17:36:31 +01:00
|
|
|
const resp = await this.db.remove(id, rev)
|
|
|
|
span.addTags({ ok: resp.ok })
|
|
|
|
return resp
|
2024-01-02 12:36:32 +01:00
|
|
|
})
|
|
|
|
}
|
|
|
|
|
2024-07-24 19:13:27 +02:00
|
|
|
bulkRemove(
|
|
|
|
documents: Document[],
|
|
|
|
opts?: { silenceErrors?: boolean }
|
|
|
|
): Promise<void> {
|
|
|
|
return tracer.trace("db.bulkRemove", span => {
|
2024-11-21 17:36:31 +01:00
|
|
|
span.addTags({
|
|
|
|
db_name: this.name,
|
|
|
|
num_docs: documents.length,
|
|
|
|
silence_errors: opts?.silenceErrors,
|
|
|
|
})
|
2024-07-24 19:13:27 +02:00
|
|
|
return this.db.bulkRemove(documents, opts)
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
2024-01-02 12:36:32 +01:00
|
|
|
put(
|
|
|
|
document: AnyDocument,
|
|
|
|
opts?: DatabasePutOpts | undefined
|
|
|
|
): Promise<DocumentInsertResponse> {
|
2024-11-21 17:36:31 +01:00
|
|
|
return tracer.trace("db.put", async span => {
|
|
|
|
span.addTags({
|
|
|
|
db_name: this.name,
|
|
|
|
doc_id: document._id,
|
|
|
|
force: opts?.force,
|
|
|
|
})
|
|
|
|
const resp = await this.db.put(document, opts)
|
|
|
|
span.addTags({ ok: resp.ok })
|
|
|
|
return resp
|
2024-01-02 12:36:32 +01:00
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
bulkDocs(documents: AnyDocument[]): Promise<DocumentBulkResponse[]> {
|
|
|
|
return tracer.trace("db.bulkDocs", span => {
|
2024-11-21 17:36:31 +01:00
|
|
|
span.addTags({ db_name: this.name, num_docs: documents.length })
|
2024-01-02 12:36:32 +01:00
|
|
|
return this.db.bulkDocs(documents)
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
2024-02-29 17:28:00 +01:00
|
|
|
allDocs<T extends Document | RowValue>(
|
2024-01-02 12:36:32 +01:00
|
|
|
params: DatabaseQueryOpts
|
|
|
|
): Promise<AllDocsResponse<T>> {
|
2024-11-21 17:36:31 +01:00
|
|
|
return tracer.trace("db.allDocs", async span => {
|
|
|
|
span.addTags({ db_name: this.name, ...params })
|
|
|
|
const resp = await this.db.allDocs<T>(params)
|
|
|
|
span.addTags({
|
|
|
|
total_rows: resp.total_rows,
|
|
|
|
rows_length: resp.rows.length,
|
|
|
|
offset: resp.offset,
|
|
|
|
})
|
|
|
|
return resp
|
2024-01-02 12:36:32 +01:00
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
query<T extends Document>(
|
|
|
|
viewName: string,
|
|
|
|
params: DatabaseQueryOpts
|
|
|
|
): Promise<AllDocsResponse<T>> {
|
2024-11-21 17:36:31 +01:00
|
|
|
return tracer.trace("db.query", async span => {
|
|
|
|
span.addTags({ db_name: this.name, view_name: viewName, ...params })
|
|
|
|
const resp = await this.db.query<T>(viewName, params)
|
|
|
|
span.addTags({
|
|
|
|
total_rows: resp.total_rows,
|
|
|
|
rows_length: resp.rows.length,
|
|
|
|
offset: resp.offset,
|
|
|
|
})
|
|
|
|
return resp
|
2024-01-02 12:36:32 +01:00
|
|
|
})
|
|
|
|
}
|
|
|
|
|
2024-11-21 17:36:31 +01:00
|
|
|
destroy(): Promise<OkResponse> {
|
|
|
|
return tracer.trace("db.destroy", async span => {
|
|
|
|
span.addTags({ db_name: this.name })
|
|
|
|
const resp = await this.db.destroy()
|
|
|
|
span.addTags({ ok: resp.ok })
|
|
|
|
return resp
|
2024-01-02 12:36:32 +01:00
|
|
|
})
|
|
|
|
}
|
|
|
|
|
2024-11-21 17:36:31 +01:00
|
|
|
compact(): Promise<OkResponse> {
|
|
|
|
return tracer.trace("db.compact", async span => {
|
|
|
|
span.addTags({ db_name: this.name })
|
|
|
|
const resp = await this.db.compact()
|
|
|
|
span.addTags({ ok: resp.ok })
|
|
|
|
return resp
|
2024-01-02 12:36:32 +01:00
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
dump(stream: Writable, opts?: DatabaseDumpOpts | undefined): Promise<any> {
|
|
|
|
return tracer.trace("db.dump", span => {
|
2024-11-21 17:36:31 +01:00
|
|
|
span.addTags({
|
|
|
|
db_name: this.name,
|
|
|
|
batch_limit: opts?.batch_limit,
|
|
|
|
batch_size: opts?.batch_size,
|
|
|
|
style: opts?.style,
|
|
|
|
timeout: opts?.timeout,
|
|
|
|
num_doc_ids: opts?.doc_ids?.length,
|
|
|
|
view: opts?.view,
|
|
|
|
})
|
2024-01-02 12:36:32 +01:00
|
|
|
return this.db.dump(stream, opts)
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
load(...args: any[]): Promise<any> {
|
|
|
|
return tracer.trace("db.load", span => {
|
2024-11-21 17:36:31 +01:00
|
|
|
span.addTags({ db_name: this.name, num_args: args.length })
|
2024-01-02 12:36:32 +01:00
|
|
|
return this.db.load(...args)
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
createIndex(...args: any[]): Promise<any> {
|
|
|
|
return tracer.trace("db.createIndex", span => {
|
2024-11-21 17:36:31 +01:00
|
|
|
span.addTags({ db_name: this.name, num_args: args.length })
|
2024-01-02 12:36:32 +01:00
|
|
|
return this.db.createIndex(...args)
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
deleteIndex(...args: any[]): Promise<any> {
|
|
|
|
return tracer.trace("db.deleteIndex", span => {
|
2024-11-21 17:36:31 +01:00
|
|
|
span.addTags({ db_name: this.name, num_args: args.length })
|
2024-01-02 12:36:32 +01:00
|
|
|
return this.db.deleteIndex(...args)
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
getIndexes(...args: any[]): Promise<any> {
|
|
|
|
return tracer.trace("db.getIndexes", span => {
|
2024-11-21 17:36:31 +01:00
|
|
|
span.addTags({ db_name: this.name, num_args: args.length })
|
2024-01-02 12:36:32 +01:00
|
|
|
return this.db.getIndexes(...args)
|
|
|
|
})
|
|
|
|
}
|
2024-02-28 18:16:36 +01:00
|
|
|
|
2024-04-10 18:36:57 +02:00
|
|
|
sql<T extends Document>(
|
|
|
|
sql: string,
|
|
|
|
parameters?: SqlQueryBinding
|
|
|
|
): Promise<T[]> {
|
2024-11-21 17:36:31 +01:00
|
|
|
return tracer.trace("db.sql", async span => {
|
|
|
|
span.addTags({ db_name: this.name, num_bindings: parameters?.length })
|
|
|
|
const resp = await this.db.sql<T>(sql, parameters)
|
|
|
|
span.addTags({ num_rows: resp.length })
|
|
|
|
return resp
|
2024-02-28 18:16:36 +01:00
|
|
|
})
|
|
|
|
}
|
2024-04-19 19:03:38 +02:00
|
|
|
|
2024-05-16 14:27:54 +02:00
|
|
|
sqlPurgeDocument(docIds: string[] | string): Promise<void> {
|
|
|
|
return tracer.trace("db.sqlPurgeDocument", span => {
|
2024-11-21 17:36:31 +01:00
|
|
|
span.addTags({
|
|
|
|
db_name: this.name,
|
|
|
|
num_docs: Array.isArray(docIds) ? docIds.length : 1,
|
|
|
|
})
|
2024-05-16 14:27:54 +02:00
|
|
|
return this.db.sqlPurgeDocument(docIds)
|
2024-04-19 19:03:38 +02:00
|
|
|
})
|
|
|
|
}
|
|
|
|
|
2024-05-16 14:27:54 +02:00
|
|
|
sqlDiskCleanup(): Promise<void> {
|
|
|
|
return tracer.trace("db.sqlDiskCleanup", span => {
|
2024-11-21 17:36:31 +01:00
|
|
|
span.addTags({ db_name: this.name })
|
2024-05-16 14:27:54 +02:00
|
|
|
return this.db.sqlDiskCleanup()
|
2024-04-19 19:03:38 +02:00
|
|
|
})
|
|
|
|
}
|
2024-01-02 12:36:32 +01:00
|
|
|
}
|