X-Git-Url: https://git.arvados.org/arvados.git/blobdiff_plain/e8d1a643cdbc3a5f4c0e5c745da58d9f7e1248d8..24a1cdaa5b5a3a98b47ab6802d00e7c4d2a848df:/services/api/app/models/container.rb diff --git a/services/api/app/models/container.rb b/services/api/app/models/container.rb index 5833c2251f..42d0ed49bf 100644 --- a/services/api/app/models/container.rb +++ b/services/api/app/models/container.rb @@ -21,7 +21,9 @@ class Container < ArvadosModel # already know how to properly treat them. attribute :secret_mounts, :jsonbHash, default: {} attribute :runtime_status, :jsonbHash, default: {} - attribute :runtime_auth_scopes, :jsonbHash, default: {} + attribute :runtime_auth_scopes, :jsonbArray, default: [] + attribute :output_storage_classes, :jsonbArray, default: lambda { Rails.configuration.DefaultStorageClasses } + attribute :output_properties, :jsonbHash, default: {} serialize :environment, Hash serialize :mounts, Hash @@ -29,6 +31,7 @@ class Container < ArvadosModel serialize :command, Array serialize :scheduling_parameters, Hash + after_find :fill_container_defaults_after_find before_validation :fill_field_defaults, :if => :new_record? before_validation :set_timestamps before_validation :check_lock @@ -76,6 +79,12 @@ class Container < ArvadosModel t.add :runtime_user_uuid t.add :runtime_auth_scopes t.add :lock_count + t.add :gateway_address + t.add :interactive_session_started + t.add :output_storage_classes + t.add :output_properties + t.add :cost + t.add :subrequests_cost end # Supported states for a container @@ -101,11 +110,11 @@ class Container < ArvadosModel end def self.full_text_searchable_columns - super - ["secret_mounts", "secret_mounts_md5", "runtime_token"] + super - ["secret_mounts", "secret_mounts_md5", "runtime_token", "gateway_address", "output_storage_classes"] end def self.searchable_columns *args - super - ["secret_mounts_md5", "runtime_token"] + super - ["secret_mounts_md5", "runtime_token", "gateway_address", "output_storage_classes"] end def logged_attributes @@ -184,7 +193,8 @@ class Container < ArvadosModel secret_mounts: req.secret_mounts, runtime_token: req.runtime_token, runtime_user_uuid: runtime_user.uuid, - runtime_auth_scopes: runtime_auth_scopes + runtime_auth_scopes: runtime_auth_scopes, + output_storage_classes: req.output_storage_classes, } end act_as_system_user do @@ -207,17 +217,22 @@ class Container < ArvadosModel # containers are suitable). def self.resolve_runtime_constraints(runtime_constraints) rc = {} - defaults = { - 'keep_cache_ram' => - Rails.configuration.Containers.DefaultKeepCacheRAM, - } - defaults.merge(runtime_constraints).each do |k, v| + runtime_constraints.each do |k, v| if v.is_a? Array rc[k] = v[0] else rc[k] = v end end + if rc['keep_cache_ram'] == 0 + rc['keep_cache_ram'] = Rails.configuration.Containers.DefaultKeepCacheRAM + end + if rc['keep_cache_disk'] == 0 and rc['keep_cache_ram'] == 0 + # If neither ram nor disk cache was specified and + # DefaultKeepCacheRAM==0, default to disk cache with size equal + # to RAM constraint (but at least 2 GiB and at most 32 GiB). + rc['keep_cache_disk'] = [[rc['ram'] || 0, 2 << 30].max, 32 << 30].min + end rc end @@ -284,7 +299,30 @@ class Container < ArvadosModel candidates = candidates.where('secret_mounts_md5 = ?', secret_mounts_md5) log_reuse_info(candidates) { "after filtering on secret_mounts_md5 #{secret_mounts_md5.inspect}" } - candidates = candidates.where_serialized(:runtime_constraints, resolve_runtime_constraints(attrs[:runtime_constraints]), md5: true) + if attrs[:runtime_constraints]['cuda'].nil? + attrs[:runtime_constraints]['cuda'] = { + 'device_count' => 0, + 'driver_version' => '', + 'hardware_capability' => '', + } + end + resolved_runtime_constraints = [resolve_runtime_constraints(attrs[:runtime_constraints])] + if resolved_runtime_constraints[0]['cuda']['device_count'] == 0 + # If no CUDA requested, extend search to include older container + # records that don't have a 'cuda' section in runtime_constraints + resolved_runtime_constraints << resolved_runtime_constraints[0].except('cuda') + end + if resolved_runtime_constraints[0]['keep_cache_disk'] == 0 + # If no disk cache requested, extend search to include older container + # records that don't have a 'keep_cache_disk' field in runtime_constraints + if resolved_runtime_constraints.length == 2 + # exclude the one that also excludes CUDA + resolved_runtime_constraints << resolved_runtime_constraints[1].except('keep_cache_disk') + end + resolved_runtime_constraints << resolved_runtime_constraints[0].except('keep_cache_disk') + end + + candidates = candidates.where_serialized(:runtime_constraints, resolved_runtime_constraints, md5: true, multivalue: true) log_reuse_info(candidates) { "after filtering on runtime_constraints #{attrs[:runtime_constraints].inspect}" } log_reuse_info { "checking for state=Complete with readable output and log..." } @@ -457,15 +495,17 @@ class Container < ArvadosModel def validate_change permitted = [:state] - progress_attrs = [:progress, :runtime_status, :log, :output] - final_attrs = [:exit_code, :finished_at] + final_attrs = [:finished_at] + progress_attrs = [:progress, :runtime_status, :subrequests_cost, :cost, + :log, :output, :output_properties, :exit_code] if self.new_record? permitted.push(:owner_uuid, :command, :container_image, :cwd, :environment, :mounts, :output_path, :priority, :runtime_constraints, :scheduling_parameters, :secret_mounts, :runtime_token, - :runtime_user_uuid, :runtime_auth_scopes) + :runtime_user_uuid, :runtime_auth_scopes, + :output_storage_classes) end case self.state @@ -476,10 +516,13 @@ class Container < ArvadosModel permitted.push :priority when Running - permitted.push :priority, *progress_attrs + permitted.push :priority, :output_properties, :gateway_address, *progress_attrs if self.state_changed? permitted.push :started_at end + if !self.interactive_session_started_was + permitted.push :interactive_session_started + end when Complete if self.state_was == Running @@ -491,7 +534,7 @@ class Container < ArvadosModel when Running permitted.push :finished_at, *progress_attrs when Queued, Locked - permitted.push :finished_at, :log, :runtime_status + permitted.push :finished_at, :log, :runtime_status, :cost end else @@ -600,7 +643,8 @@ class Container < ArvadosModel self.runtime_auth_scopes = ["all"] end - # generate a new token + # Generate a new token. This runs with admin credentials as it's done by a + # dispatcher user, so expires_at isn't enforced by API.MaxTokenLifetime. self.auth = ApiClientAuthorization. create!(user_id: User.find_by_uuid(self.runtime_user_uuid).id, api_client_id: 0, @@ -674,6 +718,31 @@ class Container < ArvadosModel end if retryable_requests.any? + scheduling_parameters = { + # partitions: empty if any are empty, else the union of all parameters + "partitions": retryable_requests + .map { |req| req.scheduling_parameters["partitions"] || [] } + .reduce { |cur, new| (cur.empty? or new.empty?) ? [] : (cur | new) }, + + # preemptible: true if all are true, else false + "preemptible": retryable_requests + .map { |req| req.scheduling_parameters["preemptible"] } + .all?, + + # max_run_time: 0 if any are 0 (unlimited), else the maximum + "max_run_time": retryable_requests + .map { |req| req.scheduling_parameters["max_run_time"] || 0 } + .reduce do |cur, new| + if cur == 0 or new == 0 + 0 + elsif new > cur + new + else + cur + end + end, + } + c_attrs = { command: self.command, cwd: self.cwd, @@ -682,7 +751,7 @@ class Container < ArvadosModel container_image: self.container_image, mounts: self.mounts, runtime_constraints: self.runtime_constraints, - scheduling_parameters: self.scheduling_parameters, + scheduling_parameters: scheduling_parameters, secret_mounts: prev_secret_mounts, runtime_token: prev_runtime_token, runtime_user_uuid: self.runtime_user_uuid, @@ -693,6 +762,7 @@ class Container < ArvadosModel cr.with_lock do leave_modified_by_user_alone do # Use row locking because this increments container_count + cr.cumulative_cost += self.cost + self.subrequests_cost cr.container_uuid = c.uuid cr.save! end