-
-
Notifications
You must be signed in to change notification settings - Fork 138
/
Copy pathPostgresMetaSchemas.ts
122 lines (115 loc) · 3.6 KB
/
PostgresMetaSchemas.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
121
122
import { ident, literal } from 'pg-format'
import { DEFAULT_SYSTEM_SCHEMAS } from './constants.js'
import schemasSql from './sql/schemas.sql'
import {
PostgresMetaResult,
PostgresSchema,
PostgresSchemaCreate,
PostgresSchemaUpdate,
} from './types.js'
export default class PostgresMetaSchemas {
query: (sql: string) => Promise<PostgresMetaResult<any>>
constructor(query: (sql: string) => Promise<PostgresMetaResult<any>>) {
this.query = query
}
async list({
includeSystemSchemas = false,
limit,
offset,
}: {
includeSystemSchemas?: boolean
limit?: number
offset?: number
} = {}): Promise<PostgresMetaResult<PostgresSchema[]>> {
let sql = schemasSql
if (!includeSystemSchemas) {
sql = `${sql} AND NOT (n.nspname IN (${DEFAULT_SYSTEM_SCHEMAS.map(literal).join(',')}))`
}
if (limit) {
sql = `${sql} LIMIT ${limit}`
}
if (offset) {
sql = `${sql} OFFSET ${offset}`
}
return await this.query(sql)
}
async retrieve({ id }: { id: number }): Promise<PostgresMetaResult<PostgresSchema>>
async retrieve({ name }: { name: string }): Promise<PostgresMetaResult<PostgresSchema>>
async retrieve({
id,
name,
}: {
id?: number
name?: string
}): Promise<PostgresMetaResult<PostgresSchema>> {
if (id) {
const sql = `${schemasSql} AND n.oid = ${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 schema with ID ${id}` } }
} else {
return { data: data[0], error }
}
} else if (name) {
const sql = `${schemasSql} AND n.nspname = ${literal(name)};`
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 schema named ${name}` } }
} else {
return { data: data[0], error }
}
} else {
return { data: null, error: { message: 'Invalid parameters on schema retrieve' } }
}
}
async create({
name,
owner = 'postgres',
}: PostgresSchemaCreate): Promise<PostgresMetaResult<PostgresSchema>> {
const sql = `CREATE SCHEMA ${ident(name)} AUTHORIZATION ${ident(owner)};`
const { error } = await this.query(sql)
if (error) {
return { data: null, error }
}
return await this.retrieve({ name })
}
async update(
id: number,
{ name, owner }: PostgresSchemaUpdate
): Promise<PostgresMetaResult<PostgresSchema>> {
const { data: old, error } = await this.retrieve({ id })
if (error) {
return { data: null, error }
}
const nameSql =
name === undefined ? '' : `ALTER SCHEMA ${ident(old!.name)} RENAME TO ${ident(name)};`
const ownerSql =
owner === undefined ? '' : `ALTER SCHEMA ${ident(old!.name)} OWNER TO ${ident(owner)};`
const sql = `BEGIN; ${ownerSql} ${nameSql} COMMIT;`
{
const { error } = await this.query(sql)
if (error) {
return { data: null, error }
}
}
return await this.retrieve({ id })
}
async remove(id: number, { cascade = false } = {}): Promise<PostgresMetaResult<PostgresSchema>> {
const { data: schema, error } = await this.retrieve({ id })
if (error) {
return { data: null, error }
}
const sql = `DROP SCHEMA ${ident(schema!.name)} ${cascade ? 'CASCADE' : 'RESTRICT'};`
{
const { error } = await this.query(sql)
if (error) {
return { data: null, error }
}
}
return { data: schema!, error: null }
}
}