driver_test.go 24 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018
  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&strict=true&"+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.db.Exec("DROP TABLE IF EXISTS test")
  105. for _, test := range tests {
  106. test(dbt)
  107. dbt.db.Exec("DROP TABLE IF EXISTS test")
  108. }
  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("CREATE TABLE test (value " + sqltype + ")")
  397. for _, test := range tests {
  398. for mode, q := range modes {
  399. dbt.mustExec("TRUNCATE test")
  400. dbt.mustExec("INSERT INTO test VALUES (?)", test.in)
  401. rows = dbt.mustQuery("SELECT value FROM test"+q.selectSuffix, q.args...)
  402. if rows.Next() {
  403. s.test(dbt, rows, test, sqltype, s.vartype, mode)
  404. } else {
  405. if err := rows.Err(); err != nil {
  406. dbt.Errorf("%s (%s %s): %v",
  407. sqltype, s.vartype, mode, err)
  408. } else {
  409. dbt.Errorf("%s (%s %s): no data",
  410. sqltype, s.vartype, mode)
  411. }
  412. }
  413. }
  414. }
  415. dbt.mustExec("DROP TABLE IF EXISTS test")
  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.db.Exec("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. func TestStrict(t *testing.T) {
  663. runTests(t, "TestStrict", func(dbt *DBTest) {
  664. dbt.mustExec("CREATE TABLE test (a TINYINT NOT NULL, b CHAR(4))")
  665. queries := [...][2]string{
  666. {"DROP TABLE IF EXISTS no_such_table", "Note 1051: Unknown table 'no_such_table'"},
  667. {"INSERT INTO test VALUES(10,'mysql'),(NULL,'test'),(300,'Open Source')",
  668. "Warning 1265: Data truncated for column 'b' at row 1\r\n" +
  669. "Warning 1048: Column 'a' cannot be null\r\n" +
  670. "Warning 1264: Out of range value for column 'a' at row 3\r\n" +
  671. "Warning 1265: Data truncated for column 'b' at row 3",
  672. },
  673. }
  674. var err error
  675. // text protocol
  676. for i := range queries {
  677. _, err = dbt.db.Exec(queries[i][0])
  678. if err == nil {
  679. dbt.Errorf("Expecteded strict error on query [text] %s", queries[i][0])
  680. } else if err.Error() != queries[i][1] {
  681. dbt.Errorf("Unexpected error on query [text] %s: %s != %s", queries[i][0], err.Error(), queries[i][1])
  682. }
  683. }
  684. var stmt *sql.Stmt
  685. // binary protocol
  686. for i := range queries {
  687. stmt, err = dbt.db.Prepare(queries[i][0])
  688. if err != nil {
  689. dbt.Error("Error on preparing query %: ", queries[i][0], err.Error())
  690. }
  691. _, err = stmt.Exec()
  692. if err == nil {
  693. dbt.Errorf("Expecteded strict error on query [binary] %s", queries[i][0])
  694. } else if err.Error() != queries[i][1] {
  695. dbt.Errorf("Unexpected error on query [binary] %s: %s != %s", queries[i][0], err.Error(), queries[i][1])
  696. }
  697. err = stmt.Close()
  698. if err != nil {
  699. dbt.Error("Error on closing stmt for query %: ", queries[i][0], err.Error())
  700. }
  701. }
  702. })
  703. }
  704. // Special cases
  705. func TestRowsClose(t *testing.T) {
  706. runTests(t, "TestRowsClose", func(dbt *DBTest) {
  707. rows, err := dbt.db.Query("SELECT 1")
  708. if err != nil {
  709. dbt.Fatal(err)
  710. }
  711. err = rows.Close()
  712. if err != nil {
  713. dbt.Fatal(err)
  714. }
  715. if rows.Next() {
  716. dbt.Fatal("Unexpected row after rows.Close()")
  717. }
  718. err = rows.Err()
  719. if err != nil {
  720. dbt.Fatal(err)
  721. }
  722. })
  723. }
  724. // dangling statements
  725. // http://code.google.com/p/go/issues/detail?id=3865
  726. func TestCloseStmtBeforeRows(t *testing.T) {
  727. runTests(t, "TestCloseStmtBeforeRows", func(dbt *DBTest) {
  728. stmt, err := dbt.db.Prepare("SELECT 1")
  729. if err != nil {
  730. dbt.Fatal(err)
  731. }
  732. rows, err := stmt.Query()
  733. if err != nil {
  734. stmt.Close()
  735. dbt.Fatal(err)
  736. }
  737. defer rows.Close()
  738. err = stmt.Close()
  739. if err != nil {
  740. dbt.Fatal(err)
  741. }
  742. if !rows.Next() {
  743. dbt.Fatal("Getting row failed")
  744. } else {
  745. err = rows.Err()
  746. if err != nil {
  747. dbt.Fatal(err)
  748. }
  749. var out bool
  750. err = rows.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. })
  759. }
  760. // It is valid to have multiple Rows for the same Stmt
  761. // http://code.google.com/p/go/issues/detail?id=3734
  762. func TestStmtMultiRows(t *testing.T) {
  763. runTests(t, "TestStmtMultiRows", func(dbt *DBTest) {
  764. stmt, err := dbt.db.Prepare("SELECT 1 UNION SELECT 0")
  765. if err != nil {
  766. dbt.Fatal(err)
  767. }
  768. rows1, err := stmt.Query()
  769. if err != nil {
  770. stmt.Close()
  771. dbt.Fatal(err)
  772. }
  773. defer rows1.Close()
  774. rows2, err := stmt.Query()
  775. if err != nil {
  776. stmt.Close()
  777. dbt.Fatal(err)
  778. }
  779. defer rows2.Close()
  780. var out bool
  781. // 1
  782. if !rows1.Next() {
  783. dbt.Fatal("1st rows1.Next failed")
  784. } else {
  785. err = rows1.Err()
  786. if err != nil {
  787. dbt.Fatal(err)
  788. }
  789. err = rows1.Scan(&out)
  790. if err != nil {
  791. dbt.Fatalf("Error on rows.Scan(): %v", err)
  792. }
  793. if out != true {
  794. dbt.Errorf("true != %t", out)
  795. }
  796. }
  797. if !rows2.Next() {
  798. dbt.Fatal("1st rows2.Next failed")
  799. } else {
  800. err = rows2.Err()
  801. if err != nil {
  802. dbt.Fatal(err)
  803. }
  804. err = rows2.Scan(&out)
  805. if err != nil {
  806. dbt.Fatalf("Error on rows.Scan(): %v", err)
  807. }
  808. if out != true {
  809. dbt.Errorf("true != %t", out)
  810. }
  811. }
  812. // 2
  813. if !rows1.Next() {
  814. dbt.Fatal("2nd rows1.Next failed")
  815. } else {
  816. err = rows1.Err()
  817. if err != nil {
  818. dbt.Fatal(err)
  819. }
  820. err = rows1.Scan(&out)
  821. if err != nil {
  822. dbt.Fatalf("Error on rows.Scan(): %v", err)
  823. }
  824. if out != false {
  825. dbt.Errorf("false != %t", out)
  826. }
  827. if rows1.Next() {
  828. dbt.Fatal("Unexpected row on rows1")
  829. }
  830. err = rows1.Close()
  831. if err != nil {
  832. dbt.Fatal(err)
  833. }
  834. }
  835. if !rows2.Next() {
  836. dbt.Fatal("2nd rows2.Next failed")
  837. } else {
  838. err = rows2.Err()
  839. if err != nil {
  840. dbt.Fatal(err)
  841. }
  842. err = rows2.Scan(&out)
  843. if err != nil {
  844. dbt.Fatalf("Error on rows.Scan(): %v", err)
  845. }
  846. if out != false {
  847. dbt.Errorf("false != %t", out)
  848. }
  849. if rows2.Next() {
  850. dbt.Fatal("Unexpected row on rows2")
  851. }
  852. err = rows2.Close()
  853. if err != nil {
  854. dbt.Fatal(err)
  855. }
  856. }
  857. })
  858. }
  859. func TestConcurrent(t *testing.T) {
  860. if readBool(os.Getenv("MYSQL_TEST_CONCURRENT")) != true {
  861. t.Log("CONCURRENT env var not set. Skipping TestConcurrent")
  862. return
  863. }
  864. runTests(t, "TestConcurrent", func(dbt *DBTest) {
  865. var max int
  866. err := dbt.db.QueryRow("SELECT @@max_connections").Scan(&max)
  867. if err != nil {
  868. dbt.Fatalf("%v", err)
  869. }
  870. dbt.Logf("Testing up to %d concurrent connections \r\n", max)
  871. canStop := false
  872. c := make(chan struct{}, max)
  873. for i := 0; i < max; i++ {
  874. go func(id int) {
  875. tx, err := dbt.db.Begin()
  876. if err != nil {
  877. canStop = true
  878. if err.Error() == "Error 1040: Too many connections" {
  879. max--
  880. return
  881. } else {
  882. dbt.Fatalf("Error on Con %d: %s", id, err.Error())
  883. }
  884. }
  885. c <- struct{}{}
  886. for !canStop {
  887. _, err = tx.Exec("SELECT 1")
  888. if err != nil {
  889. canStop = true
  890. dbt.Fatalf("Error on Con %d: %s", id, err.Error())
  891. }
  892. }
  893. err = tx.Commit()
  894. if err != nil {
  895. canStop = true
  896. dbt.Fatalf("Error on Con %d: %s", id, err.Error())
  897. }
  898. }(i)
  899. }
  900. for i := 0; i < max; i++ {
  901. <-c
  902. }
  903. canStop = true
  904. dbt.Logf("Reached %d concurrent connections \r\n", max)
  905. })
  906. }