X-Git-Url: https://git.arvados.org/arvados.git/blobdiff_plain/58a026e09bda4c1e2374347615c325007c64fac4..fd6df03fd625fe0f9126205ab324df8533fdcc79:/services/api/app/models/container_request.rb diff --git a/services/api/app/models/container_request.rb b/services/api/app/models/container_request.rb index dd3ff767dd..bbec421084 100644 --- a/services/api/app/models/container_request.rb +++ b/services/api/app/models/container_request.rb @@ -33,6 +33,7 @@ class ContainerRequest < ArvadosModel validates :command, :container_image, :output_path, :cwd, :presence => true validates :output_ttl, numericality: { only_integer: true, greater_than_or_equal_to: 0 } validates :priority, numericality: { only_integer: true, greater_than_or_equal_to: 0, less_than_or_equal_to: 1000 } + validate :validate_datatypes validate :validate_scheduling_parameters validate :validate_state_change validate :check_update_whitelist @@ -122,11 +123,13 @@ class ContainerRequest < ArvadosModel # Finalize the container request after the container has # finished/cancelled. def finalize! - out_coll = nil - log_coll = nil - c = Container.find_by_uuid(container_uuid) - ['output', 'log'].each do |out_type| - pdh = c.send(out_type) + update_collections(container: Container.find_by_uuid(container_uuid)) + update_attributes!(state: Final) + end + + def update_collections(container:, collections: ['log', 'output']) + collections.each do |out_type| + pdh = container.send(out_type) next if pdh.nil? coll_name = "Container #{out_type} for request #{uuid}" trash_at = nil @@ -140,24 +143,25 @@ class ContainerRequest < ArvadosModel end manifest = Collection.where(portable_data_hash: pdh).first.manifest_text - coll = Collection.new(owner_uuid: owner_uuid, - manifest_text: manifest, - portable_data_hash: pdh, - name: coll_name, - trash_at: trash_at, - delete_at: trash_at, - properties: { - 'type' => out_type, - 'container_request' => uuid, - }) - coll.save_with_unique_name! - if out_type == 'output' - out_coll = coll.uuid - else - log_coll = coll.uuid + coll_uuid = self.send(out_type + '_uuid') + coll = coll_uuid.nil? ? nil : Collection.where(uuid: coll_uuid).first + if !coll + coll = Collection.new( + owner_uuid: self.owner_uuid, + name: coll_name, + properties: { + 'type' => out_type, + 'container_request' => uuid, + }) end + coll.assign_attributes( + portable_data_hash: pdh, + manifest_text: manifest, + trash_at: trash_at, + delete_at: trash_at) + coll.save_with_unique_name! + self.send(out_type + '_uuid=', coll.uuid) end - update_attributes!(state: Final, output_uuid: out_coll, log_uuid: log_coll) end def self.full_text_searchable_columns @@ -228,6 +232,43 @@ class ContainerRequest < ArvadosModel end end + def validate_datatypes + command.each do |c| + if !c.is_a? String + errors.add(:command, "must be an array of strings but has entry #{c.class}") + end + end + environment.each do |k,v| + if !k.is_a?(String) || !v.is_a?(String) + errors.add(:environment, "must be an map of String to String but has entry #{k.class} to #{v.class}") + end + end + [:mounts, :secret_mounts].each do |m| + self[m].each do |k, v| + if !k.is_a?(String) || !v.is_a?(Hash) + errors.add(m, "must be an map of String to Hash but is has entry #{k.class} to #{v.class}") + end + if v["kind"].nil? + errors.add(m, "each item must have a 'kind' field") + end + [[String, ["kind", "portable_data_hash", "uuid", "device_type", + "path", "commit", "repository_name", "git_url"]], + [Integer, ["capacity"]]].each do |t, fields| + fields.each do |f| + if !v[f].nil? && !v[f].is_a?(t) + errors.add(m, "#{k}: #{f} must be a #{t} but is #{v[f].class}") + end + end + end + ["writable", "exclude_from_output"].each do |f| + if !v[f].nil? && !v[f].is_a?(TrueClass) && !v[f].is_a?(FalseClass) + errors.add(m, "#{k}: #{f} must be a #{t} but is #{v[f].class}") + end + end + end + end + end + def validate_scheduling_parameters if self.state == Committed if scheduling_parameters.include? 'partitions' and @@ -274,6 +315,10 @@ class ContainerRequest < ArvadosModel permitted.push :container_count end + if current_user.andand.is_admin + permitted.push :log_uuid + end + when Final if self.state_was == Committed # "Cancel" means setting priority=0, state=Committed @@ -321,7 +366,9 @@ class ContainerRequest < ArvadosModel c = get_requesting_container() if !c.nil? self.requesting_container_uuid = c.uuid - self.priority = c.priority>0 ? 1 : 0 + # Determine the priority of container request for the requesting + # container. + self.priority = ContainerRequest.where(container_uuid: self.requesting_container_uuid).maximum("priority") || 0 end end