mongo 2.13.0.beta1 → 2.14.0

Sign up to get free protection for your applications and to get access to all the features.
Files changed (339) hide show
  1. checksums.yaml +4 -4
  2. checksums.yaml.gz.sig +0 -0
  3. data.tar.gz.sig +1 -5
  4. data/Rakefile +50 -9
  5. data/lib/mongo.rb +13 -2
  6. data/lib/mongo/address.rb +1 -1
  7. data/lib/mongo/address/ipv4.rb +1 -1
  8. data/lib/mongo/address/ipv6.rb +1 -1
  9. data/lib/mongo/auth/aws/request.rb +31 -5
  10. data/lib/mongo/bulk_write.rb +18 -0
  11. data/lib/mongo/caching_cursor.rb +74 -0
  12. data/lib/mongo/client.rb +238 -31
  13. data/lib/mongo/cluster.rb +56 -20
  14. data/lib/mongo/cluster/sdam_flow.rb +13 -10
  15. data/lib/mongo/cluster/topology/replica_set_no_primary.rb +3 -2
  16. data/lib/mongo/cluster/topology/sharded.rb +1 -1
  17. data/lib/mongo/cluster/topology/single.rb +2 -2
  18. data/lib/mongo/collection.rb +66 -24
  19. data/lib/mongo/collection/view.rb +24 -20
  20. data/lib/mongo/collection/view/aggregation.rb +25 -4
  21. data/lib/mongo/collection/view/builder/find_command.rb +38 -18
  22. data/lib/mongo/collection/view/explainable.rb +27 -8
  23. data/lib/mongo/collection/view/iterable.rb +72 -12
  24. data/lib/mongo/collection/view/readable.rb +19 -3
  25. data/lib/mongo/collection/view/writable.rb +55 -5
  26. data/lib/mongo/crypt/encryption_io.rb +6 -6
  27. data/lib/mongo/cursor.rb +16 -3
  28. data/lib/mongo/database.rb +37 -4
  29. data/lib/mongo/database/view.rb +18 -3
  30. data/lib/mongo/distinguishing_semaphore.rb +55 -0
  31. data/lib/mongo/error.rb +5 -0
  32. data/lib/mongo/error/invalid_read_concern.rb +28 -0
  33. data/lib/mongo/error/invalid_server_auth_host.rb +22 -0
  34. data/lib/mongo/error/invalid_session.rb +2 -1
  35. data/lib/mongo/error/operation_failure.rb +11 -5
  36. data/lib/mongo/error/server_certificate_revoked.rb +22 -0
  37. data/lib/mongo/error/sessions_not_supported.rb +35 -0
  38. data/lib/mongo/error/unsupported_option.rb +14 -12
  39. data/lib/mongo/event/base.rb +6 -0
  40. data/lib/mongo/grid/file.rb +5 -0
  41. data/lib/mongo/grid/file/chunk.rb +2 -0
  42. data/lib/mongo/grid/fs_bucket.rb +15 -13
  43. data/lib/mongo/grid/stream/write.rb +9 -3
  44. data/lib/mongo/index/view.rb +3 -0
  45. data/lib/mongo/lint.rb +2 -1
  46. data/lib/mongo/logger.rb +3 -3
  47. data/lib/mongo/monitoring.rb +38 -0
  48. data/lib/mongo/monitoring/command_log_subscriber.rb +10 -2
  49. data/lib/mongo/monitoring/event/command_failed.rb +11 -0
  50. data/lib/mongo/monitoring/event/command_started.rb +37 -2
  51. data/lib/mongo/monitoring/event/command_succeeded.rb +11 -0
  52. data/lib/mongo/monitoring/event/server_closed.rb +1 -1
  53. data/lib/mongo/monitoring/event/server_description_changed.rb +27 -4
  54. data/lib/mongo/monitoring/event/server_heartbeat_failed.rb +9 -2
  55. data/lib/mongo/monitoring/event/server_heartbeat_started.rb +9 -2
  56. data/lib/mongo/monitoring/event/server_heartbeat_succeeded.rb +9 -2
  57. data/lib/mongo/monitoring/event/server_opening.rb +1 -1
  58. data/lib/mongo/monitoring/event/topology_changed.rb +1 -1
  59. data/lib/mongo/monitoring/event/topology_closed.rb +1 -1
  60. data/lib/mongo/monitoring/event/topology_opening.rb +1 -1
  61. data/lib/mongo/monitoring/publishable.rb +6 -3
  62. data/lib/mongo/monitoring/server_description_changed_log_subscriber.rb +9 -1
  63. data/lib/mongo/monitoring/topology_changed_log_subscriber.rb +1 -1
  64. data/lib/mongo/operation.rb +2 -0
  65. data/lib/mongo/operation/aggregate/result.rb +9 -8
  66. data/lib/mongo/operation/collections_info/command.rb +5 -0
  67. data/lib/mongo/operation/collections_info/result.rb +18 -1
  68. data/lib/mongo/operation/delete/bulk_result.rb +2 -0
  69. data/lib/mongo/operation/delete/result.rb +3 -0
  70. data/lib/mongo/operation/explain/command.rb +4 -0
  71. data/lib/mongo/operation/explain/legacy.rb +4 -0
  72. data/lib/mongo/operation/explain/op_msg.rb +6 -0
  73. data/lib/mongo/operation/explain/result.rb +3 -0
  74. data/lib/mongo/operation/find/legacy/result.rb +2 -0
  75. data/lib/mongo/operation/find/result.rb +13 -0
  76. data/lib/mongo/operation/get_more/result.rb +3 -0
  77. data/lib/mongo/operation/indexes/result.rb +5 -0
  78. data/lib/mongo/operation/insert/bulk_result.rb +5 -0
  79. data/lib/mongo/operation/insert/result.rb +5 -0
  80. data/lib/mongo/operation/list_collections/result.rb +5 -0
  81. data/lib/mongo/operation/map_reduce/result.rb +10 -0
  82. data/lib/mongo/operation/parallel_scan/result.rb +4 -0
  83. data/lib/mongo/operation/result.rb +35 -6
  84. data/lib/mongo/operation/shared/bypass_document_validation.rb +1 -0
  85. data/lib/mongo/operation/shared/causal_consistency_supported.rb +1 -0
  86. data/lib/mongo/operation/shared/collections_info_or_list_collections.rb +2 -0
  87. data/lib/mongo/operation/shared/executable.rb +1 -0
  88. data/lib/mongo/operation/shared/idable.rb +2 -1
  89. data/lib/mongo/operation/shared/limited.rb +1 -0
  90. data/lib/mongo/operation/shared/object_id_generator.rb +1 -0
  91. data/lib/mongo/operation/shared/result/aggregatable.rb +1 -0
  92. data/lib/mongo/operation/shared/sessions_supported.rb +1 -0
  93. data/lib/mongo/operation/shared/specifiable.rb +1 -0
  94. data/lib/mongo/operation/shared/write.rb +1 -0
  95. data/lib/mongo/operation/shared/write_concern_supported.rb +1 -0
  96. data/lib/mongo/operation/update/legacy/result.rb +7 -0
  97. data/lib/mongo/operation/update/result.rb +8 -0
  98. data/lib/mongo/operation/users_info/result.rb +3 -0
  99. data/lib/mongo/protocol/message.rb +47 -10
  100. data/lib/mongo/protocol/msg.rb +34 -1
  101. data/lib/mongo/protocol/query.rb +36 -0
  102. data/lib/mongo/protocol/serializers.rb +5 -2
  103. data/lib/mongo/query_cache.rb +242 -0
  104. data/lib/mongo/retryable.rb +8 -1
  105. data/lib/mongo/server.rb +15 -4
  106. data/lib/mongo/server/app_metadata.rb +27 -3
  107. data/lib/mongo/server/connection.rb +4 -4
  108. data/lib/mongo/server/connection_base.rb +38 -12
  109. data/lib/mongo/server/connection_common.rb +2 -2
  110. data/lib/mongo/server/connection_pool.rb +3 -0
  111. data/lib/mongo/server/description.rb +13 -1
  112. data/lib/mongo/server/monitor.rb +76 -44
  113. data/lib/mongo/server/monitor/connection.rb +57 -9
  114. data/lib/mongo/server/pending_connection.rb +14 -4
  115. data/lib/mongo/server/push_monitor.rb +173 -0
  116. data/{spec/runners/transactions/context.rb → lib/mongo/server/push_monitor/connection.rb} +9 -14
  117. data/lib/mongo/server_selector.rb +0 -1
  118. data/lib/mongo/server_selector/base.rb +583 -1
  119. data/lib/mongo/server_selector/nearest.rb +1 -6
  120. data/lib/mongo/server_selector/primary.rb +1 -6
  121. data/lib/mongo/server_selector/primary_preferred.rb +7 -10
  122. data/lib/mongo/server_selector/secondary.rb +1 -6
  123. data/lib/mongo/server_selector/secondary_preferred.rb +1 -7
  124. data/lib/mongo/session.rb +7 -1
  125. data/lib/mongo/socket.rb +26 -12
  126. data/lib/mongo/socket/ocsp_cache.rb +97 -0
  127. data/lib/mongo/socket/ocsp_verifier.rb +368 -0
  128. data/lib/mongo/socket/ssl.rb +46 -25
  129. data/lib/mongo/socket/tcp.rb +1 -1
  130. data/lib/mongo/srv/monitor.rb +7 -13
  131. data/lib/mongo/srv/resolver.rb +14 -10
  132. data/lib/mongo/timeout.rb +2 -0
  133. data/lib/mongo/topology_version.rb +9 -0
  134. data/lib/mongo/uri.rb +21 -390
  135. data/lib/mongo/uri/options_mapper.rb +582 -0
  136. data/lib/mongo/uri/srv_protocol.rb +3 -2
  137. data/lib/mongo/utils.rb +73 -0
  138. data/lib/mongo/version.rb +1 -1
  139. data/spec/NOTES.aws-auth.md +12 -7
  140. data/spec/README.aws-auth.md +2 -2
  141. data/spec/README.md +63 -1
  142. data/spec/integration/awaited_ismaster_spec.rb +28 -0
  143. data/spec/integration/bson_symbol_spec.rb +4 -2
  144. data/spec/integration/bulk_write_spec.rb +67 -0
  145. data/spec/integration/change_stream_examples_spec.rb +6 -2
  146. data/spec/integration/change_stream_spec.rb +1 -1
  147. data/spec/integration/check_clean_slate_spec.rb +16 -0
  148. data/spec/integration/client_authentication_options_spec.rb +92 -28
  149. data/spec/integration/client_construction_spec.rb +1 -0
  150. data/spec/integration/client_side_encryption/auto_encryption_bulk_writes_spec.rb +9 -5
  151. data/spec/integration/connect_single_rs_name_spec.rb +5 -2
  152. data/spec/integration/connection_pool_populator_spec.rb +4 -2
  153. data/spec/integration/connection_spec.rb +7 -4
  154. data/spec/integration/crud_spec.rb +4 -4
  155. data/spec/integration/cursor_reaping_spec.rb +54 -18
  156. data/spec/integration/docs_examples_spec.rb +6 -0
  157. data/spec/integration/fork_reconnect_spec.rb +56 -1
  158. data/spec/integration/grid_fs_bucket_spec.rb +48 -0
  159. data/spec/integration/heartbeat_events_spec.rb +4 -23
  160. data/spec/integration/ocsp_connectivity_spec.rb +26 -0
  161. data/spec/integration/ocsp_verifier_cache_spec.rb +188 -0
  162. data/spec/integration/ocsp_verifier_spec.rb +334 -0
  163. data/spec/integration/query_cache_spec.rb +1045 -0
  164. data/spec/integration/query_cache_transactions_spec.rb +190 -0
  165. data/spec/integration/read_concern_spec.rb +1 -1
  166. data/spec/integration/retryable_errors_spec.rb +1 -1
  167. data/spec/integration/retryable_writes/retryable_writes_40_and_newer_spec.rb +1 -0
  168. data/spec/integration/retryable_writes/shared/performs_legacy_retries.rb +4 -2
  169. data/spec/integration/retryable_writes/shared/performs_modern_retries.rb +3 -3
  170. data/spec/integration/retryable_writes/shared/performs_no_retries.rb +2 -2
  171. data/spec/integration/sdam_error_handling_spec.rb +122 -15
  172. data/spec/integration/sdam_events_spec.rb +80 -6
  173. data/spec/integration/sdam_prose_spec.rb +64 -0
  174. data/spec/integration/server_monitor_spec.rb +25 -1
  175. data/spec/integration/server_selection_spec.rb +36 -0
  176. data/spec/integration/size_limit_spec.rb +23 -5
  177. data/spec/integration/srv_monitoring_spec.rb +38 -3
  178. data/spec/integration/srv_spec.rb +56 -0
  179. data/spec/integration/ssl_uri_options_spec.rb +2 -2
  180. data/spec/integration/transactions_examples_spec.rb +17 -7
  181. data/spec/integration/zlib_compression_spec.rb +25 -0
  182. data/spec/lite_spec_helper.rb +20 -9
  183. data/spec/mongo/address_spec.rb +1 -1
  184. data/spec/mongo/auth/aws/request_region_spec.rb +42 -0
  185. data/spec/mongo/auth/aws/request_spec.rb +76 -0
  186. data/spec/mongo/auth/scram_spec.rb +1 -1
  187. data/spec/mongo/auth/user_spec.rb +1 -1
  188. data/spec/mongo/bulk_write_spec.rb +2 -2
  189. data/spec/mongo/caching_cursor_spec.rb +70 -0
  190. data/spec/mongo/client_construction_spec.rb +386 -3
  191. data/spec/mongo/client_encryption_spec.rb +16 -10
  192. data/spec/mongo/client_spec.rb +85 -3
  193. data/spec/mongo/cluster/topology/replica_set_spec.rb +53 -10
  194. data/spec/mongo/cluster/topology/sharded_spec.rb +1 -1
  195. data/spec/mongo/cluster/topology/single_spec.rb +19 -8
  196. data/spec/mongo/cluster/topology/unknown_spec.rb +1 -1
  197. data/spec/mongo/cluster/topology_spec.rb +1 -1
  198. data/spec/mongo/cluster_spec.rb +37 -35
  199. data/spec/mongo/collection/view/change_stream_resume_spec.rb +7 -7
  200. data/spec/mongo/collection/view/explainable_spec.rb +87 -4
  201. data/spec/mongo/collection/view/map_reduce_spec.rb +2 -0
  202. data/spec/mongo/collection/view/readable_spec.rb +36 -0
  203. data/spec/mongo/collection_spec.rb +572 -0
  204. data/spec/mongo/crypt/auto_decryption_context_spec.rb +1 -1
  205. data/spec/mongo/crypt/auto_encryption_context_spec.rb +1 -1
  206. data/spec/mongo/crypt/binary_spec.rb +1 -6
  207. data/spec/mongo/crypt/binding/binary_spec.rb +1 -6
  208. data/spec/mongo/crypt/binding/context_spec.rb +2 -7
  209. data/spec/mongo/crypt/binding/helpers_spec.rb +1 -6
  210. data/spec/mongo/crypt/binding/mongocrypt_spec.rb +2 -7
  211. data/spec/mongo/crypt/binding/status_spec.rb +1 -6
  212. data/spec/mongo/crypt/binding/version_spec.rb +1 -6
  213. data/spec/mongo/crypt/data_key_context_spec.rb +1 -1
  214. data/spec/mongo/crypt/explicit_decryption_context_spec.rb +1 -1
  215. data/spec/mongo/crypt/explicit_encryption_context_spec.rb +1 -1
  216. data/spec/mongo/crypt/status_spec.rb +1 -6
  217. data/spec/mongo/database_spec.rb +353 -8
  218. data/spec/mongo/distinguishing_semaphore_spec.rb +63 -0
  219. data/spec/mongo/error/no_server_available_spec.rb +1 -1
  220. data/spec/mongo/error/operation_failure_spec.rb +40 -0
  221. data/spec/mongo/index/view_spec.rb +148 -2
  222. data/spec/mongo/logger_spec.rb +13 -11
  223. data/spec/mongo/monitoring/event/server_closed_spec.rb +1 -1
  224. data/spec/mongo/monitoring/event/server_description_changed_spec.rb +1 -4
  225. data/spec/mongo/monitoring/event/server_opening_spec.rb +1 -1
  226. data/spec/mongo/monitoring/event/topology_changed_spec.rb +1 -1
  227. data/spec/mongo/monitoring/event/topology_closed_spec.rb +1 -1
  228. data/spec/mongo/monitoring/event/topology_opening_spec.rb +1 -1
  229. data/spec/mongo/operation/delete/op_msg_spec.rb +3 -3
  230. data/spec/mongo/operation/insert/command_spec.rb +2 -2
  231. data/spec/mongo/operation/insert/op_msg_spec.rb +3 -3
  232. data/spec/mongo/operation/read_preference_op_msg_spec.rb +1 -1
  233. data/spec/mongo/operation/update/command_spec.rb +2 -2
  234. data/spec/mongo/operation/update/op_msg_spec.rb +3 -3
  235. data/spec/mongo/protocol/msg_spec.rb +10 -0
  236. data/spec/mongo/query_cache_spec.rb +280 -0
  237. data/spec/mongo/semaphore_spec.rb +51 -0
  238. data/spec/mongo/server/app_metadata_shared.rb +82 -2
  239. data/spec/mongo/server/connection_auth_spec.rb +2 -2
  240. data/spec/mongo/server/connection_pool_spec.rb +7 -3
  241. data/spec/mongo/server/connection_spec.rb +15 -8
  242. data/spec/mongo/server/description_spec.rb +18 -0
  243. data/spec/mongo/server_selector/nearest_spec.rb +23 -23
  244. data/spec/mongo/server_selector/primary_preferred_spec.rb +26 -26
  245. data/spec/mongo/server_selector/primary_spec.rb +9 -9
  246. data/spec/mongo/server_selector/secondary_preferred_spec.rb +22 -22
  247. data/spec/mongo/server_selector/secondary_spec.rb +18 -18
  248. data/spec/mongo/server_selector_spec.rb +6 -6
  249. data/spec/mongo/session_spec.rb +35 -0
  250. data/spec/mongo/socket/ssl_spec.rb +4 -4
  251. data/spec/mongo/socket_spec.rb +1 -1
  252. data/spec/mongo/uri/srv_protocol_spec.rb +64 -33
  253. data/spec/mongo/uri_option_parsing_spec.rb +11 -11
  254. data/spec/mongo/uri_spec.rb +68 -41
  255. data/spec/mongo/utils_spec.rb +39 -0
  256. data/spec/runners/auth.rb +3 -0
  257. data/spec/runners/change_streams/test.rb +3 -3
  258. data/spec/runners/cmap.rb +1 -1
  259. data/spec/runners/command_monitoring.rb +3 -34
  260. data/spec/runners/connection_string.rb +35 -124
  261. data/spec/runners/crud/context.rb +9 -5
  262. data/spec/runners/crud/operation.rb +59 -27
  263. data/spec/runners/crud/spec.rb +0 -8
  264. data/spec/runners/crud/test.rb +1 -1
  265. data/spec/runners/crud/test_base.rb +0 -19
  266. data/spec/runners/sdam.rb +2 -2
  267. data/spec/runners/server_selection.rb +242 -28
  268. data/spec/runners/transactions.rb +12 -12
  269. data/spec/runners/transactions/operation.rb +151 -25
  270. data/spec/runners/transactions/test.rb +62 -18
  271. data/spec/shared/LICENSE +20 -0
  272. data/spec/shared/lib/mrss/child_process_helper.rb +80 -0
  273. data/spec/shared/lib/mrss/constraints.rb +303 -0
  274. data/spec/shared/lib/mrss/lite_constraints.rb +175 -0
  275. data/spec/shared/lib/mrss/spec_organizer.rb +149 -0
  276. data/spec/spec_helper.rb +3 -1
  277. data/spec/spec_tests/cmap_spec.rb +7 -3
  278. data/spec/spec_tests/command_monitoring_spec.rb +22 -12
  279. data/spec/spec_tests/crud_spec.rb +1 -1
  280. data/spec/spec_tests/data/change_streams/change-streams-errors.yml +4 -9
  281. data/spec/spec_tests/data/change_streams/change-streams-resume-whitelist.yml +66 -0
  282. data/spec/spec_tests/data/change_streams/change-streams.yml +0 -1
  283. data/spec/spec_tests/data/cmap/pool-checkout-connection.yml +6 -2
  284. data/spec/spec_tests/data/cmap/pool-create-min-size.yml +3 -0
  285. data/spec/spec_tests/data/connection_string/valid-warnings.yml +24 -0
  286. data/spec/spec_tests/data/max_staleness/ReplicaSetNoPrimary/MaxStalenessTooSmall.yml +15 -0
  287. data/spec/spec_tests/data/max_staleness/ReplicaSetNoPrimary/NoKnownServers.yml +4 -3
  288. data/spec/spec_tests/data/max_staleness/Unknown/SmallMaxStaleness.yml +1 -0
  289. data/spec/spec_tests/data/sdam_integration/cancel-server-check.yml +96 -0
  290. data/spec/spec_tests/data/sdam_integration/connectTimeoutMS.yml +88 -0
  291. data/spec/spec_tests/data/sdam_integration/find-network-error.yml +83 -0
  292. data/spec/spec_tests/data/sdam_integration/find-shutdown-error.yml +116 -0
  293. data/spec/spec_tests/data/sdam_integration/insert-network-error.yml +86 -0
  294. data/spec/spec_tests/data/sdam_integration/insert-shutdown-error.yml +115 -0
  295. data/spec/spec_tests/data/sdam_integration/isMaster-command-error.yml +168 -0
  296. data/spec/spec_tests/data/sdam_integration/isMaster-network-error.yml +162 -0
  297. data/spec/spec_tests/data/sdam_integration/isMaster-timeout.yml +229 -0
  298. data/spec/spec_tests/data/sdam_integration/rediscover-quickly-after-step-down.yml +87 -0
  299. data/spec/spec_tests/data/sdam_monitoring/discovered_standalone.yml +1 -3
  300. data/spec/spec_tests/data/sdam_monitoring/standalone.yml +2 -2
  301. data/spec/spec_tests/data/sdam_monitoring/standalone_repeated.yml +2 -2
  302. data/spec/spec_tests/data/sdam_monitoring/standalone_suppress_equal_description_changes.yml +2 -2
  303. data/spec/spec_tests/data/sdam_monitoring/standalone_to_rs_with_me_mismatch.yml +2 -2
  304. data/spec/spec_tests/data/uri_options/auth-options.yml +25 -0
  305. data/spec/spec_tests/data/uri_options/compression-options.yml +6 -3
  306. data/spec/spec_tests/data/uri_options/read-preference-options.yml +24 -0
  307. data/spec/spec_tests/data/uri_options/ruby-connection-options.yml +1 -0
  308. data/spec/spec_tests/data/uri_options/tls-options.yml +160 -4
  309. data/spec/spec_tests/dns_seedlist_discovery_spec.rb +9 -1
  310. data/spec/spec_tests/max_staleness_spec.rb +4 -142
  311. data/spec/spec_tests/retryable_reads_spec.rb +2 -2
  312. data/spec/spec_tests/sdam_integration_spec.rb +13 -0
  313. data/spec/spec_tests/sdam_monitoring_spec.rb +1 -2
  314. data/spec/spec_tests/server_selection_spec.rb +4 -116
  315. data/spec/spec_tests/uri_options_spec.rb +31 -33
  316. data/spec/stress/cleanup_spec.rb +17 -2
  317. data/spec/stress/connection_pool_stress_spec.rb +10 -8
  318. data/spec/stress/fork_reconnect_stress_spec.rb +1 -1
  319. data/spec/support/certificates/atlas-ocsp-ca.crt +28 -0
  320. data/spec/support/certificates/atlas-ocsp.crt +41 -0
  321. data/spec/support/client_registry.rb +1 -0
  322. data/spec/support/client_registry_macros.rb +11 -2
  323. data/spec/support/cluster_config.rb +4 -0
  324. data/spec/support/common_shortcuts.rb +45 -0
  325. data/spec/support/constraints.rb +6 -253
  326. data/spec/support/event_subscriber.rb +123 -33
  327. data/spec/support/keyword_struct.rb +26 -0
  328. data/spec/support/matchers.rb +16 -0
  329. data/spec/support/ocsp +1 -0
  330. data/spec/support/session_registry.rb +52 -0
  331. data/spec/support/shared/server_selector.rb +13 -1
  332. data/spec/support/spec_config.rb +60 -13
  333. data/spec/support/spec_setup.rb +1 -1
  334. data/spec/support/utils.rb +84 -1
  335. metadata +1027 -937
  336. metadata.gz.sig +0 -0
  337. data/lib/mongo/server_selector/selectable.rb +0 -560
  338. data/spec/runners/sdam_monitoring.rb +0 -89
  339. data/spec/support/lite_constraints.rb +0 -141
@@ -97,7 +97,7 @@ module Mongo
97
97
 
98
98
  # @return [ String ] NO_SRV_RECORDS Error message format string indicating that no SRV records
99
99
  # were found.
100
- NO_SRV_RECORDS = "The DNS query returned no SRV records at hostname (%s)".freeze
100
+ NO_SRV_RECORDS = "The DNS query returned no SRV records for '%s'".freeze
101
101
 
102
102
  # @return [ String ] INVALID_TXT_RECORD_OPTION Error message format string indicating that an
103
103
  # unexpected TXT record option was found.
@@ -130,6 +130,7 @@ module Mongo
130
130
  @resolver ||= Srv::Resolver.new(
131
131
  raise_on_invalid: true,
132
132
  resolv_options: options[:resolv_options],
133
+ timeout: options[:connect_timeout],
133
134
  )
134
135
  end
135
136
 
@@ -220,7 +221,7 @@ module Mongo
220
221
  raise Error::InvalidTXTRecord.new(INVALID_OPTS_VALUE_DELIM) unless opt.index(URI_OPTS_VALUE_DELIM)
221
222
  key, value = opt.split(URI_OPTS_VALUE_DELIM)
222
223
  raise Error::InvalidTXTRecord.new(INVALID_TXT_RECORD_OPTION) unless VALID_TXT_OPTIONS.include?(key.downcase)
223
- add_uri_option(key, value, txt_options)
224
+ options_mapper.add_uri_option(key, value, txt_options)
224
225
  txt_options
225
226
  end
226
227
  end
@@ -0,0 +1,73 @@
1
+ # Copyright (C) 2020 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
+ module Mongo
16
+
17
+ # @api private
18
+ module Utils
19
+
20
+ class LocalLogger
21
+ include Loggable
22
+
23
+ def initialize(**opts)
24
+ @options = opts
25
+ end
26
+
27
+ attr_reader :options
28
+ end
29
+
30
+ # @option opts [ true | false | nil | Integer ] :bg_error_backtrace
31
+ # Experimental. Set to true to log complete backtraces for errors in
32
+ # background threads. Set to false or nil to not log backtraces. Provide
33
+ # a positive integer to log up to that many backtrace lines.
34
+ # @option opts [ Logger ] :logger A custom logger to use.
35
+ # @option opts [ String ] :log_prefix A custom log prefix to use when
36
+ # logging.
37
+ module_function def warn_bg_exception(msg, exc, **opts)
38
+ bt_excerpt = excerpt_backtrace(exc, **opts)
39
+ logger = LocalLogger.new(**opts)
40
+ logger.log_warn("#{msg}: #{exc.class}: #{exc}#{bt_excerpt}")
41
+ end
42
+
43
+ # @option opts [ true | false | nil | Integer ] :bg_error_backtrace
44
+ # Experimental. Set to true to log complete backtraces for errors in
45
+ # background threads. Set to false or nil to not log backtraces. Provide
46
+ # a positive integer to log up to that many backtrace lines.
47
+ module_function def excerpt_backtrace(exc, **opts)
48
+ case lines = opts[:bg_error_backtrace]
49
+ when Integer
50
+ ":\n#{exc.backtrace[0..lines].join("\n")}"
51
+ when false, nil
52
+ nil
53
+ else
54
+ ":\n#{exc.backtrace.join("\n")}"
55
+ end
56
+ end
57
+
58
+ # Symbolizes the keys in the provided hash.
59
+ module_function def shallow_symbolize_keys(hash)
60
+ Hash[hash.map { |k, v| [k.to_sym, v] }]
61
+ end
62
+
63
+ # Stringifies the keys in the provided hash and converts underscore
64
+ # style keys to camel case style keys.
65
+ module_function def shallow_camelize_keys(hash)
66
+ Hash[hash.map { |k, v| [camelize(k), v] }]
67
+ end
68
+
69
+ module_function def camelize(sym)
70
+ sym.to_s.gsub(/_(\w)/) { $1.upcase }
71
+ end
72
+ end
73
+ end
@@ -17,5 +17,5 @@ module Mongo
17
17
  # The current version of the driver.
18
18
  #
19
19
  # @since 2.0.0
20
- VERSION = '2.13.0.beta1'
20
+ VERSION = '2.14.0'.freeze
21
21
  end
@@ -145,13 +145,16 @@ problem. Below are some of the puzzling responses I encountered:
145
145
  line) but the value is otherwise completely valid. This error has no relation
146
146
  to the "session token" or "security token" as used with temporary AWS
147
147
  credentials.
148
- - *The security token included in the request is invalid*: this error is
149
- produced when the AWS access key id, as specified in the scope part of the
150
- `Authorization` header, is not a valid access key id. In the case of
151
- non-temporary credentials being used for authentication, the error refers to
152
- a "security token" but the authentication process does not actually use a
153
- security token as this term is used in the AWS documentation describing
154
- temporary credentials.
148
+ - *The security token included in the request is invalid*: this error can be
149
+ produced in several circumstances:
150
+ - When the AWS access key id, as specified in the scope part of the
151
+ `Authorization` header, is not a valid access key id. In the case of
152
+ non-temporary credentials being used for authentication, the error refers to
153
+ a "security token" but the authentication process does not actually use a
154
+ security token as this term is used in the AWS documentation describing
155
+ temporary credentials.
156
+ - When using temporary credentials and the security token is not provided
157
+ in the STS request at all (x-amz-security-token header).
155
158
  - *Signature expired: 20200317T000000Z is now earlier than 20200317T222541Z
156
159
  (20200317T224041Z - 15 min.)*: This error happens when `x-amz-date` header
157
160
  value is the formatted date (`YYYYMMDD`) rather than the ISO8601 formatted
@@ -168,6 +171,8 @@ problem. Below are some of the puzzling responses I encountered:
168
171
  produced when temporary credentials are used and the credentials have
169
172
  expired.
170
173
 
174
+ See also [AWS documentation for STS error messages](https://docs.aws.amazon.com/STS/latest/APIReference/CommonErrors.html).
175
+
171
176
  ### Resources
172
177
 
173
178
  Generally I found Amazon's own documentation to be the best for implementing
@@ -233,7 +233,7 @@ To provision the instance for running the driver's test suite via Docker, run:
233
233
  To run the AWS auth tests using the EC2 instance role credentials, run:
234
234
 
235
235
  ./.evergreen/test-docker-remote ubuntu@$ip \
236
- MONGODB_VERSION=4.3 AUTH=aws-ec2 \
236
+ MONGODB_VERSION=4.4 AUTH=aws-ec2 \
237
237
  -s .evergreen/run-tests-aws-auth.sh \
238
238
  -a .env.private
239
239
 
@@ -299,7 +299,7 @@ If the public IP address is in the `IP` shell variable, provision the task:
299
299
 
300
300
  To run the credentials retrieval test on the ECS task, execute:
301
301
 
302
- ./.evergreen/test-remote root@$IP env AUTH=aws-ecs RVM_RUBY=ruby-2.7 MONGODB_VERSION=4.3 TEST_CMD='rspec spec/integration/aws*spec.rb' .evergreen/run-tests.sh
302
+ ./.evergreen/test-remote root@$IP env AUTH=aws-ecs RVM_RUBY=ruby-2.7 MONGODB_VERSION=4.4 TEST_CMD='rspec spec/integration/aws*spec.rb' .evergreen/run-tests.sh
303
303
 
304
304
  To run the test again without rebuilding the remote environment, execute:
305
305
 
@@ -2,6 +2,13 @@
2
2
 
3
3
  ## Quick Start
4
4
 
5
+ The test suite requires shared tooling that is stored in a separate repository
6
+ and is referenced as a submodule. After checking out the desired driver
7
+ branch, check out the matching submodules:
8
+
9
+ git submodule init
10
+ git submodule update
11
+
5
12
  To run the test suite against a local MongoDB deployment listening on port
6
13
  27017, run:
7
14
 
@@ -107,7 +114,7 @@ other tests require a sharded cluster with more than one shard. Tests requiring
107
114
  a single shard can be run against a deployment with multiple shards by
108
115
  specifying only one mongos address in MONGODB_URI.
109
116
 
110
- ## Note Regarding SSL/TLS Arguments
117
+ ## Note Regarding TLS/SSL Arguments
111
118
 
112
119
  MongoDB 4.2 (server and shell) added new command line options for setting TLS
113
120
  parameters. These options follow the naming of URI options used by both the
@@ -185,6 +192,51 @@ verification, run:
185
192
  Note that there are tests in the test suite that cover TLS verification, and
186
193
  they may fail if the test suite is run in this way.
187
194
 
195
+ ## OCSP
196
+
197
+ There are several types of OCSP tests implemented in the test suite.
198
+
199
+ OCSP unit tests are in `spec/integration/ocsp_verifier_spec.rb`. To run
200
+ these, set `OCSP_VERIFIER=1` in the environment. There must NOT be a process
201
+ running on the host port 8100 as that port will be used by the OCSP responder
202
+ launched by the tests.
203
+
204
+ For the remaining OCSP tests, the following environment variables must be set
205
+ to the possible values indicated below:
206
+
207
+ OCSP_ALGORITHM=rsa|ecdsa
208
+ OCSP_STATUS=valid|revoked|unknown
209
+ OCSP_DELEGATE=0|1
210
+ OCSP_MUST_STAPLE=0|1
211
+
212
+ These tests also require the mock OCSP responder running on the host machine
213
+ on port 8100 with the configuration that matches the environment variables
214
+ just described. Please refer to the Docker and Evergreen scripts in the
215
+ driver repository for further details.
216
+
217
+ Additionally, the server must be configured to use the appropriate server
218
+ certificate and CA certificate from the respective subdirectory of
219
+ `spec/support/ocsp`. This is easiest to achieve by using the Docker tooling
220
+ described in `.evergreen/README.md`.
221
+
222
+ OCSP connectivity tests are in `spec/integration/ocsp_connectivity.rb`.
223
+ These test the combinations described
224
+ [here](https://github.com/mongodb/specifications/blob/master/source/ocsp-support/tests/README.rst#integration-tests-permutations-to-be-tested).
225
+ To run these tests, set `OCSP_CONNECTIVITY=pass` environment variable if
226
+ the tests are expected to connect successfully or `OCSP_CONNECTIVITY=fail` if
227
+ the tests are expected to not connect.
228
+ Note that some of these configurations require OCSP responder to return
229
+ the failure response; in such configurations, ONLY the OCSP connectivity tests
230
+ may pass (since the driver may reject connections to servers when OCSP
231
+ responder returns the failure response, or OCSP verification otherwise
232
+ definitively fails).
233
+
234
+ When not running either OCSP verifier tests or OCSP connectivity tests but
235
+ when OCSP algorithm is configured, the test suite will execute normally
236
+ using the provided `MONGO_URI`. This configuration may be used to exercise
237
+ OCSP while running the full test suite. In this case, setting `OCSP_STATUS`
238
+ to `revoked` will generally cause the test suite to fail.
239
+
188
240
  ## Authentication
189
241
 
190
242
  mlaunch can configure authentication on the server:
@@ -511,6 +563,16 @@ following environment variable:
511
563
 
512
564
  FORK=1
513
565
 
566
+ OCSP tests require Python 3 with asn1crypto, oscrypto and flask packages
567
+ installed, and they require the drivers-evergreen-tools submodule to be
568
+ checked out. To run these tests, set the following environment variable:
569
+
570
+ OCSP=1
571
+
572
+ To check out the submodule, run:
573
+
574
+ git submodule update --init --recursive
575
+
514
576
  ## Debug Logging
515
577
 
516
578
  The test suite is run with the driver log level set to `WARN` by default.
@@ -0,0 +1,28 @@
1
+ require 'spec_helper'
2
+
3
+ describe 'awaited ismaster' do
4
+ min_server_fcv '4.4'
5
+
6
+ # If we send the consecutive ismasters to different mongoses,
7
+ # they have different process ids, and so the awaited one would return
8
+ # immediately.
9
+ require_no_multi_shard
10
+
11
+ let(:client) { authorized_client }
12
+
13
+ it 'waits' do
14
+ # Perform a regular ismaster to get topology version
15
+ resp = client.database.command(ismaster: 1)
16
+ doc = resp.replies.first.documents.first
17
+ tv = Mongo::TopologyVersion.new(doc['topologyVersion'])
18
+ tv.should be_a(BSON::Document)
19
+
20
+ elapsed_time = Benchmark.realtime do
21
+ resp = client.database.command(ismaster: 1,
22
+ topologyVersion: tv.to_doc, maxAwaitTimeMS: 500)
23
+ end
24
+ doc = resp.replies.first.documents.first
25
+
26
+ elapsed_time.should > 0.5
27
+ end
28
+ end
@@ -25,8 +25,10 @@ describe 'Symbol encoding to BSON' do
25
25
  end
26
26
 
27
27
  it 'round-trips symbol values using the same byte buffer' do
28
- if BSON::Environment.jruby?
29
- pending 'https://jira.mongodb.org/browse/RUBY-2128'
28
+ if BSON::Environment.jruby? && (BSON::VERSION.split('.').map(&:to_i) <=> [4, 11, 0]) < 0
29
+ skip 'This test is only relevant to bson versions that increment ByteBuffer '\
30
+ 'read and write positions separately in JRuby, as implemented in ' \
31
+ 'bson version 4.11.0. For more information, see https://jira.mongodb.org/browse/RUBY-2128'
30
32
  end
31
33
 
32
34
  Hash.from_bson(hash.to_bson).should == hash
@@ -0,0 +1,67 @@
1
+ require 'spec_helper'
2
+
3
+ describe 'Bulk writes' do
4
+ before do
5
+ authorized_collection.drop
6
+ end
7
+
8
+ context 'when bulk write is larger than 48MB' do
9
+ let(:operations) do
10
+ [ { insert_one: { text: 'a' * 1000 * 1000 } } ] * 48
11
+ end
12
+
13
+ it 'succeeds' do
14
+ expect do
15
+ authorized_collection.bulk_write(operations)
16
+ end.not_to raise_error
17
+ end
18
+ end
19
+
20
+ context 'when bulk write needs to be split' do
21
+ let(:subscriber) { EventSubscriber.new }
22
+
23
+ let(:max_bson_size) { Mongo::Server::ConnectionBase::DEFAULT_MAX_BSON_OBJECT_SIZE }
24
+
25
+ let(:insert_events) do
26
+ subscriber.command_started_events('insert')
27
+ end
28
+
29
+ let(:failed_events) do
30
+ subscriber.failed_events
31
+ end
32
+
33
+ let(:operations) do
34
+ [{ insert_one: { text: 'a' * (max_bson_size/2) } }] * 6
35
+ end
36
+
37
+ before do
38
+ authorized_client.subscribe(Mongo::Monitoring::COMMAND, subscriber)
39
+ authorized_collection.bulk_write(operations)
40
+ end
41
+
42
+ context '3.6+ server' do
43
+ min_server_fcv '3.6'
44
+
45
+ it 'splits the operations' do
46
+ # 3.6+ servers can send multiple bulk operations in one message,
47
+ # with the whole message being limited to 48m.
48
+ expect(insert_events.length).to eq(2)
49
+ end
50
+ end
51
+
52
+ context 'pre-3.6 server' do
53
+ max_server_version '3.4'
54
+
55
+ it 'splits the operations' do
56
+ # Pre-3.6 servers limit the entire message payload to the size of
57
+ # a single document which is 16m. Given our test data this means
58
+ # twice as many messages are sent.
59
+ expect(insert_events.length).to eq(4)
60
+ end
61
+ end
62
+
63
+ it 'does not have a command failed event' do
64
+ expect(failed_events).to be_empty
65
+ end
66
+ end
67
+ end
@@ -174,8 +174,10 @@ describe 'change streams examples in Ruby' do
174
174
 
175
175
  # Start Changestream Example 4
176
176
 
177
- pipeline = [ {'$match' => { '$or' => [{ 'fullDocument.username' => 'alice' },
178
- { 'operationType' => 'delete' }] } }]
177
+ pipeline = [
178
+ { "$match" => { 'fullDocument.username' => 'alice' } },
179
+ { "$addFields" => { 'newField' => 'this is an added field!' } }
180
+ ];
179
181
  cursor = inventory.watch(pipeline).to_enum
180
182
  cursor.next
181
183
 
@@ -191,6 +193,8 @@ describe 'change streams examples in Ruby' do
191
193
  expect(change['fullDocument']).not_to be_nil
192
194
  expect(change['fullDocument']['_id']).not_to be_nil
193
195
  expect(change['fullDocument']['username']).to eq('alice')
196
+ expect(change['newField']).not_to be_nil
197
+ expect(change['newField']).to eq('this is an added field!')
194
198
  expect(change['ns']).not_to be_nil
195
199
  expect(change['ns']['db']).to eq(SpecConfig.instance.test_db)
196
200
  expect(change['ns']['coll']).to eq(inventory.name)
@@ -1,7 +1,7 @@
1
1
  require 'spec_helper'
2
2
 
3
3
  describe 'Change stream integration', retry: 4 do
4
- only_mri
4
+ require_mri
5
5
  max_example_run_time 7
6
6
  min_server_fcv '3.6'
7
7
  require_topology :replica_set
@@ -0,0 +1,16 @@
1
+ require 'lite_spec_helper'
2
+
3
+ # This test can be used to manually verify that there are no leaked
4
+ # background threads - execute it after executing another test (in the same
5
+ # rspec run) that is suspected to leak background threads, such as by
6
+ # running:
7
+ #
8
+ # rspec your_spec.rb spec/integration/check_clean_slate_spec.rb
9
+
10
+ describe 'Check clean slate' do
11
+ clean_slate_for_all_if_possible
12
+
13
+ it 'checks' do
14
+ # Nothing
15
+ end
16
+ end
@@ -131,7 +131,7 @@ describe 'Client authentication options' do
131
131
  end
132
132
  end
133
133
 
134
- shared_examples_for 'an auth mechanism that doesn\'t support auth_mech_properties' do
134
+ shared_examples_for 'an auth mechanism that does not support auth_mech_properties' do
135
135
  context 'with URI options' do
136
136
  let(:credentials) { "#{user}:#{pwd}@" }
137
137
  let(:options) { "?authMechanism=#{auth_mech_string}&authMechanismProperties=CANONICALIZE_HOST_NAME:true" }
@@ -163,7 +163,7 @@ describe 'Client authentication options' do
163
163
  end
164
164
  end
165
165
 
166
- shared_examples_for 'an auth mechanism that doesn\'t support invalid auth sources' do
166
+ shared_examples_for 'an auth mechanism that does not support invalid auth sources' do
167
167
  context 'with URI options' do
168
168
  let(:credentials) { "#{user}:#{pwd}@" }
169
169
  let(:options) { "?authMechanism=#{auth_mech_string}&authSource=foo" }
@@ -199,7 +199,7 @@ describe 'Client authentication options' do
199
199
 
200
200
  it_behaves_like 'a supported auth mechanism'
201
201
  it_behaves_like 'auth mechanism that uses database or default auth source', 'admin'
202
- it_behaves_like 'an auth mechanism that doesn\'t support auth_mech_properties'
202
+ it_behaves_like 'an auth mechanism that does not support auth_mech_properties'
203
203
  end
204
204
 
205
205
  context 'with SCRAM-SHA-1 auth mechanism' do
@@ -208,7 +208,7 @@ describe 'Client authentication options' do
208
208
 
209
209
  it_behaves_like 'a supported auth mechanism'
210
210
  it_behaves_like 'auth mechanism that uses database or default auth source', 'admin'
211
- it_behaves_like 'an auth mechanism that doesn\'t support auth_mech_properties'
211
+ it_behaves_like 'an auth mechanism that does not support auth_mech_properties'
212
212
  end
213
213
 
214
214
  context 'with SCRAM-SHA-256 auth mechanism' do
@@ -217,7 +217,7 @@ describe 'Client authentication options' do
217
217
 
218
218
  it_behaves_like 'a supported auth mechanism'
219
219
  it_behaves_like 'auth mechanism that uses database or default auth source', 'admin'
220
- it_behaves_like 'an auth mechanism that doesn\'t support auth_mech_properties'
220
+ it_behaves_like 'an auth mechanism that does not support auth_mech_properties'
221
221
  end
222
222
 
223
223
  context 'with GSSAPI auth mechanism' do
@@ -227,7 +227,7 @@ describe 'Client authentication options' do
227
227
  let(:auth_mech_sym) { :gssapi }
228
228
 
229
229
  it_behaves_like 'a supported auth mechanism'
230
- it_behaves_like 'an auth mechanism that doesn\'t support invalid auth sources'
230
+ it_behaves_like 'an auth mechanism that does not support invalid auth sources'
231
231
 
232
232
  let(:auth_mech_properties) { { canonicalize_host_name: true, service_name: 'other'} }
233
233
 
@@ -256,6 +256,43 @@ describe 'Client authentication options' do
256
256
  expect(client.options[:auth_mech_properties]).to eq({ 'service_name' => 'mongodb' })
257
257
  end
258
258
  end
259
+
260
+ context 'when properties are given but not service name' do
261
+ context 'with URI options' do
262
+ let(:credentials) { "#{user}:#{pwd}@" }
263
+
264
+ context 'with default auth mech properties' do
265
+ let(:options) { '?authMechanism=GSSAPI&authMechanismProperties=service_realm:foo' }
266
+
267
+ it 'sets service name to mongodb' do
268
+ expect(client.options[:auth_mech_properties]).to eq(
269
+ 'service_name' => 'mongodb',
270
+ 'service_realm' => 'foo',
271
+ )
272
+ end
273
+ end
274
+ end
275
+
276
+ context 'with client options' do
277
+ let(:client_opts) do
278
+ {
279
+ auth_mech: :gssapi,
280
+ user: user,
281
+ password: pwd,
282
+ auth_mech_properties: {
283
+ service_realm: 'foo',
284
+ }.freeze,
285
+ }.freeze
286
+ end
287
+
288
+ it 'sets default auth mech properties' do
289
+ expect(client.options[:auth_mech_properties]).to eq(
290
+ 'service_name' => 'mongodb',
291
+ 'service_realm' => 'foo',
292
+ )
293
+ end
294
+ end
295
+ end
259
296
  end
260
297
 
261
298
  context 'with PLAIN auth mechanism' do
@@ -265,7 +302,7 @@ describe 'Client authentication options' do
265
302
  it_behaves_like 'a supported auth mechanism'
266
303
  it_behaves_like 'auth mechanism that uses database or default auth source', '$external'
267
304
  it_behaves_like 'an auth mechanism with ssl'
268
- it_behaves_like 'an auth mechanism that doesn\'t support auth_mech_properties'
305
+ it_behaves_like 'an auth mechanism that does not support auth_mech_properties'
269
306
  end
270
307
 
271
308
  context 'with MONGODB-X509 auth mechanism' do
@@ -276,8 +313,8 @@ describe 'Client authentication options' do
276
313
 
277
314
  it_behaves_like 'a supported auth mechanism'
278
315
  it_behaves_like 'an auth mechanism with ssl'
279
- it_behaves_like 'an auth mechanism that doesn\'t support auth_mech_properties'
280
- it_behaves_like 'an auth mechanism that doesn\'t support invalid auth sources'
316
+ it_behaves_like 'an auth mechanism that does not support auth_mech_properties'
317
+ it_behaves_like 'an auth mechanism that does not support invalid auth sources'
281
318
 
282
319
  context 'with URI options' do
283
320
  let(:credentials) { "#{user}@" }
@@ -402,35 +439,62 @@ describe 'Client authentication options' do
402
439
  {
403
440
  service_name: service_name,
404
441
  canonicalize_host_name: canonicalize_host_name,
405
- service_realm: service_realm
406
- }
442
+ service_realm: service_realm,
443
+ }.freeze
407
444
  end
408
445
 
409
- context 'with URI options' do
410
- let(:options) do
411
- "?authMechanismProperties=SERVICE_NAME:#{service_name}," +
412
- "CANONICALIZE_HOST_NAME:#{canonicalize_host_name}," +
413
- "SERVICE_REALM:#{service_realm}"
414
- end
415
-
446
+ shared_examples 'correctly sets auth mechanism properties on the client' do
416
447
  it 'correctly sets auth mechanism properties on the client' do
417
- expect(client.options[:auth_mech_properties]).to eq({
448
+ expect(client.options[:auth_mech_properties]).to eq(
418
449
  'service_name' => service_name,
419
450
  'canonicalize_host_name' => canonicalize_host_name,
420
- 'service_realm' => service_realm
421
- })
451
+ 'service_realm' => service_realm,
452
+ )
453
+ end
454
+ end
455
+
456
+ context 'with URI options' do
457
+ let(:options) do
458
+ "?authMechanismProperties=SERVICE_name:#{service_name}," +
459
+ "CANONICALIZE_HOST_name:#{canonicalize_host_name}," +
460
+ "SERVICE_realm:#{service_realm}"
422
461
  end
462
+
463
+ include_examples 'correctly sets auth mechanism properties on the client'
423
464
  end
424
465
 
425
466
  context 'with client options' do
426
- let(:client_opts) { { auth_mech_properties: auth_mechanism_properties } }
467
+ [:auth_mech_properties, 'auth_mech_properties'].each do |key|
427
468
 
428
- it 'correctly sets auth mechanism properties on the client' do
429
- expect(client.options[:auth_mech_properties]).to eq({
430
- 'service_name' => service_name,
431
- 'canonicalize_host_name' => canonicalize_host_name,
432
- 'service_realm' => service_realm
433
- })
469
+ context "using #{key.class} keys" do
470
+ let(:client_opts) { { key => auth_mechanism_properties } }
471
+
472
+ include_examples 'correctly sets auth mechanism properties on the client'
473
+
474
+ context 'when options are given in mixed case' do
475
+ let(:auth_mechanism_properties) do
476
+ {
477
+ service_NAME: service_name,
478
+ canonicalize_host_NAME: canonicalize_host_name,
479
+ service_REALM: service_realm,
480
+ }.freeze
481
+ end
482
+
483
+ context 'using URI and options' do
484
+
485
+ let(:client) { new_local_client_nmio(uri, client_opts) }
486
+
487
+ include_examples 'correctly sets auth mechanism properties on the client'
488
+ end
489
+
490
+ context 'using host and options' do
491
+
492
+ let(:client) { new_local_client_nmio(['localhost'], client_opts) }
493
+
494
+ include_examples 'correctly sets auth mechanism properties on the client'
495
+ end
496
+ end
497
+ end
434
498
  end
435
499
  end
436
500
  end