driver_test.go 23 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969
  1. package mysql
  2. import (
  3. "database/sql"
  4. "fmt"
  5. "io"
  6. "io/ioutil"
  7. "net"
  8. "os"
  9. "strings"
  10. "testing"
  11. "time"
  12. )
  13. var (
  14. charset string
  15. dsn string
  16. netAddr string
  17. available bool
  18. )
  19. // See https://github.com/go-sql-driver/mysql/wiki/Testing
  20. func init() {
  21. env := func(key, defaultValue string) string {
  22. if value := os.Getenv(key); value != "" {
  23. return value
  24. }
  25. return defaultValue
  26. }
  27. user := env("MYSQL_TEST_USER", "root")
  28. pass := env("MYSQL_TEST_PASS", "")
  29. prot := env("MYSQL_TEST_PROT", "tcp")
  30. addr := env("MYSQL_TEST_ADDR", "localhost:3306")
  31. dbname := env("MYSQL_TEST_DBNAME", "gotest")
  32. charset = "charset=utf8"
  33. netAddr = fmt.Sprintf("%s(%s)", prot, addr)
  34. dsn = fmt.Sprintf("%s:%s@%s/%s?timeout=30s&"+charset, user, pass, netAddr, dbname)
  35. c, err := net.Dial(prot, addr)
  36. if err == nil {
  37. available = true
  38. c.Close()
  39. }
  40. }
  41. func TestCharset(t *testing.T) {
  42. mustSetCharset := func(charsetParam, expected string) {
  43. db, err := sql.Open("mysql", strings.Replace(dsn, charset, charsetParam, 1))
  44. if err != nil {
  45. t.Fatalf("Error on Open: %v", err)
  46. }
  47. defer db.Close()
  48. dbt := &DBTest{t, db}
  49. rows := dbt.mustQuery("SELECT @@character_set_connection")
  50. defer rows.Close()
  51. if !rows.Next() {
  52. dbt.Fatalf("Error getting connection charset: %v", err)
  53. }
  54. var got string
  55. rows.Scan(&got)
  56. if got != expected {
  57. dbt.Fatalf("Expected connection charset %s but got %s", expected, got)
  58. }
  59. }
  60. if !available {
  61. t.Logf("MySQL-Server not running on %s. Skipping TestCharset", netAddr)
  62. return
  63. }
  64. // non utf8 test
  65. mustSetCharset("charset=ascii", "ascii")
  66. // when the first charset is invalid, use the second
  67. mustSetCharset("charset=none,utf8", "utf8")
  68. // when the first charset is valid, use it
  69. mustSetCharset("charset=ascii,utf8", "ascii")
  70. mustSetCharset("charset=utf8,ascii", "utf8")
  71. }
  72. func TestFailingCharset(t *testing.T) {
  73. if !available {
  74. t.Logf("MySQL-Server not running on %s. Skipping TestFailingCharset", netAddr)
  75. return
  76. }
  77. db, err := sql.Open("mysql", strings.Replace(dsn, charset, "charset=none", 1))
  78. if err != nil {
  79. t.Fatalf("Error on Open: %v", err)
  80. }
  81. defer db.Close()
  82. // run query to really establish connection...
  83. _, err = db.Exec("SELECT 1")
  84. if err == nil {
  85. db.Close()
  86. t.Fatalf("Connection must not succeed without a valid charset")
  87. }
  88. }
  89. type DBTest struct {
  90. *testing.T
  91. db *sql.DB
  92. }
  93. func runTests(t *testing.T, name string, tests ...func(dbt *DBTest)) {
  94. if !available {
  95. t.Logf("MySQL-Server not running on %s. Skipping %s", netAddr, name)
  96. return
  97. }
  98. db, err := sql.Open("mysql", dsn)
  99. if err != nil {
  100. t.Fatalf("Error connecting: %v", err)
  101. }
  102. defer db.Close()
  103. dbt := &DBTest{t, db}
  104. dbt.mustExec("DROP TABLE IF EXISTS test")
  105. for _, test := range tests {
  106. test(dbt)
  107. }
  108. dbt.mustExec("DROP TABLE IF EXISTS test")
  109. }
  110. func (dbt *DBTest) fail(method, query string, err error) {
  111. if len(query) > 300 {
  112. query = "[query too large to print]"
  113. }
  114. dbt.Fatalf("Error on %s %s: %v", method, query, err)
  115. }
  116. func (dbt *DBTest) mustExec(query string, args ...interface{}) (res sql.Result) {
  117. res, err := dbt.db.Exec(query, args...)
  118. if err != nil {
  119. dbt.fail("Exec", query, err)
  120. }
  121. return res
  122. }
  123. func (dbt *DBTest) mustQuery(query string, args ...interface{}) (rows *sql.Rows) {
  124. rows, err := dbt.db.Query(query, args...)
  125. if err != nil {
  126. dbt.fail("Query", query, err)
  127. }
  128. return rows
  129. }
  130. func TestRawBytesResultExceedsBuffer(t *testing.T) {
  131. runTests(t, "TestRawBytesResultExceedsBuffer", func(dbt *DBTest) {
  132. // defaultBufSize from buffer.go
  133. expected := strings.Repeat("abc", defaultBufSize)
  134. rows := dbt.mustQuery("SELECT '" + expected + "'")
  135. defer rows.Close()
  136. if !rows.Next() {
  137. dbt.Error("expected result, got none")
  138. }
  139. var result sql.RawBytes
  140. rows.Scan(&result)
  141. if expected != string(result) {
  142. dbt.Error("result did not match expected value")
  143. }
  144. })
  145. }
  146. func TestCRUD(t *testing.T) {
  147. runTests(t, "TestCRUD", func(dbt *DBTest) {
  148. // Create Table
  149. dbt.mustExec("CREATE TABLE test (value BOOL)")
  150. // Test for unexpected data
  151. var out bool
  152. rows := dbt.mustQuery("SELECT * FROM test")
  153. if rows.Next() {
  154. dbt.Error("unexpected data in empty table")
  155. }
  156. // Create Data
  157. res := dbt.mustExec("INSERT INTO test VALUES (1)")
  158. count, err := res.RowsAffected()
  159. if err != nil {
  160. dbt.Fatalf("res.RowsAffected() returned error: %v", err)
  161. }
  162. if count != 1 {
  163. dbt.Fatalf("Expected 1 affected row, got %d", count)
  164. }
  165. id, err := res.LastInsertId()
  166. if err != nil {
  167. dbt.Fatalf("res.LastInsertId() returned error: %v", err)
  168. }
  169. if id != 0 {
  170. dbt.Fatalf("Expected InsertID 0, got %d", id)
  171. }
  172. // Read
  173. rows = dbt.mustQuery("SELECT value FROM test")
  174. if rows.Next() {
  175. rows.Scan(&out)
  176. if true != out {
  177. dbt.Errorf("true != %t", out)
  178. }
  179. if rows.Next() {
  180. dbt.Error("unexpected data")
  181. }
  182. } else {
  183. dbt.Error("no data")
  184. }
  185. // Update
  186. res = dbt.mustExec("UPDATE test SET value = ? WHERE value = ?", false, true)
  187. count, err = res.RowsAffected()
  188. if err != nil {
  189. dbt.Fatalf("res.RowsAffected() returned error: %v", err)
  190. }
  191. if count != 1 {
  192. dbt.Fatalf("Expected 1 affected row, got %d", count)
  193. }
  194. // Check Update
  195. rows = dbt.mustQuery("SELECT value FROM test")
  196. if rows.Next() {
  197. rows.Scan(&out)
  198. if false != out {
  199. dbt.Errorf("false != %t", out)
  200. }
  201. if rows.Next() {
  202. dbt.Error("unexpected data")
  203. }
  204. } else {
  205. dbt.Error("no data")
  206. }
  207. // Delete
  208. res = dbt.mustExec("DELETE FROM test WHERE value = ?", false)
  209. count, err = res.RowsAffected()
  210. if err != nil {
  211. dbt.Fatalf("res.RowsAffected() returned error: %v", err)
  212. }
  213. if count != 1 {
  214. dbt.Fatalf("Expected 1 affected row, got %d", count)
  215. }
  216. // Check for unexpected rows
  217. res = dbt.mustExec("DELETE FROM test")
  218. count, err = res.RowsAffected()
  219. if err != nil {
  220. dbt.Fatalf("res.RowsAffected() returned error: %v", err)
  221. }
  222. if count != 0 {
  223. dbt.Fatalf("Expected 0 affected row, got %d", count)
  224. }
  225. })
  226. }
  227. func TestInt(t *testing.T) {
  228. runTests(t, "TestInt", func(dbt *DBTest) {
  229. types := [5]string{"TINYINT", "SMALLINT", "MEDIUMINT", "INT", "BIGINT"}
  230. in := int64(42)
  231. var out int64
  232. var rows *sql.Rows
  233. // SIGNED
  234. for _, v := range types {
  235. dbt.mustExec("CREATE TABLE test (value " + v + ")")
  236. dbt.mustExec("INSERT INTO test VALUES (?)", in)
  237. rows = dbt.mustQuery("SELECT value FROM test")
  238. if rows.Next() {
  239. rows.Scan(&out)
  240. if in != out {
  241. dbt.Errorf("%s: %d != %d", v, in, out)
  242. }
  243. } else {
  244. dbt.Errorf("%s: no data", v)
  245. }
  246. dbt.mustExec("DROP TABLE IF EXISTS test")
  247. }
  248. // UNSIGNED ZEROFILL
  249. for _, v := range types {
  250. dbt.mustExec("CREATE TABLE test (value " + v + " ZEROFILL)")
  251. dbt.mustExec("INSERT INTO test VALUES (?)", in)
  252. rows = dbt.mustQuery("SELECT value FROM test")
  253. if rows.Next() {
  254. rows.Scan(&out)
  255. if in != out {
  256. dbt.Errorf("%s ZEROFILL: %d != %d", v, in, out)
  257. }
  258. } else {
  259. dbt.Errorf("%s ZEROFILL: no data", v)
  260. }
  261. dbt.mustExec("DROP TABLE IF EXISTS test")
  262. }
  263. })
  264. }
  265. func TestFloat(t *testing.T) {
  266. runTests(t, "TestFloat", func(dbt *DBTest) {
  267. types := [2]string{"FLOAT", "DOUBLE"}
  268. in := float32(42.23)
  269. var out float32
  270. var rows *sql.Rows
  271. for _, v := range types {
  272. dbt.mustExec("CREATE TABLE test (value " + v + ")")
  273. dbt.mustExec("INSERT INTO test VALUES (?)", in)
  274. rows = dbt.mustQuery("SELECT value FROM test")
  275. if rows.Next() {
  276. rows.Scan(&out)
  277. if in != out {
  278. dbt.Errorf("%s: %g != %g", v, in, out)
  279. }
  280. } else {
  281. dbt.Errorf("%s: no data", v)
  282. }
  283. dbt.mustExec("DROP TABLE IF EXISTS test")
  284. }
  285. })
  286. }
  287. func TestString(t *testing.T) {
  288. runTests(t, "TestString", func(dbt *DBTest) {
  289. types := [6]string{"CHAR(255)", "VARCHAR(255)", "TINYTEXT", "TEXT", "MEDIUMTEXT", "LONGTEXT"}
  290. in := "κόσμε üöäßñóùéàâÿœ'îë Árvíztűrő いろはにほへとちりぬるを イロハニホヘト דג סקרן чащах น่าฟังเอย"
  291. var out string
  292. var rows *sql.Rows
  293. for _, v := range types {
  294. dbt.mustExec("CREATE TABLE test (value " + v + ") CHARACTER SET utf8")
  295. dbt.mustExec("INSERT INTO test VALUES (?)", in)
  296. rows = dbt.mustQuery("SELECT value FROM test")
  297. if rows.Next() {
  298. rows.Scan(&out)
  299. if in != out {
  300. dbt.Errorf("%s: %s != %s", v, in, out)
  301. }
  302. } else {
  303. dbt.Errorf("%s: no data", v)
  304. }
  305. dbt.mustExec("DROP TABLE IF EXISTS test")
  306. }
  307. // BLOB
  308. dbt.mustExec("CREATE TABLE test (id int, value BLOB) CHARACTER SET utf8")
  309. id := 2
  310. in = "Lorem ipsum dolor sit amet, consetetur sadipscing elitr, " +
  311. "sed diam nonumy eirmod tempor invidunt ut labore et dolore magna aliquyam erat, " +
  312. "sed diam voluptua. At vero eos et accusam et justo duo dolores et ea rebum. " +
  313. "Stet clita kasd gubergren, no sea takimata sanctus est Lorem ipsum dolor sit amet. " +
  314. "Lorem ipsum dolor sit amet, consetetur sadipscing elitr, " +
  315. "sed diam nonumy eirmod tempor invidunt ut labore et dolore magna aliquyam erat, " +
  316. "sed diam voluptua. At vero eos et accusam et justo duo dolores et ea rebum. " +
  317. "Stet clita kasd gubergren, no sea takimata sanctus est Lorem ipsum dolor sit amet."
  318. dbt.mustExec("INSERT INTO test VALUES (?, ?)", id, in)
  319. err := dbt.db.QueryRow("SELECT value FROM test WHERE id = ?", id).Scan(&out)
  320. if err != nil {
  321. dbt.Fatalf("Error on BLOB-Query: %v", err)
  322. } else if out != in {
  323. dbt.Errorf("BLOB: %s != %s", in, out)
  324. }
  325. })
  326. }
  327. func TestDateTime(t *testing.T) {
  328. type testmode struct {
  329. selectSuffix string
  330. args []interface{}
  331. }
  332. type timetest struct {
  333. in interface{}
  334. sOut string
  335. tOut time.Time
  336. tIsZero bool
  337. }
  338. type tester func(dbt *DBTest, rows *sql.Rows,
  339. test *timetest, sqltype, resulttype, mode string)
  340. type setup struct {
  341. vartype string
  342. dsnSuffix string
  343. test tester
  344. }
  345. var (
  346. tdate = time.Date(2012, 6, 14, 0, 0, 0, 0, time.UTC)
  347. sdate = "2012-06-14"
  348. tdatetime = time.Date(2011, 11, 20, 21, 27, 37, 0, time.UTC)
  349. sdatetime = "2011-11-20 21:27:37"
  350. tdate0 = time.Time{}
  351. sdate0 = "0000-00-00"
  352. sdatetime0 = "0000-00-00 00:00:00"
  353. modes = map[string]*testmode{
  354. "text": &testmode{},
  355. "binary": &testmode{" WHERE 1 = ?", []interface{}{1}},
  356. }
  357. timetests = map[string][]*timetest{
  358. "DATE": {
  359. {sdate, sdate, tdate, false},
  360. {sdate0, sdate0, tdate0, true},
  361. {tdate, sdate, tdate, false},
  362. {tdate0, sdate0, tdate0, true},
  363. },
  364. "DATETIME": {
  365. {sdatetime, sdatetime, tdatetime, false},
  366. {sdatetime0, sdatetime0, tdate0, true},
  367. {tdatetime, sdatetime, tdatetime, false},
  368. {tdate0, sdatetime0, tdate0, true},
  369. },
  370. }
  371. setups = []*setup{
  372. {"string", "&parseTime=false", func(
  373. dbt *DBTest, rows *sql.Rows, test *timetest, sqltype, resulttype, mode string) {
  374. var sOut string
  375. if err := rows.Scan(&sOut); err != nil {
  376. dbt.Errorf("%s (%s %s): %v", sqltype, resulttype, mode, err)
  377. } else if test.sOut != sOut {
  378. dbt.Errorf("%s (%s %s): %s != %s", sqltype, resulttype, mode, test.sOut, sOut)
  379. }
  380. }},
  381. {"time.Time", "&parseTime=true", func(
  382. dbt *DBTest, rows *sql.Rows, test *timetest, sqltype, resulttype, mode string) {
  383. var tOut time.Time
  384. if err := rows.Scan(&tOut); err != nil {
  385. dbt.Errorf("%s (%s %s): %v", sqltype, resulttype, mode, err)
  386. } else if test.tOut != tOut || test.tIsZero != tOut.IsZero() {
  387. dbt.Errorf("%s (%s %s): %s [%t] != %s [%t]", sqltype, resulttype, mode, test.tOut, test.tIsZero, tOut, tOut.IsZero())
  388. }
  389. }},
  390. }
  391. )
  392. var s *setup
  393. testTime := func(dbt *DBTest) {
  394. var rows *sql.Rows
  395. for sqltype, tests := range timetests {
  396. dbt.mustExec("DROP TABLE IF EXISTS test")
  397. dbt.mustExec("CREATE TABLE test (value " + sqltype + ")")
  398. for _, test := range tests {
  399. for mode, q := range modes {
  400. dbt.mustExec("TRUNCATE test")
  401. dbt.mustExec("INSERT INTO test VALUES (?)", test.in)
  402. rows = dbt.mustQuery("SELECT value FROM test"+q.selectSuffix, q.args...)
  403. if rows.Next() {
  404. s.test(dbt, rows, test, sqltype, s.vartype, mode)
  405. } else {
  406. if err := rows.Err(); err != nil {
  407. dbt.Errorf("%s (%s %s): %v",
  408. sqltype, s.vartype, mode, err)
  409. } else {
  410. dbt.Errorf("%s (%s %s): no data",
  411. sqltype, s.vartype, mode)
  412. }
  413. }
  414. }
  415. }
  416. }
  417. }
  418. oldDsn := dsn
  419. usedDsn := oldDsn + "&sql_mode=ALLOW_INVALID_DATES"
  420. for _, v := range setups {
  421. s = v
  422. dsn = usedDsn + s.dsnSuffix
  423. runTests(t, "TestDateTime", testTime)
  424. }
  425. dsn = oldDsn
  426. }
  427. func TestNULL(t *testing.T) {
  428. runTests(t, "TestNULL", func(dbt *DBTest) {
  429. nullStmt, err := dbt.db.Prepare("SELECT NULL")
  430. if err != nil {
  431. dbt.Fatal(err)
  432. }
  433. defer nullStmt.Close()
  434. nonNullStmt, err := dbt.db.Prepare("SELECT 1")
  435. if err != nil {
  436. dbt.Fatal(err)
  437. }
  438. defer nonNullStmt.Close()
  439. // NullBool
  440. var nb sql.NullBool
  441. // Invalid
  442. err = nullStmt.QueryRow().Scan(&nb)
  443. if err != nil {
  444. dbt.Fatal(err)
  445. }
  446. if nb.Valid {
  447. dbt.Error("Valid NullBool which should be invalid")
  448. }
  449. // Valid
  450. err = nonNullStmt.QueryRow().Scan(&nb)
  451. if err != nil {
  452. dbt.Fatal(err)
  453. }
  454. if !nb.Valid {
  455. dbt.Error("Invalid NullBool which should be valid")
  456. } else if nb.Bool != true {
  457. dbt.Errorf("Unexpected NullBool value: %t (should be true)", nb.Bool)
  458. }
  459. // NullFloat64
  460. var nf sql.NullFloat64
  461. // Invalid
  462. err = nullStmt.QueryRow().Scan(&nf)
  463. if err != nil {
  464. dbt.Fatal(err)
  465. }
  466. if nf.Valid {
  467. dbt.Error("Valid NullFloat64 which should be invalid")
  468. }
  469. // Valid
  470. err = nonNullStmt.QueryRow().Scan(&nf)
  471. if err != nil {
  472. dbt.Fatal(err)
  473. }
  474. if !nf.Valid {
  475. dbt.Error("Invalid NullFloat64 which should be valid")
  476. } else if nf.Float64 != float64(1) {
  477. dbt.Errorf("Unexpected NullFloat64 value: %f (should be 1.0)", nf.Float64)
  478. }
  479. // NullInt64
  480. var ni sql.NullInt64
  481. // Invalid
  482. err = nullStmt.QueryRow().Scan(&ni)
  483. if err != nil {
  484. dbt.Fatal(err)
  485. }
  486. if ni.Valid {
  487. dbt.Error("Valid NullInt64 which should be invalid")
  488. }
  489. // Valid
  490. err = nonNullStmt.QueryRow().Scan(&ni)
  491. if err != nil {
  492. dbt.Fatal(err)
  493. }
  494. if !ni.Valid {
  495. dbt.Error("Invalid NullInt64 which should be valid")
  496. } else if ni.Int64 != int64(1) {
  497. dbt.Errorf("Unexpected NullInt64 value: %d (should be 1)", ni.Int64)
  498. }
  499. // NullString
  500. var ns sql.NullString
  501. // Invalid
  502. err = nullStmt.QueryRow().Scan(&ns)
  503. if err != nil {
  504. dbt.Fatal(err)
  505. }
  506. if ns.Valid {
  507. dbt.Error("Valid NullString which should be invalid")
  508. }
  509. // Valid
  510. err = nonNullStmt.QueryRow().Scan(&ns)
  511. if err != nil {
  512. dbt.Fatal(err)
  513. }
  514. if !ns.Valid {
  515. dbt.Error("Invalid NullString which should be valid")
  516. } else if ns.String != `1` {
  517. dbt.Error("Unexpected NullString value:" + ns.String + " (should be `1`)")
  518. }
  519. // Insert NULL
  520. dbt.mustExec("CREATE TABLE test (dummmy1 int, value int, dummy2 int)")
  521. dbt.mustExec("INSERT INTO test VALUES (?, ?, ?)", 1, nil, 2)
  522. var out interface{}
  523. rows := dbt.mustQuery("SELECT * FROM test")
  524. if rows.Next() {
  525. rows.Scan(&out)
  526. if out != nil {
  527. dbt.Errorf("%v != nil", out)
  528. }
  529. } else {
  530. dbt.Error("no data")
  531. }
  532. })
  533. }
  534. func TestLongData(t *testing.T) {
  535. runTests(t, "TestLongData", func(dbt *DBTest) {
  536. var maxAllowedPacketSize int
  537. err := dbt.db.QueryRow("select @@max_allowed_packet").Scan(&maxAllowedPacketSize)
  538. if err != nil {
  539. dbt.Fatal(err)
  540. }
  541. maxAllowedPacketSize--
  542. // don't get too ambitious
  543. if maxAllowedPacketSize > 1<<25 {
  544. maxAllowedPacketSize = 1 << 25
  545. }
  546. dbt.mustExec("CREATE TABLE test (value LONGBLOB)")
  547. in := strings.Repeat(`0`, maxAllowedPacketSize+1)
  548. var out string
  549. var rows *sql.Rows
  550. // Long text data
  551. const nonDataQueryLen = 28 // length query w/o value
  552. inS := in[:maxAllowedPacketSize-nonDataQueryLen]
  553. dbt.mustExec("INSERT INTO test VALUES('" + inS + "')")
  554. rows = dbt.mustQuery("SELECT value FROM test")
  555. if rows.Next() {
  556. rows.Scan(&out)
  557. if inS != out {
  558. dbt.Fatalf("LONGBLOB: length in: %d, length out: %d", len(inS), len(out))
  559. }
  560. if rows.Next() {
  561. dbt.Error("LONGBLOB: unexpexted row")
  562. }
  563. } else {
  564. dbt.Fatalf("LONGBLOB: no data")
  565. }
  566. // Empty table
  567. dbt.mustExec("TRUNCATE TABLE test")
  568. // Long binary data
  569. dbt.mustExec("INSERT INTO test VALUES(?)", in)
  570. rows = dbt.mustQuery("SELECT value FROM test WHERE 1=?", 1)
  571. if rows.Next() {
  572. rows.Scan(&out)
  573. if in != out {
  574. dbt.Fatalf("LONGBLOB: length in: %d, length out: %d", len(in), len(out))
  575. }
  576. if rows.Next() {
  577. dbt.Error("LONGBLOB: unexpexted row")
  578. }
  579. } else {
  580. dbt.Fatalf("LONGBLOB: no data")
  581. }
  582. })
  583. }
  584. func TestLoadData(t *testing.T) {
  585. runTests(t, "TestLoadData", func(dbt *DBTest) {
  586. verifyLoadDataResult := func() {
  587. rows, err := dbt.db.Query("SELECT * FROM test")
  588. if err != nil {
  589. dbt.Fatal(err.Error())
  590. }
  591. i := 0
  592. values := [4]string{
  593. "a string",
  594. "a string containing a \t",
  595. "a string containing a \n",
  596. "a string containing both \t\n",
  597. }
  598. var id int
  599. var value string
  600. for rows.Next() {
  601. i++
  602. err = rows.Scan(&id, &value)
  603. if err != nil {
  604. dbt.Fatal(err.Error())
  605. }
  606. if i != id {
  607. dbt.Fatalf("%d != %d", i, id)
  608. }
  609. if values[i-1] != value {
  610. dbt.Fatalf("%s != %s", values[i-1], value)
  611. }
  612. }
  613. err = rows.Err()
  614. if err != nil {
  615. dbt.Fatal(err.Error())
  616. }
  617. if i != 4 {
  618. dbt.Fatalf("Rows count mismatch. Got %d, want 4", i)
  619. }
  620. }
  621. file, err := ioutil.TempFile("", "gotest")
  622. defer os.Remove(file.Name())
  623. if err != nil {
  624. dbt.Fatal(err)
  625. }
  626. file.WriteString("1\ta string\n2\ta string containing a \\t\n3\ta string containing a \\n\n4\ta string containing both \\t\\n\n")
  627. file.Close()
  628. dbt.mustExec("DROP TABLE IF EXISTS test")
  629. dbt.mustExec("CREATE TABLE test (id INT NOT NULL PRIMARY KEY, value TEXT NOT NULL) CHARACTER SET utf8")
  630. // Local File
  631. RegisterLocalFile(file.Name())
  632. dbt.mustExec(fmt.Sprintf("LOAD DATA LOCAL INFILE '%q' INTO TABLE test", file.Name()))
  633. verifyLoadDataResult()
  634. // negative test
  635. _, err = dbt.db.Exec("LOAD DATA LOCAL INFILE 'doesnotexist' INTO TABLE test")
  636. if err == nil {
  637. dbt.Fatal("Load non-existent file didn't fail")
  638. } else if err.Error() != "Local File 'doesnotexist' is not registered. Use the DSN parameter 'allowAllFiles=true' to allow all files" {
  639. dbt.Fatal(err.Error())
  640. }
  641. // Empty table
  642. dbt.mustExec("TRUNCATE TABLE test")
  643. // Reader
  644. RegisterReaderHandler("test", func() io.Reader {
  645. file, err = os.Open(file.Name())
  646. if err != nil {
  647. dbt.Fatal(err)
  648. }
  649. return file
  650. })
  651. dbt.mustExec("LOAD DATA LOCAL INFILE 'Reader::test' INTO TABLE test")
  652. verifyLoadDataResult()
  653. // negative test
  654. _, err = dbt.db.Exec("LOAD DATA LOCAL INFILE 'Reader::doesnotexist' INTO TABLE test")
  655. if err == nil {
  656. dbt.Fatal("Load non-existent Reader didn't fail")
  657. } else if err.Error() != "Reader 'doesnotexist' is not registered" {
  658. dbt.Fatal(err.Error())
  659. }
  660. })
  661. }
  662. // Special cases
  663. func TestRowsClose(t *testing.T) {
  664. runTests(t, "TestRowsClose", func(dbt *DBTest) {
  665. rows, err := dbt.db.Query("SELECT 1")
  666. if err != nil {
  667. dbt.Fatal(err)
  668. }
  669. err = rows.Close()
  670. if err != nil {
  671. dbt.Fatal(err)
  672. }
  673. if rows.Next() {
  674. dbt.Fatal("Unexpected row after rows.Close()")
  675. }
  676. err = rows.Err()
  677. if err != nil {
  678. dbt.Fatal(err)
  679. }
  680. })
  681. }
  682. // dangling statements
  683. // http://code.google.com/p/go/issues/detail?id=3865
  684. func TestCloseStmtBeforeRows(t *testing.T) {
  685. runTests(t, "TestCloseStmtBeforeRows", func(dbt *DBTest) {
  686. stmt, err := dbt.db.Prepare("SELECT 1")
  687. if err != nil {
  688. dbt.Fatal(err)
  689. }
  690. rows, err := stmt.Query()
  691. if err != nil {
  692. stmt.Close()
  693. dbt.Fatal(err)
  694. }
  695. defer rows.Close()
  696. err = stmt.Close()
  697. if err != nil {
  698. dbt.Fatal(err)
  699. }
  700. if !rows.Next() {
  701. dbt.Fatal("Getting row failed")
  702. } else {
  703. err = rows.Err()
  704. if err != nil {
  705. dbt.Fatal(err)
  706. }
  707. var out bool
  708. err = rows.Scan(&out)
  709. if err != nil {
  710. dbt.Fatalf("Error on rows.Scan(): %v", err)
  711. }
  712. if out != true {
  713. dbt.Errorf("true != %t", out)
  714. }
  715. }
  716. })
  717. }
  718. // It is valid to have multiple Rows for the same Stmt
  719. // http://code.google.com/p/go/issues/detail?id=3734
  720. func TestStmtMultiRows(t *testing.T) {
  721. runTests(t, "TestStmtMultiRows", func(dbt *DBTest) {
  722. stmt, err := dbt.db.Prepare("SELECT 1 UNION SELECT 0")
  723. if err != nil {
  724. dbt.Fatal(err)
  725. }
  726. rows1, err := stmt.Query()
  727. if err != nil {
  728. stmt.Close()
  729. dbt.Fatal(err)
  730. }
  731. defer rows1.Close()
  732. rows2, err := stmt.Query()
  733. if err != nil {
  734. stmt.Close()
  735. dbt.Fatal(err)
  736. }
  737. defer rows2.Close()
  738. var out bool
  739. // 1
  740. if !rows1.Next() {
  741. dbt.Fatal("1st rows1.Next failed")
  742. } else {
  743. err = rows1.Err()
  744. if err != nil {
  745. dbt.Fatal(err)
  746. }
  747. err = rows1.Scan(&out)
  748. if err != nil {
  749. dbt.Fatalf("Error on rows.Scan(): %v", err)
  750. }
  751. if out != true {
  752. dbt.Errorf("true != %t", out)
  753. }
  754. }
  755. if !rows2.Next() {
  756. dbt.Fatal("1st rows2.Next failed")
  757. } else {
  758. err = rows2.Err()
  759. if err != nil {
  760. dbt.Fatal(err)
  761. }
  762. err = rows2.Scan(&out)
  763. if err != nil {
  764. dbt.Fatalf("Error on rows.Scan(): %v", err)
  765. }
  766. if out != true {
  767. dbt.Errorf("true != %t", out)
  768. }
  769. }
  770. // 2
  771. if !rows1.Next() {
  772. dbt.Fatal("2nd rows1.Next failed")
  773. } else {
  774. err = rows1.Err()
  775. if err != nil {
  776. dbt.Fatal(err)
  777. }
  778. err = rows1.Scan(&out)
  779. if err != nil {
  780. dbt.Fatalf("Error on rows.Scan(): %v", err)
  781. }
  782. if out != false {
  783. dbt.Errorf("false != %t", out)
  784. }
  785. if rows1.Next() {
  786. dbt.Fatal("Unexpected row on rows1")
  787. }
  788. err = rows1.Close()
  789. if err != nil {
  790. dbt.Fatal(err)
  791. }
  792. }
  793. if !rows2.Next() {
  794. dbt.Fatal("2nd rows2.Next failed")
  795. } else {
  796. err = rows2.Err()
  797. if err != nil {
  798. dbt.Fatal(err)
  799. }
  800. err = rows2.Scan(&out)
  801. if err != nil {
  802. dbt.Fatalf("Error on rows.Scan(): %v", err)
  803. }
  804. if out != false {
  805. dbt.Errorf("false != %t", out)
  806. }
  807. if rows2.Next() {
  808. dbt.Fatal("Unexpected row on rows2")
  809. }
  810. err = rows2.Close()
  811. if err != nil {
  812. dbt.Fatal(err)
  813. }
  814. }
  815. })
  816. }
  817. func TestConcurrent(t *testing.T) {
  818. if os.Getenv("MYSQL_TEST_CONCURRENT") != "1" {
  819. t.Log("CONCURRENT env var not set. Skipping TestConcurrent")
  820. return
  821. }
  822. runTests(t, "TestConcurrent", func(dbt *DBTest) {
  823. var max int
  824. err := dbt.db.QueryRow("SELECT @@max_connections").Scan(&max)
  825. if err != nil {
  826. dbt.Fatalf("%v", err)
  827. }
  828. dbt.Logf("Testing up to %d concurrent connections \r\n", max)
  829. canStop := false
  830. c := make(chan struct{}, max)
  831. for i := 0; i < max; i++ {
  832. go func(id int) {
  833. tx, err := dbt.db.Begin()
  834. if err != nil {
  835. canStop = true
  836. if err.Error() == "Error 1040: Too many connections" {
  837. max--
  838. return
  839. } else {
  840. dbt.Fatalf("Error on Con %d: %s", id, err.Error())
  841. }
  842. }
  843. c <- struct{}{}
  844. for !canStop {
  845. _, err = tx.Exec("SELECT 1")
  846. if err != nil {
  847. canStop = true
  848. dbt.Fatalf("Error on Con %d: %s", id, err.Error())
  849. }
  850. }
  851. err = tx.Commit()
  852. if err != nil {
  853. canStop = true
  854. dbt.Fatalf("Error on Con %d: %s", id, err.Error())
  855. }
  856. }(i)
  857. }
  858. for i := 0; i < max; i++ {
  859. <-c
  860. }
  861. canStop = true
  862. dbt.Logf("Reached %d concurrent connections \r\n", max)
  863. })
  864. }