driver_go18_test.go 4.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190
  1. // +build go1.8
  2. package mysql
  3. import (
  4. "database/sql"
  5. "fmt"
  6. "reflect"
  7. "testing"
  8. )
  9. func TestMultiResultSet(t *testing.T) {
  10. type result struct {
  11. values [][]int
  12. columns []string
  13. }
  14. // checkRows is a helper test function to validate rows containing 3 result
  15. // sets with specific values and columns. The basic query would look like this:
  16. //
  17. // SELECT 1 AS col1, 2 AS col2 UNION SELECT 3, 4;
  18. // SELECT 0 UNION SELECT 1;
  19. // SELECT 1 AS col1, 2 AS col2, 3 AS col3 UNION SELECT 4, 5, 6;
  20. //
  21. // to distinguish test cases the first string argument is put in front of
  22. // every error or fatal message.
  23. checkRows := func(desc string, rows *sql.Rows, dbt *DBTest) {
  24. expected := []result{
  25. {
  26. values: [][]int{{1, 2}, {3, 4}},
  27. columns: []string{"col1", "col2"},
  28. },
  29. {
  30. values: [][]int{{1, 2, 3}, {4, 5, 6}},
  31. columns: []string{"col1", "col2", "col3"},
  32. },
  33. }
  34. var res1 result
  35. for rows.Next() {
  36. var res [2]int
  37. if err := rows.Scan(&res[0], &res[1]); err != nil {
  38. dbt.Fatal(err)
  39. }
  40. res1.values = append(res1.values, res[:])
  41. }
  42. cols, err := rows.Columns()
  43. if err != nil {
  44. dbt.Fatal(desc, err)
  45. }
  46. res1.columns = cols
  47. if !reflect.DeepEqual(expected[0], res1) {
  48. dbt.Error(desc, "want =", expected[0], "got =", res1)
  49. }
  50. if !rows.NextResultSet() {
  51. dbt.Fatal(desc, "expected next result set")
  52. }
  53. // ignoring one result set
  54. if !rows.NextResultSet() {
  55. dbt.Fatal(desc, "expected next result set")
  56. }
  57. var res2 result
  58. cols, err = rows.Columns()
  59. if err != nil {
  60. dbt.Fatal(desc, err)
  61. }
  62. res2.columns = cols
  63. for rows.Next() {
  64. var res [3]int
  65. if err := rows.Scan(&res[0], &res[1], &res[2]); err != nil {
  66. dbt.Fatal(desc, err)
  67. }
  68. res2.values = append(res2.values, res[:])
  69. }
  70. if !reflect.DeepEqual(expected[1], res2) {
  71. dbt.Error(desc, "want =", expected[1], "got =", res2)
  72. }
  73. if rows.NextResultSet() {
  74. dbt.Error(desc, "unexpected next result set")
  75. }
  76. if err := rows.Err(); err != nil {
  77. dbt.Error(desc, err)
  78. }
  79. }
  80. runTestsWithMultiStatement(t, dsn, func(dbt *DBTest) {
  81. rows := dbt.mustQuery(`DO 1;
  82. SELECT 1 AS col1, 2 AS col2 UNION SELECT 3, 4;
  83. DO 1;
  84. SELECT 0 UNION SELECT 1;
  85. SELECT 1 AS col1, 2 AS col2, 3 AS col3 UNION SELECT 4, 5, 6;`)
  86. defer rows.Close()
  87. checkRows("query: ", rows, dbt)
  88. })
  89. runTestsWithMultiStatement(t, dsn, func(dbt *DBTest) {
  90. queries := []string{
  91. `
  92. DROP PROCEDURE IF EXISTS test_mrss;
  93. CREATE PROCEDURE test_mrss()
  94. BEGIN
  95. DO 1;
  96. SELECT 1 AS col1, 2 AS col2 UNION SELECT 3, 4;
  97. DO 1;
  98. SELECT 0 UNION SELECT 1;
  99. SELECT 1 AS col1, 2 AS col2, 3 AS col3 UNION SELECT 4, 5, 6;
  100. END
  101. `,
  102. `
  103. DROP PROCEDURE IF EXISTS test_mrss;
  104. CREATE PROCEDURE test_mrss()
  105. BEGIN
  106. SELECT 1 AS col1, 2 AS col2 UNION SELECT 3, 4;
  107. SELECT 0 UNION SELECT 1;
  108. SELECT 1 AS col1, 2 AS col2, 3 AS col3 UNION SELECT 4, 5, 6;
  109. END
  110. `,
  111. }
  112. defer dbt.mustExec("DROP PROCEDURE IF EXISTS test_mrss")
  113. for i, query := range queries {
  114. dbt.mustExec(query)
  115. stmt, err := dbt.db.Prepare("CALL test_mrss()")
  116. if err != nil {
  117. dbt.Fatalf("%v (i=%d)", err, i)
  118. }
  119. defer stmt.Close()
  120. for j := 0; j < 2; j++ {
  121. rows, err := stmt.Query()
  122. if err != nil {
  123. dbt.Fatalf("%v (i=%d) (j=%d)", err, i, j)
  124. }
  125. checkRows(fmt.Sprintf("prepared stmt query (i=%d) (j=%d): ", i, j), rows, dbt)
  126. }
  127. }
  128. })
  129. }
  130. func TestMultiResultSetNoSelect(t *testing.T) {
  131. runTestsWithMultiStatement(t, dsn, func(dbt *DBTest) {
  132. rows := dbt.mustQuery("DO 1; DO 2;")
  133. defer rows.Close()
  134. if rows.Next() {
  135. dbt.Error("unexpected row")
  136. }
  137. if rows.NextResultSet() {
  138. dbt.Error("unexpected next result set")
  139. }
  140. if err := rows.Err(); err != nil {
  141. dbt.Error("expected nil; got ", err)
  142. }
  143. })
  144. }
  145. // tests if rows are set in a proper state if some results were ignored before
  146. // calling rows.NextResultSet.
  147. func TestSkipResults(t *testing.T) {
  148. runTests(t, dsn, func(dbt *DBTest) {
  149. rows := dbt.mustQuery("SELECT 1, 2")
  150. defer rows.Close()
  151. if !rows.Next() {
  152. dbt.Error("expected row")
  153. }
  154. if rows.NextResultSet() {
  155. dbt.Error("unexpected next result set")
  156. }
  157. if err := rows.Err(); err != nil {
  158. dbt.Error("expected nil; got ", err)
  159. }
  160. })
  161. }