1 class Link < ArvadosModel
4 include CommonApiTemplate
5 serialize :properties, Hash
6 before_create :permission_to_attach_to_objects
7 before_update :permission_to_attach_to_objects
8 after_update :maybe_invalidate_permissions_cache
9 after_create :maybe_invalidate_permissions_cache
10 after_destroy :maybe_invalidate_permissions_cache
11 attr_accessor :head_kind, :tail_kind
12 validate :name_link_has_valid_name
13 validate :name_link_owner_is_tail
15 api_accessible :user, extend: :common do |t|
26 @properties ||= Hash.new
31 if k = ArvadosModel::resource_class_for_uuid(head_uuid)
37 if k = ArvadosModel::resource_class_for_uuid(tail_uuid)
44 def permission_to_attach_to_objects
45 # Anonymous users cannot write links
46 return false if !current_user
48 # All users can write links that don't affect permissions
49 return true if self.link_class != 'permission'
51 # Administrators can grant permissions
52 return true if current_user.is_admin
54 # All users can grant permissions on objects they own
55 head_obj = self.class.
56 resource_class_for_uuid(self.head_uuid).
57 where('uuid=?',head_uuid).
60 return true if head_obj.owner_uuid == current_user.uuid
63 # Users with "can_grant" permission on an object can grant
64 # permissions on that object
65 has_grant_permission = self.class.
66 where('link_class=? AND name=? AND tail_uuid=? AND head_uuid=?',
67 'permission', 'can_grant', current_user.uuid, self.head_uuid).
69 return true if has_grant_permission
75 def maybe_invalidate_permissions_cache
76 if self.link_class == 'permission'
77 # Clearing the entire permissions cache can generate many
78 # unnecessary queries if many active users are not affected by
79 # this change. In such cases it would be better to search cached
80 # permissions for head_uuid and tail_uuid, and invalidate the
81 # cache for only those users. (This would require a browseable
83 User.invalidate_permissions_cache
87 def name_link_has_valid_name
88 if link_class == 'name'
89 unless name.is_a? String and !name.empty?
90 errors.add('name', 'must be a non-empty string')
97 def name_link_owner_is_tail
98 if link_class == 'name'
99 self.owner_uuid = tail_uuid
100 ensure_owner_uuid_is_permitted