1 # Copyright (C) The Arvados Authors. All rights reserved.
3 # SPDX-License-Identifier: AGPL-3.0
6 require 'helpers/container_test_helper'
8 class ContainerTest < ActiveSupport::TestCase
10 include ContainerTestHelper
13 command: ['echo', 'foo'],
14 container_image: 'fa3c1a9cb6783f85f2ecda037e07b8c3+167',
17 runtime_constraints: {"vcpus" => 1, "ram" => 1},
20 REUSABLE_COMMON_ATTRS = {
21 container_image: "9ae44d5792468c58bcf85ce7353c7027+124",
23 command: ["echo", "hello"],
25 runtime_constraints: {
30 "test" => {"kind" => "json"},
36 runtime_user_uuid: "zzzzz-tpzed-xurymjxw79nv3jz",
37 runtime_auth_scopes: ["all"]
40 def request_only attrs
41 attrs.reject {|k| [:runtime_user_uuid, :runtime_auth_scopes].include? k}
44 def minimal_new attrs={}
45 cr = ContainerRequest.new request_only(DEFAULT_ATTRS.merge(attrs))
46 cr.state = ContainerRequest::Committed
48 c = Container.find_by_uuid cr.container_uuid
53 def check_illegal_updates c, bad_updates
54 bad_updates.each do |u|
55 refute c.update_attributes(u), u.inspect
56 refute c.valid?, u.inspect
61 def check_illegal_modify c
62 check_illegal_updates c, [{command: ["echo", "bar"]},
63 {container_image: "arvados/apitestfixture:june10"},
65 {environment: {"FOO" => "BAR"}},
66 {mounts: {"FOO" => "BAR"}},
67 {output_path: "/tmp3"},
68 {locked_by_uuid: "zzzzz-gj3su-027z32aux8dg2s1"},
69 {auth_uuid: "zzzzz-gj3su-017z32aux8dg2s1"},
70 {runtime_constraints: {"FOO" => "BAR"}}]
73 def check_bogus_states c
74 check_illegal_updates c, [{state: nil},
78 def check_no_change_from_cancelled c
79 check_illegal_modify c
81 check_illegal_updates c, [{ priority: 3 },
82 { state: Container::Queued },
83 { state: Container::Locked },
84 { state: Container::Running },
85 { state: Container::Complete }]
88 test "Container create" do
90 c, _ = minimal_new(environment: {},
91 mounts: {"BAR" => {"kind" => "FOO"}},
94 runtime_constraints: {"vcpus" => 1, "ram" => 1})
96 check_illegal_modify c
105 test "Container valid priority" do
106 act_as_system_user do
107 c, _ = minimal_new(environment: {},
108 mounts: {"BAR" => {"kind" => "FOO"}},
111 runtime_constraints: {"vcpus" => 1, "ram" => 1})
113 assert_raises(ActiveRecord::RecordInvalid) do
133 c.priority = 1000 << 50
138 test "Container runtime_status data types" do
139 set_user_from_auth :active
142 mounts: {"BAR" => {"kind" => "FOO"}},
145 runtime_constraints: {"vcpus" => 1, "ram" => 1}
147 c, _ = minimal_new(attrs)
148 assert_equal c.runtime_status, {}
149 assert_equal Container::Queued, c.state
151 set_user_from_auth :dispatch1
152 c.update_attributes! state: Container::Locked
153 c.update_attributes! state: Container::Running
156 'error', 'errorDetail', 'warning', 'warningDetail', 'activity'
158 # String type is allowed
159 string_val = 'A string is accepted'
160 c.update_attributes! runtime_status: {k => string_val}
161 assert_equal string_val, c.runtime_status[k]
163 # Other types aren't allowed
165 42, false, [], {}, nil
166 ].each do |unallowed_val|
167 assert_raises ActiveRecord::RecordInvalid do
168 c.update_attributes! runtime_status: {k => unallowed_val}
174 test "Container runtime_status updates" do
175 set_user_from_auth :active
178 mounts: {"BAR" => {"kind" => "FOO"}},
181 runtime_constraints: {"vcpus" => 1, "ram" => 1}
183 c1, _ = minimal_new(attrs)
184 assert_equal c1.runtime_status, {}
186 assert_equal Container::Queued, c1.state
187 assert_raises ArvadosModel::PermissionDeniedError do
188 c1.update_attributes! runtime_status: {'error' => 'Oops!'}
191 set_user_from_auth :dispatch1
193 # Allow updates when state = Locked
194 c1.update_attributes! state: Container::Locked
195 c1.update_attributes! runtime_status: {'error' => 'Oops!'}
196 assert c1.runtime_status.key? 'error'
198 # Reset when transitioning from Locked to Queued
199 c1.update_attributes! state: Container::Queued
200 assert_equal c1.runtime_status, {}
202 # Allow updates when state = Running
203 c1.update_attributes! state: Container::Locked
204 c1.update_attributes! state: Container::Running
205 c1.update_attributes! runtime_status: {'error' => 'Oops!'}
206 assert c1.runtime_status.key? 'error'
208 # Don't allow updates on other states
209 c1.update_attributes! state: Container::Complete
210 assert_raises ActiveRecord::RecordInvalid do
211 c1.update_attributes! runtime_status: {'error' => 'Some other error'}
214 set_user_from_auth :active
215 c2, _ = minimal_new(attrs)
216 assert_equal c2.runtime_status, {}
217 set_user_from_auth :dispatch1
218 c2.update_attributes! state: Container::Locked
219 c2.update_attributes! state: Container::Running
220 c2.update_attributes! state: Container::Cancelled
221 assert_raises ActiveRecord::RecordInvalid do
222 c2.update_attributes! runtime_status: {'error' => 'Oops!'}
226 test "Container serialized hash attributes sorted before save" do
227 set_user_from_auth :active
228 env = {"C" => "3", "B" => "2", "A" => "1"}
229 m = {"F" => {"kind" => "3"}, "E" => {"kind" => "2"}, "D" => {"kind" => "1"}}
230 rc = {"vcpus" => 1, "ram" => 1, "keep_cache_ram" => 1}
231 c, _ = minimal_new(environment: env, mounts: m, runtime_constraints: rc)
232 assert_equal c.environment.to_json, Container.deep_sort_hash(env).to_json
233 assert_equal c.mounts.to_json, Container.deep_sort_hash(m).to_json
234 assert_equal c.runtime_constraints.to_json, Container.deep_sort_hash(rc).to_json
237 test 'deep_sort_hash on array of hashes' do
238 a = {'z' => [[{'a' => 'a', 'b' => 'b'}]]}
239 b = {'z' => [[{'b' => 'b', 'a' => 'a'}]]}
240 assert_equal Container.deep_sort_hash(a).to_json, Container.deep_sort_hash(b).to_json
243 test "find_reusable method should select higher priority queued container" do
244 Rails.configuration.Containers.LogReuseDecisions = true
245 set_user_from_auth :active
246 common_attrs = REUSABLE_COMMON_ATTRS.merge({environment:{"var" => "queued"}})
247 c_low_priority, _ = minimal_new(common_attrs.merge({use_existing:false, priority:1}))
248 c_high_priority, _ = minimal_new(common_attrs.merge({use_existing:false, priority:2}))
249 assert_not_equal c_low_priority.uuid, c_high_priority.uuid
250 assert_equal Container::Queued, c_low_priority.state
251 assert_equal Container::Queued, c_high_priority.state
252 reused = Container.find_reusable(common_attrs)
253 assert_not_nil reused
254 assert_equal reused.uuid, c_high_priority.uuid
257 test "find_reusable method should select latest completed container" do
258 set_user_from_auth :active
259 common_attrs = REUSABLE_COMMON_ATTRS.merge({environment: {"var" => "complete"}})
261 state: Container::Complete,
263 log: 'ea10d51bcf88862dbcc36eb292017dfd+45',
264 output: '1f4b0bc7583c2a7f9102c395f4ffc5e3+45'
267 c_older, _ = minimal_new(common_attrs.merge({use_existing: false}))
268 c_recent, _ = minimal_new(common_attrs.merge({use_existing: false}))
269 assert_not_equal c_older.uuid, c_recent.uuid
271 set_user_from_auth :dispatch1
272 c_older.update_attributes!({state: Container::Locked})
273 c_older.update_attributes!({state: Container::Running})
274 c_older.update_attributes!(completed_attrs)
276 c_recent.update_attributes!({state: Container::Locked})
277 c_recent.update_attributes!({state: Container::Running})
278 c_recent.update_attributes!(completed_attrs)
280 reused = Container.find_reusable(common_attrs)
281 assert_not_nil reused
282 assert_equal reused.uuid, c_older.uuid
285 test "find_reusable method should select oldest completed container when inconsistent outputs exist" do
286 set_user_from_auth :active
287 common_attrs = REUSABLE_COMMON_ATTRS.merge({environment: {"var" => "complete"}, priority: 1})
289 state: Container::Complete,
291 log: 'ea10d51bcf88862dbcc36eb292017dfd+45',
294 cr = ContainerRequest.new request_only(common_attrs)
295 cr.use_existing = false
296 cr.state = ContainerRequest::Committed
298 c_output1 = Container.where(uuid: cr.container_uuid).first
300 cr = ContainerRequest.new request_only(common_attrs)
301 cr.use_existing = false
302 cr.state = ContainerRequest::Committed
304 c_output2 = Container.where(uuid: cr.container_uuid).first
306 assert_not_equal c_output1.uuid, c_output2.uuid
308 set_user_from_auth :dispatch1
310 out1 = '1f4b0bc7583c2a7f9102c395f4ffc5e3+45'
311 log1 = collections(:real_log_collection).portable_data_hash
312 c_output1.update_attributes!({state: Container::Locked})
313 c_output1.update_attributes!({state: Container::Running})
314 c_output1.update_attributes!(completed_attrs.merge({log: log1, output: out1}))
316 out2 = 'fa7aeb5140e2848d39b416daeef4ffc5+45'
317 c_output2.update_attributes!({state: Container::Locked})
318 c_output2.update_attributes!({state: Container::Running})
319 c_output2.update_attributes!(completed_attrs.merge({log: log1, output: out2}))
321 set_user_from_auth :active
322 reused = Container.resolve(ContainerRequest.new(request_only(common_attrs)))
323 assert_equal c_output1.uuid, reused.uuid
326 test "find_reusable method should select running container by start date" do
327 set_user_from_auth :active
328 common_attrs = REUSABLE_COMMON_ATTRS.merge({environment: {"var" => "running"}})
329 c_slower, _ = minimal_new(common_attrs.merge({use_existing: false}))
330 c_faster_started_first, _ = minimal_new(common_attrs.merge({use_existing: false}))
331 c_faster_started_second, _ = minimal_new(common_attrs.merge({use_existing: false}))
332 # Confirm the 3 container UUIDs are different.
333 assert_equal 3, [c_slower.uuid, c_faster_started_first.uuid, c_faster_started_second.uuid].uniq.length
334 set_user_from_auth :dispatch1
335 c_slower.update_attributes!({state: Container::Locked})
336 c_slower.update_attributes!({state: Container::Running,
338 c_faster_started_first.update_attributes!({state: Container::Locked})
339 c_faster_started_first.update_attributes!({state: Container::Running,
341 c_faster_started_second.update_attributes!({state: Container::Locked})
342 c_faster_started_second.update_attributes!({state: Container::Running,
344 reused = Container.find_reusable(common_attrs)
345 assert_not_nil reused
346 # Selected container is the one that started first
347 assert_equal reused.uuid, c_faster_started_first.uuid
350 test "find_reusable method should select running container by progress" do
351 set_user_from_auth :active
352 common_attrs = REUSABLE_COMMON_ATTRS.merge({environment: {"var" => "running2"}})
353 c_slower, _ = minimal_new(common_attrs.merge({use_existing: false}))
354 c_faster_started_first, _ = minimal_new(common_attrs.merge({use_existing: false}))
355 c_faster_started_second, _ = minimal_new(common_attrs.merge({use_existing: false}))
356 # Confirm the 3 container UUIDs are different.
357 assert_equal 3, [c_slower.uuid, c_faster_started_first.uuid, c_faster_started_second.uuid].uniq.length
358 set_user_from_auth :dispatch1
359 c_slower.update_attributes!({state: Container::Locked})
360 c_slower.update_attributes!({state: Container::Running,
362 c_faster_started_first.update_attributes!({state: Container::Locked})
363 c_faster_started_first.update_attributes!({state: Container::Running,
365 c_faster_started_second.update_attributes!({state: Container::Locked})
366 c_faster_started_second.update_attributes!({state: Container::Running,
368 reused = Container.find_reusable(common_attrs)
369 assert_not_nil reused
370 # Selected container is the one with most progress done
371 assert_equal reused.uuid, c_faster_started_second.uuid
374 test "find_reusable method should select non-failing running container" do
375 set_user_from_auth :active
376 common_attrs = REUSABLE_COMMON_ATTRS.merge({environment: {"var" => "running2"}})
377 c_slower, _ = minimal_new(common_attrs.merge({use_existing: false}))
378 c_faster_started_first, _ = minimal_new(common_attrs.merge({use_existing: false}))
379 c_faster_started_second, _ = minimal_new(common_attrs.merge({use_existing: false}))
380 # Confirm the 3 container UUIDs are different.
381 assert_equal 3, [c_slower.uuid, c_faster_started_first.uuid, c_faster_started_second.uuid].uniq.length
382 set_user_from_auth :dispatch1
383 c_slower.update_attributes!({state: Container::Locked})
384 c_slower.update_attributes!({state: Container::Running,
386 c_faster_started_first.update_attributes!({state: Container::Locked})
387 c_faster_started_first.update_attributes!({state: Container::Running,
388 runtime_status: {'warning' => 'This is not an error'},
390 c_faster_started_second.update_attributes!({state: Container::Locked})
391 assert_equal 0, Container.where("runtime_status->'error' is not null").count
392 c_faster_started_second.update_attributes!({state: Container::Running,
393 runtime_status: {'error' => 'Something bad happened'},
395 assert_equal 1, Container.where("runtime_status->'error' is not null").count
396 reused = Container.find_reusable(common_attrs)
397 assert_not_nil reused
398 # Selected the non-failing container even if it's the one with less progress done
399 assert_equal reused.uuid, c_faster_started_first.uuid
402 test "find_reusable method should select locked container most likely to start sooner" do
403 set_user_from_auth :active
404 common_attrs = REUSABLE_COMMON_ATTRS.merge({environment: {"var" => "locked"}})
405 c_low_priority, _ = minimal_new(common_attrs.merge({use_existing: false}))
406 c_high_priority_older, _ = minimal_new(common_attrs.merge({use_existing: false}))
407 c_high_priority_newer, _ = minimal_new(common_attrs.merge({use_existing: false}))
408 # Confirm the 3 container UUIDs are different.
409 assert_equal 3, [c_low_priority.uuid, c_high_priority_older.uuid, c_high_priority_newer.uuid].uniq.length
410 set_user_from_auth :dispatch1
411 c_low_priority.update_attributes!({state: Container::Locked,
413 c_high_priority_older.update_attributes!({state: Container::Locked,
415 c_high_priority_newer.update_attributes!({state: Container::Locked,
417 reused = Container.find_reusable(common_attrs)
418 assert_not_nil reused
419 assert_equal reused.uuid, c_high_priority_older.uuid
422 test "find_reusable method should select running over failed container" do
423 set_user_from_auth :active
424 common_attrs = REUSABLE_COMMON_ATTRS.merge({environment: {"var" => "failed_vs_running"}})
425 c_failed, _ = minimal_new(common_attrs.merge({use_existing: false}))
426 c_running, _ = minimal_new(common_attrs.merge({use_existing: false}))
427 assert_not_equal c_failed.uuid, c_running.uuid
428 set_user_from_auth :dispatch1
429 c_failed.update_attributes!({state: Container::Locked})
430 c_failed.update_attributes!({state: Container::Running})
431 c_failed.update_attributes!({state: Container::Complete,
433 log: 'ea10d51bcf88862dbcc36eb292017dfd+45',
434 output: 'ea10d51bcf88862dbcc36eb292017dfd+45'})
435 c_running.update_attributes!({state: Container::Locked})
436 c_running.update_attributes!({state: Container::Running,
438 reused = Container.find_reusable(common_attrs)
439 assert_not_nil reused
440 assert_equal reused.uuid, c_running.uuid
443 test "find_reusable method should select complete over running container" do
444 set_user_from_auth :active
445 common_attrs = REUSABLE_COMMON_ATTRS.merge({environment: {"var" => "completed_vs_running"}})
446 c_completed, _ = minimal_new(common_attrs.merge({use_existing: false}))
447 c_running, _ = minimal_new(common_attrs.merge({use_existing: false}))
448 assert_not_equal c_completed.uuid, c_running.uuid
449 set_user_from_auth :dispatch1
450 c_completed.update_attributes!({state: Container::Locked})
451 c_completed.update_attributes!({state: Container::Running})
452 c_completed.update_attributes!({state: Container::Complete,
454 log: 'ea10d51bcf88862dbcc36eb292017dfd+45',
455 output: '1f4b0bc7583c2a7f9102c395f4ffc5e3+45'})
456 c_running.update_attributes!({state: Container::Locked})
457 c_running.update_attributes!({state: Container::Running,
459 reused = Container.find_reusable(common_attrs)
460 assert_not_nil reused
461 assert_equal c_completed.uuid, reused.uuid
464 test "find_reusable method should select running over locked container" do
465 set_user_from_auth :active
466 common_attrs = REUSABLE_COMMON_ATTRS.merge({environment: {"var" => "running_vs_locked"}})
467 c_locked, _ = minimal_new(common_attrs.merge({use_existing: false}))
468 c_running, _ = minimal_new(common_attrs.merge({use_existing: false}))
469 assert_not_equal c_running.uuid, c_locked.uuid
470 set_user_from_auth :dispatch1
471 c_locked.update_attributes!({state: Container::Locked})
472 c_running.update_attributes!({state: Container::Locked})
473 c_running.update_attributes!({state: Container::Running,
475 reused = Container.find_reusable(common_attrs)
476 assert_not_nil reused
477 assert_equal reused.uuid, c_running.uuid
480 test "find_reusable method should select locked over queued container" do
481 set_user_from_auth :active
482 common_attrs = REUSABLE_COMMON_ATTRS.merge({environment: {"var" => "running_vs_locked"}})
483 c_locked, _ = minimal_new(common_attrs.merge({use_existing: false}))
484 c_queued, _ = minimal_new(common_attrs.merge({use_existing: false}))
485 assert_not_equal c_queued.uuid, c_locked.uuid
486 set_user_from_auth :dispatch1
487 c_locked.update_attributes!({state: Container::Locked})
488 reused = Container.find_reusable(common_attrs)
489 assert_not_nil reused
490 assert_equal reused.uuid, c_locked.uuid
493 test "find_reusable method should not select failed container" do
494 set_user_from_auth :active
495 attrs = REUSABLE_COMMON_ATTRS.merge({environment: {"var" => "failed"}})
496 c, _ = minimal_new(attrs)
497 set_user_from_auth :dispatch1
498 c.update_attributes!({state: Container::Locked})
499 c.update_attributes!({state: Container::Running})
500 c.update_attributes!({state: Container::Complete,
502 reused = Container.find_reusable(attrs)
506 test "find_reusable with logging disabled" do
507 set_user_from_auth :active
508 Rails.logger.expects(:info).never
509 Container.find_reusable(REUSABLE_COMMON_ATTRS)
512 test "find_reusable with logging enabled" do
513 set_user_from_auth :active
514 Rails.configuration.Containers.LogReuseDecisions = true
515 Rails.logger.expects(:info).at_least(3)
516 Container.find_reusable(REUSABLE_COMMON_ATTRS)
519 def runtime_token_attr tok
520 auth = api_client_authorizations(tok)
521 {runtime_user_uuid: User.find_by_id(auth.user_id).uuid,
522 runtime_auth_scopes: auth.scopes,
523 runtime_token: auth.token}
526 test "find_reusable method with same runtime_token" do
527 set_user_from_auth :active
528 common_attrs = REUSABLE_COMMON_ATTRS.merge({use_existing:false, priority:1, environment:{"var" => "queued"}})
529 c1, _ = minimal_new(common_attrs.merge({runtime_token: api_client_authorizations(:container_runtime_token).token}))
530 assert_equal Container::Queued, c1.state
531 reused = Container.find_reusable(common_attrs.merge(runtime_token_attr(:container_runtime_token)))
532 assert_not_nil reused
533 assert_equal reused.uuid, c1.uuid
536 test "find_reusable method with different runtime_token, same user" do
537 set_user_from_auth :active
538 common_attrs = REUSABLE_COMMON_ATTRS.merge({use_existing:false, priority:1, environment:{"var" => "queued"}})
539 c1, _ = minimal_new(common_attrs.merge({runtime_token: api_client_authorizations(:crt_user).token}))
540 assert_equal Container::Queued, c1.state
541 reused = Container.find_reusable(common_attrs.merge(runtime_token_attr(:container_runtime_token)))
542 assert_not_nil reused
543 assert_equal reused.uuid, c1.uuid
546 test "find_reusable method with nil runtime_token, then runtime_token with same user" do
547 set_user_from_auth :crt_user
548 common_attrs = REUSABLE_COMMON_ATTRS.merge({use_existing:false, priority:1, environment:{"var" => "queued"}})
549 c1, _ = minimal_new(common_attrs)
550 assert_equal Container::Queued, c1.state
551 assert_equal users(:container_runtime_token_user).uuid, c1.runtime_user_uuid
552 reused = Container.find_reusable(common_attrs.merge(runtime_token_attr(:container_runtime_token)))
553 assert_not_nil reused
554 assert_equal reused.uuid, c1.uuid
557 test "find_reusable method with different runtime_token, different user" do
558 set_user_from_auth :crt_user
559 common_attrs = REUSABLE_COMMON_ATTRS.merge({use_existing:false, priority:1, environment:{"var" => "queued"}})
560 c1, _ = minimal_new(common_attrs.merge({runtime_token: api_client_authorizations(:active).token}))
561 assert_equal Container::Queued, c1.state
562 reused = Container.find_reusable(common_attrs.merge(runtime_token_attr(:container_runtime_token)))
564 assert_equal c1.uuid, reused.uuid
567 test "find_reusable method with nil runtime_token, then runtime_token with different user" do
568 set_user_from_auth :active
569 common_attrs = REUSABLE_COMMON_ATTRS.merge({use_existing:false, priority:1, environment:{"var" => "queued"}})
570 c1, _ = minimal_new(common_attrs.merge({runtime_token: nil}))
571 assert_equal Container::Queued, c1.state
572 reused = Container.find_reusable(common_attrs.merge(runtime_token_attr(:container_runtime_token)))
574 assert_equal c1.uuid, reused.uuid
577 test "find_reusable method with different runtime_token, different scope, same user" do
578 set_user_from_auth :active
579 common_attrs = REUSABLE_COMMON_ATTRS.merge({use_existing:false, priority:1, environment:{"var" => "queued"}})
580 c1, _ = minimal_new(common_attrs.merge({runtime_token: api_client_authorizations(:runtime_token_limited_scope).token}))
581 assert_equal Container::Queued, c1.state
582 reused = Container.find_reusable(common_attrs.merge(runtime_token_attr(:container_runtime_token)))
584 assert_equal c1.uuid, reused.uuid
587 test "Container running" do
588 set_user_from_auth :active
589 c, _ = minimal_new priority: 1
591 set_user_from_auth :dispatch1
592 check_illegal_updates c, [{state: Container::Running},
593 {state: Container::Complete}]
596 c.update_attributes! state: Container::Running
598 check_illegal_modify c
601 check_illegal_updates c, [{state: Container::Queued}]
604 c.update_attributes! priority: 3
607 test "Lock and unlock" do
608 set_user_from_auth :active
609 c, cr = minimal_new priority: 0
611 set_user_from_auth :dispatch1
612 assert_equal Container::Queued, c.state
614 assert_raise(ArvadosModel::LockFailedError) do
619 assert cr.update_attributes priority: 1
621 refute c.update_attributes(state: Container::Running), "not locked"
623 refute c.update_attributes(state: Container::Complete), "not locked"
626 assert c.lock, show_errors(c)
627 assert c.locked_by_uuid
630 assert_raise(ArvadosModel::LockFailedError) {c.lock}
633 assert c.unlock, show_errors(c)
634 refute c.locked_by_uuid
637 refute c.update_attributes(state: Container::Running), "not locked"
639 refute c.locked_by_uuid
642 assert c.lock, show_errors(c)
643 assert c.update_attributes(state: Container::Running), show_errors(c)
644 assert c.locked_by_uuid
647 auth_uuid_was = c.auth_uuid
649 assert_raise(ArvadosModel::LockFailedError) do
650 # Running to Locked is not allowed
654 assert_raise(ArvadosModel::InvalidStateTransitionError) do
655 # Running to Queued is not allowed
660 assert c.update_attributes(state: Container::Complete), show_errors(c)
661 refute c.locked_by_uuid
664 auth_exp = ApiClientAuthorization.find_by_uuid(auth_uuid_was).expires_at
665 assert_operator auth_exp, :<, db_current_time
667 assert_nil ApiClientAuthorization.validate(token: ApiClientAuthorization.find_by_uuid(auth_uuid_was).token)
670 test "Exceed maximum lock-unlock cycles" do
671 Rails.configuration.Containers.MaxDispatchAttempts = 3
673 set_user_from_auth :active
676 set_user_from_auth :dispatch1
677 assert_equal Container::Queued, c.state
678 assert_equal 0, c.lock_count
682 assert_equal 1, c.lock_count
683 assert_equal Container::Locked, c.state
687 assert_equal 1, c.lock_count
688 assert_equal Container::Queued, c.state
692 assert_equal 2, c.lock_count
693 assert_equal Container::Locked, c.state
697 assert_equal 2, c.lock_count
698 assert_equal Container::Queued, c.state
702 assert_equal 3, c.lock_count
703 assert_equal Container::Locked, c.state
707 assert_equal 3, c.lock_count
708 assert_equal Container::Cancelled, c.state
710 assert_raise(ArvadosModel::LockFailedError) do
711 # Cancelled to Locked is not allowed
716 test "Container queued cancel" do
717 set_user_from_auth :active
718 c, cr = minimal_new({container_count_max: 1})
719 set_user_from_auth :dispatch1
720 assert c.update_attributes(state: Container::Cancelled), show_errors(c)
721 check_no_change_from_cancelled c
723 assert_equal ContainerRequest::Final, cr.state
726 test "Container queued count" do
727 assert_equal 1, Container.readable_by(users(:active)).where(state: "Queued").count
730 test "Containers with no matching request are readable by admin" do
731 uuids = Container.includes('container_requests').where(container_requests: {uuid: nil}).collect(&:uuid)
732 assert_not_empty uuids
733 assert_empty Container.readable_by(users(:active)).where(uuid: uuids)
734 assert_not_empty Container.readable_by(users(:admin)).where(uuid: uuids)
735 assert_equal uuids.count, Container.readable_by(users(:admin)).where(uuid: uuids).count
738 test "Container locked cancel" do
739 set_user_from_auth :active
741 set_user_from_auth :dispatch1
742 assert c.lock, show_errors(c)
743 assert c.update_attributes(state: Container::Cancelled), show_errors(c)
744 check_no_change_from_cancelled c
747 test "Container locked cancel with log" do
748 set_user_from_auth :active
750 set_user_from_auth :dispatch1
751 assert c.lock, show_errors(c)
752 assert c.update_attributes(
753 state: Container::Cancelled,
754 log: collections(:real_log_collection).portable_data_hash,
756 check_no_change_from_cancelled c
759 test "Container running cancel" do
760 set_user_from_auth :active
762 set_user_from_auth :dispatch1
764 c.update_attributes! state: Container::Running
765 c.update_attributes! state: Container::Cancelled
766 check_no_change_from_cancelled c
769 test "Container create forbidden for non-admin" do
770 set_user_from_auth :active_trustedclient
771 c = Container.new DEFAULT_ATTRS
773 c.mounts = {"BAR" => "FOO"}
774 c.output_path = "/tmp"
776 c.runtime_constraints = {}
777 assert_raises(ArvadosModel::PermissionDeniedError) do
783 [Container::Queued, {state: Container::Locked}],
784 [Container::Queued, {state: Container::Running}],
785 [Container::Queued, {state: Container::Complete}],
786 [Container::Queued, {state: Container::Cancelled}],
787 [Container::Queued, {priority: 123456789}],
788 [Container::Queued, {runtime_status: {'error' => 'oops'}}],
789 [Container::Queued, {cwd: '/'}],
790 [Container::Locked, {state: Container::Running}],
791 [Container::Locked, {state: Container::Queued}],
792 [Container::Locked, {priority: 123456789}],
793 [Container::Locked, {runtime_status: {'error' => 'oops'}}],
794 [Container::Locked, {cwd: '/'}],
795 [Container::Running, {state: Container::Complete}],
796 [Container::Running, {state: Container::Cancelled}],
797 [Container::Running, {priority: 123456789}],
798 [Container::Running, {runtime_status: {'error' => 'oops'}}],
799 [Container::Running, {cwd: '/'}],
800 [Container::Complete, {state: Container::Cancelled}],
801 [Container::Complete, {priority: 123456789}],
802 [Container::Complete, {runtime_status: {'error' => 'oops'}}],
803 [Container::Complete, {cwd: '/'}],
804 [Container::Cancelled, {cwd: '/'}],
805 ].each do |start_state, updates|
806 test "Container update #{updates.inspect} when #{start_state} forbidden for non-admin" do
807 set_user_from_auth :active
809 if start_state != Container::Queued
810 set_user_from_auth :dispatch1
812 if start_state != Container::Locked
813 c.update_attributes! state: Container::Running
814 if start_state != Container::Running
815 c.update_attributes! state: start_state
819 assert_equal c.state, start_state
820 set_user_from_auth :active
821 assert_raises(ArvadosModel::PermissionDeniedError) do
822 c.update_attributes! updates
827 test "Container only set exit code on complete" do
828 set_user_from_auth :active
830 set_user_from_auth :dispatch1
832 c.update_attributes! state: Container::Running
834 check_illegal_updates c, [{exit_code: 1},
835 {exit_code: 1, state: Container::Cancelled}]
837 assert c.update_attributes(exit_code: 1, state: Container::Complete)
840 test "locked_by_uuid can update log when locked/running, and output when running" do
841 set_user_from_auth :active
842 logcoll = collections(:real_log_collection)
844 cr2 = ContainerRequest.new(DEFAULT_ATTRS)
845 cr2.state = ContainerRequest::Committed
846 act_as_user users(:active) do
849 assert_equal cr1.container_uuid, cr2.container_uuid
851 logpdh_time1 = logcoll.portable_data_hash
853 set_user_from_auth :dispatch1
855 assert_equal c.locked_by_uuid, Thread.current[:api_client_authorization].uuid
856 c.update_attributes!(log: logpdh_time1)
857 c.update_attributes!(state: Container::Running)
860 cr1log_uuid = cr1.log_uuid
861 cr2log_uuid = cr2.log_uuid
862 assert_not_nil cr1log_uuid
863 assert_not_nil cr2log_uuid
864 assert_not_equal logcoll.uuid, cr1log_uuid
865 assert_not_equal logcoll.uuid, cr2log_uuid
866 assert_not_equal cr1log_uuid, cr2log_uuid
868 logcoll.update_attributes!(manifest_text: logcoll.manifest_text + ". acbd18db4cc2f85cedef654fccc4a4d8+3 0:3:foo.txt\n")
869 logpdh_time2 = logcoll.portable_data_hash
871 assert c.update_attributes(output: collections(:collection_owned_by_active).portable_data_hash)
872 assert c.update_attributes(log: logpdh_time2)
873 assert c.update_attributes(state: Container::Complete, log: logcoll.portable_data_hash)
875 assert_equal collections(:collection_owned_by_active).portable_data_hash, c.output
876 assert_equal logpdh_time2, c.log
877 refute c.update_attributes(output: nil)
878 refute c.update_attributes(log: nil)
881 assert_equal cr1log_uuid, cr1.log_uuid
882 assert_equal cr2log_uuid, cr2.log_uuid
883 assert_equal 1, Collection.where(uuid: [cr1log_uuid, cr2log_uuid]).to_a.collect(&:portable_data_hash).uniq.length
884 assert_equal ". acbd18db4cc2f85cedef654fccc4a4d8+3 cdd549ae79fe6640fa3d5c6261d8303c+195 0:3:foo.txt 3:195:zzzzz-8i9sb-0vsrcqi7whchuil.log.txt
885 ./log\\040for\\040container\\040#{cr1.container_uuid} acbd18db4cc2f85cedef654fccc4a4d8+3 cdd549ae79fe6640fa3d5c6261d8303c+195 0:3:foo.txt 3:195:zzzzz-8i9sb-0vsrcqi7whchuil.log.txt
886 ", Collection.find_by_uuid(cr1log_uuid).manifest_text
889 ["auth_uuid", "runtime_token"].each do |tok|
890 test "#{tok} can set output, progress, runtime_status, state on running container -- but not log" do
891 if tok == "runtime_token"
892 set_user_from_auth :spectator
893 c, _ = minimal_new(container_image: "9ae44d5792468c58bcf85ce7353c7027+124",
894 runtime_token: api_client_authorizations(:active).token)
896 set_user_from_auth :active
899 set_user_from_auth :dispatch1
901 c.update_attributes! state: Container::Running
903 if tok == "runtime_token"
904 auth = ApiClientAuthorization.validate(token: c.runtime_token)
905 Thread.current[:api_client_authorization] = auth
906 Thread.current[:api_client] = auth.api_client
907 Thread.current[:token] = auth.token
908 Thread.current[:user] = auth.user
910 auth = ApiClientAuthorization.find_by_uuid(c.auth_uuid)
911 Thread.current[:api_client_authorization] = auth
912 Thread.current[:api_client] = auth.api_client
913 Thread.current[:token] = auth.token
914 Thread.current[:user] = auth.user
917 assert c.update_attributes(output: collections(:collection_owned_by_active).portable_data_hash)
918 assert c.update_attributes(runtime_status: {'warning' => 'something happened'})
919 assert c.update_attributes(progress: 0.5)
920 refute c.update_attributes(log: collections(:real_log_collection).portable_data_hash)
922 assert c.update_attributes(state: Container::Complete, exit_code: 0)
926 test "not allowed to set output that is not readable by current user" do
927 set_user_from_auth :active
929 set_user_from_auth :dispatch1
931 c.update_attributes! state: Container::Running
933 Thread.current[:api_client_authorization] = ApiClientAuthorization.find_by_uuid(c.auth_uuid)
934 Thread.current[:user] = User.find_by_id(Thread.current[:api_client_authorization].user_id)
936 assert_raises ActiveRecord::RecordInvalid do
937 c.update_attributes! output: collections(:collection_not_readable_by_active).portable_data_hash
941 test "other token cannot set output on running container" do
942 set_user_from_auth :active
944 set_user_from_auth :dispatch1
946 c.update_attributes! state: Container::Running
948 set_user_from_auth :running_to_be_deleted_container_auth
949 assert_raises(ArvadosModel::PermissionDeniedError) do
950 c.update_attributes(output: collections(:foo_file).portable_data_hash)
954 test "can set trashed output on running container" do
955 set_user_from_auth :active
957 set_user_from_auth :dispatch1
959 c.update_attributes! state: Container::Running
961 output = Collection.find_by_uuid('zzzzz-4zz18-mto52zx1s7sn3jk')
963 assert output.is_trashed
964 assert c.update_attributes output: output.portable_data_hash
965 assert c.update_attributes! state: Container::Complete
968 test "not allowed to set trashed output that is not readable by current user" do
969 set_user_from_auth :active
971 set_user_from_auth :dispatch1
973 c.update_attributes! state: Container::Running
975 output = Collection.find_by_uuid('zzzzz-4zz18-mto52zx1s7sn3jr')
977 Thread.current[:api_client_authorization] = ApiClientAuthorization.find_by_uuid(c.auth_uuid)
978 Thread.current[:user] = User.find_by_id(Thread.current[:api_client_authorization].user_id)
980 assert_raises ActiveRecord::RecordInvalid do
981 c.update_attributes! output: output.portable_data_hash
985 test "user cannot delete" do
986 set_user_from_auth :active
988 assert_raises ArvadosModel::PermissionDeniedError do
991 assert Container.find_by_uuid(c.uuid)
995 {state: Container::Complete, exit_code: 0, output: '1f4b0bc7583c2a7f9102c395f4ffc5e3+45'},
996 {state: Container::Cancelled},
997 ].each do |final_attrs|
998 test "secret_mounts and runtime_token are null after container is #{final_attrs[:state]}" do
999 set_user_from_auth :active
1000 c, cr = minimal_new(secret_mounts: {'/secret' => {'kind' => 'text', 'content' => 'foo'}},
1001 container_count_max: 1, runtime_token: api_client_authorizations(:active).token)
1002 set_user_from_auth :dispatch1
1004 c.update_attributes!(state: Container::Running)
1006 assert c.secret_mounts.has_key?('/secret')
1007 assert_equal api_client_authorizations(:active).token, c.runtime_token
1009 c.update_attributes!(final_attrs)
1011 assert_equal({}, c.secret_mounts)
1012 assert_nil c.runtime_token
1014 assert_equal({}, cr.secret_mounts)
1015 assert_nil cr.runtime_token
1016 assert_no_secrets_logged