-
Notifications
You must be signed in to change notification settings - Fork 4
/
Copy pathcustom_integration_headers.go
180 lines (148 loc) · 4.74 KB
/
custom_integration_headers.go
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
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
package database
import (
"context"
"github.com/jackc/pgtype"
"github.com/jackc/pgx/v4/pgxpool"
)
type CustomIntegrationHeadersTable struct {
*pgxpool.Pool
}
type CustomIntegrationHeader struct {
Id int `json:"id"`
IntegrationId int `json:"integration_id"`
Name string `json:"name"`
Value string `json:"value"`
}
func newCustomIntegrationHeadersTable(db *pgxpool.Pool) *CustomIntegrationHeadersTable {
return &CustomIntegrationHeadersTable{
db,
}
}
func (i CustomIntegrationHeadersTable) Schema() string {
return `
CREATE TABLE IF NOT EXISTS custom_integration_headers(
"id" SERIAL NOT NULL UNIQUE,
"integration_id" int NOT NULL,
"name" VARCHAR(32) NOT NULL,
"value" VARCHAR(255) NOT NULL,
UNIQUE("integration_id", "name"),
FOREIGN KEY("integration_id") REFERENCES custom_integrations("id") ON DELETE CASCADE,
PRIMARY KEY("id")
);
CREATE INDEX IF NOT EXISTS custom_integration_headers_integration_id ON custom_integration_headers("integration_id");
`
}
func (i *CustomIntegrationHeadersTable) GetByIntegration(ctx context.Context, integrationId int) ([]CustomIntegrationHeader, error) {
query := `SELECT "id", "integration_id", "name", "value" FROM custom_integration_headers WHERE "integration_id" = $1;`
rows, err := i.Query(ctx, query, integrationId)
if err != nil {
return nil, err
}
var headers []CustomIntegrationHeader
for rows.Next() {
var header CustomIntegrationHeader
if err := rows.Scan(&header.Id, &header.IntegrationId, &header.Name, &header.Value); err != nil {
return nil, err
}
headers = append(headers, header)
}
return headers, nil
}
// GetAll integration_id -> []CustomIntegrationHeader
func (i *CustomIntegrationHeadersTable) GetAll(ctx context.Context, integrationIds []int) (map[int][]CustomIntegrationHeader, error) {
query := `SELECT "id", "integration_id", "name", "value" FROM custom_integration_headers WHERE "integration_id" = ANY($1);`
idArray := &pgtype.Int4Array{}
if err := idArray.Set(integrationIds); err != nil {
return nil, err
}
rows, err := i.Query(ctx, query, idArray)
if err != nil {
return nil, err
}
headers := make(map[int][]CustomIntegrationHeader)
for rows.Next() {
var header CustomIntegrationHeader
if err := rows.Scan(&header.Id, &header.IntegrationId, &header.Name, &header.Value); err != nil {
return nil, err
}
if _, ok := headers[header.IntegrationId]; !ok {
headers[header.IntegrationId] = []CustomIntegrationHeader{}
}
headers[header.IntegrationId] = append(headers[header.IntegrationId], header)
}
return headers, nil
}
// Assumes that all header IDs are valid for the integration
func (i *CustomIntegrationHeadersTable) CreateOrUpdate(ctx context.Context, integrationId int, headers []CustomIntegrationHeader) ([]CustomIntegrationHeader, error) {
// The array check is weird if headers is empty
if len(headers) == 0 {
query := `DELETE FROM custom_integration_headers WHERE "integration_id" = $1;`
_, err := i.Exec(ctx, query, integrationId)
return nil, err
}
tx, err := i.Begin(ctx)
if err != nil {
return nil, err
}
defer tx.Rollback(ctx) // Does not matter if commit succeeds
query := `DELETE FROM custom_integration_headers WHERE "integration_id" = $1 AND NOT ("id" = ANY($2));`
var ids []int
for _, header := range headers {
if header.Id != 0 {
ids = append(ids, header.Id)
}
}
array := &pgtype.Int4Array{}
if err := array.Set(ids); err != nil {
return nil, err
}
if _, err := tx.Exec(ctx, query, integrationId, array); err != nil {
return nil, err
}
// Create or update new secrets
var newHeaders []CustomIntegrationHeader
for _, header := range headers {
var res CustomIntegrationHeader
if header.Id == 0 { // Create
query := `
INSERT INTO custom_integration_headers( "integration_id", "name", "value")
VALUES ($1, $2, $3)
RETURNING "id", "integration_id", "name", "value";
;`
err = tx.QueryRow(ctx, query, integrationId, header.Name, header.Value).Scan(
&res.Id,
&res.IntegrationId,
&res.Name,
&res.Value,
)
} else { // Update
query := `
UPDATE custom_integration_headers
SET "name" = $3, "value" = $4
WHERE "id" = $1 AND "integration_id" = $2;`
_, err = tx.Exec(ctx, query, header.Id, integrationId, header.Name, header.Value)
res = CustomIntegrationHeader{
Id: header.Id,
IntegrationId: integrationId,
Name: header.Name,
Value: header.Value,
}
}
if err != nil {
return nil, err
}
newHeaders = append(newHeaders, res)
}
if err := tx.Commit(ctx); err != nil {
return nil, err
}
return newHeaders, nil
}
func (i *CustomIntegrationHeadersTable) Delete(ctx context.Context, id int) (err error) {
query := `
DELETE FROM custom_integration_headers
WHERE "id" = $1;
`
_, err = i.Exec(ctx, query, id)
return
}