lib/cucumber/cucumber_expressions/tree_regexp.rb



require 'cucumber/cucumber_expressions/group_builder'
require 'cucumber/cucumber_expressions/errors'

module Cucumber
  module CucumberExpressions
    class TreeRegexp
      attr_reader :regexp, :group_builder

      def initialize(regexp)
        @regexp = regexp.is_a?(Regexp) ? regexp : Regexp.new(regexp)
        @stack = [GroupBuilder.new]
        group_start_stack = []
        last = nil
        escaping = false
        @non_capturing_maybe = false
        @name_capturing_maybe = false
        char_class = false

        @regexp.source.each_char.with_index do |c, n|
          if c == '[' && !escaping
            char_class = true
          elsif c == ']' && !escaping
            char_class = false
          elsif c == '(' && !escaping && !char_class
            @stack.push(GroupBuilder.new)
            group_start_stack.push(n + 1)
            @non_capturing_maybe = false
          elsif c == ')' && !escaping && !char_class
            gb = @stack.pop
            group_start = group_start_stack.pop
            if gb.capturing?
              gb.source = @regexp.source[group_start...n]
              @stack.last.add(gb)
            else
              gb.move_children_to(@stack.last)
            end
            end_group
          elsif c == '?' && last == '('
            @non_capturing_maybe = true
          elsif (c == '<') && @non_capturing_maybe
            @name_capturing_maybe = true
          elsif (c == ':' || c == '!' || c == '=') && last == '?' && @non_capturing_maybe
            end_non_capturing_group
          elsif (c == '=' || c == '!') && last == '<' && @name_capturing_maybe
            end_non_capturing_group
          elsif @name_capturing_maybe
            raise CucumberExpressionError.new("Named capture groups are not supported. See https://github.com/cucumber/cucumber/issues/329")
          end

          escaping = c == '\\' && !escaping
          last = c
        end
        @group_builder = @stack.pop
      end

      def match(s)
        match = @regexp.match(s)
        return nil if match.nil?
        group_indices = (0..match.length).to_a.to_enum
        @group_builder.build(match, group_indices)
      end

      private

      def end_non_capturing_group
        @stack.last.set_non_capturing!
        end_group
      end

      def end_group
        @non_capturing_maybe = false
        @name_capturing_maybe = false
      end
    end
  end
end