@@ -4,6 +4,28 @@ import { DEFAULT_SYSTEM_SCHEMAS } from './constants'
4
4
import { columnsSql } from './sql'
5
5
import { PostgresMetaResult , PostgresColumn } from './types'
6
6
7
+ interface ColumnCreationRequest {
8
+ table_id : number
9
+ name : string
10
+ type : string
11
+ default_value ?: any
12
+ default_value_format ?: 'expression' | 'literal'
13
+ is_identity ?: boolean
14
+ identity_generation ?: 'BY DEFAULT' | 'ALWAYS'
15
+ is_nullable ?: boolean
16
+ is_primary_key ?: boolean
17
+ is_unique ?: boolean
18
+ comment ?: string
19
+ check ?: string
20
+ }
21
+
22
+ interface ColumnBatchInfoRequest {
23
+ ids ?: string [ ]
24
+ names ?: string [ ]
25
+ table ?: string
26
+ schema ?: string
27
+ }
28
+
7
29
export default class PostgresMetaColumns {
8
30
query : ( sql : string ) => Promise < PostgresMetaResult < any > >
9
31
metaTables : PostgresMetaTables
@@ -57,75 +79,130 @@ export default class PostgresMetaColumns {
57
79
schema ?: string
58
80
} ) : Promise < PostgresMetaResult < PostgresColumn > > {
59
81
if ( id ) {
60
- const regexp = / ^ ( \d + ) \. ( \d + ) $ /
61
- if ( ! regexp . test ( id ) ) {
62
- return { data : null , error : { message : 'Invalid format for column ID' } }
82
+ const { data, error } = await this . batchRetrieve ( { ids : [ id ] } )
83
+ if ( data ) {
84
+ return { data : data [ 0 ] , error : null }
85
+ } else if ( error ) {
86
+ return { data : null , error : error }
87
+ }
88
+ }
89
+ if ( name && table ) {
90
+ const { data, error } = await this . batchRetrieve ( { names : [ name ] , table, schema } )
91
+ if ( data ) {
92
+ return { data : data [ 0 ] , error : null }
93
+ } else if ( error ) {
94
+ return { data : null , error : error }
63
95
}
64
- const matches = id . match ( regexp ) as RegExpMatchArray
65
- const [ tableId , ordinalPos ] = matches . slice ( 1 ) . map ( Number )
66
- const sql = `${ columnsSql } AND c.oid = ${ tableId } AND a.attnum = ${ ordinalPos } ;`
96
+ }
97
+ return { data : null , error : { message : 'Invalid parameters on column retrieve' } }
98
+ }
99
+
100
+ async batchRetrieve ( {
101
+ ids,
102
+ names,
103
+ table,
104
+ schema = 'public' ,
105
+ } : ColumnBatchInfoRequest ) : Promise < PostgresMetaResult < PostgresColumn [ ] > > {
106
+ if ( ids && ids . length > 0 ) {
107
+ const regexp = / ^ ( \d + ) \. ( \d + ) $ /
108
+ const filteringClauses = ids
109
+ . map ( ( id ) => {
110
+ if ( ! regexp . test ( id ) ) {
111
+ return { data : null , error : { message : 'Invalid format for column ID' } }
112
+ }
113
+ const matches = id . match ( regexp ) as RegExpMatchArray
114
+ const [ tableId , ordinalPos ] = matches . slice ( 1 ) . map ( Number )
115
+ return `(c.oid = ${ tableId } AND a.attnum = ${ ordinalPos } )`
116
+ } )
117
+ . join ( ' OR ' )
118
+ const sql = `${ columnsSql } AND (${ filteringClauses } );`
67
119
const { data, error } = await this . query ( sql )
68
120
if ( error ) {
69
121
return { data, error }
70
- } else if ( data . length === 0 ) {
71
- return { data : null , error : { message : `Cannot find a column with ID ${ id } ` } }
122
+ } else if ( data . length < ids . length ) {
123
+ return { data : null , error : { message : `Cannot find some of the requested columns. ` } }
72
124
} else {
73
- return { data : data [ 0 ] , error }
125
+ return { data, error }
74
126
}
75
- } else if ( name && table ) {
76
- const sql = `${ columnsSql } AND a.attname = ${ literal ( name ) } AND c.relname = ${ literal (
127
+ } else if ( names && names . length > 0 && table ) {
128
+ const filteringClauses = names . map ( ( name ) => `a.attname = ${ literal ( name ) } ` ) . join ( ' OR ' )
129
+ const sql = `${ columnsSql } AND (${ filteringClauses } ) AND c.relname = ${ literal (
77
130
table
78
131
) } AND nc.nspname = ${ literal ( schema ) } ;`
79
132
const { data, error } = await this . query ( sql )
80
133
if ( error ) {
81
134
return { data, error }
82
- } else if ( data . length === 0 ) {
135
+ } else if ( data . length < names . length ) {
83
136
return {
84
137
data : null ,
85
- error : { message : `Cannot find a column named ${ name } in table ${ schema } . ${ table } ` } ,
138
+ error : { message : `Cannot find some of the requested columns. ` } ,
86
139
}
87
140
} else {
88
- return { data : data [ 0 ] , error }
141
+ return { data, error }
89
142
}
90
143
} else {
91
144
return { data : null , error : { message : 'Invalid parameters on column retrieve' } }
92
145
}
93
146
}
94
147
95
- async create ( {
96
- table_id,
97
- name,
98
- type,
99
- default_value,
100
- default_value_format = 'literal' ,
101
- is_identity = false ,
102
- identity_generation = 'BY DEFAULT' ,
103
- // Can't pick a value as default since regular columns are nullable by default but PK columns aren't
104
- is_nullable,
105
- is_primary_key = false ,
106
- is_unique = false ,
107
- comment,
108
- check,
109
- } : {
110
- table_id : number
111
- name : string
112
- type : string
113
- default_value ?: any
114
- default_value_format ?: 'expression' | 'literal'
115
- is_identity ?: boolean
116
- identity_generation ?: 'BY DEFAULT' | 'ALWAYS'
117
- is_nullable ?: boolean
118
- is_primary_key ?: boolean
119
- is_unique ?: boolean
120
- comment ?: string
121
- check ?: string
122
- } ) : Promise < PostgresMetaResult < PostgresColumn > > {
148
+ async create ( col : ColumnCreationRequest ) : Promise < PostgresMetaResult < PostgresColumn > > {
149
+ const { data, error } = await this . batchCreate ( [ col ] )
150
+ if ( data ) {
151
+ return { data : data [ 0 ] , error : null }
152
+ } else if ( error ) {
153
+ return { data : null , error : error }
154
+ }
155
+ return { data : null , error : { message : 'Invalid params' } }
156
+ }
157
+
158
+ async batchCreate ( cols : ColumnCreationRequest [ ] ) : Promise < PostgresMetaResult < PostgresColumn [ ] > > {
159
+ if ( cols . length < 1 ) {
160
+ throw new Error ( 'no columns provided for creation' )
161
+ }
162
+ if ( [ ...new Set ( cols . map ( ( col ) => col . table_id ) ) ] . length > 1 ) {
163
+ throw new Error ( 'all columns in a single request must share the same table' )
164
+ }
165
+ const { table_id } = cols [ 0 ]
123
166
const { data, error } = await this . metaTables . retrieve ( { id : table_id } )
124
167
if ( error ) {
125
168
return { data : null , error }
126
169
}
127
170
const { name : table , schema } = data !
128
171
172
+ const sqlStrings = cols . map ( ( col ) => this . generateColumnCreationSql ( col , schema , table ) )
173
+
174
+ const sql = `BEGIN;
175
+ ${ sqlStrings . join ( '\n' ) }
176
+ COMMIT;
177
+ `
178
+ {
179
+ const { error } = await this . query ( sql )
180
+ if ( error ) {
181
+ return { data : null , error }
182
+ }
183
+ }
184
+ const names = cols . map ( ( col ) => col . name )
185
+ return await this . batchRetrieve ( { names, table, schema } )
186
+ }
187
+
188
+ generateColumnCreationSql (
189
+ {
190
+ name,
191
+ type,
192
+ default_value,
193
+ default_value_format = 'literal' ,
194
+ is_identity = false ,
195
+ identity_generation = 'BY DEFAULT' ,
196
+ // Can't pick a value as default since regular columns are nullable by default but PK columns aren't
197
+ is_nullable,
198
+ is_primary_key = false ,
199
+ is_unique = false ,
200
+ comment,
201
+ check,
202
+ } : ColumnCreationRequest ,
203
+ schema : string ,
204
+ table : string
205
+ ) {
129
206
let defaultValueClause = ''
130
207
if ( is_identity ) {
131
208
if ( default_value !== undefined ) {
@@ -159,22 +236,14 @@ export default class PostgresMetaColumns {
159
236
: `COMMENT ON COLUMN ${ ident ( schema ) } .${ ident ( table ) } .${ ident ( name ) } IS ${ literal ( comment ) } `
160
237
161
238
const sql = `
162
- BEGIN;
163
239
ALTER TABLE ${ ident ( schema ) } .${ ident ( table ) } ADD COLUMN ${ ident ( name ) } ${ typeIdent ( type ) }
164
240
${ defaultValueClause }
165
241
${ isNullableClause }
166
242
${ isPrimaryKeyClause }
167
243
${ isUniqueClause }
168
244
${ checkSql } ;
169
- ${ commentSql } ;
170
- COMMIT;`
171
- {
172
- const { error } = await this . query ( sql )
173
- if ( error ) {
174
- return { data : null , error }
175
- }
176
- }
177
- return await this . retrieve ( { name, table, schema } )
245
+ ${ commentSql } ;`
246
+ return sql
178
247
}
179
248
180
249
async update (
0 commit comments