-
-
Notifications
You must be signed in to change notification settings - Fork 138
/
Copy pathPostgresMetaMaterializedViews.ts
117 lines (112 loc) · 3.55 KB
/
PostgresMetaMaterializedViews.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
import { literal } from 'pg-format'
import { coalesceRowsToArray, filterByList } from './helpers.js'
import columnsSql from './sql/columns.sql'
import materializedViewsSql from './sql/materialized_views.sql'
import { PostgresMetaResult, PostgresMaterializedView } from './types.js'
export default class PostgresMetaMaterializedViews {
query: (sql: string) => Promise<PostgresMetaResult<any>>
constructor(query: (sql: string) => Promise<PostgresMetaResult<any>>) {
this.query = query
}
async list(options: {
includedSchemas?: string[]
excludedSchemas?: string[]
limit?: number
offset?: number
includeColumns: true
}): Promise<PostgresMetaResult<(PostgresMaterializedView & { columns: unknown[] })[]>>
async list(options?: {
includedSchemas?: string[]
excludedSchemas?: string[]
limit?: number
offset?: number
includeColumns?: boolean
}): Promise<PostgresMetaResult<(PostgresMaterializedView & { columns: never })[]>>
async list({
includedSchemas,
excludedSchemas,
limit,
offset,
includeColumns = false,
}: {
includedSchemas?: string[]
excludedSchemas?: string[]
limit?: number
offset?: number
includeColumns?: boolean
} = {}): Promise<PostgresMetaResult<PostgresMaterializedView[]>> {
let sql = generateEnrichedMaterializedViewsSql({ includeColumns })
const filter = filterByList(includedSchemas, excludedSchemas, 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<PostgresMaterializedView>>
async retrieve({
name,
schema,
}: {
name: string
schema: string
}): Promise<PostgresMetaResult<PostgresMaterializedView>>
async retrieve({
id,
name,
schema = 'public',
}: {
id?: number
name?: string
schema?: string
}): Promise<PostgresMetaResult<PostgresMaterializedView>> {
if (id) {
const sql = `${generateEnrichedMaterializedViewsSql({
includeColumns: true,
})} where materialized_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 materialized view with ID ${id}` } }
} else {
return { data: data[0], error }
}
} else if (name) {
const sql = `${generateEnrichedMaterializedViewsSql({
includeColumns: true,
})} where materialized_views.name = ${literal(
name
)} and materialized_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 materialized view named ${name} in schema ${schema}` },
}
} else {
return { data: data[0], error }
}
} else {
return { data: null, error: { message: 'Invalid parameters on materialized view retrieve' } }
}
}
}
const generateEnrichedMaterializedViewsSql = ({ includeColumns }: { includeColumns: boolean }) => `
with materialized_views as (${materializedViewsSql})
${includeColumns ? `, columns as (${columnsSql})` : ''}
select
*
${
includeColumns
? `, ${coalesceRowsToArray('columns', 'columns.table_id = materialized_views.id')}`
: ''
}
from materialized_views`