bulk_test.go 16 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507
  1. // Copyright 2012-present Oliver Eilhard. All rights reserved.
  2. // Use of this source code is governed by a MIT-license.
  3. // See http://olivere.mit-license.org/license.txt for details.
  4. package elastic
  5. import (
  6. "context"
  7. "encoding/json"
  8. "testing"
  9. )
  10. func TestBulk(t *testing.T) {
  11. client := setupTestClientAndCreateIndex(t) //, SetTraceLog(log.New(os.Stdout, "", 0)))
  12. tweet1 := tweet{User: "olivere", Message: "Welcome to Golang and Elasticsearch."}
  13. tweet2 := tweet{User: "sandrae", Message: "Dancing all night long. Yeah."}
  14. index1Req := NewBulkIndexRequest().Index(testIndexName).Type("tweet").Id("1").Doc(tweet1)
  15. index2Req := NewBulkIndexRequest().Index(testIndexName).Type("tweet").Id("2").Doc(tweet2)
  16. delete1Req := NewBulkDeleteRequest().Index(testIndexName).Type("tweet").Id("1")
  17. bulkRequest := client.Bulk()
  18. bulkRequest = bulkRequest.Add(index1Req)
  19. bulkRequest = bulkRequest.Add(index2Req)
  20. bulkRequest = bulkRequest.Add(delete1Req)
  21. if bulkRequest.NumberOfActions() != 3 {
  22. t.Errorf("expected bulkRequest.NumberOfActions %d; got %d", 3, bulkRequest.NumberOfActions())
  23. }
  24. bulkResponse, err := bulkRequest.Do(context.TODO())
  25. if err != nil {
  26. t.Fatal(err)
  27. }
  28. if bulkResponse == nil {
  29. t.Errorf("expected bulkResponse to be != nil; got nil")
  30. }
  31. if bulkRequest.NumberOfActions() != 0 {
  32. t.Errorf("expected bulkRequest.NumberOfActions %d; got %d", 0, bulkRequest.NumberOfActions())
  33. }
  34. // Document with Id="1" should not exist
  35. exists, err := client.Exists().Index(testIndexName).Type("tweet").Id("1").Do(context.TODO())
  36. if err != nil {
  37. t.Fatal(err)
  38. }
  39. if exists {
  40. t.Errorf("expected exists %v; got %v", false, exists)
  41. }
  42. // Document with Id="2" should exist
  43. exists, err = client.Exists().Index(testIndexName).Type("tweet").Id("2").Do(context.TODO())
  44. if err != nil {
  45. t.Fatal(err)
  46. }
  47. if !exists {
  48. t.Errorf("expected exists %v; got %v", true, exists)
  49. }
  50. // Update
  51. updateDoc := struct {
  52. Retweets int `json:"retweets"`
  53. }{
  54. 42,
  55. }
  56. update1Req := NewBulkUpdateRequest().Index(testIndexName).Type("tweet").Id("2").Doc(&updateDoc)
  57. bulkRequest = client.Bulk()
  58. bulkRequest = bulkRequest.Add(update1Req)
  59. if bulkRequest.NumberOfActions() != 1 {
  60. t.Errorf("expected bulkRequest.NumberOfActions %d; got %d", 1, bulkRequest.NumberOfActions())
  61. }
  62. bulkResponse, err = bulkRequest.Do(context.TODO())
  63. if err != nil {
  64. t.Fatal(err)
  65. }
  66. if bulkResponse == nil {
  67. t.Errorf("expected bulkResponse to be != nil; got nil")
  68. }
  69. if bulkRequest.NumberOfActions() != 0 {
  70. t.Errorf("expected bulkRequest.NumberOfActions %d; got %d", 0, bulkRequest.NumberOfActions())
  71. }
  72. // Document with Id="1" should have a retweets count of 42
  73. doc, err := client.Get().Index(testIndexName).Type("tweet").Id("2").Do(context.TODO())
  74. if err != nil {
  75. t.Fatal(err)
  76. }
  77. if doc == nil {
  78. t.Fatal("expected doc to be != nil; got nil")
  79. }
  80. if !doc.Found {
  81. t.Fatalf("expected doc to be found; got found = %v", doc.Found)
  82. }
  83. if doc.Source == nil {
  84. t.Fatal("expected doc source to be != nil; got nil")
  85. }
  86. var updatedTweet tweet
  87. err = json.Unmarshal(*doc.Source, &updatedTweet)
  88. if err != nil {
  89. t.Fatal(err)
  90. }
  91. if updatedTweet.Retweets != 42 {
  92. t.Errorf("expected updated tweet retweets = %v; got %v", 42, updatedTweet.Retweets)
  93. }
  94. // Update with script
  95. update2Req := NewBulkUpdateRequest().Index(testIndexName).Type("tweet").Id("2").
  96. RetryOnConflict(3).
  97. Script(NewScript("ctx._source.retweets += params.v").Param("v", 1))
  98. bulkRequest = client.Bulk()
  99. bulkRequest = bulkRequest.Add(update2Req)
  100. if bulkRequest.NumberOfActions() != 1 {
  101. t.Errorf("expected bulkRequest.NumberOfActions %d; got %d", 1, bulkRequest.NumberOfActions())
  102. }
  103. bulkResponse, err = bulkRequest.Refresh("wait_for").Do(context.TODO())
  104. if err != nil {
  105. t.Fatal(err)
  106. }
  107. if bulkResponse == nil {
  108. t.Errorf("expected bulkResponse to be != nil; got nil")
  109. }
  110. if bulkRequest.NumberOfActions() != 0 {
  111. t.Errorf("expected bulkRequest.NumberOfActions %d; got %d", 0, bulkRequest.NumberOfActions())
  112. }
  113. // Document with Id="1" should have a retweets count of 43
  114. doc, err = client.Get().Index(testIndexName).Type("tweet").Id("2").Do(context.TODO())
  115. if err != nil {
  116. t.Fatal(err)
  117. }
  118. if doc == nil {
  119. t.Fatal("expected doc to be != nil; got nil")
  120. }
  121. if !doc.Found {
  122. t.Fatalf("expected doc to be found; got found = %v", doc.Found)
  123. }
  124. if doc.Source == nil {
  125. t.Fatal("expected doc source to be != nil; got nil")
  126. }
  127. err = json.Unmarshal(*doc.Source, &updatedTweet)
  128. if err != nil {
  129. t.Fatal(err)
  130. }
  131. if updatedTweet.Retweets != 43 {
  132. t.Errorf("expected updated tweet retweets = %v; got %v", 43, updatedTweet.Retweets)
  133. }
  134. }
  135. func TestBulkWithIndexSetOnClient(t *testing.T) {
  136. client := setupTestClientAndCreateIndex(t)
  137. tweet1 := tweet{User: "olivere", Message: "Welcome to Golang and Elasticsearch."}
  138. tweet2 := tweet{User: "sandrae", Message: "Dancing all night long. Yeah."}
  139. index1Req := NewBulkIndexRequest().Index(testIndexName).Type("tweet").Id("1").Doc(tweet1)
  140. index2Req := NewBulkIndexRequest().Index(testIndexName).Type("tweet").Id("2").Doc(tweet2)
  141. delete1Req := NewBulkDeleteRequest().Index(testIndexName).Type("tweet").Id("1")
  142. bulkRequest := client.Bulk().Index(testIndexName).Type("tweet")
  143. bulkRequest = bulkRequest.Add(index1Req)
  144. bulkRequest = bulkRequest.Add(index2Req)
  145. bulkRequest = bulkRequest.Add(delete1Req)
  146. if bulkRequest.NumberOfActions() != 3 {
  147. t.Errorf("expected bulkRequest.NumberOfActions %d; got %d", 3, bulkRequest.NumberOfActions())
  148. }
  149. bulkResponse, err := bulkRequest.Do(context.TODO())
  150. if err != nil {
  151. t.Fatal(err)
  152. }
  153. if bulkResponse == nil {
  154. t.Errorf("expected bulkResponse to be != nil; got nil")
  155. }
  156. // Document with Id="1" should not exist
  157. exists, err := client.Exists().Index(testIndexName).Type("tweet").Id("1").Do(context.TODO())
  158. if err != nil {
  159. t.Fatal(err)
  160. }
  161. if exists {
  162. t.Errorf("expected exists %v; got %v", false, exists)
  163. }
  164. // Document with Id="2" should exist
  165. exists, err = client.Exists().Index(testIndexName).Type("tweet").Id("2").Do(context.TODO())
  166. if err != nil {
  167. t.Fatal(err)
  168. }
  169. if !exists {
  170. t.Errorf("expected exists %v; got %v", true, exists)
  171. }
  172. }
  173. func TestBulkRequestsSerialization(t *testing.T) {
  174. client := setupTestClientAndCreateIndex(t)
  175. tweet1 := tweet{User: "olivere", Message: "Welcome to Golang and Elasticsearch."}
  176. tweet2 := tweet{User: "sandrae", Message: "Dancing all night long. Yeah."}
  177. index1Req := NewBulkIndexRequest().Index(testIndexName).Type("tweet").Id("1").Doc(tweet1)
  178. index2Req := NewBulkIndexRequest().OpType("create").Index(testIndexName).Type("tweet").Id("2").Doc(tweet2)
  179. delete1Req := NewBulkDeleteRequest().Index(testIndexName).Type("tweet").Id("1")
  180. update2Req := NewBulkUpdateRequest().Index(testIndexName).Type("tweet").Id("2").
  181. Doc(struct {
  182. Retweets int `json:"retweets"`
  183. }{
  184. Retweets: 42,
  185. })
  186. bulkRequest := client.Bulk()
  187. bulkRequest = bulkRequest.Add(index1Req)
  188. bulkRequest = bulkRequest.Add(index2Req)
  189. bulkRequest = bulkRequest.Add(delete1Req)
  190. bulkRequest = bulkRequest.Add(update2Req)
  191. if bulkRequest.NumberOfActions() != 4 {
  192. t.Errorf("expected bulkRequest.NumberOfActions %d; got %d", 4, bulkRequest.NumberOfActions())
  193. }
  194. expected := `{"index":{"_id":"1","_index":"` + testIndexName + `","_type":"tweet"}}
  195. {"user":"olivere","message":"Welcome to Golang and Elasticsearch.","retweets":0,"created":"0001-01-01T00:00:00Z"}
  196. {"create":{"_id":"2","_index":"` + testIndexName + `","_type":"tweet"}}
  197. {"user":"sandrae","message":"Dancing all night long. Yeah.","retweets":0,"created":"0001-01-01T00:00:00Z"}
  198. {"delete":{"_id":"1","_index":"` + testIndexName + `","_type":"tweet"}}
  199. {"update":{"_id":"2","_index":"` + testIndexName + `","_type":"tweet"}}
  200. {"doc":{"retweets":42}}
  201. `
  202. got, err := bulkRequest.bodyAsString()
  203. if err != nil {
  204. t.Fatalf("expected no error, got: %v", err)
  205. }
  206. if got != expected {
  207. t.Errorf("expected\n%s\ngot:\n%s", expected, got)
  208. }
  209. // Run the bulk request
  210. bulkResponse, err := bulkRequest.Do(context.TODO())
  211. if err != nil {
  212. t.Fatal(err)
  213. }
  214. if bulkResponse == nil {
  215. t.Errorf("expected bulkResponse to be != nil; got nil")
  216. }
  217. if bulkResponse.Took == 0 {
  218. t.Errorf("expected took to be > 0; got %d", bulkResponse.Took)
  219. }
  220. if bulkResponse.Errors {
  221. t.Errorf("expected errors to be %v; got %v", false, bulkResponse.Errors)
  222. }
  223. if len(bulkResponse.Items) != 4 {
  224. t.Fatalf("expected 4 result items; got %d", len(bulkResponse.Items))
  225. }
  226. // Indexed actions
  227. indexed := bulkResponse.Indexed()
  228. if indexed == nil {
  229. t.Fatal("expected indexed to be != nil; got nil")
  230. }
  231. if len(indexed) != 1 {
  232. t.Fatalf("expected len(indexed) == %d; got %d", 1, len(indexed))
  233. }
  234. if indexed[0].Id != "1" {
  235. t.Errorf("expected indexed[0].Id == %s; got %s", "1", indexed[0].Id)
  236. }
  237. if indexed[0].Status != 201 {
  238. t.Errorf("expected indexed[0].Status == %d; got %d", 201, indexed[0].Status)
  239. }
  240. // Created actions
  241. created := bulkResponse.Created()
  242. if created == nil {
  243. t.Fatal("expected created to be != nil; got nil")
  244. }
  245. if len(created) != 1 {
  246. t.Fatalf("expected len(created) == %d; got %d", 1, len(created))
  247. }
  248. if created[0].Id != "2" {
  249. t.Errorf("expected created[0].Id == %s; got %s", "2", created[0].Id)
  250. }
  251. if created[0].Status != 201 {
  252. t.Errorf("expected created[0].Status == %d; got %d", 201, created[0].Status)
  253. }
  254. // Deleted actions
  255. deleted := bulkResponse.Deleted()
  256. if deleted == nil {
  257. t.Fatal("expected deleted to be != nil; got nil")
  258. }
  259. if len(deleted) != 1 {
  260. t.Fatalf("expected len(deleted) == %d; got %d", 1, len(deleted))
  261. }
  262. if deleted[0].Id != "1" {
  263. t.Errorf("expected deleted[0].Id == %s; got %s", "1", deleted[0].Id)
  264. }
  265. if deleted[0].Status != 200 {
  266. t.Errorf("expected deleted[0].Status == %d; got %d", 200, deleted[0].Status)
  267. }
  268. if !deleted[0].Found {
  269. t.Errorf("expected deleted[0].Found == %v; got %v", true, deleted[0].Found)
  270. }
  271. // Updated actions
  272. updated := bulkResponse.Updated()
  273. if updated == nil {
  274. t.Fatal("expected updated to be != nil; got nil")
  275. }
  276. if len(updated) != 1 {
  277. t.Fatalf("expected len(updated) == %d; got %d", 1, len(updated))
  278. }
  279. if updated[0].Id != "2" {
  280. t.Errorf("expected updated[0].Id == %s; got %s", "2", updated[0].Id)
  281. }
  282. if updated[0].Status != 200 {
  283. t.Errorf("expected updated[0].Status == %d; got %d", 200, updated[0].Status)
  284. }
  285. if updated[0].Version != 2 {
  286. t.Errorf("expected updated[0].Version == %d; got %d", 2, updated[0].Version)
  287. }
  288. // Succeeded actions
  289. succeeded := bulkResponse.Succeeded()
  290. if succeeded == nil {
  291. t.Fatal("expected succeeded to be != nil; got nil")
  292. }
  293. if len(succeeded) != 4 {
  294. t.Fatalf("expected len(succeeded) == %d; got %d", 4, len(succeeded))
  295. }
  296. // ById
  297. id1Results := bulkResponse.ById("1")
  298. if id1Results == nil {
  299. t.Fatal("expected id1Results to be != nil; got nil")
  300. }
  301. if len(id1Results) != 2 {
  302. t.Fatalf("expected len(id1Results) == %d; got %d", 2, len(id1Results))
  303. }
  304. if id1Results[0].Id != "1" {
  305. t.Errorf("expected id1Results[0].Id == %s; got %s", "1", id1Results[0].Id)
  306. }
  307. if id1Results[0].Status != 201 {
  308. t.Errorf("expected id1Results[0].Status == %d; got %d", 201, id1Results[0].Status)
  309. }
  310. if id1Results[0].Version != 1 {
  311. t.Errorf("expected id1Results[0].Version == %d; got %d", 1, id1Results[0].Version)
  312. }
  313. if id1Results[1].Id != "1" {
  314. t.Errorf("expected id1Results[1].Id == %s; got %s", "1", id1Results[1].Id)
  315. }
  316. if id1Results[1].Status != 200 {
  317. t.Errorf("expected id1Results[1].Status == %d; got %d", 200, id1Results[1].Status)
  318. }
  319. if id1Results[1].Version != 2 {
  320. t.Errorf("expected id1Results[1].Version == %d; got %d", 2, id1Results[1].Version)
  321. }
  322. }
  323. func TestFailedBulkRequests(t *testing.T) {
  324. js := `{
  325. "took" : 2,
  326. "errors" : true,
  327. "items" : [ {
  328. "index" : {
  329. "_index" : "elastic-test",
  330. "_type" : "tweet",
  331. "_id" : "1",
  332. "_version" : 1,
  333. "status" : 201
  334. }
  335. }, {
  336. "create" : {
  337. "_index" : "elastic-test",
  338. "_type" : "tweet",
  339. "_id" : "2",
  340. "_version" : 1,
  341. "status" : 423,
  342. "error" : {
  343. "type":"routing_missing_exception",
  344. "reason":"routing is required for [elastic-test2]/[comment]/[1]"
  345. }
  346. }
  347. }, {
  348. "delete" : {
  349. "_index" : "elastic-test",
  350. "_type" : "tweet",
  351. "_id" : "1",
  352. "_version" : 2,
  353. "status" : 404,
  354. "found" : false
  355. }
  356. }, {
  357. "update" : {
  358. "_index" : "elastic-test",
  359. "_type" : "tweet",
  360. "_id" : "2",
  361. "_version" : 2,
  362. "status" : 200
  363. }
  364. } ]
  365. }`
  366. var resp BulkResponse
  367. err := json.Unmarshal([]byte(js), &resp)
  368. if err != nil {
  369. t.Fatal(err)
  370. }
  371. failed := resp.Failed()
  372. if len(failed) != 2 {
  373. t.Errorf("expected %d failed items; got: %d", 2, len(failed))
  374. }
  375. }
  376. func TestBulkEstimatedSizeInBytes(t *testing.T) {
  377. client := setupTestClientAndCreateIndex(t)
  378. tweet1 := tweet{User: "olivere", Message: "Welcome to Golang and Elasticsearch."}
  379. tweet2 := tweet{User: "sandrae", Message: "Dancing all night long. Yeah."}
  380. index1Req := NewBulkIndexRequest().Index(testIndexName).Type("tweet").Id("1").Doc(tweet1)
  381. index2Req := NewBulkIndexRequest().OpType("create").Index(testIndexName).Type("tweet").Id("2").Doc(tweet2)
  382. delete1Req := NewBulkDeleteRequest().Index(testIndexName).Type("tweet").Id("1")
  383. update2Req := NewBulkUpdateRequest().Index(testIndexName).Type("tweet").Id("2").
  384. Doc(struct {
  385. Retweets int `json:"retweets"`
  386. }{
  387. Retweets: 42,
  388. })
  389. bulkRequest := client.Bulk()
  390. bulkRequest = bulkRequest.Add(index1Req)
  391. bulkRequest = bulkRequest.Add(index2Req)
  392. bulkRequest = bulkRequest.Add(delete1Req)
  393. bulkRequest = bulkRequest.Add(update2Req)
  394. if bulkRequest.NumberOfActions() != 4 {
  395. t.Errorf("expected bulkRequest.NumberOfActions %d; got %d", 4, bulkRequest.NumberOfActions())
  396. }
  397. // The estimated size of the bulk request in bytes must be at least
  398. // the length of the body request.
  399. raw, err := bulkRequest.bodyAsString()
  400. if err != nil {
  401. t.Fatal(err)
  402. }
  403. rawlen := int64(len([]byte(raw)))
  404. if got, want := bulkRequest.EstimatedSizeInBytes(), rawlen; got < want {
  405. t.Errorf("expected an EstimatedSizeInBytes = %d; got: %v", want, got)
  406. }
  407. // Reset should also reset the calculated estimated byte size
  408. bulkRequest.reset()
  409. if got, want := bulkRequest.EstimatedSizeInBytes(), int64(0); got != want {
  410. t.Errorf("expected an EstimatedSizeInBytes = %d; got: %v", want, got)
  411. }
  412. }
  413. func TestBulkEstimateSizeInBytesLength(t *testing.T) {
  414. client := setupTestClientAndCreateIndex(t)
  415. s := client.Bulk()
  416. r := NewBulkDeleteRequest().Index(testIndexName).Type("tweet").Id("1")
  417. s = s.Add(r)
  418. if got, want := s.estimateSizeInBytes(r), int64(1+len(r.String())); got != want {
  419. t.Fatalf("expected %d; got: %d", want, got)
  420. }
  421. }
  422. var benchmarkBulkEstimatedSizeInBytes int64
  423. func BenchmarkBulkEstimatedSizeInBytesWith1Request(b *testing.B) {
  424. client := setupTestClientAndCreateIndex(b)
  425. s := client.Bulk()
  426. var result int64
  427. for n := 0; n < b.N; n++ {
  428. s = s.Add(NewBulkIndexRequest().Index(testIndexName).Type("tweet").Id("1").Doc(struct{ A string }{"1"}))
  429. s = s.Add(NewBulkUpdateRequest().Index(testIndexName).Type("tweet").Id("1").Doc(struct{ A string }{"2"}))
  430. s = s.Add(NewBulkDeleteRequest().Index(testIndexName).Type("tweet").Id("1"))
  431. result = s.EstimatedSizeInBytes()
  432. s.reset()
  433. }
  434. b.ReportAllocs()
  435. benchmarkBulkEstimatedSizeInBytes = result // ensure the compiler doesn't optimize
  436. }
  437. func BenchmarkBulkEstimatedSizeInBytesWith100Requests(b *testing.B) {
  438. client := setupTestClientAndCreateIndex(b)
  439. s := client.Bulk()
  440. var result int64
  441. for n := 0; n < b.N; n++ {
  442. for i := 0; i < 100; i++ {
  443. s = s.Add(NewBulkIndexRequest().Index(testIndexName).Type("tweet").Id("1").Doc(struct{ A string }{"1"}))
  444. s = s.Add(NewBulkUpdateRequest().Index(testIndexName).Type("tweet").Id("1").Doc(struct{ A string }{"2"}))
  445. s = s.Add(NewBulkDeleteRequest().Index(testIndexName).Type("tweet").Id("1"))
  446. }
  447. result = s.EstimatedSizeInBytes()
  448. s.reset()
  449. }
  450. b.ReportAllocs()
  451. benchmarkBulkEstimatedSizeInBytes = result // ensure the compiler doesn't optimize
  452. }