bson 4.9.0 → 4.15.0

Sign up to get free protection for your applications and to get access to all the features.
Files changed (102) hide show
  1. checksums.yaml +4 -4
  2. checksums.yaml.gz.sig +0 -0
  3. data/README.md +15 -6
  4. data/ext/bson/bson-native.h +4 -0
  5. data/ext/bson/init.c +75 -23
  6. data/ext/bson/read.c +63 -11
  7. data/ext/bson/write.c +42 -3
  8. data/lib/bson/active_support.rb +1 -0
  9. data/lib/bson/array.rb +5 -1
  10. data/lib/bson/big_decimal.rb +67 -0
  11. data/lib/bson/binary.rb +8 -5
  12. data/lib/bson/boolean.rb +2 -1
  13. data/lib/bson/code.rb +2 -1
  14. data/lib/bson/code_with_scope.rb +2 -1
  15. data/lib/bson/config.rb +1 -0
  16. data/lib/bson/date.rb +1 -0
  17. data/lib/bson/date_time.rb +2 -1
  18. data/lib/bson/db_pointer.rb +2 -1
  19. data/lib/bson/dbref.rb +152 -0
  20. data/lib/bson/decimal128/builder.rb +27 -20
  21. data/lib/bson/decimal128.rb +39 -14
  22. data/lib/bson/document.rb +61 -18
  23. data/lib/bson/environment.rb +1 -0
  24. data/lib/bson/error.rb +13 -0
  25. data/lib/bson/ext_json.rb +24 -11
  26. data/lib/bson/false_class.rb +2 -1
  27. data/lib/bson/float.rb +21 -32
  28. data/lib/bson/hash.rb +18 -6
  29. data/lib/bson/int32.rb +3 -2
  30. data/lib/bson/int64.rb +3 -2
  31. data/lib/bson/integer.rb +3 -2
  32. data/lib/bson/json.rb +1 -0
  33. data/lib/bson/max_key.rb +3 -2
  34. data/lib/bson/min_key.rb +3 -2
  35. data/lib/bson/nil_class.rb +2 -1
  36. data/lib/bson/object.rb +1 -0
  37. data/lib/bson/object_id.rb +4 -3
  38. data/lib/bson/open_struct.rb +1 -0
  39. data/lib/bson/regexp.rb +24 -7
  40. data/lib/bson/registry.rb +1 -0
  41. data/lib/bson/specialized.rb +1 -0
  42. data/lib/bson/string.rb +3 -2
  43. data/lib/bson/symbol.rb +2 -1
  44. data/lib/bson/time.rb +4 -3
  45. data/lib/bson/time_with_zone.rb +1 -0
  46. data/lib/bson/timestamp.rb +7 -6
  47. data/lib/bson/true_class.rb +2 -1
  48. data/lib/bson/undefined.rb +2 -1
  49. data/lib/bson/version.rb +2 -1
  50. data/lib/bson.rb +8 -5
  51. data/spec/README.md +14 -0
  52. data/spec/bson/array_spec.rb +17 -0
  53. data/spec/bson/big_decimal_spec.rb +316 -0
  54. data/spec/bson/binary_spec.rb +1 -1
  55. data/spec/bson/binary_uuid_spec.rb +12 -0
  56. data/spec/bson/byte_buffer_read_spec.rb +59 -3
  57. data/spec/bson/byte_buffer_spec.rb +129 -6
  58. data/spec/bson/byte_buffer_write_spec.rb +96 -0
  59. data/spec/bson/date_time_spec.rb +53 -0
  60. data/spec/bson/dbref_legacy_spec.rb +169 -0
  61. data/spec/bson/dbref_spec.rb +487 -0
  62. data/spec/bson/decimal128_spec.rb +231 -0
  63. data/spec/bson/document_as_spec.rb +46 -0
  64. data/spec/bson/document_spec.rb +43 -1
  65. data/spec/bson/ext_json_parse_spec.rb +37 -0
  66. data/spec/bson/hash_as_spec.rb +57 -0
  67. data/spec/bson/hash_spec.rb +105 -0
  68. data/spec/bson/int64_spec.rb +4 -24
  69. data/spec/bson/raw_spec.rb +18 -1
  70. data/spec/bson/regexp_spec.rb +52 -0
  71. data/spec/runners/common_driver.rb +1 -1
  72. data/spec/shared/LICENSE +20 -0
  73. data/spec/shared/bin/get-mongodb-download-url +17 -0
  74. data/spec/shared/bin/s3-copy +45 -0
  75. data/spec/shared/bin/s3-upload +69 -0
  76. data/spec/shared/lib/mrss/child_process_helper.rb +80 -0
  77. data/spec/shared/lib/mrss/cluster_config.rb +231 -0
  78. data/spec/shared/lib/mrss/constraints.rb +386 -0
  79. data/spec/shared/lib/mrss/docker_runner.rb +271 -0
  80. data/spec/shared/lib/mrss/event_subscriber.rb +200 -0
  81. data/spec/shared/lib/mrss/lite_constraints.rb +191 -0
  82. data/spec/shared/lib/mrss/server_version_registry.rb +120 -0
  83. data/spec/shared/lib/mrss/spec_organizer.rb +179 -0
  84. data/spec/shared/lib/mrss/utils.rb +15 -0
  85. data/spec/shared/share/Dockerfile.erb +338 -0
  86. data/spec/shared/share/haproxy-1.conf +16 -0
  87. data/spec/shared/share/haproxy-2.conf +17 -0
  88. data/spec/shared/shlib/distro.sh +74 -0
  89. data/spec/shared/shlib/server.sh +367 -0
  90. data/spec/shared/shlib/set_env.sh +131 -0
  91. data/spec/spec_helper.rb +31 -0
  92. data/spec/spec_tests/common_driver_spec.rb +2 -1
  93. data/spec/spec_tests/data/corpus/binary.json +33 -0
  94. data/spec/spec_tests/data/corpus/dbref.json +21 -1
  95. data/spec/spec_tests/data/corpus/document.json +4 -0
  96. data/spec/spec_tests/data/corpus/regex.json +2 -2
  97. data/spec/spec_tests/data/corpus/timestamp.json +10 -0
  98. data/spec/spec_tests/data/corpus/top.json +23 -12
  99. data/spec/support/spec_config.rb +8 -1
  100. data.tar.gz.sig +0 -0
  101. metadata +168 -93
  102. metadata.gz.sig +1 -0
@@ -127,6 +127,58 @@ describe Regexp do
127
127
  expect(result).to eq(obj)
128
128
  end
129
129
  end
130
+
131
+ context "when the regexp options contains a null byte" do
132
+
133
+ let(:regexp) do
134
+ Regexp::Raw.new("pattern", "options\x00")
135
+ end
136
+
137
+ it "raises an error" do
138
+ expect do
139
+ regexp
140
+ end.to raise_error(BSON::Error::InvalidRegexpPattern, /Regexp options cannot contain a null byte/)
141
+ end
142
+ end
143
+
144
+ context "when the regexp options is an integer" do
145
+
146
+ let(:regexp) do
147
+ Regexp::Raw.new("pattern", 1)
148
+ end
149
+
150
+ it "doesn't raise an error" do
151
+ expect do
152
+ regexp
153
+ end.to_not raise_error
154
+ end
155
+ end
156
+
157
+ context "when the regexp options is an invalid type" do
158
+
159
+ let(:regexp) do
160
+ Regexp::Raw.new("pattern", [2])
161
+ end
162
+
163
+ it "raises an error" do
164
+ expect do
165
+ regexp
166
+ end.to raise_error(ArgumentError, /Regexp options must be a String, Symbol, or Integer/)
167
+ end
168
+ end
169
+ end
170
+
171
+ context "when the pattern contains a null byte" do
172
+
173
+ let(:regexp) do
174
+ Regexp::Raw.new("pattern\x00", "options")
175
+ end
176
+
177
+ it "raises an error" do
178
+ expect do
179
+ regexp
180
+ end.to raise_error(BSON::Error::InvalidRegexpPattern, /Regexp pattern cannot contain a null byte/)
181
+ end
130
182
  end
131
183
  end
132
184
  end
@@ -166,7 +166,7 @@ module BSON
166
166
  #
167
167
  # @since 4.2.0
168
168
  def reencoded_hex
169
- decoded_document.to_bson.to_s.unpack("H*").first.upcase
169
+ decoded_document.to_bson.to_s.unpack1("H*").upcase
170
170
  end
171
171
 
172
172
  # The object tested.
@@ -0,0 +1,20 @@
1
+ Copyright (c) 2020 MongoDB, Inc.
2
+
3
+ Permission is hereby granted, free of charge, to any person obtaining
4
+ a copy of this software and associated documentation files (the
5
+ "Software"), to deal in the Software without restriction, including
6
+ without limitation the rights to use, copy, modify, merge, publish,
7
+ distribute, sublicense, and/or sell copies of the Software, and to
8
+ permit persons to whom the Software is furnished to do so, subject to
9
+ the following conditions:
10
+
11
+ The above copyright notice and this permission notice shall be
12
+ included in all copies or substantial portions of the Software.
13
+
14
+ THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
15
+ EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
16
+ MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
17
+ NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE
18
+ LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION
19
+ OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
20
+ WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
@@ -0,0 +1,17 @@
1
+ #!/usr/bin/env ruby
2
+
3
+ desired_version, arch = ARGV
4
+ if arch.nil?
5
+ STDERR.puts "Usage: get-mongodb-download-url desired-version arch"
6
+ exit 1
7
+ end
8
+
9
+ $: << File.join(File.dirname(__FILE__), '../lib')
10
+ require 'mrss/server_version_registry'
11
+
12
+ begin
13
+ puts Mrss::ServerVersionRegistry.new(desired_version, arch).download_url
14
+ rescue Mrss::ServerVersionRegistry::Error => exc
15
+ STDERR.puts "Error: #{exc}"
16
+ exit 2
17
+ end
@@ -0,0 +1,45 @@
1
+ #!/usr/bin/env ruby
2
+
3
+ require 'optparse'
4
+ require 'aws-sdk-s3'
5
+
6
+ options = {}
7
+ OptionParser.new do |opts|
8
+ opts.banner = "Usage: s3-copy options"
9
+
10
+ opts.on("-r", "--region=REGION", "AWS region to use (default us-east-1)") do |v|
11
+ options[:region] = v
12
+ end
13
+
14
+ opts.on("-p", "--param=KEY=VALUE", "Specify parameter for new files") do |v|
15
+ options[:params] ||= {}
16
+ k, v = v.split('=', 2)
17
+ options[:params][k.to_sym] = v
18
+ end
19
+
20
+ opts.on("-f", "--from=BUCKET:PATH", "Bucket name and key (or path) to copy from") do |v|
21
+ options[:from] = v
22
+ end
23
+
24
+ opts.on("-t", "--to=BUCKET:PATH", "Bucket name and key (or path) to write to (may be specified more than once)") do |v|
25
+ options[:to] ||= []
26
+ options[:to] << v
27
+ end
28
+ end.parse!
29
+
30
+ ENV['AWS_REGION'] ||= options[:region] || 'us-east-1'
31
+
32
+ bucket, key = options.fetch(:from).split(':', 2)
33
+
34
+ s3 = Aws::S3::Client.new
35
+
36
+ options.fetch(:to).each do |dest|
37
+ STDERR.puts "Copying to #{dest}"
38
+ dbucket, dkey = dest.split(':', 2)
39
+ s3.copy_object(
40
+ bucket: dbucket,
41
+ key: dkey,
42
+ copy_source: "/#{bucket}/#{key}",
43
+ **options[:params] || {},
44
+ )
45
+ end
@@ -0,0 +1,69 @@
1
+ #!/usr/bin/env ruby
2
+
3
+ require 'optparse'
4
+ require 'aws-sdk-s3'
5
+
6
+ options = {}
7
+ OptionParser.new do |opts|
8
+ opts.banner = "Usage: s3-upload options"
9
+
10
+ opts.on("-r", "--region=REGION", "AWS region to use (default us-east-1)") do |v|
11
+ options[:region] = v
12
+ end
13
+
14
+ opts.on("-p", "--param=KEY=VALUE", "Specify parameter for S3 upload") do |v|
15
+ options[:params] ||= {}
16
+ k, v = v.split('=', 2)
17
+ options[:params][k.to_sym] = v
18
+ end
19
+
20
+ opts.on("-f", "--file=PATH", "Path to the file to upload, - to upload standard input") do |v|
21
+ options[:file] = v
22
+ end
23
+
24
+ opts.on("-w", "--write=BUCKET:PATH", "Bucket name and key (or path) to upload to") do |v|
25
+ options[:write] = v
26
+ end
27
+
28
+ opts.on("-c", "--copy=BUCKET:PATH", "Bucket name and key (or path) to copy to (may be specified more than once)") do |v|
29
+ options[:copy] ||= []
30
+ options[:copy] << v
31
+ end
32
+ end.parse!
33
+
34
+ ENV['AWS_REGION'] ||= options[:region] || 'us-east-1'
35
+
36
+ def upload(f, options)
37
+ s3 = Aws::S3::Client.new
38
+ write = options.fetch(:write)
39
+ STDERR.puts "Writing #{write}"
40
+ bucket, key = write.split(':', 2)
41
+ s3.put_object(
42
+ body: f.read,
43
+ bucket: bucket,
44
+ key: key,
45
+ **options[:params] || {},
46
+ )
47
+ if copy = options[:copy]
48
+ copy.each do |dest|
49
+ STDERR.puts "Copying to #{dest}"
50
+ dbucket, dkey = dest.split(':', 2)
51
+ s3.copy_object(
52
+ bucket: dbucket,
53
+ key: dkey,
54
+ copy_source: "/#{bucket}/#{key}",
55
+ **options[:params] || {},
56
+ )
57
+ end
58
+ end
59
+ end
60
+
61
+ if options[:file] == '-'
62
+ upload(STDIN, options)
63
+ elsif options[:file]
64
+ File.open(options[:file]) do |f|
65
+ upload(f, options)
66
+ end
67
+ else
68
+ upload(STDIN, options)
69
+ end
@@ -0,0 +1,80 @@
1
+ # frozen_string_literal: true
2
+ # encoding: utf-8
3
+
4
+ autoload :ChildProcess, 'childprocess'
5
+ autoload :Tempfile, 'tempfile'
6
+
7
+ module Mrss
8
+ module ChildProcessHelper
9
+ class SpawnError < StandardError; end
10
+
11
+ module_function def call(cmd, env: nil, cwd: nil)
12
+ process = ChildProcess.new(*cmd)
13
+ process.io.inherit!
14
+ if cwd
15
+ process.cwd = cwd
16
+ end
17
+ if env
18
+ env.each do |k, v|
19
+ process.environment[k.to_s] = v
20
+ end
21
+ end
22
+ process.start
23
+ process.wait
24
+ process
25
+ end
26
+
27
+ module_function def check_call(cmd, env: nil, cwd: nil)
28
+ process = call(cmd, env: env, cwd: cwd)
29
+ unless process.exit_code == 0
30
+ raise SpawnError, "Failed to execute: #{cmd}"
31
+ end
32
+ end
33
+
34
+ module_function def get_output(cmd, env: nil, cwd: nil)
35
+ process = ChildProcess.new(*cmd)
36
+ process.io.inherit!
37
+ if cwd
38
+ process.cwd = cwd
39
+ end
40
+ if env
41
+ env.each do |k, v|
42
+ process.environment[k.to_s] = v
43
+ end
44
+ end
45
+
46
+ output = ''
47
+ r, w = IO.pipe
48
+
49
+ begin
50
+ process.io.stdout = w
51
+ process.start
52
+ w.close
53
+
54
+ thread = Thread.new do
55
+ begin
56
+ loop do
57
+ output << r.readpartial(16384)
58
+ end
59
+ rescue EOFError
60
+ end
61
+ end
62
+
63
+ process.wait
64
+ thread.join
65
+ ensure
66
+ r.close
67
+ end
68
+
69
+ [process, output]
70
+ end
71
+
72
+ module_function def check_output(*args)
73
+ process, output = get_output(*args)
74
+ unless process.exit_code == 0
75
+ raise SpawnError,"Failed to execute: #{args}"
76
+ end
77
+ output
78
+ end
79
+ end
80
+ end
@@ -0,0 +1,231 @@
1
+ # frozen_string_literal: true
2
+ # encoding: utf-8
3
+
4
+ # ClusterConfig requires ClientRegistry class provided by the host project.
5
+
6
+ require 'singleton'
7
+
8
+ module Mrss
9
+ class ClusterConfig
10
+ include Singleton
11
+ include RSpec::Core::Pending
12
+
13
+ def single_server?
14
+ determine_cluster_config
15
+ @single_server
16
+ end
17
+
18
+ def sharded_ish?
19
+ determine_cluster_config
20
+ @topology == :sharded || @topology == :load_balanced
21
+ end
22
+
23
+ def replica_set_name
24
+ determine_cluster_config
25
+ @replica_set_name
26
+ end
27
+
28
+ def server_version
29
+ determine_cluster_config
30
+ @server_version
31
+ end
32
+
33
+ def enterprise?
34
+ determine_cluster_config
35
+ @enterprise
36
+ end
37
+
38
+ def short_server_version
39
+ server_version.split('.')[0..1].join('.')
40
+ end
41
+
42
+ def fcv
43
+ determine_cluster_config
44
+ @fcv
45
+ end
46
+
47
+ # Per https://jira.mongodb.org/browse/SERVER-39052, working with FCV
48
+ # in sharded topologies is annoying. Also, FCV doesn't exist in servers
49
+ # less than 3.4. This method returns FCV on 3.4+ servers when in single
50
+ # or RS topologies, and otherwise returns the major.minor server version.
51
+ def fcv_ish
52
+ if server_version.nil?
53
+ raise "Deployment server version not known - check that connection to deployment succeeded"
54
+ end
55
+
56
+ if server_version >= '3.4' && !sharded_ish?
57
+ fcv
58
+ else
59
+ if short_server_version == '4.1'
60
+ '4.2'
61
+ else
62
+ short_server_version
63
+ end
64
+ end
65
+ end
66
+
67
+ # @return [ Mongo::Address ] The address of the primary in the deployment.
68
+ def primary_address
69
+ determine_cluster_config
70
+ @primary_address
71
+ end
72
+
73
+ def primary_address_str
74
+ determine_cluster_config
75
+ @primary_address.seed
76
+ end
77
+
78
+ def primary_address_host
79
+ both = primary_address_str
80
+ both.split(':').first
81
+ end
82
+
83
+ def primary_address_port
84
+ both = primary_address_str
85
+ both.split(':')[1] || 27017
86
+ end
87
+
88
+ def primary_description
89
+ determine_cluster_config
90
+ @primary_description
91
+ end
92
+
93
+ def server_parameters
94
+ determine_cluster_config
95
+ @server_parameters
96
+ end
97
+
98
+ # Try running a command on the admin database to see if the mongod was
99
+ # started with auth.
100
+ def auth_enabled?
101
+ if @auth_enabled.nil?
102
+ @auth_enabled = begin
103
+ basic_client.use(:admin).command(getCmdLineOpts: 1).first["argv"].include?("--auth")
104
+ rescue => e
105
+ e.message =~ /(not authorized)|(unauthorized)|(no users authenticated)|(requires authentication)/
106
+ end
107
+ end
108
+ @auth_enabled
109
+ end
110
+
111
+ def topology
112
+ determine_cluster_config
113
+ @topology
114
+ end
115
+
116
+ def storage_engine
117
+ @storage_engine ||= begin
118
+ # 2.6 does not have wired tiger
119
+ if short_server_version == '2.6'
120
+ :mmapv1
121
+ else
122
+ client = ClientRegistry.instance.global_client('root_authorized')
123
+ if sharded_ish?
124
+ shards = client.use(:admin).command(listShards: 1).first
125
+ if shards['shards'].empty?
126
+ raise 'Shards are empty'
127
+ end
128
+ shard = shards['shards'].first
129
+ address_str = shard['host'].sub(/^.*\//, '').sub(/,.*/, '')
130
+ client = ClusterTools.instance.direct_client(address_str,
131
+ SpecConfig.instance.test_options.merge(SpecConfig.instance.auth_options).merge(connect: :direct))
132
+ end
133
+ rv = client.use(:admin).command(serverStatus: 1).first
134
+ rv = rv['storageEngine']['name']
135
+ rv_map = {
136
+ 'wiredTiger' => :wired_tiger,
137
+ 'mmapv1' => :mmapv1,
138
+ }
139
+ rv_map[rv] || rv
140
+ end
141
+ end
142
+ end
143
+
144
+ # This method returns an alternate address for connecting to the configured
145
+ # deployment. For example, if the replica set is configured with nodes at
146
+ # of localhost:27017 and so on, this method will return 127.0.0.:27017.
147
+ #
148
+ # Note that the "alternate" refers to replica set configuration, not the
149
+ # addresses specified in test suite configuration. If the deployment topology
150
+ # is not a replica set, "alternate" refers to test suite configuration as
151
+ # this is the only configuration available.
152
+ def alternate_address
153
+ @alternate_address ||= begin
154
+ address = primary_address_host
155
+ str = case address
156
+ when '127.0.0.1'
157
+ 'localhost'
158
+ when /^(\d+\.){3}\d+$/
159
+ skip 'This test requires a hostname or 127.0.0.1 as address'
160
+ else
161
+ # We don't know if mongod is listening on ipv4 or ipv6, in principle.
162
+ # Our tests use ipv4, so hardcode that for now.
163
+ # To support both we need to try both addresses which will make this
164
+ # test more complicated.
165
+ #
166
+ # JRuby chokes on primary_address_port as the port (e.g. 27017).
167
+ # Since the port does not actually matter, use a common port like 80.
168
+ resolved_address = Addrinfo.getaddrinfo(address, 80, Socket::PF_INET).first.ip_address
169
+ if resolved_address.include?(':')
170
+ "[#{resolved_address}]"
171
+ else
172
+ resolved_address
173
+ end
174
+ end + ":#{primary_address_port}"
175
+ Mongo::Address.new(str)
176
+ end
177
+ end
178
+
179
+ private
180
+
181
+ def determine_cluster_config
182
+ return if @primary_address
183
+
184
+ # Run all commands to figure out the cluster configuration from the same
185
+ # client. This is somewhat wasteful when running a single test, but reduces
186
+ # test runtime for the suite overall because all commands are sent on the
187
+ # same connection rather than each command connecting to the cluster by
188
+ # itself.
189
+ client = ClientRegistry.instance.global_client('root_authorized')
190
+
191
+ primary = client.cluster.next_primary
192
+ @primary_address = primary.address
193
+ @primary_description = primary.description
194
+ @replica_set_name = client.cluster.topology.replica_set_name
195
+
196
+ @topology ||= begin
197
+ topology = client.cluster.topology.class.name.sub(/.*::/, '')
198
+ topology = topology.gsub(/([A-Z])/) { |match| '_' + match.downcase }.sub(/^_/, '')
199
+ if topology =~ /^replica_set/
200
+ topology = 'replica_set'
201
+ end
202
+ topology.to_sym
203
+ end
204
+
205
+ @single_server = client.cluster.servers_list.length == 1
206
+
207
+ build_info = client.database.command(buildInfo: 1).first
208
+
209
+ @server_version = build_info['version']
210
+ @enterprise = build_info['modules'] && build_info['modules'].include?('enterprise')
211
+
212
+ @server_parameters = begin
213
+ client.use(:admin).command(getParameter: '*').first
214
+ rescue => e
215
+ STDERR.puts("WARNING: Failed to obtain server parameters: #{e.class}: #{e.message}")
216
+ {}
217
+ end
218
+
219
+ if !sharded_ish? && short_server_version >= '3.4'
220
+ rv = @server_parameters['featureCompatibilityVersion']
221
+ @fcv = rv['version'] || rv
222
+ end
223
+ end
224
+
225
+ def basic_client
226
+ # Do not cache the result here so that if the client gets closed,
227
+ # client registry reconnects it in subsequent tests
228
+ ClientRegistry.instance.global_client('basic')
229
+ end
230
+ end
231
+ end