forked from joncrlsn/pgdiff
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathfunction.go
200 lines (170 loc) · 6.01 KB
/
function.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
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
//
// Copyright (c) 2017 Jon Carlson. All rights reserved.
// Use of this source code is governed by an MIT-style
// license that can be found in the LICENSE file.
//
package main
import (
"bytes"
"database/sql"
"fmt"
"github.com/joncrlsn/misc"
"github.com/joncrlsn/pgutil"
"sort"
"strings"
"text/template"
)
var (
functionSqlTemplate = initFunctionSqlTemplate()
)
// Initializes the Sql template
func initFunctionSqlTemplate() *template.Template {
sql := `
SELECT n.nspname AS schema_name
, {{if eq $.DbSchema "*" }}n.nspname || '.' || {{end}}p.proname AS compare_name
, p.proname AS function_name
, p.oid::regprocedure AS fancy
, t.typname AS return_type
, pg_get_functiondef(p.oid) AS definition
FROM pg_proc AS p
JOIN pg_type t ON (p.prorettype = t.oid)
JOIN pg_namespace n ON (n.oid = p.pronamespace)
JOIN pg_language l ON (p.prolang = l.oid AND l.lanname IN ('c','plpgsql', 'sql'))
WHERE true
{{if eq $.DbSchema "*" }}
AND n.nspname NOT LIKE 'pg_%'
AND n.nspname <> 'information_schema'
{{else}}
AND n.nspname = '{{$.DbSchema}}'
{{end}};
`
t := template.New("FunctionSqlTmpl")
template.Must(t.Parse(sql))
return t
}
// ==================================
// FunctionRows definition
// ==================================
// FunctionRows is a sortable slice of string maps
type FunctionRows []map[string]string
func (slice FunctionRows) Len() int {
return len(slice)
}
func (slice FunctionRows) Less(i, j int) bool {
return slice[i]["compare_name"] < slice[j]["compare_name"]
}
func (slice FunctionRows) Swap(i, j int) {
slice[i], slice[j] = slice[j], slice[i]
}
// FunctionSchema holds a channel streaming function information from one of the databases as well as
// a reference to the current row of data we're viewing.
//
// FunctionSchema implements the Schema interface defined in pgdiff.go
type FunctionSchema struct {
rows FunctionRows
rowNum int
done bool
}
// get returns the value from the current row for the given key
func (c *FunctionSchema) get(key string) string {
if c.rowNum >= len(c.rows) {
return ""
}
return c.rows[c.rowNum][key]
}
// NextRow increments the rowNum and tells you whether or not there are more
func (c *FunctionSchema) NextRow() bool {
if c.rowNum >= len(c.rows)-1 {
c.done = true
}
c.rowNum = c.rowNum + 1
return !c.done
}
// Compare tells you, in one pass, whether or not the first row matches, is less than, or greater than the second row
func (c *FunctionSchema) Compare(obj interface{}) int {
c2, ok := obj.(*FunctionSchema)
if !ok {
fmt.Println("Error!!!, Compare(obj) needs a FunctionSchema instance", c2)
return +999
}
val := misc.CompareStrings(c.get("compare_name"), c2.get("compare_name"))
//fmt.Printf("-- Compared %v: %s with %s \n", val, c.get("function_name"), c2.get("function_name"))
return val
}
// Add returns SQL to create the function
func (c FunctionSchema) Add() {
fmt.Println("-- Add")
// If we are comparing two different schemas against each other, we need to do some
// modification of the first function definition so we create it in the right schema
functionDef := c.get("definition")
if dbInfo1.DbSchema != dbInfo2.DbSchema {
functionDef = strings.Replace(
functionDef,
fmt.Sprintf("FUNCTION %s.%s(", c.get("schema_name"), c.get("function_name")),
fmt.Sprintf("FUNCTION %s.%s(", dbInfo2.DbSchema, c.get("function_name")),
-1)
}
fmt.Println("-- STATEMENT-BEGIN")
fmt.Println(functionDef, ";")
fmt.Println("-- STATEMENT-END")
}
// Drop returns SQL to drop the function
func (c FunctionSchema) Drop() {
fmt.Println("-- Drop")
fmt.Println("-- Note that CASCADE in the statement below will also drop any triggers depending on this function.")
fmt.Println("-- Also, if there are two functions with this name, you will want to add arguments to identify the correct one to drop.")
fmt.Println("-- (See http://www.postgresql.org/docs/9.4/interactive/sql-dropfunction.html) ")
fmt.Printf("DROP FUNCTION %s.%s CASCADE;\n", c.get("schema_name"), c.get("function_name"))
}
// Change handles the case where the function names match, but the definition does not
func (c FunctionSchema) Change(obj interface{}) {
fmt.Println("-- Change")
c2, ok := obj.(*FunctionSchema)
if !ok {
fmt.Println("Error!!!, Change needs a FunctionSchema instance", c2)
}
if c.get("definition") != c2.get("definition") {
fmt.Println("-- This function is different so we'll recreate it:")
// If we are comparing two different schemas against each other, we need to do some
// modification of the first function definition so we create it in the right schema
functionDef := c.get("definition")
if dbInfo1.DbSchema != dbInfo2.DbSchema {
functionDef = strings.Replace(
functionDef,
fmt.Sprintf("FUNCTION %s.%s(", c.get("schema_name"), c.get("function_name")),
fmt.Sprintf("FUNCTION %s.%s(", dbInfo2.DbSchema, c.get("function_name")),
-1)
}
// The definition column has everything needed to rebuild the function
fmt.Println("-- STATEMENT-BEGIN")
fmt.Printf("%s;\n", functionDef)
fmt.Println("-- STATEMENT-END")
}
}
// ==================================
// Functions
// ==================================
// compareFunctions outputs SQL to make the functions match between DBs
func compareFunctions(conn1 *sql.DB, conn2 *sql.DB) {
buf1 := new(bytes.Buffer)
functionSqlTemplate.Execute(buf1, dbInfo1)
buf2 := new(bytes.Buffer)
functionSqlTemplate.Execute(buf2, dbInfo2)
rowChan1, _ := pgutil.QueryStrings(conn1, buf1.String())
rowChan2, _ := pgutil.QueryStrings(conn2, buf2.String())
rows1 := make(FunctionRows, 0)
for row := range rowChan1 {
rows1 = append(rows1, row)
}
sort.Sort(rows1)
rows2 := make(FunctionRows, 0)
for row := range rowChan2 {
rows2 = append(rows2, row)
}
sort.Sort(rows2)
// We must explicitly type this as Schema here
var schema1 Schema = &FunctionSchema{rows: rows1, rowNum: -1}
var schema2 Schema = &FunctionSchema{rows: rows2, rowNum: -1}
// Compare the functions
doDiff(schema1, schema2)
}