X-Git-Url: https://git.arvados.org/arvados.git/blobdiff_plain/b02ee3ad4768636bc85097ae0c76b36a38a61964..5dbc1ae3d451f904654a2a61e5df620808ac175d:/services/api/app/models/arvados_model.rb diff --git a/services/api/app/models/arvados_model.rb b/services/api/app/models/arvados_model.rb index f5d20b1fef..93d5b9a023 100644 --- a/services/api/app/models/arvados_model.rb +++ b/services/api/app/models/arvados_model.rb @@ -2,13 +2,16 @@ # # SPDX-License-Identifier: AGPL-3.0 +require 'arvados_model_updates' require 'has_uuid' require 'record_filters' require 'serializers' +require 'request_error' class ArvadosModel < ActiveRecord::Base self.abstract_class = true + include ArvadosModelUpdates include CurrentApiClient # current_user, current_api_client, etc. include DbCurrentTime extend RecordFilters @@ -38,37 +41,37 @@ class ArvadosModel < ActiveRecord::Base class_name: 'Link', primary_key: :uuid) - class PermissionDeniedError < StandardError + class PermissionDeniedError < RequestError def http_status 403 end end - class AlreadyLockedError < StandardError + class AlreadyLockedError < RequestError def http_status 422 end end - class LockFailedError < StandardError + class LockFailedError < RequestError def http_status 422 end end - class InvalidStateTransitionError < StandardError + class InvalidStateTransitionError < RequestError def http_status 422 end end - class UnauthorizedError < StandardError + class UnauthorizedError < RequestError def http_status 401 end end - class UnresolvableContainerError < StandardError + class UnresolvableContainerError < RequestError def http_status 422 end @@ -237,7 +240,7 @@ class ArvadosModel < ActiveRecord::Base end.compact.uniq end - # Return a query with read permissions restricted to the union of of the + # Return a query with read permissions restricted to the union of the # permissions of the members of users_list, i.e. if something is readable by # any user in users_list, it will be readable in the query returned by this # function. @@ -255,63 +258,76 @@ class ArvadosModel < ActiveRecord::Base # Collect the UUIDs of the authorized users. sql_table = kwargs.fetch(:table_name, table_name) include_trash = kwargs.fetch(:include_trash, false) - query_on = kwargs.fetch(:query_on, self) + include_old_versions = kwargs.fetch(:include_old_versions, false) - sql_conds = [] + sql_conds = nil user_uuids = users_list.map { |u| u.uuid } - User.install_view('permission') + exclude_trashed_records = "" + if !include_trash and (sql_table == "groups" or sql_table == "collections") then + # Only include records that are not explicitly trashed + exclude_trashed_records = "AND #{sql_table}.is_trashed = false" + end - # Check if any of the users are admin. if users_list.select { |u| u.is_admin }.any? + # Admin skips most permission checks, but still want to filter on trashed items. if !include_trash - # exclude rows that are explicitly trashed. - if self.column_names.include? "owner_uuid" - sql_conds += ["NOT EXISTS(SELECT 1 - FROM permission_view - WHERE trashed = 1 AND - (#{sql_table}.uuid = target_uuid OR #{sql_table}.owner_uuid = target_uuid))"] - else - sql_conds += ["NOT EXISTS(SELECT 1 - FROM permission_view - WHERE trashed = 1 AND - (#{sql_table}.uuid = target_uuid))"] + if sql_table != "api_client_authorizations" + # Only include records where the owner is not trashed + sql_conds = "#{sql_table}.owner_uuid NOT IN (SELECT target_uuid FROM #{PERMISSION_VIEW} "+ + "WHERE trashed = 1) #{exclude_trashed_records}" end end else - # Can read object (evidently a group or user) whose UUID is listed - # explicitly in user_uuids. - sql_conds += ["#{sql_table}.uuid IN (:user_uuids)"] - - if include_trash - trashed_check = "" - else - trashed_check = "trashed = 0 AND" + trashed_check = "" + if !include_trash then + trashed_check = "AND trashed = 0" end - perm_check = "perm_level >= 1" - - if self.column_names.include? "owner_uuid" and sql_table != "groups" - owner_check = "OR #{sql_table}.owner_uuid IN (:user_uuids) OR (target_uuid = #{sql_table}.owner_uuid AND target_owner_uuid IS NOT NULL)" - else - owner_check = "" + # Note: it is possible to combine the direct_check and + # owner_check into a single EXISTS() clause, however it turns + # out query optimizer doesn't like it and forces a sequential + # table scan. Constructing the query with separate EXISTS() + # clauses enables it to use the index. + # + # see issue 13208 for details. + + # Match a direct read permission link from the user to the record uuid + direct_check = "#{sql_table}.uuid IN (SELECT target_uuid FROM #{PERMISSION_VIEW} "+ + "WHERE user_uuid IN (:user_uuids) AND perm_level >= 1 #{trashed_check})" + + # Match a read permission link from the user to the record's owner_uuid + owner_check = "" + if sql_table != "api_client_authorizations" and sql_table != "groups" then + owner_check = "OR #{sql_table}.owner_uuid IN (SELECT target_uuid FROM #{PERMISSION_VIEW} "+ + "WHERE user_uuid IN (:user_uuids) AND perm_level >= 1 #{trashed_check} AND target_owner_uuid IS NOT NULL) " end - sql_conds += ["EXISTS(SELECT 1 FROM permission_view "+ - "WHERE user_uuid IN (:user_uuids) AND #{trashed_check} #{perm_check} AND (target_uuid = #{sql_table}.uuid #{owner_check}))"] - + links_cond = "" if sql_table == "links" # Match any permission link that gives one of the authorized # users some permission _or_ gives anyone else permission to # view one of the authorized users. - sql_conds += ["(#{sql_table}.link_class IN (:permission_link_classes) AND "+ - "(#{sql_table}.head_uuid IN (:user_uuids) OR #{sql_table}.tail_uuid IN (:user_uuids)))"] + links_cond = "OR (#{sql_table}.link_class IN (:permission_link_classes) AND "+ + "(#{sql_table}.head_uuid IN (:user_uuids) OR #{sql_table}.tail_uuid IN (:user_uuids)))" + end + + sql_conds = "(#{direct_check} #{owner_check} #{links_cond}) #{exclude_trashed_records}" + + end + + if !include_old_versions && sql_table == "collections" + exclude_old_versions = "#{sql_table}.uuid = #{sql_table}.current_version_uuid" + if sql_conds.nil? + sql_conds = exclude_old_versions + else + sql_conds += " AND #{exclude_old_versions}" end end - query_on.where(sql_conds.join(' OR '), - user_uuids: user_uuids, - permission_link_classes: ['permission', 'resources']) + self.where(sql_conds, + user_uuids: user_uuids, + permission_link_classes: ['permission', 'resources']) end def save_with_unique_name! @@ -356,7 +372,13 @@ class ArvadosModel < ActiveRecord::Base end self[:name] = new_name - self[:uuid] = nil if uuid_was.nil? && !uuid.nil? + if uuid_was.nil? && !uuid.nil? + self[:uuid] = nil + if self.is_a? Collection + # Reset so that is assigned to the new UUID + self[:current_version_uuid] = nil + end + end conn.exec_query 'SAVEPOINT save_with_unique_name' retry ensure @@ -371,15 +393,16 @@ class ArvadosModel < ActiveRecord::Base def self.full_text_searchable_columns self.columns.select do |col| - col.type == :string or col.type == :text + [:string, :text, :jsonb].include?(col.type) end.map(&:name) end def self.full_text_tsvector parts = full_text_searchable_columns.collect do |column| - "coalesce(#{column},'')" + cast = serialized_attributes[column] ? '::text' : '' + "coalesce(#{column}#{cast},'')" end - "to_tsvector('english', #{parts.join(" || ' ' || ")})" + "to_tsvector('english', substr(#{parts.join(" || ' ' || ")}, 0, 8000))" end def self.apply_filters query, filters @@ -529,11 +552,13 @@ class ArvadosModel < ActiveRecord::Base def update_modified_by_fields current_time = db_current_time - self.created_at = created_at_was || current_time + self.created_at ||= created_at_was || current_time self.updated_at = current_time self.owner_uuid ||= current_default_owner if self.respond_to? :owner_uuid= - self.modified_at = current_time - self.modified_by_user_uuid = current_user ? current_user.uuid : nil + if !anonymous_updater + self.modified_by_user_uuid = current_user ? current_user.uuid : nil + self.modified_at = current_time + end self.modified_by_client_uuid = current_api_client ? current_api_client.uuid : nil true end @@ -586,16 +611,24 @@ class ArvadosModel < ActiveRecord::Base end end - def self.where_serialized(colname, value) + def self.where_serialized(colname, value, md5: false) + colsql = colname.to_s + if md5 + colsql = "md5(#{colsql})" + end if value.empty? # rails4 stores as null, rails3 stored as serialized [] or {} - sql = "#{colname.to_s} is null or #{colname.to_s} IN (?)" + sql = "#{colsql} is null or #{colsql} IN (?)" sorted = value else - sql = "#{colname.to_s} IN (?)" + sql = "#{colsql} IN (?)" sorted = deep_sort_hash(value) end - where(sql, [sorted.to_yaml, SafeJSON.dump(sorted)]) + params = [sorted.to_yaml, SafeJSON.dump(sorted)] + if md5 + params = params.map { |x| Digest::MD5.hexdigest(x) } + end + where(sql, params) end Serializer = { @@ -747,42 +780,57 @@ class ArvadosModel < ActiveRecord::Base if self == ArvadosModel # If called directly as ArvadosModel.find_by_uuid rather than via subclass, # delegate to the appropriate subclass based on the given uuid. - self.resource_class_for_uuid(uuid).unscoped.find_by_uuid(uuid) + self.resource_class_for_uuid(uuid).find_by_uuid(uuid) else super end end + def is_audit_logging_enabled? + return !(Rails.configuration.max_audit_log_age.to_i == 0 && + Rails.configuration.max_audit_log_delete_batch.to_i > 0) + end + def log_start_state - @old_attributes = Marshal.load(Marshal.dump(attributes)) - @old_logged_attributes = Marshal.load(Marshal.dump(logged_attributes)) + if is_audit_logging_enabled? + @old_attributes = Marshal.load(Marshal.dump(attributes)) + @old_logged_attributes = Marshal.load(Marshal.dump(logged_attributes)) + end end def log_change(event_type) - log = Log.new(event_type: event_type).fill_object(self) - yield log - log.save! - log_start_state + if is_audit_logging_enabled? + log = Log.new(event_type: event_type).fill_object(self) + yield log + log.save! + log_start_state + end end def log_create - log_change('create') do |log| - log.fill_properties('old', nil, nil) - log.update_to self + if is_audit_logging_enabled? + log_change('create') do |log| + log.fill_properties('old', nil, nil) + log.update_to self + end end end def log_update - log_change('update') do |log| - log.fill_properties('old', etag(@old_attributes), @old_logged_attributes) - log.update_to self + if is_audit_logging_enabled? + log_change('update') do |log| + log.fill_properties('old', etag(@old_attributes), @old_logged_attributes) + log.update_to self + end end end def log_destroy - log_change('delete') do |log| - log.fill_properties('old', etag(@old_attributes), @old_logged_attributes) - log.update_to nil + if is_audit_logging_enabled? + log_change('delete') do |log| + log.fill_properties('old', etag(@old_attributes), @old_logged_attributes) + log.update_to nil + end end end end