mongoid 7.1.6 → 7.1.7
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.
- checksums.yaml +4 -4
- checksums.yaml.gz.sig +0 -0
- data/lib/mongoid/criteria/queryable/selector.rb +0 -4
- data/lib/mongoid/document.rb +3 -2
- data/lib/mongoid/interceptable.rb +3 -1
- data/lib/mongoid/version.rb +1 -1
- data/spec/app/models/customer.rb +11 -0
- data/spec/app/models/customer_address.rb +12 -0
- data/spec/app/models/dictionary.rb +6 -0
- data/spec/integration/app_spec.rb +35 -2
- data/spec/integration/callbacks_models.rb +49 -0
- data/spec/integration/callbacks_spec.rb +216 -0
- data/spec/lite_spec_helper.rb +1 -1
- data/spec/mongoid/association/embedded/embedded_in/proxy_spec.rb +50 -0
- data/spec/mongoid/atomic/paths_spec.rb +41 -0
- data/spec/mongoid/criteria/queryable/selectable_logical_spec.rb +36 -0
- data/spec/shared/lib/mrss/constraints.rb +18 -0
- data/spec/shared/lib/mrss/docker_runner.rb +262 -0
- data/spec/shared/lib/mrss/server_version_registry.rb +69 -0
- data/spec/shared/share/Dockerfile.erb +229 -0
- data/spec/shared/shlib/distro.sh +73 -0
- data/spec/shared/shlib/server.sh +270 -0
- data/spec/shared/shlib/set_env.sh +128 -0
- data.tar.gz.sig +0 -0
- metadata +32 -12
- metadata.gz.sig +0 -0
@@ -268,5 +268,46 @@ describe Mongoid::Atomic::Paths do
|
|
268
268
|
end
|
269
269
|
end
|
270
270
|
end
|
271
|
+
|
272
|
+
context "when the same class is embedded in multiple associations" do
|
273
|
+
|
274
|
+
let(:customer) do
|
275
|
+
Customer.new
|
276
|
+
end
|
277
|
+
|
278
|
+
context "assignment after saving" do
|
279
|
+
|
280
|
+
it "correctly sets the association for the embedded class" do
|
281
|
+
pending 'MONGOID-5039'
|
282
|
+
|
283
|
+
customer.home_address = CustomerAddress.new
|
284
|
+
customer.work_address = CustomerAddress.new
|
285
|
+
|
286
|
+
expect(customer.home_address.atomic_path).to eq("home_address")
|
287
|
+
expect(customer.work_address.atomic_path).to eq("work_address")
|
288
|
+
|
289
|
+
customer.save!
|
290
|
+
|
291
|
+
customer.home_address = CustomerAddress.new
|
292
|
+
customer.work_address = CustomerAddress.new
|
293
|
+
|
294
|
+
expect(customer.home_address.atomic_path).to eq("home_address")
|
295
|
+
expect(customer.work_address.atomic_path).to eq("work_address")
|
296
|
+
end
|
297
|
+
end
|
298
|
+
|
299
|
+
context "inverse assignment" do
|
300
|
+
|
301
|
+
it "correctly returns the path for each embedded class" do
|
302
|
+
pending 'MONGOID-5039'
|
303
|
+
|
304
|
+
customer.work_address = CustomerAddress.new
|
305
|
+
customer.work_address.addressable = customer
|
306
|
+
|
307
|
+
expect(customer.home_address.atomic_path).to eq("home_address")
|
308
|
+
expect(customer.work_address.atomic_path).to eq("work_address")
|
309
|
+
end
|
310
|
+
end
|
311
|
+
end
|
271
312
|
end
|
272
313
|
end
|
@@ -1447,6 +1447,42 @@ describe Mongoid::Criteria::Queryable::Selectable do
|
|
1447
1447
|
end
|
1448
1448
|
end
|
1449
1449
|
end
|
1450
|
+
|
1451
|
+
context 'when using multiple criteria and symbol operators' do
|
1452
|
+
context 'when using fields that meaningfully evolve values' do
|
1453
|
+
|
1454
|
+
let(:query) do
|
1455
|
+
Dictionary.any_of({a: 1}, :published.gt => Date.new(2020, 2, 3))
|
1456
|
+
end
|
1457
|
+
|
1458
|
+
it 'generates the expected query' do
|
1459
|
+
query.selector.should == {'$or' => [
|
1460
|
+
{'a' => 1},
|
1461
|
+
# Date instance is converted to a Time instance in local time,
|
1462
|
+
# because we are querying on a Time field and dates are interpreted
|
1463
|
+
# in local time when assigning to Time fields
|
1464
|
+
{'published' => {'$gt' => Time.local(2020, 2, 3)}},
|
1465
|
+
]}
|
1466
|
+
end
|
1467
|
+
end
|
1468
|
+
|
1469
|
+
context 'when using fields that do not meaningfully evolve values' do
|
1470
|
+
|
1471
|
+
let(:query) do
|
1472
|
+
Dictionary.any_of({a: 1}, :submitted_on.gt => Date.new(2020, 2, 3))
|
1473
|
+
end
|
1474
|
+
|
1475
|
+
it 'generates the expected query' do
|
1476
|
+
query.selector.should == {'$or' => [
|
1477
|
+
{'a' => 1},
|
1478
|
+
# Date instance is converted to a Time instance in UTC,
|
1479
|
+
# because we are querying on a Date field and dates are interpreted
|
1480
|
+
# in UTC when persisted as dates by Mongoid
|
1481
|
+
{'submitted_on' => {'$gt' => Time.utc(2020, 2, 3)}},
|
1482
|
+
]}
|
1483
|
+
end
|
1484
|
+
end
|
1485
|
+
end
|
1450
1486
|
end
|
1451
1487
|
|
1452
1488
|
describe "#not" do
|
@@ -156,6 +156,24 @@ module Mrss
|
|
156
156
|
end
|
157
157
|
end
|
158
158
|
|
159
|
+
def require_zstd_compression
|
160
|
+
before(:all) do
|
161
|
+
compressors = SpecConfig.instance.compressors
|
162
|
+
unless compressors && compressors.include?('zstd')
|
163
|
+
skip "Zstd compression is not enabled"
|
164
|
+
end
|
165
|
+
end
|
166
|
+
end
|
167
|
+
|
168
|
+
def require_no_zstd_compression
|
169
|
+
before(:all) do
|
170
|
+
compressors = SpecConfig.instance.compressors
|
171
|
+
if compressors && compressors.include?('zstd')
|
172
|
+
skip "Zstd compression is enabled"
|
173
|
+
end
|
174
|
+
end
|
175
|
+
end
|
176
|
+
|
159
177
|
def require_no_compression
|
160
178
|
before(:all) do
|
161
179
|
if SpecConfig.instance.compressors
|
@@ -0,0 +1,262 @@
|
|
1
|
+
require 'optparse'
|
2
|
+
require 'erb'
|
3
|
+
autoload :Dotenv, 'dotenv'
|
4
|
+
|
5
|
+
module Mrss
|
6
|
+
autoload :ServerVersionRegistry, 'mrss/server_version_registry'
|
7
|
+
|
8
|
+
class DockerRunner
|
9
|
+
def initialize(**opts)
|
10
|
+
# These options are required:
|
11
|
+
opts.fetch(:image_tag)
|
12
|
+
opts.fetch(:dockerfile_path)
|
13
|
+
opts.fetch(:default_script)
|
14
|
+
opts.fetch(:project_lib_subdir)
|
15
|
+
|
16
|
+
@options = opts
|
17
|
+
end
|
18
|
+
|
19
|
+
attr_reader :options
|
20
|
+
|
21
|
+
def run
|
22
|
+
process_arguments
|
23
|
+
unless @options[:exec_only]
|
24
|
+
create_dockerfile
|
25
|
+
create_image
|
26
|
+
end
|
27
|
+
if @options[:mongo_only]
|
28
|
+
run_deployment
|
29
|
+
else
|
30
|
+
run_tests
|
31
|
+
end
|
32
|
+
end
|
33
|
+
|
34
|
+
private
|
35
|
+
|
36
|
+
def process_arguments
|
37
|
+
#@options = {}
|
38
|
+
OptionParser.new do |opts|
|
39
|
+
opts.banner = "Usage: test-on-docker [-d distro] [evergreen_key=value ...]"
|
40
|
+
|
41
|
+
opts.on("-a", "--add-env=PATH", "Load environment variables from PATH in .env format") do |path|
|
42
|
+
@options[:extra_env] ||= {}
|
43
|
+
unless File.exist?(path)
|
44
|
+
raise "-a option references nonexistent file #{path}"
|
45
|
+
end
|
46
|
+
Dotenv.parse(path).each do |k, v|
|
47
|
+
@options[:extra_env][k] = v
|
48
|
+
end
|
49
|
+
end
|
50
|
+
|
51
|
+
opts.on("-d", "--distro=DISTRO", "Distro to use") do |v|
|
52
|
+
@options[:distro] = v
|
53
|
+
end
|
54
|
+
|
55
|
+
opts.on('-e', '--exec-only', 'Execute tests using existing Dockerfile (for offline user)') do |v|
|
56
|
+
@options[:exec_only] = v
|
57
|
+
end
|
58
|
+
|
59
|
+
opts.on('-m', '--mongo-only=PORT', 'Start the MongoDB deployment and expose it to host on ports starting with PORT') do |v|
|
60
|
+
@options[:mongo_only] = v.to_i
|
61
|
+
end
|
62
|
+
|
63
|
+
opts.on('-p', '--preload', 'Preload Ruby toolchain and server binaries in docker') do |v|
|
64
|
+
@options[:preload] = v
|
65
|
+
end
|
66
|
+
|
67
|
+
opts.on('-s', '--script=SCRIPT', 'Test script to invoke') do |v|
|
68
|
+
@options[:script] = v
|
69
|
+
end
|
70
|
+
|
71
|
+
opts.on('-i', '--interactive', 'Interactive mode - disable per-test timeouts') do |v|
|
72
|
+
@options[:interactive] = v
|
73
|
+
end
|
74
|
+
end.parse!
|
75
|
+
|
76
|
+
@env = Hash[ARGV.map do |arg|
|
77
|
+
arg.split('=', 2)
|
78
|
+
end]
|
79
|
+
|
80
|
+
@env['RVM_RUBY'] ||= 'ruby-2.7'
|
81
|
+
unless ruby =~ /^j?ruby-/
|
82
|
+
raise "RVM_RUBY option is not in expected format: #{ruby}"
|
83
|
+
end
|
84
|
+
|
85
|
+
@env['MONGODB_VERSION'] ||= '4.4'
|
86
|
+
end
|
87
|
+
|
88
|
+
def create_dockerfile
|
89
|
+
template_path = File.join(File.dirname(__FILE__), '../../share/Dockerfile.erb')
|
90
|
+
result = ERB.new(File.read(template_path)).result(binding)
|
91
|
+
File.open(dockerfile_path, 'w') do |f|
|
92
|
+
f << result
|
93
|
+
end
|
94
|
+
end
|
95
|
+
|
96
|
+
def image_tag
|
97
|
+
options.fetch(:image_tag)
|
98
|
+
end
|
99
|
+
|
100
|
+
def dockerfile_path
|
101
|
+
options.fetch(:dockerfile_path)
|
102
|
+
end
|
103
|
+
|
104
|
+
def create_image
|
105
|
+
run_command(['docker', 'build',
|
106
|
+
'-t', image_tag,
|
107
|
+
'-f', dockerfile_path,
|
108
|
+
'.'])
|
109
|
+
end
|
110
|
+
|
111
|
+
BASE_TEST_COMMAND = %w(docker run -i --tmpfs /tmpfs:exec).freeze
|
112
|
+
|
113
|
+
def run_tests
|
114
|
+
run_command(BASE_TEST_COMMAND + tty_arg + extra_env + [image_tag] +
|
115
|
+
script.split(/\s+/))
|
116
|
+
end
|
117
|
+
|
118
|
+
def run_deployment
|
119
|
+
run_command(BASE_TEST_COMMAND + tty_arg + extra_env + [
|
120
|
+
'-e', %q`TEST_CMD=watch -x bash -c "ps awwxu |egrep 'mongo|ocsp'"`,
|
121
|
+
'-e', 'BIND_ALL=true',
|
122
|
+
] + port_forwards + [image_tag] + script.split(/\s+/))
|
123
|
+
end
|
124
|
+
|
125
|
+
def tty_arg
|
126
|
+
tty = File.open('/dev/stdin') do |f|
|
127
|
+
f.isatty
|
128
|
+
end
|
129
|
+
if tty
|
130
|
+
%w(-t --init)
|
131
|
+
else
|
132
|
+
[]
|
133
|
+
end
|
134
|
+
end
|
135
|
+
|
136
|
+
def extra_env
|
137
|
+
if @options[:extra_env]
|
138
|
+
@options[:extra_env].map do |k, v|
|
139
|
+
# Here the value must not be escaped
|
140
|
+
['-e', "#{k}=#{v}"]
|
141
|
+
end.flatten
|
142
|
+
else
|
143
|
+
[]
|
144
|
+
end
|
145
|
+
end
|
146
|
+
|
147
|
+
def port_forwards
|
148
|
+
args = (0...num_exposed_ports).map do |i|
|
149
|
+
host_port = @options[:mongo_only] + i
|
150
|
+
container_port = 27017 + i
|
151
|
+
['-p', "#{host_port}:#{container_port}"]
|
152
|
+
end.flatten
|
153
|
+
|
154
|
+
if @env['OCSP_ALGORITHM'] && !@env['OCSP_VERIFIER']
|
155
|
+
args += %w(-p 8100:8100)
|
156
|
+
end
|
157
|
+
|
158
|
+
args
|
159
|
+
end
|
160
|
+
|
161
|
+
def run_command(cmd)
|
162
|
+
if pid = fork
|
163
|
+
Process.wait(pid)
|
164
|
+
unless $?.exitstatus == 0
|
165
|
+
raise "Process exited with code #{$?.exitstatus}"
|
166
|
+
end
|
167
|
+
else
|
168
|
+
exec(*cmd)
|
169
|
+
end
|
170
|
+
end
|
171
|
+
|
172
|
+
def distro
|
173
|
+
@options[:distro] || 'ubuntu1604'
|
174
|
+
end
|
175
|
+
|
176
|
+
BASE_IMAGES = {
|
177
|
+
'debian81' => 'debian:jessie',
|
178
|
+
'debian92' => 'debian:stretch',
|
179
|
+
'ubuntu1404' => 'ubuntu:trusty',
|
180
|
+
'ubuntu1604' => 'ubuntu:xenial',
|
181
|
+
'ubuntu1804' => 'ubuntu:bionic',
|
182
|
+
'rhel62' => 'centos:6',
|
183
|
+
'rhel70' => 'centos:7',
|
184
|
+
}.freeze
|
185
|
+
|
186
|
+
def base_image
|
187
|
+
BASE_IMAGES[distro] or raise "Unknown distro: #{distro}"
|
188
|
+
end
|
189
|
+
|
190
|
+
def ruby
|
191
|
+
@env['RVM_RUBY']
|
192
|
+
end
|
193
|
+
|
194
|
+
def ruby_head?
|
195
|
+
ruby == 'ruby-head'
|
196
|
+
end
|
197
|
+
|
198
|
+
def server_version
|
199
|
+
@env['MONGODB_VERSION']
|
200
|
+
end
|
201
|
+
|
202
|
+
def script
|
203
|
+
@options[:script] || options.fetch(:default_script)
|
204
|
+
end
|
205
|
+
|
206
|
+
def debian?
|
207
|
+
distro =~ /debian|ubuntu/
|
208
|
+
end
|
209
|
+
|
210
|
+
def preload?
|
211
|
+
!!@options[:preload]
|
212
|
+
end
|
213
|
+
|
214
|
+
def interactive?
|
215
|
+
!!@options[:interactive]
|
216
|
+
end
|
217
|
+
|
218
|
+
def project_lib_subdir
|
219
|
+
options.fetch(:project_lib_subdir)
|
220
|
+
end
|
221
|
+
|
222
|
+
def server_download_url
|
223
|
+
@server_download_url ||= ServerVersionRegistry.new(server_version, distro).download_url
|
224
|
+
end
|
225
|
+
|
226
|
+
def libmongocrypt_path
|
227
|
+
case distro
|
228
|
+
when /ubuntu1604/
|
229
|
+
"./ubuntu1604/nocrypto/lib64/libmongocrypt.so"
|
230
|
+
when /ubuntu1804/
|
231
|
+
"./ubuntu1804-64/nocrypto/lib64/libmongocrypt.so"
|
232
|
+
when /debian92/
|
233
|
+
"./debian92/nocrypto/lib64/libmongocrypt.so"
|
234
|
+
else
|
235
|
+
raise "This script does not support running FLE tests on #{distro}. Use ubuntu1604, ubuntu1804 or debian92 instead"
|
236
|
+
end
|
237
|
+
end
|
238
|
+
|
239
|
+
def expose?
|
240
|
+
!!@options[:mongo_only]
|
241
|
+
end
|
242
|
+
|
243
|
+
def fle?
|
244
|
+
%w(1 true yes).include?(@env['FLE']&.downcase)
|
245
|
+
end
|
246
|
+
|
247
|
+
def num_exposed_ports
|
248
|
+
case @env['TOPOLOGY'] || 'standalone'
|
249
|
+
when 'standalone'
|
250
|
+
1
|
251
|
+
when 'replica-set'
|
252
|
+
3
|
253
|
+
when 'sharded-cluster'
|
254
|
+
if @env['SINGLE_MONGOS']
|
255
|
+
1
|
256
|
+
else
|
257
|
+
2
|
258
|
+
end
|
259
|
+
end
|
260
|
+
end
|
261
|
+
end
|
262
|
+
end
|
@@ -0,0 +1,69 @@
|
|
1
|
+
autoload :JSON, 'json'
|
2
|
+
require 'open-uri'
|
3
|
+
|
4
|
+
module Mrss
|
5
|
+
class ServerVersionRegistry
|
6
|
+
def initialize(desired_version, arch)
|
7
|
+
@desired_version, @arch = desired_version, arch
|
8
|
+
end
|
9
|
+
|
10
|
+
attr_reader :desired_version, :arch
|
11
|
+
|
12
|
+
def download_url
|
13
|
+
@download_url ||= begin
|
14
|
+
info = JSON.load(uri_open('http://downloads.mongodb.org/current.json').read)
|
15
|
+
version = info['versions'].detect do |version|
|
16
|
+
version['version'].start_with?(desired_version) &&
|
17
|
+
!version['version'].include?('-') &&
|
18
|
+
# Sometimes the download situation is borked and there is a release
|
19
|
+
# with no downloads... skip those.
|
20
|
+
!version['downloads'].empty?
|
21
|
+
end
|
22
|
+
# Allow RC releases if there isn't a GA release.
|
23
|
+
version ||= info['versions'].detect do |version|
|
24
|
+
version['version'].start_with?(desired_version) &&
|
25
|
+
# Sometimes the download situation is borked and there is a release
|
26
|
+
# with no downloads... skip those.
|
27
|
+
!version['downloads'].empty?
|
28
|
+
end
|
29
|
+
if version.nil?
|
30
|
+
info = JSON.load(URI.parse('http://downloads.mongodb.org/full.json').open.read)
|
31
|
+
versions = info['versions'].select do |version|
|
32
|
+
version['version'].start_with?(desired_version) &&
|
33
|
+
!version['downloads'].empty?
|
34
|
+
end
|
35
|
+
# Get rid of rc, beta etc. versions if there is a GA release.
|
36
|
+
if versions.any? { |version| !version.include?('-') }
|
37
|
+
versions.delete_if do |version|
|
38
|
+
version['version'].include?('-')
|
39
|
+
end
|
40
|
+
end
|
41
|
+
# Versions are ordered with newest first, take the first one i.e. the most
|
42
|
+
# recent one.
|
43
|
+
version = versions.first
|
44
|
+
if version.nil?
|
45
|
+
STDERR.puts "Error: no version #{desired_version}"
|
46
|
+
exit 2
|
47
|
+
end
|
48
|
+
end
|
49
|
+
dl = version['downloads'].detect do |dl|
|
50
|
+
dl['archive']['url'].index("enterprise-#{arch}") &&
|
51
|
+
dl['arch'] == 'x86_64'
|
52
|
+
end
|
53
|
+
unless dl
|
54
|
+
STDERR.puts "Error: no download for #{arch} for #{version['version']}"
|
55
|
+
exit 2
|
56
|
+
end
|
57
|
+
url = dl['archive']['url']
|
58
|
+
end
|
59
|
+
end
|
60
|
+
|
61
|
+
def uri_open(*args)
|
62
|
+
if RUBY_VERSION < '2.5'
|
63
|
+
open(*args)
|
64
|
+
else
|
65
|
+
URI.open(*args)
|
66
|
+
end
|
67
|
+
end
|
68
|
+
end
|
69
|
+
end
|