-
-
Notifications
You must be signed in to change notification settings - Fork 138
/
Copy pathPostgresMetaViews.ts
120 lines (115 loc) · 3.48 KB
/
PostgresMetaViews.ts
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
import { literal } from 'pg-format'
import { DEFAULT_SYSTEM_SCHEMAS } from './constants.js'
import { coalesceRowsToArray, filterByList } from './helpers.js'
import columnsSql from './sql/columns.sql'
import viewsSql from './sql/views.sql'
import { PostgresMetaResult, PostgresView } from './types.js'
export default class PostgresMetaViews {
query: (sql: string) => Promise<PostgresMetaResult<any>>
constructor(query: (sql: string) => Promise<PostgresMetaResult<any>>) {
this.query = query
}
async list(options: {
includeSystemSchemas?: boolean
includedSchemas?: string[]
excludedSchemas?: string[]
limit?: number
offset?: number
includeColumns: false
}): Promise<PostgresMetaResult<(PostgresView & { columns: never })[]>>
async list(options?: {
includeSystemSchemas?: boolean
includedSchemas?: string[]
excludedSchemas?: string[]
limit?: number
offset?: number
includeColumns?: boolean
}): Promise<PostgresMetaResult<(PostgresView & { columns: unknown[] })[]>>
async list({
includeSystemSchemas = false,
includedSchemas,
excludedSchemas,
limit,
offset,
includeColumns = true,
}: {
includeSystemSchemas?: boolean
includedSchemas?: string[]
excludedSchemas?: string[]
limit?: number
offset?: number
includeColumns?: boolean
} = {}): Promise<PostgresMetaResult<PostgresView[]>> {
let sql = generateEnrichedViewsSql({ includeColumns })
const filter = filterByList(
includedSchemas,
excludedSchemas,
!includeSystemSchemas ? DEFAULT_SYSTEM_SCHEMAS : undefined
)
if (filter) {
sql += ` where schema ${filter}`
}
if (limit) {
sql += ` limit ${limit}`
}
if (offset) {
sql += ` offset ${offset}`
}
return await this.query(sql)
}
async retrieve({ id }: { id: number }): Promise<PostgresMetaResult<PostgresView>>
async retrieve({
name,
schema,
}: {
name: string
schema: string
}): Promise<PostgresMetaResult<PostgresView>>
async retrieve({
id,
name,
schema = 'public',
}: {
id?: number
name?: string
schema?: string
}): Promise<PostgresMetaResult<PostgresView>> {
if (id) {
const sql = `${generateEnrichedViewsSql({
includeColumns: true,
})} where views.id = ${literal(id)};`
const { data, error } = await this.query(sql)
if (error) {
return { data, error }
} else if (data.length === 0) {
return { data: null, error: { message: `Cannot find a view with ID ${id}` } }
} else {
return { data: data[0], error }
}
} else if (name) {
const sql = `${generateEnrichedViewsSql({
includeColumns: true,
})} where views.name = ${literal(name)} and views.schema = ${literal(schema)};`
const { data, error } = await this.query(sql)
if (error) {
return { data, error }
} else if (data.length === 0) {
return {
data: null,
error: { message: `Cannot find a view named ${name} in schema ${schema}` },
}
} else {
return { data: data[0], error }
}
} else {
return { data: null, error: { message: 'Invalid parameters on view retrieve' } }
}
}
}
const generateEnrichedViewsSql = ({ includeColumns }: { includeColumns: boolean }) => `
with views as (${viewsSql})
${includeColumns ? `, columns as (${columnsSql})` : ''}
select
*
${includeColumns ? `, ${coalesceRowsToArray('columns', 'columns.table_id = views.id')}` : ''}
from views`