forked from minus5/gofreetds
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathexecutesql.go
153 lines (137 loc) · 3.9 KB
/
executesql.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
package freetds
import (
"database/sql/driver"
"fmt"
"regexp"
"strings"
"time"
)
const statusRow string = `;
select cast(coalesce(scope_identity(), -1) as bigint) last_insert_id,
cast(@@rowcount as bigint) rows_affected
`
const statusRowSybase125 string = `
select cast(coalesce(@@IDENTITY, -1) as int) last_insert_id,
cast(@@rowcount as int) rows_affected
`
//Execute sql query with arguments.
//? in query are arguments placeholders.
// ExecuteSql("select * from authors where au_fname = ?", "John")
func (conn *Conn) ExecuteSql(query string, params ...driver.Value) ([]*Result, error) {
if conn.sybaseMode() {
return conn.executeSqlSybase125(query, params...)
}
statement, numParams := query2Statement(query)
if numParams != len(params) {
return nil, fmt.Errorf("Incorrect number of params, expecting %d got %d", numParams, len(params))
}
paramDef, paramVal, err := parseParams(params...)
if err != nil {
return nil, err
}
statement += statusRow
sql := fmt.Sprintf("exec sp_executesql N'%s', N'%s', %s", statement, paramDef, paramVal)
if numParams == 0 {
sql = fmt.Sprintf("exec sp_executesql N'%s'", statement)
}
return conn.Exec(sql)
}
func (conn *Conn) executeSqlSybase125(query string, params ...driver.Value) ([]*Result, error) {
statement, numParams := query2Statement(query)
if numParams != len(params) {
return nil, fmt.Errorf("Incorrect number of params, expecting %d got %d", numParams, len(params))
}
statement += statusRowSybase125
sql := strings.Replace(query, "?", "$bindkey", -1)
re, _ := regexp.Compile(`(?P<bindkey>\$bindkey)`)
matches := re.FindAllSubmatchIndex([]byte(sql), -1)
for i, _ := range matches {
_, escapedValue, _ := go2SqlDataType(params[i])
sql = fmt.Sprintf("%s", strings.Replace(sql, "$bindkey", escapedValue, 1))
}
if numParams == 0 {
sql = fmt.Sprintf("%s", statement)
}
return conn.Exec(sql)
}
//converts query to SqlServer statement for sp_executesql
//replaces ? in query with params @p1, @p2, ...
//returns statement and number of params
func query2Statement(query string) (string, int) {
parts := strings.Split(query, "?")
var statement string
numParams := len(parts) - 1
statement = parts[0]
for i, part := range parts {
if i > 0 {
statement = fmt.Sprintf("%s@p%d%s", statement, i, part)
}
}
return quote(statement), numParams
}
func parseParams(params ...driver.Value) (string, string, error) {
paramDef := ""
paramVal := ""
for i, param := range params {
if i > 0 {
paramVal += ", "
paramDef += ", "
}
sqlType, sqlValue, err := go2SqlDataType(param)
if err != nil {
return "", "", err
}
paramName := fmt.Sprintf("@p%d", i+1)
paramDef += fmt.Sprintf("%s %s", paramName, sqlType)
paramVal += fmt.Sprintf("%s=%s", paramName, sqlValue)
}
return paramDef, paramVal, nil
}
func quote(in string) string {
return strings.Replace(in, "'", "''", -1)
}
func go2SqlDataType(value interface{}) (string, string, error) {
max := func(a int, b int) int {
if a > b {
return a
}
return b
}
strValue := fmt.Sprintf("%v", value)
switch t := value.(type) {
case bool:
bitStrValue := "0"
if strValue == "true" {
bitStrValue = "1"
}
return "bit", bitStrValue, nil
case uint8, int8:
return "tinyint", strValue, nil
case uint16, int16:
return "smallint", strValue, nil
case uint32, int32, int:
return "int", strValue, nil
case uint64, int64:
return "bigint", strValue, nil
case float32, float64:
return "real", strValue, nil
case string:
{
}
case time.Time:
{
strValue = t.Format(time.RFC3339Nano)
return "datetimeoffset", fmt.Sprintf("'%s'", quote(strValue)), nil
}
case []byte:
{
b, _ := value.([]byte)
return fmt.Sprintf("varbinary (%d)", max(1, len(b))),
fmt.Sprintf("0x%x", b), nil
}
default:
return "", "", fmt.Errorf("unknown dataType %T", t)
}
return fmt.Sprintf("nvarchar (%d)", max(1, len(strValue))),
fmt.Sprintf("'%s'", quote(strValue)), nil
}