X-Git-Url: https://git.arvados.org/arvados.git/blobdiff_plain/3da41130c997f36f462feb1b34efbec53697d91e..0f537bcaa60b8a1496010bc9d4a943484e69081c:/services/api/app/models/link.rb diff --git a/services/api/app/models/link.rb b/services/api/app/models/link.rb index 8e17ce697e..ad7800fe67 100644 --- a/services/api/app/models/link.rb +++ b/services/api/app/models/link.rb @@ -1,30 +1,43 @@ +# Copyright (C) The Arvados Authors. All rights reserved. +# +# SPDX-License-Identifier: AGPL-3.0 + class Link < ArvadosModel - include AssignUuid + include HasUuid include KindAndEtag include CommonApiTemplate - serialize :properties, Hash + + # Posgresql JSONB columns should NOT be declared as serialized, Rails 5 + # already know how to properly treat them. + attribute :properties, :jsonbHash, default: {} + before_create :permission_to_attach_to_objects before_update :permission_to_attach_to_objects after_update :maybe_invalidate_permissions_cache after_create :maybe_invalidate_permissions_cache after_destroy :maybe_invalidate_permissions_cache - - attr_accessor :head - attr_accessor :tail + validate :name_links_are_obsolete api_accessible :user, extend: :common do |t| t.add :tail_uuid t.add :link_class t.add :name t.add :head_uuid - t.add :head, :if => :head - t.add :tail, :if => :tail + t.add :head_kind + t.add :tail_kind t.add :properties end - def properties - @properties ||= Hash.new - super + def head_kind + if k = ArvadosModel::resource_class_for_uuid(head_uuid) + k.kind + end + end + + def tail_kind + if k = ArvadosModel::resource_class_for_uuid(tail_uuid) + k.kind + end end protected @@ -39,22 +52,13 @@ class Link < ArvadosModel # Administrators can grant permissions return true if current_user.is_admin - # All users can grant permissions on objects they own - head_obj = self.class. - kind_class(self.head_uuid). - where('uuid=?',head_uuid). - first - if head_obj - return true if head_obj.owner_uuid == current_user.uuid - end + head_obj = ArvadosModel.find_by_uuid(head_uuid) + + # No permission links can be pointed to past collection versions + return false if head_obj.is_a?(Collection) && head_obj.current_version_uuid != head_uuid - # Users with "can_grant" permission on an object can grant - # permissions on that object - has_grant_permission = self.class. - where('link_class=? AND name=? AND tail_uuid=? AND head_uuid=?', - 'permission', 'can_grant', current_user.uuid, self.head_uuid). - count > 0 - return true if has_grant_permission + # All users can grant permissions on objects they own or can manage + return true if current_user.can?(manage: head_obj) # Default = deny. false @@ -71,4 +75,31 @@ class Link < ArvadosModel User.invalidate_permissions_cache end end + + def name_links_are_obsolete + if link_class == 'name' + errors.add('name', 'Name links are obsolete') + false + else + true + end + end + + # A user is permitted to create, update or modify a permission link + # if and only if they have "manage" permission on the object + # indicated by the permission link's head_uuid. + # + # All other links are treated as regular ArvadosModel objects. + # + def ensure_owner_uuid_is_permitted + if link_class == 'permission' + ob = ArvadosModel.find_by_uuid(head_uuid) + raise PermissionDeniedError unless current_user.can?(manage: ob) + # All permission links should be owned by the system user. + self.owner_uuid = system_user_uuid + return true + else + super + end + end end