1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556 |
- # Copyright (C) 2012-2025 Zammad Foundation, https://zammad-foundation.org/
- # Abstract base class for various "types" of ticket access.
- #
- # Do NOT instantiate directly; instead,
- # choose the appropriate subclass from below
- # (see commit message for details).
- class TicketPolicy < ApplicationPolicy
- class BaseScope < ApplicationPolicy::Scope
- # overwrite PunditPolicy#initialize to make `context` optional and use Ticket as default
- def initialize(user, context = Ticket)
- super
- end
- def resolve # rubocop:disable Metrics/AbcSize
- raise NoMethodError, <<~ERR.chomp if instance_of?(TicketPolicy::BaseScope)
- specify an access type using a subclass of TicketPolicy::BaseScope
- ERR
- sql = []
- bind = []
- if user.permissions?('ticket.agent')
- sql.push('group_id IN (?)')
- bind.push(user.group_ids_access(self.class::ACCESS_TYPE))
- end
- if user.permissions?('ticket.customer')
- sql.push('tickets.customer_id = ?')
- bind.push(user.id)
- if user.all_organization_ids.present?
- Organization.where(id: user.all_organization_ids).select(&:shared).each do |organization|
- sql.push('tickets.organization_id = ?')
- bind.push(organization.id)
- end
- end
- end
- # The report permission can access all tickets.
- if sql.empty? && !user.permissions?('report')
- sql.push '0 = 1' # Forbid unlimited access for all other permissions.
- end
- scope.where sql.join(' OR '), *bind
- end
- # #resolve is UNDEFINED BEHAVIOR for the abstract base class (but not its subclasses)
- def respond_to?(*args)
- return false if args.first.to_s == 'resolve' && instance_of?(TicketPolicy::BaseScope)
- super
- end
- end
- end
|