// Go MySQL Driver - A MySQL-Driver for Go's database/sql package // // Copyright 2013 The Go-MySQL-Driver Authors. All rights reserved. // // This Source Code Form is subject to the terms of the Mozilla Public // License, v. 2.0. If a copy of the MPL was not distributed with this file, // You can obtain one at http://mozilla.org/MPL/2.0/. package mysql import ( "bytes" "database/sql" "database/sql/driver" "encoding/binary" "testing" "time" ) func TestScanNullTime(t *testing.T) { var scanTests = []struct { in interface{} error bool valid bool time time.Time }{ {tDate, false, true, tDate}, {sDate, false, true, tDate}, {[]byte(sDate), false, true, tDate}, {tDateTime, false, true, tDateTime}, {sDateTime, false, true, tDateTime}, {[]byte(sDateTime), false, true, tDateTime}, {tDate0, false, true, tDate0}, {sDate0, false, true, tDate0}, {[]byte(sDate0), false, true, tDate0}, {sDateTime0, false, true, tDate0}, {[]byte(sDateTime0), false, true, tDate0}, {"", true, false, tDate0}, {"1234", true, false, tDate0}, {0, true, false, tDate0}, } var nt = NullTime{} var err error for _, tst := range scanTests { err = nt.Scan(tst.in) if (err != nil) != tst.error { t.Errorf("%v: expected error status %t, got %t", tst.in, tst.error, (err != nil)) } if nt.Valid != tst.valid { t.Errorf("%v: expected valid status %t, got %t", tst.in, tst.valid, nt.Valid) } if nt.Time != tst.time { t.Errorf("%v: expected time %v, got %v", tst.in, tst.time, nt.Time) } } } func TestLengthEncodedInteger(t *testing.T) { var integerTests = []struct { num uint64 encoded []byte }{ {0x0000000000000000, []byte{0x00}}, {0x0000000000000012, []byte{0x12}}, {0x00000000000000fa, []byte{0xfa}}, {0x0000000000000100, []byte{0xfc, 0x00, 0x01}}, {0x0000000000001234, []byte{0xfc, 0x34, 0x12}}, {0x000000000000ffff, []byte{0xfc, 0xff, 0xff}}, {0x0000000000010000, []byte{0xfd, 0x00, 0x00, 0x01}}, {0x0000000000123456, []byte{0xfd, 0x56, 0x34, 0x12}}, {0x0000000000ffffff, []byte{0xfd, 0xff, 0xff, 0xff}}, {0x0000000001000000, []byte{0xfe, 0x00, 0x00, 0x00, 0x01, 0x00, 0x00, 0x00, 0x00}}, {0x123456789abcdef0, []byte{0xfe, 0xf0, 0xde, 0xbc, 0x9a, 0x78, 0x56, 0x34, 0x12}}, {0xffffffffffffffff, []byte{0xfe, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff}}, } for _, tst := range integerTests { num, isNull, numLen := readLengthEncodedInteger(tst.encoded) if isNull { t.Errorf("%x: expected %d, got NULL", tst.encoded, tst.num) } if num != tst.num { t.Errorf("%x: expected %d, got %d", tst.encoded, tst.num, num) } if numLen != len(tst.encoded) { t.Errorf("%x: expected size %d, got %d", tst.encoded, len(tst.encoded), numLen) } encoded := appendLengthEncodedInteger(nil, num) if !bytes.Equal(encoded, tst.encoded) { t.Errorf("%v: expected %x, got %x", num, tst.encoded, encoded) } } } func TestFormatBinaryDateTime(t *testing.T) { rawDate := [11]byte{} binary.LittleEndian.PutUint16(rawDate[:2], 1978) // years rawDate[2] = 12 // months rawDate[3] = 30 // days rawDate[4] = 15 // hours rawDate[5] = 46 // minutes rawDate[6] = 23 // seconds binary.LittleEndian.PutUint32(rawDate[7:], 987654) // microseconds expect := func(expected string, inlen, outlen uint8) { actual, _ := formatBinaryDateTime(rawDate[:inlen], outlen) bytes, ok := actual.([]byte) if !ok { t.Errorf("formatBinaryDateTime must return []byte, was %T", actual) } if string(bytes) != expected { t.Errorf( "expected %q, got %q for length in %d, out %d", expected, actual, inlen, outlen, ) } } expect("0000-00-00", 0, 10) expect("0000-00-00 00:00:00", 0, 19) expect("1978-12-30", 4, 10) expect("1978-12-30 15:46:23", 7, 19) expect("1978-12-30 15:46:23.987654", 11, 26) } func TestFormatBinaryTime(t *testing.T) { expect := func(expected string, src []byte, outlen uint8) { actual, _ := formatBinaryTime(src, outlen) bytes, ok := actual.([]byte) if !ok { t.Errorf("formatBinaryDateTime must return []byte, was %T", actual) } if string(bytes) != expected { t.Errorf( "expected %q, got %q for src=%q and outlen=%d", expected, actual, src, outlen) } } // binary format: // sign (0: positive, 1: negative), days(4), hours, minutes, seconds, micro(4) // Zeros expect("00:00:00", []byte{}, 8) expect("00:00:00.0", []byte{}, 10) expect("00:00:00.000000", []byte{}, 15) // Without micro(4) expect("12:34:56", []byte{0, 0, 0, 0, 0, 12, 34, 56}, 8) expect("-12:34:56", []byte{1, 0, 0, 0, 0, 12, 34, 56}, 8) expect("12:34:56.00", []byte{0, 0, 0, 0, 0, 12, 34, 56}, 11) expect("24:34:56", []byte{0, 1, 0, 0, 0, 0, 34, 56}, 8) expect("-99:34:56", []byte{1, 4, 0, 0, 0, 3, 34, 56}, 8) expect("103079215103:34:56", []byte{0, 255, 255, 255, 255, 23, 34, 56}, 8) // With micro(4) expect("12:34:56.00", []byte{0, 0, 0, 0, 0, 12, 34, 56, 99, 0, 0, 0}, 11) expect("12:34:56.000099", []byte{0, 0, 0, 0, 0, 12, 34, 56, 99, 0, 0, 0}, 15) } 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 } func TestAtomicBool(t *testing.T) { var ab atomicBool if ab.IsSet() { t.Fatal("Expected value to be false") } ab.Set(true) if ab.value != 1 { t.Fatal("Set(true) did not set value to 1") } if !ab.IsSet() { t.Fatal("Expected value to be true") } ab.Set(true) if !ab.IsSet() { t.Fatal("Expected value to be true") } ab.Set(false) if ab.value != 0 { t.Fatal("Set(false) did not set value to 0") } if ab.IsSet() { t.Fatal("Expected value to be false") } ab.Set(false) if ab.IsSet() { t.Fatal("Expected value to be false") } if ab.TrySet(false) { t.Fatal("Expected TrySet(false) to fail") } if !ab.TrySet(true) { t.Fatal("Expected TrySet(true) to succeed") } if !ab.IsSet() { t.Fatal("Expected value to be true") } ab.Set(true) if !ab.IsSet() { t.Fatal("Expected value to be true") } if ab.TrySet(true) { t.Fatal("Expected TrySet(true) to fail") } if !ab.TrySet(false) { t.Fatal("Expected TrySet(false) to succeed") } if ab.IsSet() { t.Fatal("Expected value to be false") } ab._noCopy.Lock() // we've "tested" it ¯\_(ツ)_/¯ } func TestAtomicError(t *testing.T) { var ae atomicError if ae.Value() != nil { t.Fatal("Expected value to be nil") } ae.Set(ErrMalformPkt) if v := ae.Value(); v != ErrMalformPkt { if v == nil { t.Fatal("Value is still nil") } t.Fatal("Error did not match") } ae.Set(ErrPktSync) if ae.Value() == ErrMalformPkt { t.Fatal("Error still matches old error") } if v := ae.Value(); v != ErrPktSync { t.Fatal("Error did not match") } } func TestIsolationLevelMapping(t *testing.T) { data := []struct { level driver.IsolationLevel expected string }{ { level: driver.IsolationLevel(sql.LevelReadCommitted), expected: "READ COMMITTED", }, { level: driver.IsolationLevel(sql.LevelRepeatableRead), expected: "REPEATABLE READ", }, { level: driver.IsolationLevel(sql.LevelReadUncommitted), expected: "READ UNCOMMITTED", }, { level: driver.IsolationLevel(sql.LevelSerializable), expected: "SERIALIZABLE", }, } for i, td := range data { if actual, err := mapIsolationLevel(td.level); actual != td.expected || err != nil { t.Fatal(i, td.expected, actual, err) } } // check unsupported mapping expectedErr := "mysql: unsupported isolation level: 7" actual, err := mapIsolationLevel(driver.IsolationLevel(sql.LevelLinearizable)) if actual != "" || err == nil { t.Fatal("Expected error on unsupported isolation level") } if err.Error() != expectedErr { t.Fatalf("Expected error to be %q, got %q", expectedErr, err) } }