mongo 2.1.0.beta → 2.1.0.rc0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (253) hide show
  1. checksums.yaml +4 -4
  2. checksums.yaml.gz.sig +0 -0
  3. data.tar.gz.sig +0 -0
  4. data/Rakefile +2 -2
  5. data/lib/mongo.rb +2 -3
  6. data/lib/mongo/address.rb +7 -5
  7. data/lib/mongo/address/unix.rb +2 -2
  8. data/lib/mongo/auth/ldap/conversation.rb +6 -2
  9. data/lib/mongo/auth/scram/conversation.rb +8 -2
  10. data/lib/mongo/auth/user/view.rb +21 -0
  11. data/lib/mongo/bulk_write.rb +155 -23
  12. data/lib/mongo/bulk_write/combineable.rb +51 -0
  13. data/lib/mongo/bulk_write/ordered_combiner.rb +55 -0
  14. data/lib/mongo/bulk_write/result.rb +61 -8
  15. data/lib/mongo/bulk_write/result_combiner.rb +117 -0
  16. data/lib/mongo/bulk_write/transformable.rb +117 -0
  17. data/lib/mongo/bulk_write/unordered_combiner.rb +52 -0
  18. data/lib/mongo/bulk_write/validatable.rb +62 -0
  19. data/lib/mongo/client.rb +7 -3
  20. data/lib/mongo/cluster.rb +3 -3
  21. data/lib/mongo/cluster/topology/replica_set.rb +8 -6
  22. data/lib/mongo/cluster/topology/unknown.rb +5 -2
  23. data/lib/mongo/collection.rb +75 -4
  24. data/lib/mongo/collection/view.rb +1 -2
  25. data/lib/mongo/collection/view/aggregation.rb +13 -8
  26. data/lib/mongo/collection/view/immutable.rb +6 -6
  27. data/lib/mongo/collection/view/iterable.rb +13 -4
  28. data/lib/mongo/collection/view/map_reduce.rb +22 -17
  29. data/lib/mongo/collection/view/readable.rb +121 -70
  30. data/lib/mongo/cursor.rb +5 -1
  31. data/lib/mongo/database.rb +3 -3
  32. data/lib/mongo/database/view.rb +1 -1
  33. data/lib/mongo/error.rb +7 -0
  34. data/lib/mongo/{bulk_write/unordered_bulk_write.rb → error/closed_stream.rb} +12 -21
  35. data/lib/mongo/{bulk_write/ordered_bulk_write.rb → error/extra_file_chunk.rb} +13 -27
  36. data/lib/mongo/error/file_not_found.rb +37 -0
  37. data/lib/mongo/error/invalid_file.rb +2 -2
  38. data/lib/mongo/error/invalid_file_revision.rb +37 -0
  39. data/lib/mongo/error/invalid_uri.rb +5 -4
  40. data/lib/mongo/error/missing_file_chunk.rb +38 -0
  41. data/lib/mongo/error/operation_failure.rb +1 -1
  42. data/lib/mongo/error/unchangeable_collection_option.rb +38 -0
  43. data/lib/mongo/error/unexpected_chunk_length.rb +39 -0
  44. data/lib/mongo/grid.rb +2 -1
  45. data/lib/mongo/grid/file.rb +12 -9
  46. data/lib/mongo/grid/file/chunk.rb +6 -6
  47. data/lib/mongo/grid/file/{metadata.rb → info.rb} +41 -39
  48. data/lib/mongo/grid/fs_bucket.rb +441 -0
  49. data/lib/mongo/grid/stream.rb +64 -0
  50. data/lib/mongo/grid/stream/read.rb +208 -0
  51. data/lib/mongo/grid/stream/write.rb +187 -0
  52. data/lib/mongo/index/view.rb +1 -1
  53. data/lib/mongo/loggable.rb +34 -57
  54. data/lib/mongo/logger.rb +16 -78
  55. data/lib/mongo/monitoring.rb +1 -5
  56. data/lib/mongo/monitoring/command_log_subscriber.rb +35 -17
  57. data/lib/mongo/monitoring/event/command_succeeded.rb +20 -1
  58. data/lib/mongo/monitoring/publishable.rb +22 -12
  59. data/lib/mongo/operation.rb +3 -6
  60. data/lib/mongo/operation/commands.rb +24 -0
  61. data/lib/mongo/operation/{aggregate.rb → commands/aggregate.rb} +3 -41
  62. data/lib/mongo/operation/{aggregate → commands/aggregate}/result.rb +0 -0
  63. data/lib/mongo/operation/commands/collections_info.rb +66 -0
  64. data/lib/mongo/operation/{command.rb → commands/command.rb} +2 -18
  65. data/lib/mongo/operation/commands/indexes.rb +70 -0
  66. data/lib/mongo/operation/commands/list_collections.rb +54 -0
  67. data/lib/mongo/operation/commands/list_collections/result.rb +112 -0
  68. data/lib/mongo/operation/commands/list_indexes.rb +56 -0
  69. data/lib/mongo/operation/commands/list_indexes/result.rb +115 -0
  70. data/lib/mongo/operation/{map_reduce.rb → commands/map_reduce.rb} +3 -41
  71. data/lib/mongo/operation/{map_reduce → commands/map_reduce}/result.rb +0 -0
  72. data/lib/mongo/operation/{parallel_scan.rb → commands/parallel_scan.rb} +3 -23
  73. data/lib/mongo/operation/{parallel_scan → commands/parallel_scan}/result.rb +0 -0
  74. data/lib/mongo/operation/commands/user_query.rb +69 -0
  75. data/lib/mongo/operation/commands/users_info.rb +53 -0
  76. data/lib/mongo/operation/commands/users_info/result.rb +36 -0
  77. data/lib/mongo/operation/executable.rb +4 -68
  78. data/lib/mongo/operation/kill_cursors.rb +3 -3
  79. data/lib/mongo/operation/read.rb +0 -4
  80. data/lib/mongo/operation/read/get_more.rb +2 -22
  81. data/lib/mongo/operation/read/query.rb +2 -21
  82. data/lib/mongo/operation/{read_preferrable.rb → read_preference.rb} +3 -2
  83. data/lib/mongo/operation/specifiable.rb +24 -0
  84. data/lib/mongo/operation/write.rb +2 -0
  85. data/lib/mongo/operation/write/bulk.rb +6 -3
  86. data/lib/mongo/operation/write/bulk/bulkable.rb +82 -0
  87. data/lib/mongo/operation/write/bulk/delete.rb +71 -0
  88. data/lib/mongo/operation/write/bulk/delete/result.rb +74 -0
  89. data/lib/mongo/operation/write/bulk/insert.rb +96 -0
  90. data/lib/mongo/operation/write/bulk/insert/result.rb +129 -0
  91. data/lib/mongo/operation/write/bulk/legacy_mergable.rb +87 -0
  92. data/lib/mongo/operation/write/bulk/mergable.rb +71 -0
  93. data/lib/mongo/operation/write/bulk/update.rb +81 -0
  94. data/lib/mongo/operation/write/bulk/update/result.rb +174 -0
  95. data/lib/mongo/operation/write/command/create_index.rb +0 -1
  96. data/lib/mongo/operation/write/command/create_user.rb +0 -1
  97. data/lib/mongo/operation/write/command/delete.rb +0 -1
  98. data/lib/mongo/operation/write/command/drop_index.rb +0 -1
  99. data/lib/mongo/operation/write/command/insert.rb +0 -1
  100. data/lib/mongo/operation/write/command/remove_user.rb +0 -1
  101. data/lib/mongo/operation/write/command/update.rb +0 -1
  102. data/lib/mongo/operation/write/command/update_user.rb +0 -1
  103. data/lib/mongo/operation/write/command/writable.rb +13 -18
  104. data/lib/mongo/operation/write/create_index.rb +4 -27
  105. data/lib/mongo/operation/write/create_user.rb +4 -30
  106. data/lib/mongo/operation/write/delete.rb +5 -28
  107. data/lib/mongo/operation/write/drop_index.rb +3 -3
  108. data/lib/mongo/operation/write/gle.rb +48 -0
  109. data/lib/mongo/operation/write/idable.rb +5 -0
  110. data/lib/mongo/operation/write/insert.rb +2 -24
  111. data/lib/mongo/operation/write/remove_user.rb +4 -27
  112. data/lib/mongo/operation/write/update.rb +4 -32
  113. data/lib/mongo/operation/write/update_user.rb +4 -30
  114. data/lib/mongo/operation/write/write_command_enabled.rb +53 -0
  115. data/lib/mongo/options/mapper.rb +4 -2
  116. data/lib/mongo/protocol/delete.rb +68 -3
  117. data/lib/mongo/protocol/get_more.rb +54 -2
  118. data/lib/mongo/protocol/insert.rb +59 -1
  119. data/lib/mongo/protocol/kill_cursors.rb +53 -4
  120. data/lib/mongo/protocol/message.rb +12 -12
  121. data/lib/mongo/protocol/query.rb +139 -65
  122. data/lib/mongo/protocol/reply.rb +69 -1
  123. data/lib/mongo/protocol/update.rb +70 -1
  124. data/lib/mongo/server/connection.rb +11 -3
  125. data/lib/mongo/server/description.rb +29 -0
  126. data/lib/mongo/server/description/features.rb +2 -1
  127. data/lib/mongo/server/monitor.rb +2 -2
  128. data/lib/mongo/server_selector.rb +14 -10
  129. data/lib/mongo/server_selector/selectable.rb +24 -22
  130. data/lib/mongo/socket.rb +6 -3
  131. data/lib/mongo/socket/tcp.rb +2 -2
  132. data/lib/mongo/socket/unix.rb +5 -8
  133. data/lib/mongo/uri.rb +243 -139
  134. data/lib/mongo/version.rb +1 -1
  135. data/spec/mongo/address/unix_spec.rb +1 -1
  136. data/spec/mongo/address_spec.rb +25 -0
  137. data/spec/mongo/auth/ldap/conversation_spec.rb +43 -0
  138. data/spec/mongo/auth/user/view_spec.rb +26 -1
  139. data/spec/mongo/bulk_write/ordered_combiner_spec.rb +271 -0
  140. data/spec/mongo/bulk_write/unordered_combiner_spec.rb +239 -0
  141. data/spec/mongo/bulk_write_spec.rb +332 -166
  142. data/spec/mongo/client_spec.rb +25 -0
  143. data/spec/mongo/cluster/topology/replica_set_spec.rb +2 -0
  144. data/spec/mongo/collection/view/aggregation_spec.rb +65 -0
  145. data/spec/mongo/collection/view/immutable_spec.rb +103 -0
  146. data/spec/mongo/collection/view/map_reduce_spec.rb +98 -3
  147. data/spec/mongo/collection/view/readable_spec.rb +17 -30
  148. data/spec/mongo/collection/view_spec.rb +233 -7
  149. data/spec/mongo/collection_spec.rb +360 -18
  150. data/spec/mongo/command_monitoring_spec.rb +51 -0
  151. data/spec/mongo/connection_string_spec.rb +137 -0
  152. data/spec/mongo/database_spec.rb +27 -11
  153. data/spec/mongo/grid/file/chunk_spec.rb +5 -5
  154. data/spec/mongo/grid/file/{metadata_spec.rb → info_spec.rb} +29 -17
  155. data/spec/mongo/grid/file_spec.rb +8 -8
  156. data/spec/mongo/grid/fs_bucket_spec.rb +1020 -0
  157. data/spec/mongo/grid/stream/read_spec.rb +275 -0
  158. data/spec/mongo/grid/stream/write_spec.rb +440 -0
  159. data/spec/mongo/grid/stream_spec.rb +48 -0
  160. data/spec/mongo/gridfs_spec.rb +50 -0
  161. data/spec/mongo/logger_spec.rb +0 -40
  162. data/spec/mongo/monitoring/command_log_subscriber_spec.rb +76 -0
  163. data/spec/mongo/operation/{aggregate_spec.rb → commands/aggregate_spec.rb} +0 -42
  164. data/spec/mongo/operation/{read → commands}/collections_info_spec.rb +1 -1
  165. data/spec/mongo/operation/{command_spec.rb → commands/command_spec.rb} +0 -0
  166. data/spec/mongo/operation/{read → commands}/indexes_spec.rb +1 -1
  167. data/spec/mongo/operation/{map_reduce_spec.rb → commands/map_reduce_spec.rb} +0 -18
  168. data/spec/mongo/operation/kill_cursors_spec.rb +1 -1
  169. data/spec/mongo/operation/{read_preferrable_spec.rb → read_preference_spec.rb} +11 -11
  170. data/spec/mongo/operation/write/bulk/{bulk_delete_spec.rb → delete_spec.rb} +1 -12
  171. data/spec/mongo/operation/write/bulk/{bulk_insert_spec.rb → insert_spec.rb} +1 -12
  172. data/spec/mongo/operation/write/bulk/{bulk_update_spec.rb → update_spec.rb} +1 -12
  173. data/spec/mongo/operation/write/insert_spec.rb +0 -11
  174. data/spec/mongo/protocol/kill_cursors_spec.rb +5 -3
  175. data/spec/mongo/server/description_spec.rb +42 -0
  176. data/spec/mongo/server/monitor_spec.rb +21 -0
  177. data/spec/mongo/server_discovery_and_monitoring_spec.rb +1 -0
  178. data/spec/mongo/server_selection_spec.rb +3 -3
  179. data/spec/mongo/server_selector/nearest_spec.rb +34 -27
  180. data/spec/mongo/server_selector/primary_preferred_spec.rb +31 -30
  181. data/spec/mongo/server_selector/primary_spec.rb +14 -13
  182. data/spec/mongo/server_selector/secondary_preferred_spec.rb +27 -26
  183. data/spec/mongo/server_selector/secondary_spec.rb +23 -22
  184. data/spec/mongo/server_selector_spec.rb +87 -24
  185. data/spec/mongo/socket/unix_spec.rb +52 -0
  186. data/spec/mongo/uri_spec.rb +251 -39
  187. data/spec/spec_helper.rb +11 -4
  188. data/spec/support/authorization.rb +4 -5
  189. data/spec/support/command_monitoring.rb +365 -0
  190. data/spec/support/command_monitoring/bulkWrite.yml +73 -0
  191. data/spec/support/command_monitoring/command.yml +42 -0
  192. data/spec/support/command_monitoring/deleteMany.yml +55 -0
  193. data/spec/support/command_monitoring/deleteOne.yml +55 -0
  194. data/spec/support/command_monitoring/find.yml +219 -0
  195. data/spec/support/command_monitoring/insertMany.yml +81 -0
  196. data/spec/support/command_monitoring/insertOne.yml +51 -0
  197. data/spec/support/command_monitoring/updateMany.yml +67 -0
  198. data/spec/support/command_monitoring/updateOne.yml +95 -0
  199. data/spec/support/connection_string.rb +228 -0
  200. data/spec/support/connection_string_tests/invalid-uris.yml +193 -0
  201. data/spec/support/connection_string_tests/valid-auth.yml +256 -0
  202. data/spec/support/connection_string_tests/valid-host_identifiers.yml +121 -0
  203. data/spec/support/connection_string_tests/valid-options.yml +30 -0
  204. data/spec/support/connection_string_tests/valid-unix_socket-absolute.yml +197 -0
  205. data/spec/support/connection_string_tests/valid-unix_socket-relative.yml +213 -0
  206. data/spec/support/connection_string_tests/valid-warnings.yml +55 -0
  207. data/spec/support/crud.rb +3 -1
  208. data/spec/support/crud/read.rb +14 -10
  209. data/spec/support/crud/write.rb +36 -9
  210. data/spec/support/gridfs.rb +637 -0
  211. data/spec/support/gridfs_tests/delete.yml +157 -0
  212. data/spec/support/gridfs_tests/download.yml +210 -0
  213. data/spec/support/gridfs_tests/download_by_name.yml +113 -0
  214. data/spec/support/gridfs_tests/upload.yml +158 -0
  215. data/spec/support/sdam/rs/equal_electionids.yml +1 -2
  216. data/spec/support/sdam/rs/new_primary_new_electionid.yml +0 -3
  217. data/spec/support/sdam/rs/primary_mismatched_me.yml +37 -0
  218. data/spec/support/sdam/rs/primary_to_no_primary_mismatched_me.yml +75 -0
  219. data/spec/support/sdam/rs/secondary_mismatched_me.yml +37 -0
  220. data/spec/support/sdam/single/direct_connection_rsarbiter.yml +1 -1
  221. data/spec/support/sdam/single/direct_connection_rsprimary.yml +1 -1
  222. data/spec/support/sdam/single/direct_connection_rssecondary.yml +1 -1
  223. data/spec/support/sdam/single/direct_connection_slave.yml +1 -1
  224. data/spec/support/sdam/single/direct_connection_standalone.yml +1 -1
  225. data/spec/support/sdam/single/not_ok_response.yml +0 -1
  226. data/spec/support/server_discovery_and_monitoring.rb +3 -1
  227. data/spec/support/server_selection.rb +3 -1
  228. data/spec/support/shared/bulk_write.rb +192 -0
  229. data/spec/support/shared/server_selector.rb +21 -12
  230. metadata +147 -57
  231. metadata.gz.sig +0 -0
  232. data/lib/mongo/bulk_write/bulk_writable.rb +0 -252
  233. data/lib/mongo/bulk_write/deletable.rb +0 -57
  234. data/lib/mongo/bulk_write/insertable.rb +0 -49
  235. data/lib/mongo/bulk_write/replacable.rb +0 -58
  236. data/lib/mongo/bulk_write/updatable.rb +0 -69
  237. data/lib/mongo/grid/fs.rb +0 -146
  238. data/lib/mongo/operation/list_collections/result.rb +0 -114
  239. data/lib/mongo/operation/list_indexes/result.rb +0 -118
  240. data/lib/mongo/operation/read/collections_info.rb +0 -68
  241. data/lib/mongo/operation/read/indexes.rb +0 -69
  242. data/lib/mongo/operation/read/list_collections.rb +0 -76
  243. data/lib/mongo/operation/read/list_indexes.rb +0 -78
  244. data/lib/mongo/operation/write/bulk/bulk_delete.rb +0 -145
  245. data/lib/mongo/operation/write/bulk/bulk_delete/result.rb +0 -75
  246. data/lib/mongo/operation/write/bulk/bulk_insert.rb +0 -132
  247. data/lib/mongo/operation/write/bulk/bulk_insert/result.rb +0 -130
  248. data/lib/mongo/operation/write/bulk/bulk_mergable.rb +0 -67
  249. data/lib/mongo/operation/write/bulk/bulk_update.rb +0 -154
  250. data/lib/mongo/operation/write/bulk/bulk_update/result.rb +0 -174
  251. data/lib/mongo/operation/write/bulk/legacy_bulk_mergable.rb +0 -83
  252. data/spec/mongo/grid/fs_spec.rb +0 -160
  253. data/spec/mongo/loggable_spec.rb +0 -63
data/spec/spec_helper.rb CHANGED
@@ -24,6 +24,9 @@ SERVER_DISCOVERY_TESTS = Dir.glob("#{CURRENT_PATH}/support/sdam/**/*.yml")
24
24
  SERVER_SELECTION_RTT_TESTS = Dir.glob("#{CURRENT_PATH}/support/server_selection/rtt/*.yml")
25
25
  SERVER_SELECTION_TESTS = Dir.glob("#{CURRENT_PATH}/support/server_selection/selection/**/*.yml")
26
26
  CRUD_TESTS = Dir.glob("#{CURRENT_PATH}/support/crud_tests/**/*.yml")
27
+ COMMAND_MONITORING_TESTS = Dir.glob("#{CURRENT_PATH}/support/command_monitoring/**/*.yml")
28
+ CONNECTION_STRING_TESTS = Dir.glob("#{CURRENT_PATH}/support/connection_string_tests/*.yml")
29
+ GRIDFS_TESTS = Dir.glob("#{CURRENT_PATH}/support/gridfs_tests/*.yml")
27
30
 
28
31
  SSL_CERTS_DIR = "#{CURRENT_PATH}/support/certificates"
29
32
  CLIENT_PEM = "#{SSL_CERTS_DIR}/client.pem"
@@ -33,6 +36,9 @@ CRL_PEM = "#{SSL_CERTS_DIR}/crl.pem"
33
36
 
34
37
  require 'mongo'
35
38
 
39
+ Mongo::Logger.logger = Logger.new($stdout)
40
+ Mongo::Logger.logger.level = Logger::INFO
41
+
36
42
  require 'support/travis'
37
43
  require 'support/matchers'
38
44
  require 'support/authorization'
@@ -40,9 +46,9 @@ require 'support/server_discovery_and_monitoring'
40
46
  require 'support/server_selection_rtt'
41
47
  require 'support/server_selection'
42
48
  require 'support/crud'
43
-
44
- Mongo::Logger.logger = Logger.new($stdout)
45
- Mongo::Logger.logger.level = Logger::INFO
49
+ require 'support/command_monitoring'
50
+ require 'support/connection_string'
51
+ require 'support/gridfs'
46
52
 
47
53
  RSpec.configure do |config|
48
54
  config.color = true
@@ -57,6 +63,7 @@ RSpec.configure do |config|
57
63
  # database. This user will need to be authenticated in order to add any
58
64
  # more users to any other databases.
59
65
  ADMIN_UNAUTHORIZED_CLIENT.database.users.create(ROOT_USER)
66
+ ADMIN_UNAUTHORIZED_CLIENT.close
60
67
  rescue Exception => e
61
68
  end
62
69
  begin
@@ -67,7 +74,7 @@ RSpec.configure do |config|
67
74
  unless write_command_enabled?
68
75
  # If we are on versions less than 2.6, we need to create a user for
69
76
  # each database, since the users are not stored in the admin database
70
- # but in the system.users collection on the datbases themselves. Also,
77
+ # but in the system.users collection on the databases themselves. Also,
71
78
  # roles in versions lower than 2.6 can only be strings, not hashes.
72
79
  begin ADMIN_AUTHORIZED_TEST_CLIENT.database.users.create(TEST_READ_WRITE_USER); rescue; end
73
80
  end
@@ -144,8 +144,7 @@ AUTHORIZED_CLIENT = Mongo::Client.new(
144
144
  # @since 2.0.0
145
145
  UNAUTHORIZED_CLIENT = Mongo::Client.new(
146
146
  ADDRESSES,
147
- TEST_OPTIONS.merge(
148
- database: TEST_DB)
147
+ TEST_OPTIONS.merge(database: TEST_DB, monitoring: false)
149
148
  )
150
149
 
151
150
  # Provides an unauthorized mongo client on the admin database, for use in
@@ -154,8 +153,7 @@ UNAUTHORIZED_CLIENT = Mongo::Client.new(
154
153
  # @since 2.0.0
155
154
  ADMIN_UNAUTHORIZED_CLIENT = Mongo::Client.new(
156
155
  ADDRESSES,
157
- TEST_OPTIONS.merge(
158
- database: Mongo::Database::ADMIN)
156
+ TEST_OPTIONS.merge(database: Mongo::Database::ADMIN, monitoring: false)
159
157
  )
160
158
 
161
159
  # Get an authorized client on the test database logged in as the admin
@@ -166,7 +164,8 @@ ADMIN_AUTHORIZED_TEST_CLIENT = ADMIN_UNAUTHORIZED_CLIENT.with(
166
164
  user: ROOT_USER.name,
167
165
  password: ROOT_USER.password,
168
166
  database: TEST_DB,
169
- auth_source: Mongo::Database::ADMIN
167
+ auth_source: Mongo::Database::ADMIN,
168
+ monitoring: false
170
169
  )
171
170
 
172
171
  module Authorization
@@ -0,0 +1,365 @@
1
+ # Copyright (C) 2014-2015 MongoDB, Inc.
2
+ #
3
+ # Licensed under the Apache License, Version 2.0 (the "License");
4
+ # you may not use this file except in compliance with the License.
5
+ # You may obtain a copy of the License at
6
+ #
7
+ # http://www.apache.org/licenses/LICENSE-2.0
8
+ #
9
+ # Unless required by applicable law or agreed to in writing, software
10
+ # distributed under the License is distributed on an "AS IS" BASIS,
11
+ # WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
12
+ # See the License for the specific language governing permissions and
13
+ # limitations under the License.
14
+ #
15
+
16
+ RSpec::Matchers.define :match_command_name do |expectation|
17
+
18
+ match do |event|
19
+ expect(event.command_name.to_s).to eq(expectation.command_name.to_s)
20
+ end
21
+ end
22
+
23
+ RSpec::Matchers.define :match_database_name do |expectation|
24
+
25
+ match do |event|
26
+ expect(event.database_name.to_s).to eq(expectation.database_name.to_s)
27
+ end
28
+ end
29
+
30
+ RSpec::Matchers.define :generate_request_id do |expectation|
31
+
32
+ match do |event|
33
+ expect(event.request_id).to be > 0
34
+ end
35
+ end
36
+
37
+ RSpec::Matchers.define :generate_operation_id do |expectation|
38
+
39
+ match do |event|
40
+ expect(event.request_id).to be > 0
41
+ end
42
+ end
43
+
44
+ RSpec::Matchers.define :match_command do |expectation|
45
+ include Mongo::CommandMonitoring::Matchable
46
+
47
+ match do |event|
48
+ data_matches?(event.command, expectation.event_data['command'])
49
+ end
50
+ end
51
+
52
+ RSpec::Matchers.define :match_reply do |expectation|
53
+ include Mongo::CommandMonitoring::Matchable
54
+
55
+ match do |event|
56
+ data_matches?(event.reply, expectation.event_data['reply'])
57
+ end
58
+ end
59
+
60
+ RSpec::Matchers.define :match_command_started_event do |expectation|
61
+
62
+ match do |event|
63
+ expect(event).to match_command_name(expectation)
64
+ expect(event).to match_database_name(expectation)
65
+ expect(event).to generate_operation_id
66
+ expect(event).to generate_request_id
67
+ expect(event).to match_command(expectation)
68
+ end
69
+ end
70
+
71
+ RSpec::Matchers.define :match_command_succeeded_event do |expectation|
72
+
73
+ match do |event|
74
+ expect(event).to match_command_name(expectation)
75
+ expect(event).to generate_operation_id
76
+ expect(event).to generate_request_id
77
+ expect(event).to match_reply(expectation)
78
+ end
79
+ end
80
+
81
+ RSpec::Matchers.define :match_command_failed_event do |expectation|
82
+
83
+ match do |event|
84
+ expect(event).to match_command_name(expectation)
85
+ expect(event).to generate_operation_id
86
+ expect(event).to generate_request_id
87
+ end
88
+ end
89
+
90
+ module Mongo
91
+ module CommandMonitoring
92
+
93
+ # Matchers common behaviour.
94
+ #
95
+ # @since 2.1.0
96
+ module Matchable
97
+
98
+ # Determine if the data matches.
99
+ #
100
+ # @example Does the data match?
101
+ # matchable.data_matches?(actual, expected)
102
+ #
103
+ # @param [ Object ] actual The actual data.
104
+ # @param [ Object ] expected The expected data.
105
+ #
106
+ # @return [ true, false ] If the data matches.
107
+ #
108
+ # @since 2.1.0
109
+ def data_matches?(actual, expected)
110
+ case expected
111
+ when ::Hash, BSON::Document then
112
+ hash_matches?(actual, expected)
113
+ when ::Array
114
+ array_matches?(actual, expected)
115
+ else
116
+ value_matches?(actual, expected)
117
+ end
118
+ end
119
+
120
+ # Determine if the hash matches.
121
+ #
122
+ # @example Does the hash match?
123
+ # matchable.hash_matches?(actual, expected)
124
+ #
125
+ # @param [ Hash ] actual The actual hash.
126
+ # @param [ Hash ] expected The expected hash.
127
+ #
128
+ # @return [ true, false ] If the hash matches.
129
+ #
130
+ # @since 2.1.0
131
+ def hash_matches?(actual, expected)
132
+ if expected.keys.first == '$numberLong'
133
+ converted = expected.values.first.to_i
134
+ (actual == converted) || actual >= 0
135
+ else
136
+ expected.each do |key, value|
137
+ return false unless data_matches?(actual[key], value)
138
+ end
139
+ end
140
+ end
141
+
142
+ # Determine if an array matches.
143
+ #
144
+ # @example Does the array match?
145
+ # matchable.array_matches?(actual, expected)
146
+ #
147
+ # @param [ Array ] actual The actual array.
148
+ # @param [ Array ] expected The expected array.
149
+ #
150
+ # @return [ true, false ] If the array matches.
151
+ #
152
+ # @since 2.1.0
153
+ def array_matches?(actual, expected)
154
+ expected.each_with_index do |value, i|
155
+ # @todo: Durran: fix for kill cursors replies
156
+ if actual
157
+ return false unless data_matches?(actual[i], value)
158
+ end
159
+ end
160
+ end
161
+
162
+ # Check if a value matches.
163
+ #
164
+ # @example Does a value match.
165
+ # matchable.value_matches?(actual, expected)
166
+ #
167
+ # @param [ Object ] actual The actual value.
168
+ # @param [ Object ] expected The expected object.
169
+ #
170
+ # @return [ true, false ] If the value matches.
171
+ #
172
+ # @since 2.1.0
173
+ def value_matches?(actual, expected)
174
+ case expected
175
+ when '42', 42 then
176
+ actual > 0
177
+ when '' then
178
+ !actual.nil?
179
+ else
180
+ actual == expected
181
+ end
182
+ end
183
+ end
184
+
185
+ # Represents a command monitoring spec in its entirety.
186
+ #
187
+ # @since 2.1.0
188
+ class Spec
189
+
190
+ # Create the spec.
191
+ #
192
+ # @example Create the spec.
193
+ # Spec.new('/path/to/test')
194
+ #
195
+ # @param [ String ] file The yaml test file.
196
+ #
197
+ # @since 2.1.0
198
+ def initialize(file)
199
+ file = File.new(file)
200
+ @spec = YAML.load(ERB.new(file.read).result)
201
+ file.close
202
+ @data = @spec['data']
203
+ @tests = @spec['tests']
204
+ end
205
+
206
+ # Get all the tests in the spec.
207
+ #
208
+ # @example Get all the tests.
209
+ # spec.tests
210
+ #
211
+ # @return [ Array<Test> ] The tests.
212
+ def tests
213
+ @tests.map do |test|
214
+ Test.new(@data, test)
215
+ end
216
+ end
217
+ end
218
+
219
+ # Represents an individual command monitoring test.
220
+ #
221
+ # @since 2.1.0
222
+ class Test
223
+
224
+ # @return [ String ] description The test description.
225
+ attr_reader :description
226
+
227
+ # @return [ Array<Expectation> ] The expectations.
228
+ attr_reader :expectations
229
+
230
+ # Create the new test.
231
+ #
232
+ # @example Create the test.
233
+ # Test.new(data, test)
234
+ #
235
+ # @param [ Array<Hash> ] data The test data.
236
+ # @param [ Hash ] The test itself.
237
+ #
238
+ # @since 2.1.0
239
+ def initialize(data, test)
240
+ @data = data
241
+ @description = test['description']
242
+ @operation = Mongo::CRUD::Operation.get(test['operation'])
243
+ @expectations = test['expectations'].map{ |e| Expectation.new(e) }
244
+ end
245
+
246
+ # Run the test against the provided collection.
247
+ #
248
+ # @example Run the test.
249
+ # test.run(collection)
250
+ #
251
+ # @param [ Mongo::Collection ] collection The collection.
252
+ #
253
+ # @since 2.1.0
254
+ def run(collection)
255
+ collection.insert_many(@data)
256
+ @operation.execute(collection)
257
+ end
258
+ end
259
+
260
+ # Encapsulates expectation behaviour.
261
+ #
262
+ # @since 2.1.0
263
+ class Expectation
264
+
265
+ # @return [ String ] event_type The type of expected event.
266
+ attr_reader :event_type
267
+
268
+ # @return [ Hash ] event_data The event data.
269
+ attr_reader :event_data
270
+
271
+ # Get the expected command name.
272
+ #
273
+ # @example Get the expected command name.
274
+ # expectation.command_name
275
+ #
276
+ # @return [ String ] The command name.
277
+ #
278
+ # @since 2.1.0
279
+ def command_name
280
+ @event_data['command_name']
281
+ end
282
+
283
+ # Get the expected database name.
284
+ #
285
+ # @example Get the expected database name.
286
+ # expectation.database_name
287
+ #
288
+ # @return [ String ] The database name.
289
+ #
290
+ # @since 2.1.0
291
+ def database_name
292
+ @event_data['database_name']
293
+ end
294
+
295
+ # Get a readable event name.
296
+ #
297
+ # @example Get the event name.
298
+ # expectation.event_name
299
+ #
300
+ # @return [ String ] The event name.
301
+ #
302
+ # @since 2.1.0
303
+ def event_name
304
+ event_type.gsub('_', ' ')
305
+ end
306
+
307
+ # Create the new expectation.
308
+ #
309
+ # @example Create the new expectation.
310
+ # Expectation.new(expectation)
311
+ #
312
+ # @param [ Hash ] expectation The expectation.
313
+ #
314
+ # @since 2.1.0
315
+ def initialize(expectation)
316
+ @event_type = expectation.keys.first
317
+ @event_data = expectation[@event_type]
318
+ end
319
+
320
+ # Get the name of the matcher.
321
+ #
322
+ # @example Get the matcher name.
323
+ # expectation.matcher
324
+ #
325
+ # @return [ String ] The matcher name.
326
+ #
327
+ # @since 2.1.0
328
+ def matcher
329
+ "match_#{event_type}"
330
+ end
331
+ end
332
+
333
+ # The test subscriber to track the events.
334
+ #
335
+ # @since 2.1.0
336
+ class TestSubscriber
337
+
338
+ def started(event)
339
+ command_started_event[event.command_name] = event
340
+ end
341
+
342
+ def succeeded(event)
343
+ command_succeeded_event[event.command_name] = event
344
+ end
345
+
346
+ def failed(event)
347
+ command_failed_event[event.command_name] = event
348
+ end
349
+
350
+ private
351
+
352
+ def command_started_event
353
+ @started_events ||= BSON::Document.new
354
+ end
355
+
356
+ def command_succeeded_event
357
+ @succeeded_events ||= BSON::Document.new
358
+ end
359
+
360
+ def command_failed_event
361
+ @failed_events ||= BSON::Document.new
362
+ end
363
+ end
364
+ end
365
+ end
@@ -0,0 +1,73 @@
1
+ data:
2
+ - { _id: 1, x: 11 }
3
+ - { _id: 2, x: 22 }
4
+ - { _id: 3, x: 33 }
5
+
6
+ collection_name: &collection_name "test"
7
+ database_name: &database_name "ruby-driver"
8
+
9
+ tests:
10
+ -
11
+ description: "A successful mixed bulk write"
12
+ operation:
13
+ name: "bulkWrite"
14
+ arguments:
15
+ requests:
16
+ - insertOne:
17
+ document: { _id: 4, x: 44 }
18
+ - updateOne:
19
+ filter: { _id: 3 }
20
+ update: { $set: { x: 333 } }
21
+ expectations:
22
+ -
23
+ command_started_event:
24
+ command:
25
+ insert: *collection_name
26
+ documents:
27
+ - { _id: 4, x: 44 }
28
+ ordered: true
29
+ command_name: "insert"
30
+ database_name: *database_name
31
+ -
32
+ command_succeeded_event:
33
+ reply: { ok: 1.0, n: 1 }
34
+ command_name: "insert"
35
+ -
36
+ command_started_event:
37
+ command:
38
+ update: *collection_name
39
+ updates:
40
+ - { q: {_id: 3 }, u: { $set: { x: 333 } }, upsert: false, multi: false }
41
+ ordered: true
42
+ command_name: "update"
43
+ database_name: *database_name
44
+ -
45
+ command_succeeded_event:
46
+ reply: { ok: 1.0, n: 1 }
47
+ command_name: "update"
48
+ -
49
+ description: "A successful unordered bulk write with an unacknowledged write concern"
50
+ comment: "On a 2.4 server, no GLE is sent and requires a client-side manufactored reply"
51
+ operation:
52
+ name: "bulkWrite"
53
+ arguments:
54
+ requests:
55
+ - insertOne:
56
+ document: { _id: 4, x: 44 }
57
+ ordered: false
58
+ writeConcern: { w: 0 }
59
+ expectations:
60
+ -
61
+ command_started_event:
62
+ command:
63
+ insert: *collection_name
64
+ documents:
65
+ - { _id: 4, x: 44 }
66
+ ordered: false
67
+ writeConcern: { w: 0 }
68
+ command_name: "insert"
69
+ database_name: *database_name
70
+ -
71
+ command_succeeded_event:
72
+ reply: { ok: 1.0 }
73
+ command_name: "insert"