about summary refs log tree commit diff
path: root/app/lib/scope_transformer.rb
blob: adcb711f8ad2d1e7613b7682909522b1a500c89b (plain) (blame)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
# frozen_string_literal: true

class ScopeTransformer < Parslet::Transform
  class Scope
    DEFAULT_TERM   = 'all'
    DEFAULT_ACCESS = %w(read write).freeze

    attr_reader :namespace, :term

    def initialize(scope)
      @namespace = scope[:namespace]&.to_s
      @access    = scope[:access] ? [scope[:access].to_s] : DEFAULT_ACCESS.dup
      @term      = scope[:term]&.to_s || DEFAULT_TERM
    end

    def key
      @key ||= [@namespace, @term].compact.join('/')
    end

    def access
      @access.join('/')
    end

    def merge(other_scope)
      clone.merge!(other_scope)
    end

    def merge!(other_scope)
      raise ArgumentError unless other_scope.namespace == namespace && other_scope.term == term

      @access.concat(other_scope.instance_variable_get(:@access))
      @access.uniq!
      @access.sort!

      self
    end
  end

  rule(scope: subtree(:scope)) { Scope.new(scope) }
end