-
Notifications
You must be signed in to change notification settings - Fork 2.1k
/
charset.go
99 lines (85 loc) · 2.92 KB
/
charset.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
/*
Copyright 2019 The Vitess Authors.
Licensed under the Apache License, Version 2.0 (the "License");
you may not use this file except in compliance with the License.
You may obtain a copy of the License at
http://www.apache.org/licenses/LICENSE-2.0
Unless required by applicable law or agreed to in writing, software
distributed under the License is distributed on an "AS IS" BASIS,
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
See the License for the specific language governing permissions and
limitations under the License.
*/
package mysql
import (
"fmt"
"strconv"
binlogdatapb "vitess.io/vitess/go/vt/proto/binlogdata"
"vitess.io/vitess/go/vt/proto/vtrpc"
"vitess.io/vitess/go/vt/vterrors"
)
// This file contains utility methods for Conn objects. Only useful on the client
// side.
// ExecuteFetchMap returns a map from column names to cell data for a query
// that should return exactly 1 row.
func ExecuteFetchMap(conn *Conn, query string) (map[string]string, error) {
qr, err := conn.ExecuteFetch(query, 1, true)
if err != nil {
return nil, err
}
if len(qr.Rows) != 1 {
return nil, vterrors.Errorf(vtrpc.Code_OUT_OF_RANGE, "query %#v returned %d rows, expected 1", query, len(qr.Rows))
}
if len(qr.Fields) != len(qr.Rows[0]) {
return nil, vterrors.Errorf(vtrpc.Code_INTERNAL, "query %#v returned %d column names, expected %d", query, len(qr.Fields), len(qr.Rows[0]))
}
rowMap := make(map[string]string)
for i, value := range qr.Rows[0] {
rowMap[qr.Fields[i].Name] = value.ToString()
}
return rowMap, nil
}
// GetCharset returns the current numerical values of the per-session character
// set variables.
func GetCharset(conn *Conn) (*binlogdatapb.Charset, error) {
// character_set_client
row, err := ExecuteFetchMap(conn, "SHOW COLLATION WHERE `charset`=@@session.character_set_client AND `default`='Yes'")
if err != nil {
return nil, err
}
client, err := strconv.ParseInt(row["Id"], 10, 16)
if err != nil {
return nil, err
}
// collation_connection
row, err = ExecuteFetchMap(conn, "SHOW COLLATION WHERE `collation`=@@session.collation_connection")
if err != nil {
return nil, err
}
connection, err := strconv.ParseInt(row["Id"], 10, 16)
if err != nil {
return nil, err
}
// collation_server
row, err = ExecuteFetchMap(conn, "SHOW COLLATION WHERE `collation`=@@session.collation_server")
if err != nil {
return nil, err
}
server, err := strconv.ParseInt(row["Id"], 10, 16)
if err != nil {
return nil, err
}
return &binlogdatapb.Charset{
Client: int32(client),
Conn: int32(connection),
Server: int32(server),
}, nil
}
// SetCharset changes the per-session character set variables.
func SetCharset(conn *Conn, cs *binlogdatapb.Charset) error {
sql := fmt.Sprintf(
"SET @@session.character_set_client=%d, @@session.collation_connection=%d, @@session.collation_server=%d",
cs.Client, cs.Conn, cs.Server)
_, err := conn.ExecuteFetch(sql, 1, false)
return err
}