driver_test.go 23 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971
  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) mustExec(query string, args ...interface{}) (res sql.Result) {
  111. res, err := dbt.db.Exec(query, args...)
  112. if err != nil {
  113. if len(query) > 300 {
  114. query = "[query too large to print]"
  115. }
  116. dbt.Fatalf("Error on Exec %s: %v", query, err)
  117. }
  118. return res
  119. }
  120. func (dbt *DBTest) mustQuery(query string, args ...interface{}) (rows *sql.Rows) {
  121. rows, err := dbt.db.Query(query, args...)
  122. if err != nil {
  123. if len(query) > 300 {
  124. query = "[query too large to print]"
  125. }
  126. dbt.Fatalf("Error on Query %s: %v", 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",
  379. sqltype, resulttype, mode, test.sOut, sOut)
  380. }
  381. }},
  382. {"time.Time", "&parseTime=true", func(
  383. dbt *DBTest, rows *sql.Rows, test *timetest, sqltype, resulttype, mode string) {
  384. var tOut time.Time
  385. if err := rows.Scan(&tOut); err != nil {
  386. dbt.Errorf("%s (%s %s): %v",
  387. sqltype, resulttype, mode, err)
  388. } else if test.tOut != tOut || test.tIsZero != tOut.IsZero() {
  389. dbt.Errorf("%s (%s %s): %s [%t] != %s [%t]",
  390. sqltype, resulttype, mode, test.tOut, test.tIsZero, tOut, tOut.IsZero())
  391. }
  392. }},
  393. }
  394. )
  395. var s *setup
  396. testTime := func(dbt *DBTest) {
  397. var rows *sql.Rows
  398. for sqltype, tests := range timetests {
  399. dbt.mustExec("DROP TABLE IF EXISTS test")
  400. dbt.mustExec("CREATE TABLE test (value " + sqltype + ")")
  401. for _, test := range tests {
  402. for mode, q := range modes {
  403. dbt.mustExec("TRUNCATE test")
  404. dbt.mustExec("INSERT INTO test VALUES (?)", test.in)
  405. rows = dbt.mustQuery("SELECT value FROM test"+q.selectSuffix, q.args...)
  406. if rows.Next() {
  407. s.test(dbt, rows, test, sqltype, s.vartype, mode)
  408. } else {
  409. if err := rows.Err(); err != nil {
  410. dbt.Errorf("%s (%s %s): %v",
  411. sqltype, s.vartype, mode, err)
  412. } else {
  413. dbt.Errorf("%s (%s %s): no data",
  414. sqltype, s.vartype, mode)
  415. }
  416. }
  417. }
  418. }
  419. }
  420. }
  421. oldDsn := dsn
  422. usedDsn := oldDsn + "&sql_mode=ALLOW_INVALID_DATES"
  423. for _, v := range setups {
  424. s = v
  425. dsn = usedDsn + s.dsnSuffix
  426. runTests(t, "TestDateTime", testTime)
  427. }
  428. dsn = oldDsn
  429. }
  430. func TestNULL(t *testing.T) {
  431. runTests(t, "TestNULL", func(dbt *DBTest) {
  432. nullStmt, err := dbt.db.Prepare("SELECT NULL")
  433. if err != nil {
  434. dbt.Fatal(err)
  435. }
  436. defer nullStmt.Close()
  437. nonNullStmt, err := dbt.db.Prepare("SELECT 1")
  438. if err != nil {
  439. dbt.Fatal(err)
  440. }
  441. defer nonNullStmt.Close()
  442. // NullBool
  443. var nb sql.NullBool
  444. // Invalid
  445. err = nullStmt.QueryRow().Scan(&nb)
  446. if err != nil {
  447. dbt.Fatal(err)
  448. }
  449. if nb.Valid {
  450. dbt.Error("Valid NullBool which should be invalid")
  451. }
  452. // Valid
  453. err = nonNullStmt.QueryRow().Scan(&nb)
  454. if err != nil {
  455. dbt.Fatal(err)
  456. }
  457. if !nb.Valid {
  458. dbt.Error("Invalid NullBool which should be valid")
  459. } else if nb.Bool != true {
  460. dbt.Errorf("Unexpected NullBool value: %t (should be true)", nb.Bool)
  461. }
  462. // NullFloat64
  463. var nf sql.NullFloat64
  464. // Invalid
  465. err = nullStmt.QueryRow().Scan(&nf)
  466. if err != nil {
  467. dbt.Fatal(err)
  468. }
  469. if nf.Valid {
  470. dbt.Error("Valid NullFloat64 which should be invalid")
  471. }
  472. // Valid
  473. err = nonNullStmt.QueryRow().Scan(&nf)
  474. if err != nil {
  475. dbt.Fatal(err)
  476. }
  477. if !nf.Valid {
  478. dbt.Error("Invalid NullFloat64 which should be valid")
  479. } else if nf.Float64 != float64(1) {
  480. dbt.Errorf("Unexpected NullFloat64 value: %f (should be 1.0)", nf.Float64)
  481. }
  482. // NullInt64
  483. var ni sql.NullInt64
  484. // Invalid
  485. err = nullStmt.QueryRow().Scan(&ni)
  486. if err != nil {
  487. dbt.Fatal(err)
  488. }
  489. if ni.Valid {
  490. dbt.Error("Valid NullInt64 which should be invalid")
  491. }
  492. // Valid
  493. err = nonNullStmt.QueryRow().Scan(&ni)
  494. if err != nil {
  495. dbt.Fatal(err)
  496. }
  497. if !ni.Valid {
  498. dbt.Error("Invalid NullInt64 which should be valid")
  499. } else if ni.Int64 != int64(1) {
  500. dbt.Errorf("Unexpected NullInt64 value: %d (should be 1)", ni.Int64)
  501. }
  502. // NullString
  503. var ns sql.NullString
  504. // Invalid
  505. err = nullStmt.QueryRow().Scan(&ns)
  506. if err != nil {
  507. dbt.Fatal(err)
  508. }
  509. if ns.Valid {
  510. dbt.Error("Valid NullString which should be invalid")
  511. }
  512. // Valid
  513. err = nonNullStmt.QueryRow().Scan(&ns)
  514. if err != nil {
  515. dbt.Fatal(err)
  516. }
  517. if !ns.Valid {
  518. dbt.Error("Invalid NullString which should be valid")
  519. } else if ns.String != `1` {
  520. dbt.Error("Unexpected NullString value:" + ns.String + " (should be `1`)")
  521. }
  522. // Insert NULL
  523. dbt.mustExec("CREATE TABLE test (dummmy1 int, value int, dummy2 int)")
  524. dbt.mustExec("INSERT INTO test VALUES (?, ?, ?)", 1, nil, 2)
  525. var out interface{}
  526. rows := dbt.mustQuery("SELECT * FROM test")
  527. if rows.Next() {
  528. rows.Scan(&out)
  529. if out != nil {
  530. dbt.Errorf("%v != nil", out)
  531. }
  532. } else {
  533. dbt.Error("no data")
  534. }
  535. })
  536. }
  537. func TestLongData(t *testing.T) {
  538. runTests(t, "TestLongData", func(dbt *DBTest) {
  539. var maxAllowedPacketSize int
  540. err := dbt.db.QueryRow("select @@max_allowed_packet").Scan(&maxAllowedPacketSize)
  541. if err != nil {
  542. dbt.Fatal(err)
  543. }
  544. maxAllowedPacketSize--
  545. // don't get too ambitious
  546. if maxAllowedPacketSize > 1<<25 {
  547. maxAllowedPacketSize = 1 << 25
  548. }
  549. dbt.mustExec("CREATE TABLE test (value LONGBLOB)")
  550. in := strings.Repeat(`0`, maxAllowedPacketSize+1)
  551. var out string
  552. var rows *sql.Rows
  553. // Long text data
  554. const nonDataQueryLen = 28 // length query w/o value
  555. inS := in[:maxAllowedPacketSize-nonDataQueryLen]
  556. dbt.mustExec("INSERT INTO test VALUES('" + inS + "')")
  557. rows = dbt.mustQuery("SELECT value FROM test")
  558. if rows.Next() {
  559. rows.Scan(&out)
  560. if inS != out {
  561. dbt.Fatalf("LONGBLOB: length in: %d, length out: %d", len(inS), len(out))
  562. }
  563. if rows.Next() {
  564. dbt.Error("LONGBLOB: unexpexted row")
  565. }
  566. } else {
  567. dbt.Fatalf("LONGBLOB: no data")
  568. }
  569. // Empty table
  570. dbt.mustExec("TRUNCATE TABLE test")
  571. // Long binary data
  572. dbt.mustExec("INSERT INTO test VALUES(?)", in)
  573. rows = dbt.mustQuery("SELECT value FROM test WHERE 1=?", 1)
  574. if rows.Next() {
  575. rows.Scan(&out)
  576. if in != out {
  577. dbt.Fatalf("LONGBLOB: length in: %d, length out: %d", len(in), len(out))
  578. }
  579. if rows.Next() {
  580. dbt.Error("LONGBLOB: unexpexted row")
  581. }
  582. } else {
  583. dbt.Fatalf("LONGBLOB: no data")
  584. }
  585. })
  586. }
  587. func TestLoadData(t *testing.T) {
  588. runTests(t, "TestLoadData", func(dbt *DBTest) {
  589. verifyLoadDataResult := func() {
  590. rows, err := dbt.db.Query("SELECT * FROM test")
  591. if err != nil {
  592. dbt.Fatal(err.Error())
  593. }
  594. i := 0
  595. values := [4]string{
  596. "a string",
  597. "a string containing a \t",
  598. "a string containing a \n",
  599. "a string containing both \t\n",
  600. }
  601. var id int
  602. var value string
  603. for rows.Next() {
  604. i++
  605. err = rows.Scan(&id, &value)
  606. if err != nil {
  607. dbt.Fatal(err.Error())
  608. }
  609. if i != id {
  610. dbt.Fatalf("%d != %d", i, id)
  611. }
  612. if values[i-1] != value {
  613. dbt.Fatalf("%s != %s", values[i-1], value)
  614. }
  615. }
  616. err = rows.Err()
  617. if err != nil {
  618. dbt.Fatal(err.Error())
  619. }
  620. if i != 4 {
  621. dbt.Fatalf("Rows count mismatch. Got %d, want 4", i)
  622. }
  623. }
  624. file, err := ioutil.TempFile("", "gotest")
  625. defer os.Remove(file.Name())
  626. if err != nil {
  627. dbt.Fatal(err)
  628. }
  629. 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")
  630. file.Close()
  631. dbt.mustExec("DROP TABLE IF EXISTS test")
  632. dbt.mustExec("CREATE TABLE test (id INT NOT NULL PRIMARY KEY, value TEXT NOT NULL) CHARACTER SET utf8")
  633. // Local File
  634. RegisterLocalFile(file.Name())
  635. dbt.mustExec(fmt.Sprintf("LOAD DATA LOCAL INFILE '%q' INTO TABLE test", file.Name()))
  636. verifyLoadDataResult()
  637. // negative test
  638. _, err = dbt.db.Exec("LOAD DATA LOCAL INFILE 'doesnotexist' INTO TABLE test")
  639. if err == nil {
  640. dbt.Fatal("Load non-existent file didn't fail")
  641. } else if err.Error() != "Local File 'doesnotexist' is not registered. Use the DSN parameter 'allowAllFiles=true' to allow all files" {
  642. dbt.Fatal(err.Error())
  643. }
  644. // Empty table
  645. dbt.mustExec("TRUNCATE TABLE test")
  646. // Reader
  647. RegisterReaderHandler("test", func() io.Reader {
  648. file, err = os.Open(file.Name())
  649. if err != nil {
  650. dbt.Fatal(err)
  651. }
  652. return file
  653. })
  654. dbt.mustExec("LOAD DATA LOCAL INFILE 'Reader::test' INTO TABLE test")
  655. verifyLoadDataResult()
  656. // negative test
  657. _, err = dbt.db.Exec("LOAD DATA LOCAL INFILE 'Reader::doesnotexist' INTO TABLE test")
  658. if err == nil {
  659. dbt.Fatal("Load non-existent Reader didn't fail")
  660. } else if err.Error() != "Reader 'doesnotexist' is not registered" {
  661. dbt.Fatal(err.Error())
  662. }
  663. })
  664. }
  665. // Special cases
  666. func TestRowsClose(t *testing.T) {
  667. runTests(t, "TestRowsClose", func(dbt *DBTest) {
  668. rows, err := dbt.db.Query("SELECT 1")
  669. if err != nil {
  670. dbt.Fatal(err)
  671. }
  672. err = rows.Close()
  673. if err != nil {
  674. dbt.Fatal(err)
  675. }
  676. if rows.Next() {
  677. dbt.Fatal("Unexpected row after rows.Close()")
  678. }
  679. err = rows.Err()
  680. if err != nil {
  681. dbt.Fatal(err)
  682. }
  683. })
  684. }
  685. // dangling statements
  686. // http://code.google.com/p/go/issues/detail?id=3865
  687. func TestCloseStmtBeforeRows(t *testing.T) {
  688. runTests(t, "TestCloseStmtBeforeRows", func(dbt *DBTest) {
  689. stmt, err := dbt.db.Prepare("SELECT 1")
  690. if err != nil {
  691. dbt.Fatal(err)
  692. }
  693. rows, err := stmt.Query()
  694. if err != nil {
  695. stmt.Close()
  696. dbt.Fatal(err)
  697. }
  698. defer rows.Close()
  699. err = stmt.Close()
  700. if err != nil {
  701. dbt.Fatal(err)
  702. }
  703. if !rows.Next() {
  704. dbt.Fatal("Getting row failed")
  705. } else {
  706. err = rows.Err()
  707. if err != nil {
  708. dbt.Fatal(err)
  709. }
  710. var out bool
  711. err = rows.Scan(&out)
  712. if err != nil {
  713. dbt.Fatalf("Error on rows.Scan(): %v", err)
  714. }
  715. if out != true {
  716. dbt.Errorf("true != %t", out)
  717. }
  718. }
  719. })
  720. }
  721. // It is valid to have multiple Rows for the same Stmt
  722. // http://code.google.com/p/go/issues/detail?id=3734
  723. func TestStmtMultiRows(t *testing.T) {
  724. runTests(t, "TestStmtMultiRows", func(dbt *DBTest) {
  725. stmt, err := dbt.db.Prepare("SELECT 1 UNION SELECT 0")
  726. if err != nil {
  727. dbt.Fatal(err)
  728. }
  729. rows1, err := stmt.Query()
  730. if err != nil {
  731. stmt.Close()
  732. dbt.Fatal(err)
  733. }
  734. defer rows1.Close()
  735. rows2, err := stmt.Query()
  736. if err != nil {
  737. stmt.Close()
  738. dbt.Fatal(err)
  739. }
  740. defer rows2.Close()
  741. var out bool
  742. // 1
  743. if !rows1.Next() {
  744. dbt.Fatal("1st rows1.Next failed")
  745. } else {
  746. err = rows1.Err()
  747. if err != nil {
  748. dbt.Fatal(err)
  749. }
  750. err = rows1.Scan(&out)
  751. if err != nil {
  752. dbt.Fatalf("Error on rows.Scan(): %v", err)
  753. }
  754. if out != true {
  755. dbt.Errorf("true != %t", out)
  756. }
  757. }
  758. if !rows2.Next() {
  759. dbt.Fatal("1st rows2.Next failed")
  760. } else {
  761. err = rows2.Err()
  762. if err != nil {
  763. dbt.Fatal(err)
  764. }
  765. err = rows2.Scan(&out)
  766. if err != nil {
  767. dbt.Fatalf("Error on rows.Scan(): %v", err)
  768. }
  769. if out != true {
  770. dbt.Errorf("true != %t", out)
  771. }
  772. }
  773. // 2
  774. if !rows1.Next() {
  775. dbt.Fatal("2nd rows1.Next failed")
  776. } else {
  777. err = rows1.Err()
  778. if err != nil {
  779. dbt.Fatal(err)
  780. }
  781. err = rows1.Scan(&out)
  782. if err != nil {
  783. dbt.Fatalf("Error on rows.Scan(): %v", err)
  784. }
  785. if out != false {
  786. dbt.Errorf("false != %t", out)
  787. }
  788. if rows1.Next() {
  789. dbt.Fatal("Unexpected row on rows1")
  790. }
  791. err = rows1.Close()
  792. if err != nil {
  793. dbt.Fatal(err)
  794. }
  795. }
  796. if !rows2.Next() {
  797. dbt.Fatal("2nd rows2.Next failed")
  798. } else {
  799. err = rows2.Err()
  800. if err != nil {
  801. dbt.Fatal(err)
  802. }
  803. err = rows2.Scan(&out)
  804. if err != nil {
  805. dbt.Fatalf("Error on rows.Scan(): %v", err)
  806. }
  807. if out != false {
  808. dbt.Errorf("false != %t", out)
  809. }
  810. if rows2.Next() {
  811. dbt.Fatal("Unexpected row on rows2")
  812. }
  813. err = rows2.Close()
  814. if err != nil {
  815. dbt.Fatal(err)
  816. }
  817. }
  818. })
  819. }
  820. func TestConcurrent(t *testing.T) {
  821. if os.Getenv("MYSQL_TEST_CONCURRENT") != "1" {
  822. t.Log("CONCURRENT env var not set. Skipping TestConcurrent")
  823. return
  824. }
  825. runTests(t, "TestConcurrent", func(dbt *DBTest) {
  826. var max int
  827. err := dbt.db.QueryRow("SELECT @@max_connections").Scan(&max)
  828. if err != nil {
  829. dbt.Fatalf("%v", err)
  830. }
  831. dbt.Logf("Testing up to %d concurrent connections \r\n", max)
  832. canStop := false
  833. c := make(chan struct{}, max)
  834. for i := 0; i < max; i++ {
  835. go func(id int) {
  836. tx, err := dbt.db.Begin()
  837. if err != nil {
  838. canStop = true
  839. if err.Error() == "Error 1040: Too many connections" {
  840. max--
  841. return
  842. } else {
  843. dbt.Fatalf("Error on Con %d: %s", id, err.Error())
  844. }
  845. }
  846. c <- struct{}{}
  847. for !canStop {
  848. _, err = tx.Exec("SELECT 1")
  849. if err != nil {
  850. canStop = true
  851. dbt.Fatalf("Error on Con %d: %s", id, err.Error())
  852. }
  853. }
  854. err = tx.Commit()
  855. if err != nil {
  856. canStop = true
  857. dbt.Fatalf("Error on Con %d: %s", id, err.Error())
  858. }
  859. }(i)
  860. }
  861. for i := 0; i < max; i++ {
  862. <-c
  863. }
  864. canStop = true
  865. dbt.Logf("Reached %d concurrent connections \r\n", max)
  866. })
  867. }