|
|
@@ -22,19 +22,19 @@ var testDSNs = []struct {
|
|
|
out string
|
|
|
loc *time.Location
|
|
|
}{
|
|
|
- {"username:password@protocol(address)/dbname?param=value", "&{user:username passwd:password net:protocol addr:address dbname:dbname params:map[param:value] loc:%p tls:<nil> timeout:0 collation:33 allowAllFiles:false allowOldPasswords:false clientFoundRows:false columnsWithAlias:false}", time.UTC},
|
|
|
- {"username:password@protocol(address)/dbname?param=value&columnsWithAlias=true", "&{user:username passwd:password net:protocol addr:address dbname:dbname params:map[param:value] loc:%p tls:<nil> timeout:0 collation:33 allowAllFiles:false allowOldPasswords:false clientFoundRows:false columnsWithAlias:true}", time.UTC},
|
|
|
- {"user@unix(/path/to/socket)/dbname?charset=utf8", "&{user:user passwd: net:unix addr:/path/to/socket dbname:dbname params:map[charset:utf8] loc:%p tls:<nil> timeout:0 collation:33 allowAllFiles:false allowOldPasswords:false clientFoundRows:false columnsWithAlias:false}", time.UTC},
|
|
|
- {"user:password@tcp(localhost:5555)/dbname?charset=utf8&tls=true", "&{user:user passwd:password net:tcp addr:localhost:5555 dbname:dbname params:map[charset:utf8] loc:%p tls:<nil> timeout:0 collation:33 allowAllFiles:false allowOldPasswords:false clientFoundRows:false columnsWithAlias:false}", time.UTC},
|
|
|
- {"user:password@tcp(localhost:5555)/dbname?charset=utf8mb4,utf8&tls=skip-verify", "&{user:user passwd:password net:tcp addr:localhost:5555 dbname:dbname params:map[charset:utf8mb4,utf8] loc:%p tls:<nil> timeout:0 collation:33 allowAllFiles:false allowOldPasswords:false clientFoundRows:false columnsWithAlias:false}", time.UTC},
|
|
|
- {"user:password@/dbname?loc=UTC&timeout=30s&allowAllFiles=1&clientFoundRows=true&allowOldPasswords=TRUE&collation=utf8mb4_unicode_ci", "&{user:user passwd:password net:tcp addr:127.0.0.1:3306 dbname:dbname params:map[] loc:%p tls:<nil> timeout:30000000000 collation:224 allowAllFiles:true allowOldPasswords:true clientFoundRows:true columnsWithAlias:false}", time.UTC},
|
|
|
- {"user:p@ss(word)@tcp([de:ad:be:ef::ca:fe]:80)/dbname?loc=Local", "&{user:user passwd:p@ss(word) net:tcp addr:[de:ad:be:ef::ca:fe]:80 dbname:dbname params:map[] loc:%p tls:<nil> timeout:0 collation:33 allowAllFiles:false allowOldPasswords:false clientFoundRows:false columnsWithAlias:false}", time.Local},
|
|
|
- {"/dbname", "&{user: passwd: net:tcp addr:127.0.0.1:3306 dbname:dbname params:map[] loc:%p tls:<nil> timeout:0 collation:33 allowAllFiles:false allowOldPasswords:false clientFoundRows:false columnsWithAlias:false}", time.UTC},
|
|
|
- {"@/", "&{user: passwd: net:tcp addr:127.0.0.1:3306 dbname: params:map[] loc:%p tls:<nil> timeout:0 collation:33 allowAllFiles:false allowOldPasswords:false clientFoundRows:false columnsWithAlias:false}", time.UTC},
|
|
|
- {"/", "&{user: passwd: net:tcp addr:127.0.0.1:3306 dbname: params:map[] loc:%p tls:<nil> timeout:0 collation:33 allowAllFiles:false allowOldPasswords:false clientFoundRows:false columnsWithAlias:false}", time.UTC},
|
|
|
- {"", "&{user: passwd: net:tcp addr:127.0.0.1:3306 dbname: params:map[] loc:%p tls:<nil> timeout:0 collation:33 allowAllFiles:false allowOldPasswords:false clientFoundRows:false columnsWithAlias:false}", time.UTC},
|
|
|
- {"user:p@/ssword@/", "&{user:user passwd:p@/ssword net:tcp addr:127.0.0.1:3306 dbname: params:map[] loc:%p tls:<nil> timeout:0 collation:33 allowAllFiles:false allowOldPasswords:false clientFoundRows:false columnsWithAlias:false}", time.UTC},
|
|
|
- {"unix/?arg=%2Fsome%2Fpath.ext", "&{user: passwd: net:unix addr:/tmp/mysql.sock dbname: params:map[arg:/some/path.ext] loc:%p tls:<nil> timeout:0 collation:33 allowAllFiles:false allowOldPasswords:false clientFoundRows:false columnsWithAlias:false}", time.UTC},
|
|
|
+ {"username:password@protocol(address)/dbname?param=value", "&{user:username passwd:password net:protocol addr:address dbname:dbname params:map[param:value] loc:%p tls:<nil> timeout:0 collation:33 allowAllFiles:false allowOldPasswords:false clientFoundRows:false columnsWithAlias:false interpolateParams:false}", time.UTC},
|
|
|
+ {"username:password@protocol(address)/dbname?param=value&columnsWithAlias=true", "&{user:username passwd:password net:protocol addr:address dbname:dbname params:map[param:value] loc:%p tls:<nil> timeout:0 collation:33 allowAllFiles:false allowOldPasswords:false clientFoundRows:false columnsWithAlias:true interpolateParams:false}", time.UTC},
|
|
|
+ {"user@unix(/path/to/socket)/dbname?charset=utf8", "&{user:user passwd: net:unix addr:/path/to/socket dbname:dbname params:map[charset:utf8] loc:%p tls:<nil> timeout:0 collation:33 allowAllFiles:false allowOldPasswords:false clientFoundRows:false columnsWithAlias:false interpolateParams:false}", time.UTC},
|
|
|
+ {"user:password@tcp(localhost:5555)/dbname?charset=utf8&tls=true", "&{user:user passwd:password net:tcp addr:localhost:5555 dbname:dbname params:map[charset:utf8] loc:%p tls:<nil> timeout:0 collation:33 allowAllFiles:false allowOldPasswords:false clientFoundRows:false columnsWithAlias:false interpolateParams:false}", time.UTC},
|
|
|
+ {"user:password@tcp(localhost:5555)/dbname?charset=utf8mb4,utf8&tls=skip-verify", "&{user:user passwd:password net:tcp addr:localhost:5555 dbname:dbname params:map[charset:utf8mb4,utf8] loc:%p tls:<nil> timeout:0 collation:33 allowAllFiles:false allowOldPasswords:false clientFoundRows:false columnsWithAlias:false interpolateParams:false}", time.UTC},
|
|
|
+ {"user:password@/dbname?loc=UTC&timeout=30s&allowAllFiles=1&clientFoundRows=true&allowOldPasswords=TRUE&collation=utf8mb4_unicode_ci", "&{user:user passwd:password net:tcp addr:127.0.0.1:3306 dbname:dbname params:map[] loc:%p tls:<nil> timeout:30000000000 collation:224 allowAllFiles:true allowOldPasswords:true clientFoundRows:true columnsWithAlias:false interpolateParams:false}", time.UTC},
|
|
|
+ {"user:p@ss(word)@tcp([de:ad:be:ef::ca:fe]:80)/dbname?loc=Local", "&{user:user passwd:p@ss(word) net:tcp addr:[de:ad:be:ef::ca:fe]:80 dbname:dbname params:map[] loc:%p tls:<nil> timeout:0 collation:33 allowAllFiles:false allowOldPasswords:false clientFoundRows:false columnsWithAlias:false interpolateParams:false}", time.Local},
|
|
|
+ {"/dbname", "&{user: passwd: net:tcp addr:127.0.0.1:3306 dbname:dbname params:map[] loc:%p tls:<nil> timeout:0 collation:33 allowAllFiles:false allowOldPasswords:false clientFoundRows:false columnsWithAlias:false interpolateParams:false}", time.UTC},
|
|
|
+ {"@/", "&{user: passwd: net:tcp addr:127.0.0.1:3306 dbname: params:map[] loc:%p tls:<nil> timeout:0 collation:33 allowAllFiles:false allowOldPasswords:false clientFoundRows:false columnsWithAlias:false interpolateParams:false}", time.UTC},
|
|
|
+ {"/", "&{user: passwd: net:tcp addr:127.0.0.1:3306 dbname: params:map[] loc:%p tls:<nil> timeout:0 collation:33 allowAllFiles:false allowOldPasswords:false clientFoundRows:false columnsWithAlias:false interpolateParams:false}", time.UTC},
|
|
|
+ {"", "&{user: passwd: net:tcp addr:127.0.0.1:3306 dbname: params:map[] loc:%p tls:<nil> timeout:0 collation:33 allowAllFiles:false allowOldPasswords:false clientFoundRows:false columnsWithAlias:false interpolateParams:false}", time.UTC},
|
|
|
+ {"user:p@/ssword@/", "&{user:user passwd:p@/ssword net:tcp addr:127.0.0.1:3306 dbname: params:map[] loc:%p tls:<nil> timeout:0 collation:33 allowAllFiles:false allowOldPasswords:false clientFoundRows:false columnsWithAlias:false interpolateParams:false}", time.UTC},
|
|
|
+ {"unix/?arg=%2Fsome%2Fpath.ext", "&{user: passwd: net:unix addr:/tmp/mysql.sock dbname: params:map[arg:/some/path.ext] loc:%p tls:<nil> timeout:0 collation:33 allowAllFiles:false allowOldPasswords:false clientFoundRows:false columnsWithAlias:false interpolateParams:false}", time.UTC},
|
|
|
}
|
|
|
|
|
|
func TestDSNParser(t *testing.T) {
|
|
|
@@ -116,6 +116,43 @@ func TestDSNWithCustomTLS(t *testing.T) {
|
|
|
DeregisterTLSConfig("utils_test")
|
|
|
}
|
|
|
|
|
|
+func TestDSNUnsafeCollation(t *testing.T) {
|
|
|
+ _, err := parseDSN("/dbname?collation=gbk_chinese_ci&interpolateParams=true")
|
|
|
+ if err != errInvalidDSNUnsafeCollation {
|
|
|
+ t.Error("Expected %v, Got %v", errInvalidDSNUnsafeCollation, err)
|
|
|
+ }
|
|
|
+
|
|
|
+ _, err = parseDSN("/dbname?collation=gbk_chinese_ci&interpolateParams=false")
|
|
|
+ if err != nil {
|
|
|
+ t.Error("Expected %v, Got %v", nil, err)
|
|
|
+ }
|
|
|
+
|
|
|
+ _, err = parseDSN("/dbname?collation=gbk_chinese_ci")
|
|
|
+ if err != nil {
|
|
|
+ t.Error("Expected %v, Got %v", nil, err)
|
|
|
+ }
|
|
|
+
|
|
|
+ _, err = parseDSN("/dbname?collation=ascii_bin&interpolateParams=true")
|
|
|
+ if err != nil {
|
|
|
+ t.Error("Expected %v, Got %v", nil, err)
|
|
|
+ }
|
|
|
+
|
|
|
+ _, err = parseDSN("/dbname?collation=latin1_german1_ci&interpolateParams=true")
|
|
|
+ if err != nil {
|
|
|
+ t.Error("Expected %v, Got %v", nil, err)
|
|
|
+ }
|
|
|
+
|
|
|
+ _, err = parseDSN("/dbname?collation=utf8_general_ci&interpolateParams=true")
|
|
|
+ if err != nil {
|
|
|
+ t.Error("Expected %v, Got %v", nil, err)
|
|
|
+ }
|
|
|
+
|
|
|
+ _, err = parseDSN("/dbname?collation=utf8mb4_general_ci&interpolateParams=true")
|
|
|
+ if err != nil {
|
|
|
+ t.Error("Expected %v, Got %v", nil, err)
|
|
|
+ }
|
|
|
+}
|
|
|
+
|
|
|
func BenchmarkParseDSN(b *testing.B) {
|
|
|
b.ReportAllocs()
|
|
|
|
|
|
@@ -252,3 +289,58 @@ func TestFormatBinaryDateTime(t *testing.T) {
|
|
|
expect("1978-12-30 15:46:23", 7, 19)
|
|
|
expect("1978-12-30 15:46:23.987654", 11, 26)
|
|
|
}
|
|
|
+
|
|
|
+func TestEscapeBackslash(t *testing.T) {
|
|
|
+ expect := func(expected, value string) {
|
|
|
+ actual := string(escapeBytesBackslash([]byte{}, []byte(value)))
|
|
|
+ if actual != expected {
|
|
|
+ t.Errorf(
|
|
|
+ "expected %s, got %s",
|
|
|
+ expected, actual,
|
|
|
+ )
|
|
|
+ }
|
|
|
+
|
|
|
+ actual = string(escapeStringBackslash([]byte{}, value))
|
|
|
+ if actual != expected {
|
|
|
+ t.Errorf(
|
|
|
+ "expected %s, got %s",
|
|
|
+ expected, actual,
|
|
|
+ )
|
|
|
+ }
|
|
|
+ }
|
|
|
+
|
|
|
+ expect("foo\\0bar", "foo\x00bar")
|
|
|
+ expect("foo\\nbar", "foo\nbar")
|
|
|
+ expect("foo\\rbar", "foo\rbar")
|
|
|
+ expect("foo\\Zbar", "foo\x1abar")
|
|
|
+ expect("foo\\\"bar", "foo\"bar")
|
|
|
+ expect("foo\\\\bar", "foo\\bar")
|
|
|
+ expect("foo\\'bar", "foo'bar")
|
|
|
+}
|
|
|
+
|
|
|
+func TestEscapeQuotes(t *testing.T) {
|
|
|
+ expect := func(expected, value string) {
|
|
|
+ actual := string(escapeBytesQuotes([]byte{}, []byte(value)))
|
|
|
+ if actual != expected {
|
|
|
+ t.Errorf(
|
|
|
+ "expected %s, got %s",
|
|
|
+ expected, actual,
|
|
|
+ )
|
|
|
+ }
|
|
|
+
|
|
|
+ actual = string(escapeStringQuotes([]byte{}, value))
|
|
|
+ if actual != expected {
|
|
|
+ t.Errorf(
|
|
|
+ "expected %s, got %s",
|
|
|
+ expected, actual,
|
|
|
+ )
|
|
|
+ }
|
|
|
+ }
|
|
|
+
|
|
|
+ expect("foo\x00bar", "foo\x00bar") // not affected
|
|
|
+ expect("foo\nbar", "foo\nbar") // not affected
|
|
|
+ expect("foo\rbar", "foo\rbar") // not affected
|
|
|
+ expect("foo\x1abar", "foo\x1abar") // not affected
|
|
|
+ expect("foo''bar", "foo'bar") // affected
|
|
|
+ expect("foo\"bar", "foo\"bar") // not affected
|
|
|
+}
|