mongo 2.15.1 → 2.16.0.alpha1

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 (326) hide show
  1. checksums.yaml +4 -4
  2. checksums.yaml.gz.sig +0 -0
  3. data/README.md +1 -1
  4. data/lib/mongo/bulk_write.rb +2 -2
  5. data/lib/mongo/client.rb +45 -5
  6. data/lib/mongo/cluster/periodic_executor.rb +4 -3
  7. data/lib/mongo/cluster/reapers/cursor_reaper.rb +76 -43
  8. data/lib/mongo/cluster/sdam_flow.rb +9 -3
  9. data/lib/mongo/cluster/topology/base.rb +13 -9
  10. data/lib/mongo/cluster/topology/load_balanced.rb +102 -0
  11. data/lib/mongo/cluster/topology.rb +28 -8
  12. data/lib/mongo/cluster.rb +136 -51
  13. data/lib/mongo/collection/view/aggregation.rb +5 -10
  14. data/lib/mongo/collection/view/builder/aggregation.rb +6 -5
  15. data/lib/mongo/collection/view/builder/map_reduce.rb +12 -49
  16. data/lib/mongo/collection/view/builder.rb +0 -4
  17. data/lib/mongo/collection/view/iterable.rb +42 -18
  18. data/lib/mongo/collection/view/map_reduce.rb +39 -15
  19. data/lib/mongo/collection/view/readable.rb +60 -51
  20. data/lib/mongo/collection/view/writable.rb +178 -175
  21. data/lib/mongo/collection/view.rb +15 -21
  22. data/lib/mongo/collection.rb +13 -13
  23. data/lib/mongo/cursor/kill_spec.rb +38 -0
  24. data/lib/mongo/cursor.rb +62 -31
  25. data/lib/mongo/database/view.rb +1 -1
  26. data/lib/mongo/error/bad_load_balancer_target.rb +26 -0
  27. data/lib/mongo/error/missing_service_id.rb +26 -0
  28. data/lib/mongo/error/no_service_connection_available.rb +49 -0
  29. data/lib/mongo/error/notable.rb +7 -0
  30. data/lib/mongo/error.rb +3 -0
  31. data/lib/mongo/grid/fs_bucket.rb +21 -2
  32. data/lib/mongo/id.rb +7 -5
  33. data/lib/mongo/index/view.rb +22 -41
  34. data/lib/mongo/monitoring/event/cmap/pool_cleared.rb +7 -4
  35. data/lib/mongo/monitoring/event/command_failed.rb +1 -1
  36. data/lib/mongo/monitoring/event/command_started.rb +2 -0
  37. data/lib/mongo/monitoring/publishable.rb +2 -2
  38. data/lib/mongo/operation/aggregate/command.rb +8 -0
  39. data/lib/mongo/operation/context.rb +19 -1
  40. data/lib/mongo/operation/count/command.rb +6 -0
  41. data/lib/mongo/operation/count/op_msg.rb +6 -0
  42. data/lib/mongo/operation/create/command.rb +7 -1
  43. data/lib/mongo/operation/create/op_msg.rb +7 -0
  44. data/lib/mongo/operation/create_index/command.rb +17 -1
  45. data/lib/mongo/operation/create_index/op_msg.rb +17 -4
  46. data/lib/mongo/operation/delete/command.rb +6 -3
  47. data/lib/mongo/operation/delete/legacy.rb +9 -2
  48. data/lib/mongo/operation/delete/op_msg.rb +8 -1
  49. data/lib/mongo/operation/distinct/command.rb +6 -0
  50. data/lib/mongo/operation/distinct/op_msg.rb +7 -0
  51. data/lib/mongo/operation/explain/command.rb +13 -1
  52. data/lib/mongo/operation/explain/legacy.rb +12 -5
  53. data/lib/mongo/operation/explain/op_msg.rb +9 -1
  54. data/lib/mongo/operation/find/builder/command.rb +110 -0
  55. data/lib/mongo/{collection/view → operation/find}/builder/flags.rb +10 -14
  56. data/lib/mongo/operation/find/builder/legacy.rb +123 -0
  57. data/lib/mongo/{collection/view → operation/find}/builder/modifiers.rb +31 -25
  58. data/lib/mongo/{cursor → operation/find}/builder.rb +4 -4
  59. data/lib/mongo/operation/find/command.rb +9 -0
  60. data/lib/mongo/operation/find/legacy.rb +10 -1
  61. data/lib/mongo/operation/find/op_msg.rb +12 -0
  62. data/lib/mongo/operation/find.rb +1 -0
  63. data/lib/mongo/operation/get_more/command.rb +1 -0
  64. data/lib/mongo/operation/get_more/command_builder.rb +38 -0
  65. data/lib/mongo/operation/get_more/op_msg.rb +1 -0
  66. data/lib/mongo/operation/get_more.rb +1 -0
  67. data/lib/mongo/operation/kill_cursors/command.rb +8 -0
  68. data/lib/mongo/operation/kill_cursors/command_builder.rb +35 -0
  69. data/lib/mongo/operation/kill_cursors/legacy.rb +2 -1
  70. data/lib/mongo/operation/kill_cursors/op_msg.rb +10 -0
  71. data/lib/mongo/operation/kill_cursors.rb +1 -0
  72. data/lib/mongo/operation/map_reduce/command.rb +8 -0
  73. data/lib/mongo/operation/map_reduce/op_msg.rb +1 -1
  74. data/lib/mongo/operation/shared/executable.rb +15 -1
  75. data/lib/mongo/operation/shared/polymorphic_operation.rb +1 -1
  76. data/lib/mongo/operation/shared/read_preference_supported.rb +3 -1
  77. data/lib/mongo/operation/shared/response_handling.rb +1 -0
  78. data/lib/mongo/operation/shared/sessions_supported.rb +12 -12
  79. data/lib/mongo/operation/shared/specifiable.rb +11 -29
  80. data/lib/mongo/operation/shared/validatable.rb +87 -0
  81. data/lib/mongo/operation/shared/write.rb +1 -1
  82. data/lib/mongo/operation/update/command.rb +6 -3
  83. data/lib/mongo/operation/update/legacy.rb +19 -11
  84. data/lib/mongo/operation/update/op_msg.rb +7 -4
  85. data/lib/mongo/operation/write_command/command.rb +51 -0
  86. data/lib/mongo/operation/write_command/op_msg.rb +43 -0
  87. data/lib/mongo/operation/write_command.rb +32 -0
  88. data/lib/mongo/operation.rb +10 -0
  89. data/lib/mongo/protocol/query.rb +35 -18
  90. data/lib/mongo/server/connection.rb +25 -3
  91. data/lib/mongo/server/connection_base.rb +12 -1
  92. data/lib/mongo/server/connection_common.rb +38 -1
  93. data/lib/mongo/server/connection_pool/generation_manager.rb +71 -0
  94. data/lib/mongo/server/connection_pool.rb +100 -27
  95. data/lib/mongo/server/description/features.rb +17 -16
  96. data/lib/mongo/server/description/load_balancer.rb +33 -0
  97. data/lib/mongo/server/description.rb +85 -6
  98. data/lib/mongo/server/monitor/connection.rb +5 -6
  99. data/lib/mongo/server/monitor.rb +2 -1
  100. data/lib/mongo/server/pending_connection.rb +47 -31
  101. data/lib/mongo/server.rb +73 -26
  102. data/lib/mongo/server_selector/base.rb +5 -1
  103. data/lib/mongo/session/session_pool.rb +11 -0
  104. data/lib/mongo/session.rb +21 -1
  105. data/lib/mongo/socket/ocsp_verifier.rb +6 -37
  106. data/lib/mongo/uri/options_mapper.rb +1 -0
  107. data/lib/mongo/uri/srv_protocol.rb +6 -8
  108. data/lib/mongo/uri.rb +18 -0
  109. data/lib/mongo/utils.rb +0 -7
  110. data/lib/mongo/version.rb +1 -1
  111. data/mongo.gemspec +1 -1
  112. data/spec/integration/auth_spec.rb +31 -1
  113. data/spec/integration/awaited_ismaster_spec.rb +1 -1
  114. data/spec/integration/bulk_write_spec.rb +1 -1
  115. data/spec/integration/change_stream_spec.rb +3 -3
  116. data/spec/integration/client_construction_spec.rb +54 -0
  117. data/spec/integration/client_side_encryption/auto_encryption_bulk_writes_spec.rb +1 -1
  118. data/spec/integration/client_side_encryption/auto_encryption_command_monitoring_spec.rb +1 -1
  119. data/spec/integration/client_side_encryption/bson_size_limit_spec.rb +1 -1
  120. data/spec/integration/client_side_encryption/data_key_spec.rb +1 -1
  121. data/spec/integration/client_spec.rb +2 -0
  122. data/spec/integration/command_monitoring_spec.rb +1 -1
  123. data/spec/integration/command_spec.rb +1 -1
  124. data/spec/integration/connection_spec.rb +52 -35
  125. data/spec/integration/crud_spec.rb +174 -1
  126. data/spec/integration/cursor_pinning_spec.rb +121 -0
  127. data/spec/integration/cursor_reaping_spec.rb +8 -4
  128. data/spec/integration/fork_reconnect_spec.rb +1 -5
  129. data/spec/integration/get_more_spec.rb +1 -1
  130. data/spec/integration/heartbeat_events_spec.rb +1 -1
  131. data/spec/integration/map_reduce_spec.rb +77 -0
  132. data/spec/integration/query_cache_spec.rb +2 -2
  133. data/spec/integration/query_cache_transactions_spec.rb +1 -1
  134. data/spec/integration/read_concern_spec.rb +1 -1
  135. data/spec/integration/read_preference_spec.rb +1 -1
  136. data/spec/integration/reconnect_spec.rb +30 -12
  137. data/spec/integration/retryable_errors_spec.rb +1 -1
  138. data/spec/integration/retryable_writes/retryable_writes_36_and_older_spec.rb +1 -1
  139. data/spec/integration/retryable_writes/retryable_writes_40_and_newer_spec.rb +1 -1
  140. data/spec/integration/sdam_error_handling_spec.rb +5 -3
  141. data/spec/integration/sdam_events_spec.rb +35 -19
  142. data/spec/integration/sdam_prose_spec.rb +1 -1
  143. data/spec/integration/server_monitor_spec.rb +1 -0
  144. data/spec/integration/server_selector_spec.rb +22 -5
  145. data/spec/integration/server_spec.rb +2 -0
  146. data/spec/integration/srv_monitoring_spec.rb +1 -1
  147. data/spec/integration/step_down_spec.rb +1 -1
  148. data/spec/integration/transaction_pinning_spec.rb +120 -0
  149. data/spec/integration/versioned_api_examples_spec.rb +45 -0
  150. data/spec/integration/x509_auth_spec.rb +1 -1
  151. data/spec/lite_spec_helper.rb +1 -2
  152. data/spec/mongo/address/unix_spec.rb +1 -0
  153. data/spec/mongo/auth/cr_spec.rb +2 -3
  154. data/spec/mongo/auth/ldap_spec.rb +2 -3
  155. data/spec/mongo/auth/scram_spec.rb +2 -3
  156. data/spec/mongo/auth/user/view_spec.rb +1 -1
  157. data/spec/mongo/auth/x509_spec.rb +2 -3
  158. data/spec/mongo/bulk_write_spec.rb +3 -3
  159. data/spec/mongo/client_construction_spec.rb +259 -28
  160. data/spec/mongo/client_spec.rb +6 -4
  161. data/spec/mongo/cluster/cursor_reaper_spec.rb +36 -21
  162. data/spec/mongo/cluster/periodic_executor_spec.rb +3 -1
  163. data/spec/mongo/cluster_spec.rb +44 -3
  164. data/spec/mongo/collection/view/aggregation_spec.rb +1 -1
  165. data/spec/mongo/collection/view/builder/find_command_spec.rb +4 -0
  166. data/spec/mongo/collection/view/builder/op_query_spec.rb +4 -0
  167. data/spec/mongo/collection/view/map_reduce_spec.rb +1 -1
  168. data/spec/mongo/collection_crud_spec.rb +7 -2
  169. data/spec/mongo/collection_ddl_spec.rb +1 -1
  170. data/spec/mongo/collection_spec.rb +1 -1
  171. data/spec/mongo/cursor/builder/get_more_command_spec.rb +4 -0
  172. data/spec/mongo/cursor/builder/op_get_more_spec.rb +4 -0
  173. data/spec/mongo/cursor_spec.rb +15 -5
  174. data/spec/mongo/database_spec.rb +15 -15
  175. data/spec/mongo/error/operation_failure_heavy_spec.rb +1 -1
  176. data/spec/mongo/grid/fs_bucket_spec.rb +18 -12
  177. data/spec/mongo/grid/stream/write_spec.rb +3 -9
  178. data/spec/mongo/grid/stream_spec.rb +1 -1
  179. data/spec/mongo/index/view_spec.rb +2 -2
  180. data/spec/mongo/operation/delete/op_msg_spec.rb +1 -1
  181. data/spec/mongo/{collection/view → operation/find}/builder/flags_spec.rb +2 -2
  182. data/spec/mongo/{collection/view → operation/find}/builder/modifiers_spec.rb +2 -2
  183. data/spec/mongo/operation/find/legacy_spec.rb +1 -0
  184. data/spec/mongo/operation/insert/bulk_spec.rb +1 -1
  185. data/spec/mongo/operation/insert/op_msg_spec.rb +1 -1
  186. data/spec/mongo/operation/kill_cursors_spec.rb +4 -1
  187. data/spec/mongo/operation/read_preference_legacy_spec.rb +4 -0
  188. data/spec/mongo/operation/read_preference_op_msg_spec.rb +2 -0
  189. data/spec/mongo/operation/update/bulk_spec.rb +1 -1
  190. data/spec/mongo/operation/update/op_msg_spec.rb +1 -1
  191. data/spec/mongo/query_cache_spec.rb +6 -2
  192. data/spec/mongo/server/connection_common_spec.rb +62 -11
  193. data/spec/mongo/server/connection_pool_spec.rb +73 -7
  194. data/spec/mongo/server/connection_spec.rb +138 -43
  195. data/spec/mongo/server/description_spec.rb +1 -1
  196. data/spec/mongo/server/monitor_spec.rb +4 -3
  197. data/spec/mongo/session/session_pool_spec.rb +42 -10
  198. data/spec/mongo/session_transaction_spec.rb +15 -30
  199. data/spec/mongo/socket/unix_spec.rb +1 -0
  200. data/spec/mongo/uri_option_parsing_spec.rb +38 -5
  201. data/spec/runners/change_streams/test.rb +1 -1
  202. data/spec/runners/cmap.rb +1 -1
  203. data/spec/runners/connection_string.rb +7 -3
  204. data/spec/runners/crud/operation.rb +5 -3
  205. data/spec/runners/crud/requirement.rb +1 -0
  206. data/spec/runners/crud.rb +1 -1
  207. data/spec/runners/sdam.rb +2 -1
  208. data/spec/runners/transactions/test.rb +2 -2
  209. data/spec/runners/unified/assertions.rb +2 -3
  210. data/spec/runners/unified/event_subscriber.rb +2 -2
  211. data/spec/runners/unified/test.rb +3 -0
  212. data/spec/runners/unified.rb +1 -1
  213. data/spec/shared/lib/mrss/constraints.rb +11 -5
  214. data/spec/shared/lib/mrss/event_subscriber.rb +200 -0
  215. data/spec/shared/lib/mrss/server_version_registry.rb +17 -12
  216. data/spec/shared/share/Dockerfile.erb +2 -1
  217. data/spec/shared/shlib/server.sh +70 -20
  218. data/spec/spec_tests/change_streams_spec.rb +1 -1
  219. data/spec/spec_tests/cmap_spec.rb +4 -1
  220. data/spec/spec_tests/command_monitoring_spec.rb +2 -2
  221. data/spec/spec_tests/data/command_monitoring/find.yml +9 -9
  222. data/spec/spec_tests/data/crud/read/aggregate-collation.yml +2 -1
  223. data/spec/spec_tests/data/crud/read/aggregate-out.yml +1 -0
  224. data/spec/spec_tests/data/crud/read/count-collation.yml +2 -1
  225. data/spec/spec_tests/data/crud/read/distinct-collation.yml +2 -1
  226. data/spec/spec_tests/data/crud/read/find-collation.yml +2 -1
  227. data/spec/spec_tests/data/crud/write/bulkWrite-collation.yml +2 -1
  228. data/spec/spec_tests/data/crud/write/deleteMany-collation.yml +2 -1
  229. data/spec/spec_tests/data/crud/write/deleteOne-collation.yml +2 -1
  230. data/spec/spec_tests/data/crud/write/findOneAndDelete-collation.yml +3 -2
  231. data/spec/spec_tests/data/crud/write/findOneAndReplace-collation.yml +2 -1
  232. data/spec/spec_tests/data/crud/write/findOneAndUpdate-collation.yml +3 -2
  233. data/spec/spec_tests/data/crud/write/replaceOne-collation.yml +3 -2
  234. data/spec/spec_tests/data/crud/write/updateMany-collation.yml +2 -1
  235. data/spec/spec_tests/data/crud/write/updateOne-collation.yml +2 -1
  236. data/spec/spec_tests/data/load_balancers/event-monitoring.yml +99 -0
  237. data/spec/spec_tests/data/load_balancers/lb-connection-establishment.yml +36 -0
  238. data/spec/spec_tests/data/load_balancers/non-lb-connection-establishment.yml +56 -0
  239. data/spec/spec_tests/data/load_balancers/server-selection.yml +50 -0
  240. data/spec/spec_tests/data/retryable_reads/changeStreams-client.watch-serverErrors.yml +1 -1
  241. data/spec/spec_tests/data/retryable_reads/changeStreams-client.watch.yml +1 -1
  242. data/spec/spec_tests/data/retryable_reads/changeStreams-db.coll.watch-serverErrors.yml +1 -1
  243. data/spec/spec_tests/data/retryable_reads/changeStreams-db.coll.watch.yml +1 -1
  244. data/spec/spec_tests/data/retryable_reads/changeStreams-db.watch-serverErrors.yml +1 -1
  245. data/spec/spec_tests/data/retryable_reads/changeStreams-db.watch.yml +1 -1
  246. data/spec/spec_tests/data/retryable_reads/mapReduce.yml +3 -1
  247. data/spec/spec_tests/data/sdam/load-balanced/discover_load_balancer.yml +25 -0
  248. data/spec/spec_tests/data/sdam_integration/find-network-error.yml +2 -0
  249. data/spec/spec_tests/data/sdam_integration/find-shutdown-error.yml +2 -0
  250. data/spec/spec_tests/data/sdam_integration/hello-command-error.yml +3 -1
  251. data/spec/spec_tests/data/sdam_integration/hello-network-error.yml +3 -1
  252. data/spec/spec_tests/data/sdam_integration/hello-timeout.yml +2 -0
  253. data/spec/spec_tests/data/sdam_integration/insert-network-error.yml +2 -0
  254. data/spec/spec_tests/data/sdam_integration/insert-shutdown-error.yml +2 -0
  255. data/spec/spec_tests/data/sdam_monitoring/load_balancer.yml +65 -0
  256. data/spec/spec_tests/data/seed_list_discovery/load-balanced/loadBalanced-directConnection.yml +13 -0
  257. data/spec/spec_tests/data/seed_list_discovery/load-balanced/loadBalanced-replicaSet-errors.yml +6 -0
  258. data/spec/spec_tests/data/seed_list_discovery/load-balanced/loadBalanced-true-multiple-hosts.yml +5 -0
  259. data/spec/spec_tests/data/seed_list_discovery/load-balanced/loadBalanced-true-txt.yml +10 -0
  260. data/spec/spec_tests/data/{dns_seedlist_discovery → seed_list_discovery/replica-set}/direct-connection-false.yml +0 -0
  261. data/spec/spec_tests/data/{dns_seedlist_discovery → seed_list_discovery/replica-set}/direct-connection-true.yml +0 -0
  262. data/spec/spec_tests/data/seed_list_discovery/replica-set/encoded-userinfo-and-db.yml +15 -0
  263. data/spec/spec_tests/data/seed_list_discovery/replica-set/loadBalanced-false-txt.yml +10 -0
  264. data/spec/spec_tests/data/{dns_seedlist_discovery → seed_list_discovery/replica-set}/longer-parent-in-return.yml +0 -0
  265. data/spec/spec_tests/data/{dns_seedlist_discovery → seed_list_discovery/replica-set}/misformatted-option.yml +0 -0
  266. data/spec/spec_tests/data/{dns_seedlist_discovery → seed_list_discovery/replica-set}/no-results.yml +0 -0
  267. data/spec/spec_tests/data/{dns_seedlist_discovery → seed_list_discovery/replica-set}/not-enough-parts.yml +0 -0
  268. data/spec/spec_tests/data/{dns_seedlist_discovery → seed_list_discovery/replica-set}/one-result-default-port.yml +0 -0
  269. data/spec/spec_tests/data/{dns_seedlist_discovery → seed_list_discovery/replica-set}/one-txt-record-multiple-strings.yml +0 -0
  270. data/spec/spec_tests/data/{dns_seedlist_discovery → seed_list_discovery/replica-set}/one-txt-record.yml +0 -0
  271. data/spec/spec_tests/data/{dns_seedlist_discovery → seed_list_discovery/replica-set}/parent-part-mismatch1.yml +0 -0
  272. data/spec/spec_tests/data/{dns_seedlist_discovery → seed_list_discovery/replica-set}/parent-part-mismatch2.yml +0 -0
  273. data/spec/spec_tests/data/{dns_seedlist_discovery → seed_list_discovery/replica-set}/parent-part-mismatch3.yml +0 -0
  274. data/spec/spec_tests/data/{dns_seedlist_discovery → seed_list_discovery/replica-set}/parent-part-mismatch4.yml +0 -0
  275. data/spec/spec_tests/data/{dns_seedlist_discovery → seed_list_discovery/replica-set}/parent-part-mismatch5.yml +0 -0
  276. data/spec/spec_tests/data/{dns_seedlist_discovery → seed_list_discovery/replica-set}/returned-parent-too-short.yml +0 -0
  277. data/spec/spec_tests/data/{dns_seedlist_discovery → seed_list_discovery/replica-set}/returned-parent-wrong.yml +0 -0
  278. data/spec/spec_tests/data/{dns_seedlist_discovery → seed_list_discovery/replica-set}/two-results-default-port.yml +0 -0
  279. data/spec/spec_tests/data/{dns_seedlist_discovery → seed_list_discovery/replica-set}/two-results-nonstandard-port.yml +0 -0
  280. data/spec/spec_tests/data/{dns_seedlist_discovery → seed_list_discovery/replica-set}/two-txt-records.yml +0 -0
  281. data/spec/spec_tests/data/{dns_seedlist_discovery → seed_list_discovery/replica-set}/txt-record-not-allowed-option.yml +0 -0
  282. data/spec/spec_tests/data/{dns_seedlist_discovery → seed_list_discovery/replica-set}/txt-record-with-overridden-ssl-option.yml +0 -0
  283. data/spec/spec_tests/data/{dns_seedlist_discovery → seed_list_discovery/replica-set}/txt-record-with-overridden-uri-option.yml +0 -0
  284. data/spec/spec_tests/data/{dns_seedlist_discovery → seed_list_discovery/replica-set}/txt-record-with-unallowed-option.yml +0 -0
  285. data/spec/spec_tests/data/seed_list_discovery/replica-set/uri-with-admin-database.yml +13 -0
  286. data/spec/spec_tests/data/seed_list_discovery/replica-set/uri-with-auth.yml +12 -0
  287. data/spec/spec_tests/data/{dns_seedlist_discovery → seed_list_discovery/replica-set}/uri-with-port.yml +0 -0
  288. data/spec/spec_tests/data/{dns_seedlist_discovery → seed_list_discovery/replica-set}/uri-with-two-hosts.yml +0 -0
  289. data/spec/spec_tests/data/transactions/retryable-abort-errorLabels.yml +2 -0
  290. data/spec/spec_tests/data/transactions/retryable-abort.yml +2 -0
  291. data/spec/spec_tests/data/transactions/retryable-commit-errorLabels.yml +2 -0
  292. data/spec/spec_tests/data/transactions/retryable-commit.yml +2 -0
  293. data/spec/spec_tests/data/transactions/retryable-writes.yml +2 -0
  294. data/spec/spec_tests/data/uri_options/connection-options.yml +60 -0
  295. data/spec/spec_tests/load_balancers_spec.rb +15 -0
  296. data/spec/spec_tests/retryable_reads_spec.rb +2 -2
  297. data/spec/spec_tests/retryable_writes_spec.rb +1 -1
  298. data/spec/spec_tests/sdam_integration_spec.rb +1 -1
  299. data/spec/spec_tests/sdam_monitoring_spec.rb +10 -5
  300. data/spec/spec_tests/sdam_spec.rb +1 -1
  301. data/spec/spec_tests/seed_list_discovery_spec.rb +118 -0
  302. data/spec/spec_tests/uri_options_spec.rb +4 -4
  303. data/spec/stress/fork_reconnect_stress_spec.rb +1 -5
  304. data/spec/support/certificates/atlas-ocsp-ca.crt +82 -90
  305. data/spec/support/certificates/atlas-ocsp.crt +127 -122
  306. data/spec/support/common_shortcuts.rb +2 -3
  307. data/spec/support/matchers.rb +13 -0
  308. data/spec/support/shared/auth_context.rb +16 -0
  309. data/spec/support/shared/session.rb +2 -2
  310. data/spec/support/spec_config.rb +9 -10
  311. data/spec/support/using_hash.rb +31 -0
  312. data/spec/support/utils.rb +1 -1
  313. data.tar.gz.sig +0 -0
  314. metadata +120 -76
  315. metadata.gz.sig +0 -0
  316. data/lib/mongo/collection/view/builder/find_command.rb +0 -173
  317. data/lib/mongo/collection/view/builder/op_query.rb +0 -94
  318. data/lib/mongo/cursor/builder/get_more_command.rb +0 -80
  319. data/lib/mongo/cursor/builder/kill_cursors_command.rb +0 -111
  320. data/lib/mongo/cursor/builder/op_get_more.rb +0 -64
  321. data/lib/mongo/cursor/builder/op_kill_cursors.rb +0 -106
  322. data/lib/mongo/server/context.rb +0 -72
  323. data/spec/mongo/cursor/builder/op_kill_cursors_spec.rb +0 -66
  324. data/spec/runners/unified/using_hash.rb +0 -34
  325. data/spec/spec_tests/dns_seedlist_discovery_spec.rb +0 -79
  326. data/spec/support/event_subscriber.rb +0 -221
@@ -32,13 +32,16 @@ module Mongo
32
32
  include BypassDocumentValidation
33
33
  include ExecutableNoValidate
34
34
  include PolymorphicResult
35
+ include Validatable
35
36
 
36
37
  private
37
38
 
38
39
  def selector(connection)
39
- { update: coll_name,
40
- updates: send(IDENTIFIER),
41
- ordered: ordered? }
40
+ {
41
+ update: coll_name,
42
+ updates: validate_updates(connection, send(IDENTIFIER)),
43
+ ordered: ordered?,
44
+ }
42
45
  end
43
46
 
44
47
  def message(connection)
@@ -30,24 +30,32 @@ module Mongo
30
30
  include Specifiable
31
31
  include Executable
32
32
  include PolymorphicResult
33
+ include Validatable
33
34
 
34
35
  private
35
36
 
36
- def selector
37
- send(IDENTIFIER).first
37
+ def selector(connection)
38
+ # This returns the first update.
39
+ # The driver only puts one update into the list normally, so this
40
+ # doesn't discard operations.
41
+ send(IDENTIFIER).first.tap do |selector|
42
+ validate_find_options(connection, selector)
43
+ end
38
44
  end
39
45
 
40
46
  def message(connection)
47
+ selector = selector(connection)
48
+
41
49
  flags = []
42
50
  flags << :multi_update if selector[Operation::MULTI]
43
51
  flags << :upsert if selector[Operation::UPSERT]
44
52
 
45
53
  Protocol::Update.new(
46
- db_name,
47
- coll_name,
48
- selector[Operation::Q],
49
- selector[Operation::U],
50
- flags.empty? ? {} : { flags: flags }
54
+ db_name,
55
+ coll_name,
56
+ selector[Operation::Q],
57
+ selector[Operation::U],
58
+ flags.empty? ? {} : { flags: flags },
51
59
  )
52
60
  end
53
61
 
@@ -55,10 +63,10 @@ module Mongo
55
63
  wc = write_concern || WriteConcern.get(WriteConcern::DEFAULT)
56
64
  if gle_message = wc.get_last_error
57
65
  Protocol::Query.new(
58
- db_name,
59
- Database::COMMAND,
60
- gle_message,
61
- options.merge(limit: -1)
66
+ db_name,
67
+ Database::COMMAND,
68
+ gle_message,
69
+ options.merge(limit: -1),
62
70
  )
63
71
  end
64
72
  end
@@ -29,17 +29,20 @@ module Mongo
29
29
  include ExecutableNoValidate
30
30
  include ExecutableTransactionLabel
31
31
  include PolymorphicResult
32
+ include Validatable
32
33
 
33
34
  private
34
35
 
35
36
  def selector(connection)
36
- { update: coll_name,
37
- Protocol::Msg::DATABASE_IDENTIFIER => db_name,
38
- ordered: ordered? }
37
+ {
38
+ update: coll_name,
39
+ ordered: ordered?,
40
+ }
39
41
  end
40
42
 
41
43
  def message(connection)
42
- section = Protocol::Msg::Section1.new(IDENTIFIER, send(IDENTIFIER))
44
+ updates = validate_updates(connection, send(IDENTIFIER))
45
+ section = Protocol::Msg::Section1.new(IDENTIFIER, updates)
43
46
  Protocol::Msg.new(flags, {}, command(connection), section)
44
47
  end
45
48
  end
@@ -0,0 +1,51 @@
1
+ # frozen_string_literal: true
2
+ # encoding: utf-8
3
+
4
+ # Copyright (C) 2021 MongoDB Inc.
5
+ #
6
+ # Licensed under the Apache License, Version 2.0 (the "License");
7
+ # you may not use this file except in compliance with the License.
8
+ # You may obtain a copy of the License at
9
+ #
10
+ # http://www.apache.org/licenses/LICENSE-2.0
11
+ #
12
+ # Unless required by applicable law or agreed to in writing, software
13
+ # distributed under the License is distributed on an "AS IS" BASIS,
14
+ # WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
15
+ # See the License for the specific language governing permissions and
16
+ # limitations under the License.
17
+
18
+ module Mongo
19
+ module Operation
20
+ class WriteCommand
21
+
22
+ # A MongoDB write command operation sent as a command message.
23
+ #
24
+ # @api private
25
+ class Command
26
+ include Specifiable
27
+ include Executable
28
+ include Limited
29
+ include ReadPreferenceSupported
30
+ include Validatable
31
+
32
+ private
33
+
34
+ def selector(connection)
35
+ super.tap do |selector|
36
+ if selector.key?(:findAndModify)
37
+ validate_find_options(connection, selector)
38
+ end
39
+ if wc = spec[:write_concern]
40
+ selector[:writeConcern] = wc.options
41
+ end
42
+ end
43
+ end
44
+
45
+ def message(connection)
46
+ Protocol::Query.new(db_name, Database::COMMAND, command(connection), options(connection))
47
+ end
48
+ end
49
+ end
50
+ end
51
+ end
@@ -0,0 +1,43 @@
1
+ # frozen_string_literal: true
2
+ # encoding: utf-8
3
+
4
+ # Copyright (C) 2021 MongoDB Inc.
5
+ #
6
+ # Licensed under the Apache License, Version 2.0 (the "License");
7
+ # you may not use this file except in compliance with the License.
8
+ # You may obtain a copy of the License at
9
+ #
10
+ # http://www.apache.org/licenses/LICENSE-2.0
11
+ #
12
+ # Unless required by applicable law or agreed to in writing, software
13
+ # distributed under the License is distributed on an "AS IS" BASIS,
14
+ # WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
15
+ # See the License for the specific language governing permissions and
16
+ # limitations under the License.
17
+
18
+ module Mongo
19
+ module Operation
20
+ class WriteCommand
21
+
22
+ # A MongoDB write command operation sent as an op message.
23
+ #
24
+ # @api private
25
+ class OpMsg < OpMsgBase
26
+ include Validatable
27
+
28
+ private
29
+
30
+ def selector(connection)
31
+ super.tap do |selector|
32
+ if selector.key?(:findAndModify)
33
+ validate_find_options(connection, selector)
34
+ end
35
+ if wc = spec[:write_concern]
36
+ selector[:writeConcern] = wc.options
37
+ end
38
+ end
39
+ end
40
+ end
41
+ end
42
+ end
43
+ end
@@ -0,0 +1,32 @@
1
+ # frozen_string_literal: true
2
+ # encoding: utf-8
3
+
4
+ # Copyright (C) 2021 MongoDB Inc.
5
+ #
6
+ # Licensed under the Apache License, Version 2.0 (the "License");
7
+ # you may not use this file except in compliance with the License.
8
+ # You may obtain a copy of the License at
9
+ #
10
+ # http://www.apache.org/licenses/LICENSE-2.0
11
+ #
12
+ # Unless required by applicable law or agreed to in writing, software
13
+ # distributed under the License is distributed on an "AS IS" BASIS,
14
+ # WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
15
+ # See the License for the specific language governing permissions and
16
+ # limitations under the License.
17
+
18
+ require 'mongo/operation/write_command/command'
19
+ require 'mongo/operation/write_command/op_msg'
20
+
21
+ module Mongo
22
+ module Operation
23
+
24
+ # A MongoDB general command operation.
25
+ #
26
+ # @api private
27
+ class WriteCommand
28
+ include Specifiable
29
+ include OpMsgOrCommand
30
+ end
31
+ end
32
+ end
@@ -20,12 +20,14 @@ require 'mongo/operation/shared/causal_consistency_supported'
20
20
  require 'mongo/operation/shared/write'
21
21
  require 'mongo/operation/shared/idable'
22
22
  require 'mongo/operation/shared/specifiable'
23
+ require 'mongo/operation/shared/validatable'
23
24
  require 'mongo/operation/shared/object_id_generator'
24
25
  require 'mongo/operation/shared/op_msg_or_command'
25
26
  require 'mongo/operation/shared/op_msg_or_find_command'
26
27
 
27
28
  require 'mongo/operation/op_msg_base'
28
29
  require 'mongo/operation/command'
30
+ require 'mongo/operation/write_command'
29
31
  require 'mongo/operation/aggregate'
30
32
  require 'mongo/operation/result'
31
33
  require 'mongo/operation/collections_info'
@@ -54,6 +56,14 @@ require 'mongo/operation/drop_index'
54
56
 
55
57
  module Mongo
56
58
 
59
+ # This module encapsulates all of the operation classes defined by the driver.
60
+ #
61
+ # The operation classes take Ruby options as constructor parameters.
62
+ # For example, :read contains read preference and :read_concern contains read
63
+ # concern, whereas server commands use readConcern field for the read
64
+ # concern and read preference is passed as $readPreference or secondaryOk
65
+ # wire protocol flag bit.
66
+ #
57
67
  # @api private
58
68
  module Operation
59
69
 
@@ -64,13 +64,21 @@ module Mongo
64
64
  def initialize(database, collection, selector, options = {})
65
65
  @database = database
66
66
  @namespace = "#{database}.#{collection}"
67
+ if selector.nil?
68
+ raise ArgumentError, 'Selector cannot be nil'
69
+ end
67
70
  @selector = selector
68
71
  @options = options
69
72
  @project = options[:project]
70
73
  @limit = determine_limit
71
74
  @skip = options[:skip] || 0
72
75
  @flags = options[:flags] || []
73
- @upconverter = Upconverter.new(collection, selector, options, flags)
76
+ @upconverter = Upconverter.new(
77
+ collection,
78
+ BSON::Document.new(selector),
79
+ BSON::Document.new(options),
80
+ flags,
81
+ )
74
82
  super
75
83
  end
76
84
 
@@ -224,7 +232,7 @@ module Mongo
224
232
  }.freeze
225
233
 
226
234
  SPECIAL_FIELD_MAPPINGS = {
227
- :$readPreference => 'readPreference',
235
+ :$readPreference => '$readPreference',
228
236
  :$orderby => 'sort',
229
237
  :$hint => 'hint',
230
238
  :$comment => 'comment',
@@ -249,16 +257,6 @@ module Mongo
249
257
  :partial => 'allowPartialResults'
250
258
  }.freeze
251
259
 
252
- # Find command constant.
253
- #
254
- # @since 2.1.0
255
- FIND = 'find'.freeze
256
-
257
- # Filter attribute constant.
258
- #
259
- # @since 2.1.0
260
- FILTER = 'filter'.freeze
261
-
262
260
  # @return [ String ] collection The name of the collection.
263
261
  attr_reader :collection
264
262
 
@@ -283,6 +281,15 @@ module Mongo
283
281
  #
284
282
  # @since 2.1.0
285
283
  def initialize(collection, filter, options, flags)
284
+ # Although the docstring claims both hashes and BSON::Documents
285
+ # are acceptable, this class expects the filter and options to
286
+ # contain symbol keys which isn't what the operation layer produces.
287
+ unless BSON::Document === filter
288
+ raise ArgumentError, 'Filter must provide indifferent access'
289
+ end
290
+ unless BSON::Document === options
291
+ raise ArgumentError, 'Options must provide indifferent access'
292
+ end
286
293
  @collection = collection
287
294
  @filter = filter
288
295
  @options = options
@@ -311,7 +318,7 @@ module Mongo
311
318
  #
312
319
  # @since 2.1.0
313
320
  def command_name
314
- ((filter[:$query] || !command?) ? FIND : filter.keys.first).to_s
321
+ ((filter[:$query] || !command?) ? :find : filter.keys.first).to_s
315
322
  end
316
323
 
317
324
  private
@@ -333,15 +340,25 @@ module Mongo
333
340
  end
334
341
 
335
342
  def find_command
336
- document = BSON::Document.new
337
- document.store(FIND, collection)
338
- document.store(FILTER, query_filter)
343
+ document = BSON::Document.new(
344
+ find: collection,
345
+ filter: query_filter,
346
+ )
339
347
  OPTION_MAPPINGS.each do |legacy, option|
340
348
  document.store(option, options[legacy]) unless options[legacy].nil?
341
349
  end
342
- Mongo::Lint.validate_camel_case_read_preference(filter['readPreference'])
350
+ if Lint.enabled?
351
+ filter.each do |k, v|
352
+ unless String === k
353
+ raise Error::LintError, "All keys in filter must be strings: #{filter.inspect}"
354
+ end
355
+ end
356
+ end
357
+ Lint.validate_camel_case_read_preference(filter['readPreference'])
343
358
  SPECIAL_FIELD_MAPPINGS.each do |special, normal|
344
- document.store(normal, filter[special]) unless filter[special].nil?
359
+ unless (v = filter[special]).nil?
360
+ document.store(normal, v)
361
+ end
345
362
  end
346
363
  FLAG_MAPPINGS.each do |legacy, flag|
347
364
  document.store(flag, true) if flags.include?(legacy)
@@ -86,8 +86,12 @@ module Mongo
86
86
  # @param [ Mongo::Server ] server The server the connection is for.
87
87
  # @param [ Hash ] options The connection options.
88
88
  #
89
- # @option options [ Integer ] :generation Connection pool's generation
90
- # for this connection.
89
+ # @option options [ Integer ] :generation The generation of this
90
+ # connection. The generation should only be specified in this option
91
+ # when not in load-balancing mode, and it should be the generation
92
+ # of the connection pool when the connection is created. In
93
+ # load-balancing mode, the generation is set on the connection
94
+ # after the handshake completes.
91
95
  # @option options [ Hash ] :server_api The requested server API version.
92
96
  # This hash can have the following items:
93
97
  # - *:version* -- string
@@ -96,6 +100,10 @@ module Mongo
96
100
  #
97
101
  # @since 2.0.0
98
102
  def initialize(server, options = {})
103
+ if server.load_balancer? && options[:generation]
104
+ raise ArgumentError, "Generation cannot be set when server is a load balancer"
105
+ end
106
+
99
107
  @id = server.next_connection_id
100
108
  @monitoring = server.monitoring
101
109
  @options = options.freeze
@@ -173,6 +181,15 @@ module Mongo
173
181
  # authenticated and be usable.
174
182
  @socket, @description, @compressor = do_connect
175
183
 
184
+ if server.load_balancer?
185
+ if Lint.enabled?
186
+ unless service_id
187
+ raise Error::InternalDriverError, "The connection is to a load balancer and it must have service_id set here, but does not"
188
+ end
189
+ end
190
+ @generation = connection_pool.generation_manager.generation(service_id: service_id)
191
+ end
192
+
176
193
  publish_cmap_event(
177
194
  Monitoring::Event::Cmap::ConnectionReady.new(address, id)
178
195
  )
@@ -314,7 +331,12 @@ module Mongo
314
331
  yield
315
332
  rescue Error::SocketError => e
316
333
  @error = e
317
- @server.unknown!(generation: e.generation, stop_push_monitor: true)
334
+ @server.unknown!(
335
+ generation: e.generation,
336
+ # or description.service_id?
337
+ service_id: e.service_id,
338
+ stop_push_monitor: true,
339
+ )
318
340
  raise
319
341
  rescue Error::SocketTimeoutError => e
320
342
  @error = e
@@ -88,12 +88,20 @@ module Mongo
88
88
  :max_message_size,
89
89
  :mongos?
90
90
 
91
+ # @return [ nil | Object ] The service id, if any.
92
+ def service_id
93
+ description&.service_id
94
+ end
95
+
91
96
  # Connection pool generation from which this connection was created.
92
97
  # May be nil.
93
98
  #
94
99
  # @return [ Integer | nil ] Connection pool generation.
95
100
  def generation
96
- options[:generation]
101
+ # If the connection is to a load balancer, @generation is set
102
+ # after handshake completes. If the connection is to another server
103
+ # type, generation is specified during connection creation.
104
+ @generation || options[:generation]
97
105
  end
98
106
 
99
107
  def app_metadata
@@ -167,6 +175,7 @@ module Mongo
167
175
  socket_object_id: socket.object_id, connection_id: id,
168
176
  connection_generation: generation,
169
177
  server_connection_id: description.server_connection_id,
178
+ service_id: description.service_id,
170
179
  )
171
180
  start = Utils.monotonic_time
172
181
  result = nil
@@ -184,6 +193,7 @@ module Mongo
184
193
  command_failed(nil, address, operation_id, message.payload,
185
194
  e.message, total_duration,
186
195
  started_event: started_event,
196
+ service_id: description.service_id,
187
197
  )
188
198
  raise
189
199
  else
@@ -191,6 +201,7 @@ module Mongo
191
201
  command_completed(result, address, operation_id, message.payload,
192
202
  total_duration,
193
203
  started_event: started_event,
204
+ service_id: description.service_id,
194
205
  )
195
206
  end
196
207
  if result && context.decrypt?
@@ -57,13 +57,15 @@ module Mongo
57
57
  # @param [ Server::AppMetadata ] app_metadata Application metadata
58
58
  # @param [ BSON::Document ] speculative_auth_doc The speculative
59
59
  # authentication document, if any.
60
+ # @param [ true | false ] load_balancer Whether the connection is to
61
+ # a load balancer.
60
62
  # @param server_api [ Hash | nil ] server_api Server API version.
61
63
  #
62
64
  # @return [BSON::Document] Document that should be sent to a server
63
65
  # for handshake purposes.
64
66
  #
65
67
  # @api private
66
- def handshake_document(app_metadata, speculative_auth_doc: nil, server_api: nil)
68
+ def handshake_document(app_metadata, speculative_auth_doc: nil, load_balancer: false, server_api: nil)
67
69
  serv_api = app_metadata.server_api || server_api
68
70
  document = if serv_api
69
71
  HELLO_DOC.merge(Utils.transform_server_api(serv_api))
@@ -74,6 +76,33 @@ module Mongo
74
76
  if speculative_auth_doc
75
77
  doc.update(speculativeAuthenticate: speculative_auth_doc)
76
78
  end
79
+ if load_balancer
80
+ doc.update(loadBalanced: true)
81
+ end
82
+ end
83
+ end
84
+
85
+ # Build a command that should be used for connection handshake.
86
+ #
87
+ # @param [ BSON::Document ] handshake_document Document that should be
88
+ # sent to a server for handshake purpose.
89
+ #
90
+ # @return [ Protocol::Message ] Command that should be sent to a server
91
+ # for handshake purposes.
92
+ #
93
+ # @api private
94
+ def handshake_command(handshake_document)
95
+ if handshake_document['apiVersion']
96
+ Protocol::Msg.new(
97
+ [], {}, handshake_document.merge({'$db' => Database::ADMIN})
98
+ )
99
+ else
100
+ Protocol::Query.new(
101
+ Database::ADMIN,
102
+ Database::COMMAND,
103
+ handshake_document,
104
+ :limit => -1
105
+ )
77
106
  end
78
107
  end
79
108
 
@@ -125,10 +154,18 @@ module Mongo
125
154
  if respond_to?(:id)
126
155
  note << ", connection #{generation}:#{id}"
127
156
  end
157
+ # Non-monitoring connections have service id.
158
+ # Monitoring connections do not.
159
+ if respond_to?(:service_id) && service_id
160
+ note << ", service id #{service_id}"
161
+ end
128
162
  e.add_note(note)
129
163
  if respond_to?(:generation)
130
164
  # Non-monitoring connections
131
165
  e.generation = generation
166
+ if respond_to?(:description)
167
+ e.service_id = service_id
168
+ end
132
169
  end
133
170
  raise e
134
171
  end
@@ -0,0 +1,71 @@
1
+ # frozen_string_literal: true
2
+ # encoding: utf-8
3
+
4
+ # Copyright (C) 2021 MongoDB Inc.
5
+ #
6
+ # Licensed under the Apache License, Version 2.0 (the "License");
7
+ # you may not use this file except in compliance with the License.
8
+ # You may obtain a copy of the License at
9
+ #
10
+ # http://www.apache.org/licenses/LICENSE-2.0
11
+ #
12
+ # Unless required by applicable law or agreed to in writing, software
13
+ # distributed under the License is distributed on an "AS IS" BASIS,
14
+ # WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
15
+ # See the License for the specific language governing permissions and
16
+ # limitations under the License.
17
+
18
+ module Mongo
19
+ class Server
20
+ class ConnectionPool
21
+
22
+ # @api private
23
+ class GenerationManager
24
+
25
+ def initialize(server:)
26
+ @map = Hash.new { |hash, key| hash[key] = 1 }
27
+ @server = server
28
+ @lock = Mutex.new
29
+ end
30
+
31
+ attr_reader :server
32
+
33
+ def generation(service_id: nil)
34
+ if service_id
35
+ unless server.load_balancer?
36
+ raise ArgumentError, "Generation scoping to services is only available in load-balanced mode, but the server at #{server.address} is not a load balancer"
37
+ end
38
+ else
39
+ if server.load_balancer?
40
+ raise ArgumentError, "The server at #{server.address} is a load balancer and therefore does not have a single global generation"
41
+ end
42
+ end
43
+ @lock.synchronize do
44
+ @map[service_id]
45
+ end
46
+ end
47
+
48
+ def bump(service_id: nil)
49
+ @lock.synchronize do
50
+ if service_id
51
+ @map[service_id] += 1
52
+ else
53
+ # When service id is not supplied, one of two things may be
54
+ # happening;
55
+ #
56
+ # 1. The pool is not to a load balancer, in which case we only
57
+ # need to increment the generation for the nil service_id.
58
+ # 2. The pool is to a load balancer, in which case we need to
59
+ # increment the generation for each service.
60
+ #
61
+ # Incrementing everything in the map accomplishes both tasks.
62
+ @map.each do |k, v|
63
+ @map[k] += 1
64
+ end
65
+ end
66
+ end
67
+ end
68
+ end
69
+ end
70
+ end
71
+ end