RSpec.describe Regexp, "#examples" do def self.examples_exist_and_match(*regexps) regexps.each do |regexp| it do regexp_examples = regexp.examples expect(regexp_examples).not_to be_empty regexp_examples.each { |example| expect(example).to match(/\A(?:#{regexp.source})\z/) } # Note: /\A...\z/ is used, to prevent misleading examples from passing the test. # For example, we don't want things like: # /a*/.examples to include "xyz" # /a|b/.examples to include "bad" end end end def self.examples_raise_illegal_syntax_error(*regexps) regexps.each do |regexp| it do expect{regexp.examples}.to raise_error RegexpExamples::IllegalSyntaxError end end end def self.examples_raise_unsupported_syntax_error(*regexps) regexps.each do |regexp| it do expect{regexp.examples}.to raise_error RegexpExamples::UnsupportedSyntaxError end end end def self.examples_are_empty(*regexps) regexps.each do |regexp| it do expect(regexp.examples).to be_empty end end end context 'returns matching strings' do context "for basic repeaters" do examples_exist_and_match( /a/, /a*/, /a+/, /a?/, /a{1}/, /a{1,}/, /a{,2}/, /a{1,2}/ ) end context "for basic groups" do examples_exist_and_match( /[a]/, /(a)/, /a|b/, /./ ) end context "for complex char groups (square brackets)" do examples_exist_and_match( /[abc]/, /[a-c]/, /[abc-e]/, /[^a-zA-Z]/, /[\w]/, /[]]/, # TODO: How to suppress annoying warnings on this test? /[\]]/, /[\\]/, /[\\\]]/, /[\n-\r]/, /[\-]/, /[%-+]/, # This regex is "supposed to" match some surprising things!!! /['-.]/ # Test to ensure no "infinite loop" on character set expansion ) end context "for complex multi groups" do examples_exist_and_match( /(normal)/, /(?:nocapture)/, /(?namedgroup)/, /(?namedgroup) \k/ ) end context "for escaped characters" do examples_exist_and_match( /\w/, /\W/, /\s/, /\S/, /\d/, /\D/, /\h/, /\H/, /\t/, /\n/, /\f/, /\a/, /\v/, /\e/, /[\b]/ ) end context "for backreferences" do examples_exist_and_match( /(repeat) \1/, /(ref1) (ref2) \1 \2/, /((ref2)ref1) \1 \2/, /((ref1and2)) \1 \2/, /(one)(two)(three)(four)(five)(six)(seven)(eight)(nine)(ten) \10\9\8\7\6\5\4\3\2\1/, /(a?(b?(c?(d?(e?)))))/ ) end context "for complex patterns" do # Longer combinations of the above examples_exist_and_match( /https?:\/\/(www\.)github\.com/, /(I(N(C(E(P(T(I(O(N)))))))))*/, /[\w]{1}/, /((a?b*c+)) \1/, /((a?b*c+)?) \1/, /a|b|c|d/, /a+|b*|c?/, /one|two|three/ ) end context "for illegal syntax" do examples_raise_illegal_syntax_error( /(?=lookahead)/, /(?!neglookahead)/, /(?<=lookbehind)/, /(? ... \g*)/, /[[:space:]]/ ) end context "for control characters" do examples_exist_and_match( /\ca/, /\cZ/, /\c9/, /\c[/, /\c#/, /\c?/, /\C-a/, /\C-&/ ) end context "for escape sequences" do examples_exist_and_match( /\x42/, /\x1D/, /\x3word/, /#{"\x80".force_encoding("ASCII-8BIT")}/ ) end context "for unicode sequences" do examples_exist_and_match( /\u6829/, /\uabcd/, /\u{42}word/ ) end context "for empty character sets" do examples_are_empty( /[^\d\D]/, /[^\w\W]/, /[^\s\S]/, /[^\h\H]/, /[^\D0-9]/, /[^\Wa-zA-Z0-9_]/, /[^\d\D]*/, /[^\d\D]+/, /[^\d\D]{2}/, /[^\d\D]word/ ) end end end