driver_test.go 23 KB

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