cassandra_test.go 8.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306
  1. // Copyright (c) 2012 The gocql Authors. All rights reserved.
  2. // Use of this source code is governed by a BSD-style
  3. // license that can be found in the LICENSE file.
  4. package gocql
  5. import (
  6. "bytes"
  7. "flag"
  8. "reflect"
  9. "sort"
  10. "strings"
  11. "sync"
  12. "testing"
  13. "time"
  14. )
  15. var (
  16. flagCluster = flag.String("cluster", "127.0.0.1", "a comma-separated list of host:port tuples")
  17. flagProto = flag.Int("proto", 2, "protcol version")
  18. flagCQL = flag.String("cql", "3.0.0", "CQL version")
  19. )
  20. var initOnce sync.Once
  21. func createSession(t *testing.T) *Session {
  22. cluster := NewCluster(strings.Split(*flagCluster, ",")...)
  23. cluster.ProtoVersion = *flagProto
  24. cluster.CQLVersion = *flagCQL
  25. cluster.Authenticator = PasswordAuthenticator{
  26. Username: "cassandra",
  27. Password: "cassandra",
  28. }
  29. session, err := cluster.CreateSession()
  30. if err != nil {
  31. t.Fatal("createSession:", err)
  32. }
  33. initOnce.Do(func() {
  34. // Drop and re-create the keyspace once. Different tests should use their own
  35. // individual tables, but can assume that the table does not exist before.
  36. if err := session.Query(`DROP KEYSPACE gocql_test`).Exec(); err != nil {
  37. t.Log("drop keyspace:", err)
  38. }
  39. if err := session.Query(`CREATE KEYSPACE gocql_test
  40. WITH replication = {
  41. 'class' : 'SimpleStrategy',
  42. 'replication_factor' : 1
  43. }`).Exec(); err != nil {
  44. t.Fatal("create keyspace:", err)
  45. }
  46. })
  47. if err := session.Query(`USE gocql_test`).Exec(); err != nil {
  48. t.Fatal("createSession:", err)
  49. }
  50. return session
  51. }
  52. func TestEmptyHosts(t *testing.T) {
  53. cluster := NewCluster()
  54. if session, err := cluster.CreateSession(); err == nil {
  55. session.Close()
  56. t.Error("expected err, got nil")
  57. }
  58. }
  59. func TestCRUD(t *testing.T) {
  60. session := createSession(t)
  61. defer session.Close()
  62. if err := session.Query(`CREATE TABLE page (
  63. title varchar,
  64. revid timeuuid,
  65. body varchar,
  66. views bigint,
  67. protected boolean,
  68. modified timestamp,
  69. tags set<varchar>,
  70. attachments map<varchar, text>,
  71. PRIMARY KEY (title, revid)
  72. )`).Exec(); err != nil {
  73. t.Fatal("create table:", err)
  74. }
  75. for _, page := range pageTestData {
  76. if err := session.Query(`INSERT INTO page
  77. (title, revid, body, views, protected, modified, tags, attachments)
  78. VALUES (?, ?, ?, ?, ?, ?, ?, ?)`,
  79. page.Title, page.RevId, page.Body, page.Views, page.Protected,
  80. page.Modified, page.Tags, page.Attachments).Exec(); err != nil {
  81. t.Fatal("insert:", err)
  82. }
  83. }
  84. var count int
  85. if err := session.Query("SELECT COUNT(*) FROM page").Scan(&count); err != nil {
  86. t.Error("select count:", err)
  87. }
  88. if count != len(pageTestData) {
  89. t.Errorf("count: expected %d, got %d\n", len(pageTestData), count)
  90. }
  91. for _, original := range pageTestData {
  92. page := new(Page)
  93. err := session.Query(`SELECT title, revid, body, views, protected, modified,
  94. tags, attachments
  95. FROM page WHERE title = ? AND revid = ? LIMIT 1`,
  96. original.Title, original.RevId).Scan(&page.Title, &page.RevId,
  97. &page.Body, &page.Views, &page.Protected, &page.Modified, &page.Tags,
  98. &page.Attachments)
  99. if err != nil {
  100. t.Error("select page:", err)
  101. continue
  102. }
  103. sort.Sort(sort.StringSlice(page.Tags))
  104. sort.Sort(sort.StringSlice(original.Tags))
  105. if !reflect.DeepEqual(page, original) {
  106. t.Errorf("page: expected %#v, got %#v\n", original, page)
  107. }
  108. }
  109. }
  110. func TestTracing(t *testing.T) {
  111. session := createSession(t)
  112. defer session.Close()
  113. if err := session.Query(`CREATE TABLE trace (id int primary key)`).Exec(); err != nil {
  114. t.Fatal("create:", err)
  115. }
  116. buf := &bytes.Buffer{}
  117. trace := NewTraceWriter(session, buf)
  118. if err := session.Query(`INSERT INTO trace (id) VALUES (?)`, 42).Trace(trace).Exec(); err != nil {
  119. t.Error("insert:", err)
  120. } else if buf.Len() == 0 {
  121. t.Error("insert: failed to obtain any tracing")
  122. }
  123. buf.Reset()
  124. var value int
  125. if err := session.Query(`SELECT id FROM trace WHERE id = ?`, 42).Trace(trace).Scan(&value); err != nil {
  126. t.Error("select:", err)
  127. } else if value != 42 {
  128. t.Errorf("value: expected %d, got %d", 42, value)
  129. } else if buf.Len() == 0 {
  130. t.Error("select: failed to obtain any tracing")
  131. }
  132. }
  133. func TestPaging(t *testing.T) {
  134. if *flagProto == 1 {
  135. t.Skip("Paging not supported. Please use Cassandra >= 2.0")
  136. }
  137. session := createSession(t)
  138. defer session.Close()
  139. if err := session.Query("CREATE TABLE large (id int primary key)").Exec(); err != nil {
  140. t.Fatal("create table:", err)
  141. }
  142. for i := 0; i < 100; i++ {
  143. if err := session.Query("INSERT INTO large (id) VALUES (?)", i).Exec(); err != nil {
  144. t.Fatal("insert:", err)
  145. }
  146. }
  147. iter := session.Query("SELECT id FROM large").PageSize(10).Iter()
  148. var id int
  149. count := 0
  150. for iter.Scan(&id) {
  151. count++
  152. }
  153. if err := iter.Close(); err != nil {
  154. t.Fatal("close:", err)
  155. }
  156. if count != 100 {
  157. t.Fatalf("expected %d, got %d", 100, count)
  158. }
  159. }
  160. func TestCAS(t *testing.T) {
  161. if *flagProto == 1 {
  162. t.Skip("lightweight transactions not supported. Please use Cassandra >= 2.0")
  163. }
  164. session := createSession(t)
  165. defer session.Close()
  166. if err := session.Query(`CREATE TABLE cas_table (
  167. title varchar,
  168. revid timeuuid,
  169. PRIMARY KEY (title, revid)
  170. )`).Exec(); err != nil {
  171. t.Fatal("create:", err)
  172. }
  173. title, revid := "baz", TimeUUID()
  174. var titleCAS string
  175. var revidCAS UUID
  176. if applied, err := session.Query(`INSERT INTO cas_table (title, revid)
  177. VALUES (?, ?) IF NOT EXISTS`,
  178. title, revid).ScanCAS(&titleCAS, &revidCAS); err != nil {
  179. t.Fatal("insert:", err)
  180. } else if !applied {
  181. t.Fatal("insert should have been applied")
  182. }
  183. if applied, err := session.Query(`INSERT INTO cas_table (title, revid)
  184. VALUES (?, ?) IF NOT EXISTS`,
  185. title, revid).ScanCAS(&titleCAS, &revidCAS); err != nil {
  186. t.Fatal("insert:", err)
  187. } else if applied {
  188. t.Fatal("insert should not have been applied")
  189. } else if title != titleCAS || revid != revidCAS {
  190. t.Fatalf("expected %s/%v but got %s/%v", title, revid, titleCAS, revidCAS)
  191. }
  192. }
  193. func TestBatch(t *testing.T) {
  194. if *flagProto == 1 {
  195. t.Skip("atomic batches not supported. Please use Cassandra >= 2.0")
  196. }
  197. session := createSession(t)
  198. defer session.Close()
  199. if err := session.Query(`CREATE TABLE batch_table (id int primary key)`).Exec(); err != nil {
  200. t.Fatal("create table:", err)
  201. }
  202. batch := NewBatch(LoggedBatch)
  203. for i := 0; i < 100; i++ {
  204. batch.Query(`INSERT INTO batch_table (id) VALUES (?)`, i)
  205. }
  206. if err := session.ExecuteBatch(batch); err != nil {
  207. t.Fatal("execute batch:", err)
  208. }
  209. count := 0
  210. if err := session.Query(`SELECT COUNT(*) FROM batch_table`).Scan(&count); err != nil {
  211. t.Fatal("select count:", err)
  212. } else if count != 100 {
  213. t.Fatalf("count: expected %d, got %d\n", 100, count)
  214. }
  215. }
  216. // TestBatchLimit tests gocql to make sure batch operations larger than the maximum
  217. // statement limit are not submitted to a cassandra node.
  218. func TestBatchLimit(t *testing.T) {
  219. if *flagProto == 1 {
  220. t.Skip("atomic batches not supported. Please use Cassandra >= 2.0")
  221. }
  222. session := createSession(t)
  223. defer session.Close()
  224. if err := session.Query(`CREATE TABLE batch_table2 (id int primary key)`).Exec(); err != nil {
  225. t.Fatal("create table:", err)
  226. }
  227. batch := NewBatch(LoggedBatch)
  228. for i := 0; i < 65537; i++ {
  229. batch.Query(`INSERT INTO batch_table2 (id) VALUES (?)`, i)
  230. }
  231. if err := session.ExecuteBatch(batch); err == nil {
  232. t.Fatal("gocql attempted to execute a batch larger than the support limit of statements.")
  233. }
  234. }
  235. type Page struct {
  236. Title string
  237. RevId UUID
  238. Body string
  239. Views int64
  240. Protected bool
  241. Modified time.Time
  242. Tags []string
  243. Attachments map[string]Attachment
  244. }
  245. type Attachment []byte
  246. var pageTestData = []*Page{
  247. &Page{
  248. Title: "Frontpage",
  249. RevId: TimeUUID(),
  250. Body: "Welcome to this wiki page!",
  251. Modified: time.Date(2013, time.August, 13, 9, 52, 3, 0, time.UTC),
  252. Tags: []string{"start", "important", "test"},
  253. Attachments: map[string]Attachment{
  254. "logo": Attachment("\x00company logo\x00"),
  255. "favicon": Attachment("favicon.ico"),
  256. },
  257. },
  258. &Page{
  259. Title: "Foobar",
  260. RevId: TimeUUID(),
  261. Body: "foo::Foo f = new foo::Foo(foo::Foo::INIT);",
  262. Modified: time.Date(2013, time.August, 13, 9, 52, 3, 0, time.UTC),
  263. },
  264. }