// Written in the D programming language. // Regular Expressions. /** $(WEB digitalmars.com/ctg/regular.html, Regular expressions) are a powerful method of string pattern matching. The regular expression language used in this library is the same as that commonly used, however, some of the very advanced forms may behave slightly differently. The standard observed is the $(WEB www.ecma-international.org/publications/standards/Ecma-262.htm, ECMA standard) for regular expressions. $(D std.regex) is designed to work only with valid UTF strings as input - UTF8 ($(D char)), UTF16 ($(D wchar)), or UTF32 ($(D dchar)). To validate untrusted input, use $(D std.utf.validate()). In the following guide, $(D pattern[]) refers to a $(WEB digitalmars.com/ctg/regular.html, regular expression). The $(D attributes[]) refers to a string controlling the interpretation of the regular expression. It consists of a sequence of one or more of the following characters: $(BOOKTABLE Attribute Characters, $(TR $(TH Attribute) $(TH Action)) $(TR $(TD $(B g)) $(TD global; repeat over the whole input string)) $(TR $(TD $(B i)) $(TD case insensitive)) $(TR $(TD $(B m)) $(TD treat as multiple lines separated by newlines))) The $(D format[]) string has the formatting characters: $(BOOKTABLE Formatting Characters, $(TR $(TH Format) $(TH Replaced With)) $(TR $(TD $(B $$)) $(TD $)) $(TR$(TD $(B $&)) $(TD The matched substring.)) $(TR $(TD $(B $`)) $(TD The portion of string that precedes the matched substring.)) $(TR $(TD $(B $')) $(TD The portion of string that follows the matched substring.)) $(TR $(TD $(B $(DOLLAR))$(I n)) $(TD The $(I n)th capture, where $(I n) is a single digit 1-9 and $$(I n) is not followed by a decimal digit.)) $(TR $(TD $(B $(DOLLAR))$(I nn)) $(TD The $(I nn)th capture, where $(I nn) is a two-digit decimal number 01-99. If $(I nn)th capture is undefined or more than the number of parenthesized subexpressions, use the empty string instead.))) Any other $ are left as is. References: $(WEB en.wikipedia.org/wiki/Regular_expressions, Wikipedia) Macros: WIKI = StdRegex DOLLAR = $ Copyright: Copyright Digital Mars 2000 - 2009. License: Boost License 1.0. Authors: $(WEB digitalmars.com, Walter Bright), $(WEB erdani.org, Andrei Alexandrescu) Source: $(PHOBOSSRC std/_regex.d) */ /* Copyright Digital Mars 2000 - 2009. Distributed under the Boost Software License, Version 1.0. (See accompanying file LICENSE_1_0.txt or copy at http://www.boost.org/LICENSE_1_0.txt) */ /* Escape sequences: \nnn starts out a 1, 2 or 3 digit octal sequence, where n is an octal digit. If nnn is larger than 0377, then the 3rd digit is not part of the sequence and is not consumed. For maximal portability, use exactly 3 digits. \xXX starts out a 1 or 2 digit hex sequence. X is a hex character. If the first character after the \x is not a hex character, the value of the sequence is 'x' and the XX are not consumed. For maximal portability, use exactly 2 digits. \uUUUU is a unicode sequence. There are exactly 4 hex characters after the \u, if any are not, then the value of the sequence is 'u', and the UUUU are not consumed. Character classes: [a-b], where a is greater than b, will produce an error. References: http://www.unicode.org/unicode/reports/tr18/ */ module std.regex; //debug = std_regex; // uncomment to turn on debugging writef's import core.stdc.stdlib; import core.stdc.string; import std.stdio; import std.string; import std.ctype; import std.outbuffer; import std.bitmanip; import std.utf; import std.algorithm; import std.array; import std.traits; import std.typecons; import std.typetuple; import std.range; import std.conv; import std.functional; unittest { auto r = regex("abc"w); auto m = match("abc"w, r); if (!m.empty) return; writeln(m.pre); writeln(m.hit); writeln(m.post); r.printProgram; assert(false); } /** Regular expression to extract an _email address. References: $(WEB regular-expressions.info/_email.html, How to Find or Validate an Email Address); $(WEB tools.ietf.org/html/rfc2822#section-3.4.1, RFC 2822 Internet Message Format) */ enum string email = r"[a-zA-Z0-9._%+-]+@[a-zA-Z0-9.-]+\.[a-zA-Z]{2,4}"; unittest { assert(match("asdassfsd@", regex(email)).empty); assert(!match("andrei@metalanguage.com", regex(email)).empty); } /** Regular expression to extract a _url */ enum string url = r"(([h|H][t|T]|[f|F])[t|T][p|P]([s|S]?)\:\/\/|~/|/)?" r"([\w]+:\w+@)?(([a-zA-Z]{1}([\w\-]+\.)+([\w]{2,5}))" r"(:[\d]{1,5})?)?((/?\w+/)+|/?)" r"(\w+\.[\w]{3,4})?([,]\w+)*((\?\w+=\w+)?(&\w+=\w+)*([,]\w*)*)?"; unittest { assert(!match("http://www.erdani.org/asd/sd?asd#eds", regex(url)).empty); } /**************************************************** A $(D Regex) stores a regular expression engine. A $(D Regex) object is constructed from a string and compiled into an internal format for performance. The type parameter $(D E) specifies the character type recognized by the regular expression. Currently $(D char), $(D wchar), and $(D dchar) are supported. The encoding of the regex string and of the recognized strings must be the same. This object will be mostly used via a call to the $(D regex) function, which automatically deduces the character type. Example: Declare two variables and assign to them a $(D Regex) object. The first matches UTF-8 strings, the second matches UTF-32 strings and also has the global option set. --- auto r = regex("pattern"); auto s = regex(r"p[1-5]\s*"w, "g"); --- */ struct Regex(E) if (is(E == Unqual!E)) { private: struct regmatch_t { size_t startIdx, endIdx; } enum REA { global = 1, // has the g attribute ignoreCase = 2, // has the i attribute multiline = 4, // if treat as multiple lines separated by // newlines, or as a single line dotmatchlf = 8, // if . matches \n } enum uint inf = ~0u; uint re_nsub; // number of parenthesized subexpression matches uint nCounters; //current counter (internal), number of counters ubyte attributes; immutable(ubyte)[] program; // pattern[] compiled into regular // expression program // Opcodes enum : ubyte { REend, // end of program REchar, // single character REichar, // single character, case insensitive REdchar, // single UCS character REidchar, // single wide character, case insensitive REanychar, // any character REanystar, // ".*?" REanystarg, // ".*" REstring, // string of characters REistring, // string of characters, case insensitive REtestbit, // any in bitmap, non-consuming REbit, // any in the bit map REnotbit, // any not in the bit map RErange, // any in the string REnotrange, // any not in the string REor, // a | b REplus, // 1 or more REstar, // 0 or more REquest, // 0 or 1 REcounter, // begining of repetition REloopg, // loop on body of repetition (greedy) REloop, // ditto non-greedy REbol, // begining of line REeol, // end of line REsave, // save submatch position i.e. "(" & ")" REgoto, // goto offset REret, // end of subprogram RElookahead, REneglookahead, RElookbehind, REneglookbehind, REwordboundary, REnotwordboundary, REdigit, REnotdigit, REspace, REnotspace, REword, REnotword, REbackref, }; public: // @@@BUG Should be a constructor but template constructors don't work // private void initialize(String)(String pattern, string attributes) // { // compile(pattern, attributes); // } /** Construct a $(D Regex) object. Compile pattern with $(D attributes) into an internal form for fast execution. Params: pattern = regular expression attributes = The _attributes (g, i, and m accepted) Throws: $(D Exception) if there are any compilation errors. */ this(String)(String pattern, string attributes = null) { compile(pattern, attributes); } unittest { debug(std_regex) writefln("regex.opCall.unittest()"); auto r1 = Regex("hello", "m"); string msg; try { auto r2 = Regex("hello", "q"); assert(0); } catch (Exception ree) { msg = ree.toString(); //writefln("message: %s", ree); } assert(std.algorithm.countUntil(msg, "unrecognized attribute") >= 0); } /** Returns the number of parenthesized captures */ uint captures() const { return re_nsub; } /* ******************************** * Throws Exception on error */ public void compile(String)(String pattern, string attributes) { this.attributes = 0; foreach (c; attributes) { REA att; switch (c) { case 'g': att = REA.global; break; case 'i': att = REA.ignoreCase; break; case 'm': att = REA.multiline; break; default: error("unrecognized attribute"); assert(0); } if (this.attributes & att) { error("redundant attribute"); assert(0); } this.attributes |= att; } uint oldre_nsub = re_nsub; re_nsub = 0; auto buf = new OutBuffer; buf.reserve(pattern.length * 8); size_t p = 0; parseRegex(pattern, p, buf); if (p < pattern.length) { error("unmatched ')'"); } re_nsub /= 2; //start & ends -> pairs postprocess(buf.data); program = cast(immutable(ubyte)[]) buf.data; buf.data = null; delete buf; } void error(string msg) { debug(std_regex) writefln("error: %s", msg); throw new Exception(msg); } //adjust jumps, after removing instructions at 'place' void fixup(ubyte[] prog, size_t place, uint change) { for (size_t pc=0;pc place) *dest -= change; } pc += 1 + 2*uint.sizeof; break; case REloop, REloopg: //jump back if (pc > place) { auto dest = cast(uint *)&prog[pc + 1 + 2*uint.sizeof]; if (pc + *dest > place) *dest += change; } pc += 1 + 3*uint.sizeof; break; case REneglookahead://jump or call forward case RElookahead: case REor: case REgoto: if (pc < place) { auto dest = cast(uint *)&prog[pc+1]; if (pc + *dest > place) *dest -= change; } pc += 1 + uint.sizeof; break; case REret: case REanychar: case REanystarg: case REanystar: case REbol: case REeol: case REwordboundary: case REnotwordboundary: case REdigit: case REnotdigit: case REspace: case REnotspace: case REword: case REnotword: pc++; break; case REchar: case REichar: case REbackref: pc += 2; break; case REdchar: case REidchar: pc += 1 + dchar.sizeof; break; case REstring: case REistring: auto len = *cast(size_t *)&prog[pc + 1]; assert(len % E.sizeof == 0); pc += 1 + size_t.sizeof + len; break; case REtestbit: case REbit: case REnotbit: auto pu = cast(ushort *)&prog[pc + 1]; auto len = pu[1]; pc += 1 + 2 * ushort.sizeof + len; break; case RErange: case REnotrange: auto len = *cast(uint *)&prog[pc + 1]; pc += 1 + uint.sizeof + len; break; case REsave: pc += 1 + uint.sizeof; break; default: writeln("%d",prog[pc]); assert(0); } } } //Fixup counter numbers, simplify instructions private void postprocess(ubyte[] prog) { uint counter = 0; size_t len; ushort* pu; nCounters = 0; size_t pc = 0; for (;;) { switch (prog[pc]) { case REend: return; case REcounter: size_t offs = pc + 1 + 2*uint.sizeof; bool anyloop = counter == 0 && prog[offs] == REanychar && (prog[offs+1] == REloop || prog[offs+1] == REloopg); uint* puint = cast(uint*)&prog[offs+2]; if (anyloop && puint[0] == 0 && puint[1] == inf) { prog[pc] = prog[offs+1] == REloop ? REanystar : REanystarg; uint change = 2*(1 + uint.sizeof) + 1 + 3*uint.sizeof - 1; std.array.replaceInPlace(prog, pc + 1, pc + change + 1, cast(ubyte[])[]); fixup(prog, pc, change); pc++; } else { *cast(uint *)&prog[pc+1] = counter; counter++; nCounters = max(nCounters, counter); pc += 1 + 2*uint.sizeof; } break; case REloop, REloopg: counter--; pc += 1 + 3*uint.sizeof; break; case REret: case REanychar: case REbol: case REeol: case REwordboundary: case REnotwordboundary: case REdigit: case REnotdigit: case REspace: case REnotspace: case REword: case REnotword: pc++; break; case REbackref: case REchar: case REichar: pc += 2; break; case REdchar: case REidchar: pc += 1 + dchar.sizeof; break; case REstring: case REistring: len = *cast(size_t *)&prog[pc+1]; assert(len % E.sizeof == 0); pc += 1 + size_t.sizeof + len; break; case REtestbit: case REbit: case REnotbit: pu = cast(ushort *)&prog[pc+1]; len = pu[1]; pc += 1 + 2 * ushort.sizeof + len; break; case RErange: case REnotrange: len = *cast(uint *)&prog[pc+1]; pc += 1 + uint.sizeof + len; break; case REneglookahead: case RElookahead: case REor: case REgoto: pc += 1 + uint.sizeof; break; case REsave: pc += 1 + uint.sizeof; break; default: assert(0); } } } /* =================== Compiler ================== */ void parseRegex(String)(String pattern, ref size_t p, OutBuffer buf) { auto offset = buf.offset; for (;;) { assert(p <= pattern.length); if (p == pattern.length) { buf.write(REend); return; } switch (pattern[p]) { case ')': return; case '|': p++; auto gotooffset = buf.offset; buf.write(REgoto); buf.write(cast(uint)0); immutable uint len1 = cast(uint) (buf.offset - offset); buf.spread(offset, 1 + uint.sizeof); gotooffset += 1 + uint.sizeof; parseRegex(pattern, p, buf); immutable len2 = cast(uint) (buf.offset - (gotooffset + 1 + uint.sizeof)); buf.data[offset] = REor; (cast(uint *)&buf.data[offset + 1])[0] = len1; (cast(uint *)&buf.data[gotooffset + 1])[0] = len2; break; default: parsePiece(pattern, p, buf); break; } } } void parsePiece(String)(String pattern, ref size_t p, OutBuffer buf) { uint n; uint m; debug(std_regex) { auto sss = pattern[p .. pattern.length]; writefln("parsePiece() '%s'", sss); } size_t offset = buf.offset; size_t plength = pattern.length; parseAtom(pattern, p, buf); if (p == plength) return; switch (pattern[p]) { case '*': n = 0; m = inf; break; case '+': n = 1; m = inf; break; case '?': n = 0; m = 1; break; case '{': // {n} {n,} {n,m} p++; if (p == plength || !isdigit(pattern[p])) error("badly formed {n,m}"); auto src = pattern[p..$]; n = parse!uint(src); p = plength - src.length; if (pattern[p] == '}') // {n} { m = n; break; } if (pattern[p] != ',') error("',' expected in {n,m}"); p++; if (p == plength) error("unexpected end of pattern in {n,m}"); if (pattern[p] == /*{*/ '}') // {n,} { m = inf; break; } if (!isdigit(pattern[p])) error("badly formed {n,m}"); src = pattern[p..$]; m = parse!uint(src); p = plength - src.length; if (pattern[p] != /*{*/ '}') error("unmatched '}' in {n,m}"); break; default: return; } p++; uint len = cast(uint)(buf.offset - offset); if (p < plength && pattern[p] == '?') { buf.write(REloop); p++; } else buf.write(REloopg); buf.write(cast(uint)n); buf.write(cast(uint)m); buf.write(cast(uint)len);//set jump back buf.spread(offset, (1 + 2*uint.sizeof)); buf.data[offset] = REcounter; *(cast(uint*)&buf.data[offset+1]) = 0;//reserve counter num *(cast(uint*)&buf.data[offset+5]) = len; return; } void parseAtom(String)(String pattern, ref size_t p, OutBuffer buf) { ubyte op; size_t offset; E c; debug(std_regex) { auto sss = pattern[p .. pattern.length]; writefln("parseAtom() '%s'", sss); } if (p >= pattern.length) return; c = pattern[p]; switch (c) { case '*': case '+': case '?': error("*+? not allowed in atom"); assert(0); case '(': p++; if (pattern[p] != '?') { buf.write(REsave); buf.write(2 + re_nsub); //handle nested groups uint end = re_nsub; re_nsub += 2; parseRegex(pattern, p, buf); buf.write(REsave); buf.write(2 + end + 1); } else if (pattern.length > p+1) { p++; switch (pattern[p]) { case ':': p++; parseRegex(pattern, p, buf); break; case '=': case '!': buf.write(pattern[p] == '=' ? RElookahead : REneglookahead); offset = buf.offset; buf.write(cast(uint)0); // reserve space for length p++; parseRegex(pattern, p, buf); *cast(uint *)&buf.data[offset] = cast(uint)(buf.offset - (offset + uint.sizeof)+1); buf.write(REret); break; default: error("any of :=! expected after '(?'"); assert(0); } } else { error("any of :=! expected after '(?'"); assert(0); } if (p == pattern.length || pattern[p] != ')') { error("')' expected"); assert(0); } p++; break; case '[': parseRange(pattern, p, buf); break; case '.': p++; buf.write(REanychar); break; case '^': p++; buf.write(REbol); break; case '$': p++; buf.write(REeol); break; case '\\': p++; if (p == pattern.length) { error("no character past '\\'"); assert(0); } c = pattern[p]; switch (c) { case 'b': op = REwordboundary; goto Lop; case 'B': op = REnotwordboundary; goto Lop; case 'd': op = REdigit; goto Lop; case 'D': op = REnotdigit; goto Lop; case 's': op = REspace; goto Lop; case 'S': op = REnotspace; goto Lop; case 'w': op = REword; goto Lop; case 'W': op = REnotword; goto Lop; Lop: buf.write(op); p++; break; case 'f': case 'n': case 'r': case 't': case 'v': case 'c': case 'x': case 'u': case '0': c = cast(char)escape(pattern, p); goto Lbyte; case '1': case '2': case '3': case '4': case '5': case '6': case '7': case '8': case '9': c -= '1'; if (c < re_nsub) { buf.write(REbackref); buf.write(cast(ubyte)c); } else error("no matching back reference"); p++; break; default: p++; goto Lbyte; } break; default: p++; Lbyte: op = REchar; if (attributes & REA.ignoreCase) { if (isalpha(c)) { op = REichar; c = cast(char)std.ctype.toupper(c); } } if (op == REchar && c <= 0xFF) { // Look ahead and see if we can make this into // an REstring sizediff_t q = p; sizediff_t len; for (; q < pattern.length; ++q) { auto qc = pattern[q]; switch (qc) { case '{': case '*': case '+': case '?': if (q == p) goto Lchar; q--; break; case '(': case ')': case '|': case '[': case ']': case '.': case '^': case '$': case '\\': case '}': break; default: continue; } break; } len = q - p; if (len > 0) { debug(std_regex) writefln("writing string len %d, c = '%s'" ", pattern[p] = '%s'", len+1, c, pattern[p]); buf.reserve(5 + (1 + len) * E.sizeof); buf.write((attributes & REA.ignoreCase) ? REistring : REstring); //auto narrow = to!string(pattern[p .. p + len]); buf.write(E.sizeof * (len + 1)); //buf.write(narrow.length + 1); buf.write(c); buf.write(pattern[p .. p + len]); //buf.write(narrow); p = q; break; } } if (c >= 0x80) { debug(std_regex) writefln("dchar"); // Convert to dchar opcode op = (op == REchar) ? REdchar : REidchar; buf.write(op); buf.write(c); } else { Lchar: debug(std_regex) writefln("It's an REchar '%s'", c); buf.write(op); buf.write(cast(char)c); } break; } } struct Range { size_t maxc; size_t maxb; OutBuffer buf; ubyte* base; BitArray bits; this(OutBuffer buf) { this.buf = buf; if (buf.data.length) this.base = &buf.data[buf.offset]; } void setbitmax(size_t u) { //writefln("setbitmax(x%x), maxc = x%x", u, maxc); if (u <= maxc) return; maxc = u; auto b = u / 8; if (b >= maxb) { size_t u2 = base ? base - &buf.data[0] : 0; buf.fill0(b - maxb + 1); base = &buf.data[u2]; maxb = b + 1; //bits = (cast(bit*)this.base)[0 .. maxc + 1]; bits.ptr = cast(size_t*)this.base; } bits.len = maxc + 1; } void setbit2(size_t u) { setbitmax(u + 1); //writefln("setbit2 [x%02x] |= x%02xn", u >> 3, 1 << (u & 7)); bits[u] = 1; } } int parseRange(String)(in String pattern, ref size_t p, OutBuffer buf) { int c; int c2; uint i; uint cmax = 0x7F; p++; ubyte op = REbit; if (p == pattern.length) goto Lerr; if (pattern[p] == '^') { p++; op = REnotbit; if (p == pattern.length) goto Lerr; } buf.write(op); auto offset = buf.offset; buf.write(cast(uint)0); // reserve space for length buf.reserve(128 / 8); auto r = Range(buf); if (op == REnotbit) r.setbit2(0); switch (pattern[p]) { case ']': case '-': c = pattern[p]; p++; r.setbit2(c); break; default: break; } enum RS { start, rliteral, dash }; auto rs = RS.start; for (;;) { if (p == pattern.length) goto Lerr; switch (pattern[p]) { case ']': switch (rs) { case RS.dash: r.setbit2('-'); goto case; case RS.rliteral: r.setbit2(c); break; case RS.start: break; default: assert(0); } p++; break; case '\\': p++; r.setbitmax(cmax); if (p == pattern.length) goto Lerr; switch (pattern[p]) { case 'd': for (i = '0'; i <= '9'; i++) r.bits[i] = 1; goto Lrs; case 'D': for (i = 1; i < '0'; i++) r.bits[i] = 1; for (i = '9' + 1; i <= cmax; i++) r.bits[i] = 1; goto Lrs; case 's': for (i = 0; i <= cmax; i++) if (isspace(i)) r.bits[i] = 1; goto Lrs; case 'S': for (i = 1; i <= cmax; i++) if (!isspace(i)) r.bits[i] = 1; goto Lrs; case 'w': for (i = 0; i <= cmax; i++) if (isword(cast(E) i)) r.bits[i] = 1; goto Lrs; case 'W': for (i = 1; i <= cmax; i++) if (!isword(cast(E) i)) r.bits[i] = 1; goto Lrs; Lrs: switch (rs) { case RS.dash: r.setbit2('-'); goto case; case RS.rliteral: r.setbit2(c); break; default: break; } rs = RS.start; continue; default: break; } c2 = escape(pattern, p); goto Lrange; case '-': p++; if (rs == RS.start) goto Lrange; else if (rs == RS.rliteral) rs = RS.dash; else if (rs == RS.dash) { r.setbit2(c); r.setbit2('-'); rs = RS.start; } continue; default: c2 = pattern[p]; p++; Lrange: switch (rs) { case RS.rliteral: r.setbit2(c); goto case; case RS.start: c = c2; rs = RS.rliteral; break; case RS.dash: if (c > c2) { error("inverted range in character class"); return 0; } r.setbitmax(c2); for (; c <= c2; c++) r.bits[c] = 1; rs = RS.start; break; default: assert(0); } continue; } break; } if (attributes & REA.ignoreCase) { // BUG: what about dchar? r.setbitmax(0x7F); for (c = 'a'; c <= 'z'; c++) { if (r.bits[c]) r.bits[c + 'A' - 'a'] = 1; else if (r.bits[c + 'A' - 'a']) r.bits[c] = 1; } } //writefln("maxc = %d, maxb = %d",r.maxc,r.maxb); (cast(ushort *)&buf.data[offset])[0] = cast(ushort)r.maxc; (cast(ushort *)&buf.data[offset])[1] = cast(ushort)r.maxb; return 1; Lerr: error("invalid range"); return 0; } int escape(String)(in String pattern, ref size_t p) in { assert(p < pattern.length); } body { int c; int i; E tc; c = pattern[p]; // none of the cases are multibyte switch (c) { case 'b': c = '\b'; break; case 'f': c = '\f'; break; case 'n': c = '\n'; break; case 'r': c = '\r'; break; case 't': c = '\t'; break; case 'v': c = '\v'; break; // BUG: Perl does \a and \e too, should we? case 'c': ++p; if (p == pattern.length) goto Lretc; c = pattern[p]; // Note: we are deliberately not allowing dchar letters if (!(('a' <= c && c <= 'z') || ('A' <= c && c <= 'Z'))) { Lcerr: error("letter expected following \\c"); return 0; } c &= 0x1F; break; case '0': case '1': case '2': case '3': case '4': case '5': case '6': case '7': c -= '0'; for (i = 0; i < 2; i++) { p++; if (p == pattern.length) goto Lretc; tc = pattern[p]; if ('0' <= tc && tc <= '7') { c = c * 8 + (tc - '0'); // Treat overflow as if last // digit was not an octal digit if (c >= 0xFF) { c >>= 3; return c; } } else return c; } break; case 'x': c = 0; for (i = 0; i < 2; i++) { p++; if (p == pattern.length) goto Lretc; tc = pattern[p]; if ('0' <= tc && tc <= '9') c = c * 16 + (tc - '0'); else if ('a' <= tc && tc <= 'f') c = c * 16 + (tc - 'a' + 10); else if ('A' <= tc && tc <= 'F') c = c * 16 + (tc - 'A' + 10); else if (i == 0) // if no hex digits after \x { // Not a valid \xXX sequence return 'x'; } else return c; } break; case 'u': c = 0; for (i = 0; i < 4; i++) { p++; if (p == pattern.length) goto Lretc; tc = pattern[p]; if ('0' <= tc && tc <= '9') c = c * 16 + (tc - '0'); else if ('a' <= tc && tc <= 'f') c = c * 16 + (tc - 'a' + 10); else if ('A' <= tc && tc <= 'F') c = c * 16 + (tc - 'A' + 10); else { // Not a valid \uXXXX sequence p -= i; return 'u'; } } break; default: break; } p++; Lretc: return c; } // BUG: should this include '$'? private int isword(dchar c) { return isalnum(c) || c == '_'; } void printProgram(const(ubyte)[] prog = null) { if (!prog) prog = program; //debug(std_regex) { size_t len; uint n; uint m; ushort *pu; uint *puint; char[] str; writefln("printProgram()"); for (uint pc = 0; pc < prog.length; ) { writef("%3d: ", pc); switch (prog[pc]) { case REchar: writefln("\tREchar '%s'", cast(char)prog[pc + 1]); pc += 1 + cast(uint)char.sizeof; break; case REichar: writefln("\tREichar '%s'", cast(char)prog[pc + 1]); pc += 1 + cast(uint)char.sizeof; break; case REdchar: writefln("\tREdchar '%s'", *cast(dchar *)&prog[pc + 1]); pc += 1 + cast(uint)dchar.sizeof; break; case REidchar: writefln("\tREidchar '%s'", *cast(dchar *)&prog[pc + 1]); pc += 1 + cast(uint)dchar.sizeof; break; case REanychar: writefln("\tREanychar"); pc++; break; case REstring: len = *cast(size_t *)&prog[pc + 1]; assert(len % E.sizeof == 0); len /= E.sizeof; writef("\tREstring x%x*%d, ", len, E.sizeof); auto es = cast(E*) (&prog[pc + 1 + size_t.sizeof]); foreach (e; es[0 .. len]) { writef("'%s' ", e); } writefln(""); pc += 1 + cast(uint)size_t.sizeof + len * E.sizeof; break; case REistring: len = *cast(size_t *)&prog[pc + 1]; assert(len % E.sizeof == 0); len /= E.sizeof; writef("\tREistring x%x*%d, ", len, E.sizeof); auto es = cast(E*) (&prog[pc + 1 + size_t.sizeof]); foreach (e; es[0 .. len]) { writef("'%s' ", e); } writefln(""); pc += 1 + cast(uint)size_t.sizeof + len * E.sizeof; break; case REtestbit: pu = cast(ushort *)&prog[pc + 1]; writef("\tREtestbit %d, %d: ", pu[0], pu[1]); len = pu[1]; { ubyte * b = cast(ubyte*)pu; foreach (i; 0 .. len) { writef(" %x", b[i]); } writeln(); } pc += 1 + 2 * cast(uint)ushort.sizeof + len; break; case REbit: pu = cast(ushort *)&prog[pc + 1]; len = pu[1]; writef("\tREbit cmax=%02x, len=%d:", pu[0], len); for (n = 0; n < len; n++) writef(" %02x", prog[pc + 1 + 2 * ushort.sizeof + n]); writefln(""); pc += 1 + 2 * cast(uint)ushort.sizeof + len; break; case REnotbit: pu = cast(ushort *)&prog[pc + 1]; writefln("\tREnotbit %d, %d", pu[0], pu[1]); len = pu[1]; pc += 1 + 2 * cast(uint)ushort.sizeof + len; break; case RErange: len = *cast(uint *)&prog[pc + 1]; writefln("\tRErange %d", len); // BUG: REAignoreCase? pc += 1 + cast(uint)uint.sizeof + len; break; case REnotrange: len = *cast(uint *)&prog[pc + 1]; writefln("\tREnotrange %d", len); // BUG: REAignoreCase? pc += 1 + cast(uint)uint.sizeof + len; break; case REbol: writefln("\tREbol"); pc++; break; case REeol: writefln("\tREeol"); pc++; break; case REor: len = *cast(uint *)&prog[pc + 1]; writefln("\tREor %d, pc=>%d", len, pc + 1 + uint.sizeof + len); pc += 1 + cast(uint)uint.sizeof; break; case REgoto: len = *cast(uint *)&prog[pc + 1]; writefln("\tREgoto %d, pc=>%d", len, pc + 1 + uint.sizeof + len); pc += 1 + cast(uint)uint.sizeof; break; case REanystar: case REanystarg: writefln("\tREanystar%s", prog[pc] == REanystarg ? "g":""); pc++; break; case REcounter: // n, len puint = cast(uint *)&prog[pc + 1]; n = puint[0]; len = puint[1]; writefln("\tREcounter n=%u pc=>%d", n, pc + 1 + 2*uint.sizeof + len); pc += 1 + cast(uint)2*uint.sizeof; break; case REloop: case REloopg: //n, m, len puint = cast(uint *)&prog[pc + 1]; n = puint[0]; m = puint[1]; len = puint[2]; writefln("\tREloop%s min=%u max=%u pc=>%u", prog[pc] == REloopg ? "g" : "", n, m, pc-len); pc += 1 + cast(uint)uint.sizeof*3; break; case REsave: // n n = *cast(uint *)&prog[pc + 1]; writefln("\tREsave %s n=%d ", n % 2 ? "end" :"start", n/2); pc += 1 + cast(uint)uint.sizeof; break; case RElookahead: // len, () len = *cast(uint *)&prog[pc + 1]; writefln("\tRElookahead len=%d, pc=>%d", len, pc + 1 + uint.sizeof + len); pc += 1 + cast(uint)uint.sizeof; break; case REneglookahead: // len, () len = *cast(uint *)&prog[pc + 1]; writefln("\tREneglookahead len=%d, pc=>%d", len, pc + 1 + uint.sizeof + len); pc += 1 + cast(uint)uint.sizeof; break; case REend: writefln("\tREend"); return; case REret: writefln("\tREret"); pc++; break; case REwordboundary: writefln("\tREwordboundary"); pc++; break; case REnotwordboundary: writefln("\tREnotwordboundary"); pc++; break; case REdigit: writefln("\tREdigit"); pc++; break; case REnotdigit: writefln("\tREnotdigit"); pc++; break; case REspace: writefln("\tREspace"); pc++; break; case REnotspace: writefln("\tREnotspace"); pc++; break; case REword: writefln("\tREword"); pc++; break; case REnotword: writefln("\tREnotword"); pc++; break; case REbackref: writefln("\tREbackref %d", prog[1]); pc += 2; break; default: assert(0); } } } } } /// Ditto Regex!(Unqual!(ElementEncodingType!String)) regex(String) (String pattern, string flags = null) if (isSomeString!String) { alias Unqual!(ElementEncodingType!String) Char; alias immutable(Char)[] IString; static Tuple!(IString, string) lastReq = tuple(cast(IString)[],"\u0001");//most unlikely static typeof(return) lastResult; if (lastReq[0] == pattern && lastReq[1] == flags) { // cache hit return lastResult; } auto result = typeof(return)(pattern, flags); lastReq[0] = to!IString(pattern); lastReq[1] = flags; lastResult = result; return result; } /** $(D RegexMatch) is the type returned by a call to $(D match). It stores the matching state and can be inspected and iterated. */ struct RegexMatch(Range = string) { alias typeof(Range.init[0]) E; // Engine alias .Regex!(Unqual!E) Regex; private alias Regex.regmatch_t regmatch_t; enum stackSize = 32*1024; /** Get or set the engine of the match. */ public Regex engine; // the string to search Range input; size_t src; // current source index in input[] size_t src_start; // starting index for match in input[] regmatch_t[] pmatch; // array [engine.re_nsub + 1] uint curCounter; uint[] counters; //array [engine.counter] /* Build a RegexMatch from an engine. */ private this(Regex engine) { this.engine = engine; pmatch.length = engine.re_nsub + 1; counters.length = engine.nCounters; pmatch[0].startIdx = -1; pmatch[0].endIdx = -1; } /* Build a RegexMatch from an engine and an input. */ private this(Regex engine, Range input) { this.engine = engine; pmatch.length = engine.re_nsub + 1; pmatch[0].startIdx = -1; pmatch[0].endIdx = -1; counters.length = engine.nCounters; this.input = input; // amorsate test; } /* Copy zis. */ this(this) { pmatch = pmatch.dup; } // ref auto opSlice() // { // return this; // } /** Range primitives that allow incremental matching against a string. Example: --- import std.stdio; import std.regex; void main() { foreach(m; match("abcabcabab", regex("ab"))) { writefln("%s[%s]%s", m.pre, m.hit, m.post); } } // Prints: // [ab]cabcabab // abc[ab]cabab // abcabc[ab]ab // abcabcab[ab] --- */ bool empty() const { return pmatch[0].startIdx == pmatch[0].startIdx.max; } /// Ditto void popFront() { assert(!empty); test; } /// Ditto RegexMatch!(Range) front() { return this; } /// Ditto static if (isForwardRange!Range) { @property typeof(this) save() { auto ret = this; ret.input = input.save; return ret; } } unittest { // @@@BUG@@@ This doesn't work if a client module uses -unittest // uint i; // foreach (m; match(to!(Range)("abcabcabab"), regex(to!(Range)("ab")))) // { // ++i; // assert(m.hit == "ab"); // //writefln("%s[%s]%s", m.pre, m.hit, m.post); // } // assert(i == 4); } unittest { // @@@BUG@@@ This doesn't work if a client module uses -unittest // debug(std_regex) writefln("regex.search.unittest()"); // int i; // //foreach(m; RegexMatch("ab").search("abcabcabab")) // foreach(m; .match("abcabcabab", regex("ab"))) // { // auto s = std.string.format("%s[%s]%s", m.pre, m.hit, m.post); // if (i == 0) assert(s == "[ab]cabcabab"); // else if (i == 1) assert(s == "abc[ab]cabab"); // else if (i == 2) assert(s == "abcabc[ab]ab"); // else if (i == 3) assert(s == "abcabcab[ab]"); // else assert(0); // i++; // } // assert(i == 4); } struct Captures { private Range input; private regmatch_t[] matches; ref auto opSlice() { return this; } @property bool empty() { return matches.empty; } @property Range front() { return input[matches[0].startIdx .. matches[0].endIdx]; } void popFront() { matches.popFront; } @property Range back() { return input[matches[$-1].startIdx .. matches[$-1].endIdx]; } void popBack() { matches.popBack; } @property typeof(this) save() { return this; } @property size_t length() { return matches.length; } Range opIndex(size_t n) { assert(n < length, text("length = ", length, ", requested match = ", n)); return input[matches[n].startIdx .. matches[n].endIdx]; } } /****************** Retrieve the captured parenthesized matches, in the form of a random-access range. The first element in the range is always the full match. Example: ---- foreach (m; match("abracadabra", "(.)a(.)")) { foreach (c; m.captures) write(c, ';'); writeln(); } // writes: // rac;r;c; // dab;d;b; ---- */ public Captures captures() { return Captures(input, empty ? [] : pmatch); } unittest { // @@@BUG@@@ This doesn't work if a client module uses -unittest // auto app = appender!string(); // foreach (m; match("abracadabra", "(.)a(.)")) // { // assert(m.captures.length == 3); // foreach (c; m.captures) // app.put(c), app.put(';'); // } // assert(app.data == "rac;r;c;dab;d;b;"); } /******************* Returns the slice of the input that precedes the matched substring. */ public Range pre() { return input[0 .. pmatch[0].startIdx != pmatch[0].startIdx.max ? pmatch[0].startIdx : $]; } /** The matched portion of the input. */ public Range hit() { assert(pmatch[0].startIdx <= pmatch[0].endIdx && pmatch[0].endIdx <= input.length, text(pmatch[0].startIdx, " .. ", pmatch[0].endIdx, " vs. ", input.length)); return input[pmatch[0].startIdx .. pmatch[0].endIdx]; } /******************* Returns the slice of the input that follows the matched substring. */ public Range post() { return input[pmatch[0].endIdx < $ ? pmatch[0].endIdx : $ .. $]; } /** Returns $(D hit) (converted to $(D string) if necessary). */ string toString() { return to!string(hit); } /* ************************************************ * Find regular expression matches in s[]. Replace those matches * with a new string composed of format[] merged with the result of the * matches. * If global, replace all matches. Otherwise, replace first match. * Returns: the new string */ private Range replaceAll(String)(String format) { auto result = input; size_t lastindex = 0; size_t offset = 0; for (;;) { if (!test(lastindex)) break; auto so = pmatch[0].startIdx; auto eo = pmatch[0].endIdx; auto replacement = replace(format); /+ // Optimize by using std.string.replace if possible - Dave Fladebo auto slice = result[offset + so .. offset + eo]; if (attributes & REA.global && // global, so replace all !(attributes & REA.ignoreCase) && // not ignoring case !(attributes & REA.multiline) && // not multiline pattern == slice && // simple pattern // (exact match, no // special characters) format == replacement) // simple format, not $ formats { debug(std_regex) { auto sss = result[offset + so .. offset + eo]; writefln("pattern: %s, slice: %s, format: %s, replacement: %s", pattern, sss, format, replacement); } result = std.string.replace(result, slice, replacement); break; } +/ result = replaceSlice(result, result[offset + so .. offset + eo], replacement); if (engine.attributes & engine.REA.global) { offset += replacement.length - (eo - so); if (lastindex == eo) lastindex++; // always consume some source else lastindex = eo; } else break; } return result; } /* * Test s[] starting at startindex against regular expression. * Returns: 0 for no match, !=0 for match */ private bool test(size_t startindex = size_t.max) { if (startindex == size_t.max) { if (pmatch[0].endIdx != pmatch[0].endIdx.max) { startindex = pmatch[0].endIdx; if (startindex >= input.length) { pmatch[0].startIdx = pmatch[0].startIdx.max; pmatch[0].endIdx = pmatch[0].endIdx.max; return false; // fail } if (pmatch[0].endIdx == pmatch[0].startIdx) startindex += std.utf.stride(input, pmatch[0].endIdx); } else startindex = 0; } debug (regex) writefln("Regex.test(input[] = '%s', startindex = %d)", input, startindex); //engine.printProgram(engine.program); pmatch[0].startIdx = -1; pmatch[0].endIdx = -1; // First character optimization Unqual!(typeof(Range.init[0])) firstc = 0; if (engine.program[0] == engine.REchar) { firstc = engine.program[1]; if (engine.attributes & engine.REA.ignoreCase && isalpha(firstc)) firstc = 0; } ubyte* pmemory = cast(ubyte *)alloca(stackSize); ubyte[] memory = pmemory ? pmemory[0..stackSize] : new ubyte [stackSize]; for (;; ++startindex) { if (firstc) { if (startindex == input.length) { break; // no match } if (input[startindex] != firstc) { startindex++; if (!chr(startindex, firstc)) // 1st char not found break; // no match } } foreach (i; 1 .. engine.re_nsub + 1)//subs considered empty matches { pmatch[i].startIdx = 0; pmatch[i].endIdx = 0; } src_start = src = startindex; if (trymatch(0, memory)) { pmatch[0].startIdx = startindex; pmatch[0].endIdx = src; return true; } // If possible match must start at beginning, we are done if (engine.program[0] == engine.REbol || engine.program[0] == engine.REanystarg) { if (!(engine.attributes & engine.REA.multiline)) break; // Scan for the next \n if (!chr(startindex, '\n')) break; // no match if '\n' not found } if (startindex == input.length) break; debug(std_regex) { auto sss = input[startindex + 1 .. input.length]; writefln("Starting new try: '%s'", sss); } } pmatch[0].startIdx = pmatch[0].startIdx.max; pmatch[0].endIdx = pmatch[0].endIdx.max; return false; // no match } /** Returns whether string $(D_PARAM s) matches $(D_PARAM this). */ //alias test opEquals; private bool chr(ref size_t si, E c) { for (; si < input.length; si++) { if (input[si] == c) return 1; } return 0; } private static sizediff_t icmp(E[] a, E[] b) { static if (is(Unqual!(E) == char)) { return .icmp(a, b); } else static if (is(E : dchar)) { for (size_t i, j;; ++i, ++j) { if (j == b.length) return i != a.length; if (i == a.length) return -1; immutable x = std.uni.toUniLower(a[i]), y = std.uni.toUniLower(b[j]); if (x == y) continue; return x - y; } } else { for (size_t i, j;; ++i, ++j) { if (j == b.length) return i != a.length; if (i == a.length) return -1; immutable x = a[i], y = b[j]; if (x == y) continue; return x - y; } } } /* ************************************************* * Match input against a section of the program[]. * Returns: * 1 if successful match * 0 no match */ private bool trymatch(uint pc, ubyte[] memory) { /* * All variables related to position in input are size_t * almost anything else reasonably fits into uint */ uint pop; size_t lastState = 0; //top of backtrack stack uint matchesToSave = 0; //number of currently used entries in pmatch size_t[] trackers; struct StateTail { //this structure is preceeded by all matches, then by all counters size_t src; uint pc, counter, matches, size; } bool backtrack() { if (lastState == 0) return false; auto tail = cast(StateTail *)&memory[lastState - StateTail.sizeof]; pc = tail.pc; src = tail.src; matchesToSave = tail.matches; curCounter = tail.counter; lastState -= tail.size; debug(std_regex) writefln("\tBacktracked pc=>%d src='%s'", pc, input[src..$]); auto matchPtr = cast(regmatch_t*)&memory[lastState]; pmatch[1..matchesToSave+1] = matchPtr[0..matchesToSave]; pmatch[matchesToSave+1..$] = regmatch_t(0, 0);//remove any stale matches here if (!counters.empty) { auto counterPtr = cast(uint*)(matchPtr+matchesToSave); counters[0..curCounter+1] = counterPtr[0..curCounter+1]; } return true; } void memoize(uint newpc) { auto stateSize = (counters.empty ? 0 : (curCounter+1)*uint.sizeof) + matchesToSave*regmatch_t.sizeof; if (memory.length < lastState + stateSize + StateTail.sizeof) memory.length += memory.length; //reallocates on heap auto matchPtr = cast(regmatch_t*)&memory[lastState]; matchPtr[0..matchesToSave] = pmatch[1..matchesToSave+1]; if (!counters.empty) { auto counterPtr = cast(uint*)(matchPtr + matchesToSave); counterPtr[0..curCounter+1] = counters[0..curCounter+1]; } lastState += stateSize; auto tail = cast(StateTail *) &memory[lastState]; tail.pc = newpc; tail.src = src; tail.matches = matchesToSave; tail.counter = curCounter; tail.size = cast(uint)(stateSize + StateTail.sizeof); lastState += StateTail.sizeof; } debug(std_regex) { auto sss = input[src .. input.length]; writefln("Regex.trymatch(pc = %d, src = '%s')", pc, sss); } auto srcsave = src; for (;;) { //writefln("\top = %d", program[pc]); switch (engine.program[pc]) { case engine.REchar: if (src == input.length) goto Lnomatch; debug(std_regex) writefln("\tREchar '%s', src = '%s'", engine.program[pc + 1], input[src]); if (engine.program[pc + 1] != input[src]) goto Lnomatch; src++; pc += 1 + cast(uint)char.sizeof; break; case engine.REichar: if (src == input.length) goto Lnomatch; debug(std_regex) writefln("\tREichar '%s', src = '%s'", engine.program[pc + 1], input[src]); size_t c1 = engine.program[pc + 1]; size_t c2 = input[src]; if (c1 != c2) { if (islower(cast(E) c2)) c2 = std.ctype.toupper(cast(E) c2); else goto Lnomatch; if (c1 != c2) goto Lnomatch; } src++; pc += 1 + cast(uint)char.sizeof; break; case engine.REdchar: if (src == input.length) goto Lnomatch; debug(std_regex) writefln("\tREdchar '%s', src = '%s'", *(cast(dchar *)&engine.program[pc + 1]), input[src]); if (*(cast(dchar *)&engine.program[pc + 1]) != input[src]) goto Lnomatch; src++; pc += 1 + cast(uint)dchar.sizeof; break; case engine.REidchar: if (src == input.length) goto Lnomatch; debug(std_regex) writefln("\tREidchar '%s', src = '%s'", *(cast(dchar *)&engine.program[pc + 1]), input[src]); size_t c1 = *(cast(dchar *)&engine.program[pc + 1]); size_t c2 = input[src]; if (c1 != c2) { if (islower(cast(E) c2)) c2 = std.ctype.toupper(cast(E) c2); else goto Lnomatch; if (c1 != c2) goto Lnomatch; } src++; pc += 1 + cast(uint)dchar.sizeof; break; case engine.REanychar: debug(std_regex) writefln("\tREanychar"); if (src == input.length) goto Lnomatch; if (!(engine.attributes & engine.REA.dotmatchlf) && input[src] == '\n') goto Lnomatch; src += std.utf.stride(input, src); pc++; break; case engine.REstring: auto len = *cast(size_t *)&engine.program[pc + 1]; assert(len % E.sizeof == 0); len /= E.sizeof; debug(std_regex) { auto sssa = (&engine.program[pc + 1 + size_t.sizeof])[0 .. len]; writefln("\tREstring x%x, '%s'", len, sssa); } if (src + len > input.length) goto Lnomatch; if (memcmp(&engine.program[pc + 1 + size_t.sizeof], &input[src], len * E.sizeof)) goto Lnomatch; src += len; pc += 1 + size_t.sizeof + cast(uint)len * E.sizeof; break; case engine.REistring: auto len = *cast(size_t *)&engine.program[pc + 1]; assert(len % E.sizeof == 0); len /= E.sizeof; debug(std_regex) { auto sssa = (&engine.program[pc + 1 + size_t.sizeof])[0 .. len]; writefln("\tREistring x%x, '%s'", len, sssa); } if (src + len > input.length) goto Lnomatch; if (icmp( (cast(E*)&engine.program[pc+1+size_t.sizeof])[0..len], input[src .. src + len])) goto Lnomatch; src += len; pc += 1 + size_t.sizeof + cast(uint)len * E.sizeof; break; case engine.REtestbit: auto pu = (cast(ushort *)&engine.program[pc + 1]); if (src == input.length) goto Lnomatch; debug(std_regex) writefln("\tREtestbit %d, %d, '%s', x%02x", pu[0], pu[1], input[src], input[src]); auto len = pu[1]; size_t c1 = input[src]; if (c1 <= pu[0] && !((&(engine.program[pc + 1 + 4]))[c1 >> 3] & (1 << (c1 & 7)))) goto Lnomatch; pc += 1 + 2 * cast(uint)ushort.sizeof + len; break; case engine.REbit: if (src == input.length) goto Lnomatch; auto pu = (cast(ushort *)&engine.program[pc + 1]); debug(std_regex) writefln("\tREbit %d, %d, '%s'", pu[0], pu[1], input[src]); auto len = pu[1]; size_t c1 = input[src]; if (c1 > pu[0]) goto Lnomatch; if (!((&engine.program[pc + 1 + 4])[c1 >> 3] & (1 << (c1 & 7)))) goto Lnomatch; src++; pc += 1 + 2 * cast(uint)ushort.sizeof + len; break; case engine.REnotbit: if (src == input.length) goto Lnomatch; auto pu = (cast(ushort *)&engine.program[pc + 1]); debug(std_regex) writefln("\tREnotbit %d, %d, '%s'", pu[0], pu[1], input[src]); auto len = pu[1]; size_t c1 = input[src]; if (c1 <= pu[0] && ((&engine.program[pc + 1 + 4])[c1 >> 3] & (1 << (c1 & 7)))) goto Lnomatch; src++; pc += 1 + 2 * cast(uint)ushort.sizeof + len; break; case engine.RErange: auto len = *cast(uint *)&engine.program[pc + 1]; debug(std_regex) writefln("\tRErange %d", len); if (src == input.length) goto Lnomatch; // BUG: REA.ignoreCase? if (memchr(cast(char*)&engine.program[pc + 1 + uint.sizeof], input[src], len) == null) goto Lnomatch; src++; pc += 1 + cast(uint)uint.sizeof + len; break; case engine.REnotrange: auto len = *cast(uint *)&engine.program[pc + 1]; debug(std_regex) writefln("\tREnotrange %d", len); if (src == input.length) goto Lnomatch; // BUG: REA.ignoreCase? if (memchr(cast(char*)&engine.program[pc + 1 + uint.sizeof], input[src], len) != null) goto Lnomatch; src++; pc += 1 + cast(uint)uint.sizeof + len; break; case engine.REbol: debug(std_regex) writefln("\tREbol"); if (src == 0) { } else if (engine.attributes & engine.REA.multiline) { if (input[src - 1] != '\n') goto Lnomatch; } else goto Lnomatch; pc++; break; case engine.REeol: debug(std_regex) writefln("\tREeol"); if (src == input.length) { } else if (engine.attributes & engine.REA.multiline && input[src] == '\n') src++; else goto Lnomatch; pc++; break; case engine.REor: auto len = (cast(uint *)&engine.program[pc + 1])[0]; debug(std_regex) writefln("\tREor %d", len); pop = pc + 1 + cast(uint)uint.sizeof; memoize(pop+len); // remember 2nd branch pc = pop; // proceed with 1st branch break; case engine.REgoto: debug(std_regex) writefln("\tREgoto"); auto len = (cast(uint *)&engine.program[pc + 1])[0]; pc += 1 + cast(uint)uint.sizeof + len; break; case engine.REanystar: pc++; for (;;) { if (src == input.length) break; if (!(engine.attributes & engine.REA.dotmatchlf) && input[src] == '\n') break; if (trymatch(pc, memory[lastState..$])) return true; src += std.utf.stride(input, src); } break; case engine.REanystarg: debug(std_regex) writefln("\tREanystar"); pc++; auto ss = src; if (engine.attributes & engine.REA.dotmatchlf) src = input.length; else { auto p = memchr(input.ptr+src,'\n', input.length-src); src = p ? p - &input[src] : input.length; } while (src > ss) { if (trymatch(pc, memory[lastState..$])) return true; if (trymatch(pc, memory[lastState..$])) return true; src -= strideBack(input, src); } break; case engine.REcounter: // n auto puint = cast(uint *)&engine.program[pc + 1]; curCounter = puint[0]; auto len = puint[1]; counters[curCounter] = 0; if (trackers.empty) { auto ptracker = cast(size_t *)alloca(counters.length*size_t.sizeof); if (ptracker) trackers = ptracker[0..counters.length]; else trackers = new size_t[counters.length]; } trackers[curCounter] = size_t.max; pc += len + 1 + cast(uint)2*uint.sizeof; break; case engine.REloop: case engine.REloopg: // n, m, len auto puint = cast(uint *)&engine.program[pc + 1]; auto n = puint[0]; auto m = puint[1]; auto len = puint[2]; debug(std_regex) writefln("\tREloop%s min=%u, max=%u pc=>%d", (engine.program[pc] == engine.REloopg) ? "g" : "", n, m, pc - len); if (counters[curCounter] < n) { counters[curCounter]++; pc = pc - len; break; } else if (counters[curCounter] == m || trackers[curCounter] == src) {//proceed with outer loops curCounter--; pc += 1 + cast(uint)uint.sizeof*3; break; } counters[curCounter]++; if (engine.program[pc] == engine.REloop) { memoize(pc-len); //memoize next step of loop curCounter--; pc += 1 + cast(uint)uint.sizeof*3; // proceed with outer loop } else // maximal munch { curCounter--; memoize(pc + 1 + cast(uint)uint.sizeof*3); curCounter++; pc = pc - len; //move on with the loop trackers[curCounter] = src; } break; case engine.REsave: // n debug(std_regex) writefln("\tREsave"); auto n = *cast(uint *)&engine.program[pc + 1]; (cast(size_t*)pmatch)[n] = src; debug(std_regex) { if (n % 2) writefln("\tmatch # %d at %d .. %d", n/2, pmatch[n/2].startIdx, pmatch[n/2].endIdx); } matchesToSave = max(n/2, matchesToSave); pc += cast(uint)uint.sizeof+1; break; case engine.RElookahead: case engine.REneglookahead: // len, () debug(std_regex) writef("\t%s", engine.program[pc] == engine.RElookahead ? "RElookahead" : "REneglookahead"); auto len = *cast(uint*)&engine.program[pc+1]; pop = pc + 1 + cast(uint)uint.sizeof; bool invert = engine.program[pc] == engine.REneglookahead ? true : false; auto tmp_match = trymatch(pop, memory[lastState..$]); //inverse the match if negative lookahead tmp_match = tmp_match ^ invert; if (!tmp_match) goto Lnomatch; pc = pop + len; break; case engine.REret: debug(std_regex) writefln("\tREret"); src = srcsave; return 1; case engine.REend: debug(std_regex) writefln("\tREend"); return 1; // successful match case engine.REwordboundary: debug(std_regex) writefln("\tREwordboundary"); if (src > 0 && src < input.length) { size_t c1 = input[src - 1]; size_t c2 = input[src]; if (!((engine.isword(cast(E)c1) && !engine.isword(cast(E)c2)) || (!engine.isword(cast(E)c1) && engine.isword(cast(E)c2)))) goto Lnomatch; } pc++; break; case engine.REnotwordboundary: debug(std_regex) writefln("\tREnotwordboundary"); if (src == 0 || src == input.length) goto Lnomatch; size_t c1 = input[src - 1]; size_t c2 = input[src]; if ( (engine.isword(cast(E)c1) && !engine.isword(cast(E)c2)) || (!engine.isword(cast(E)c1) && engine.isword(cast(E)c2)) ) goto Lnomatch; pc++; break; case engine.REdigit: debug(std_regex) writefln("\tREdigit"); if (src == input.length) goto Lnomatch; if (!isdigit(input[src])) goto Lnomatch; src++; pc++; break; case engine.REnotdigit: debug(std_regex) writefln("\tREnotdigit"); if (src == input.length) goto Lnomatch; if (isdigit(input[src])) goto Lnomatch; src++; pc++; break; case engine.REspace: debug(std_regex) writefln("\tREspace"); if (src == input.length) goto Lnomatch; if (!isspace(input[src])) goto Lnomatch; src++; pc++; break; case engine.REnotspace: debug(std_regex) writefln("\tREnotspace"); if (src == input.length) goto Lnomatch; if (isspace(input[src])) goto Lnomatch; src++; pc++; break; case engine.REword: debug(std_regex) writefln("\tREword"); if (src == input.length) goto Lnomatch; if (!engine.isword(input[src])) goto Lnomatch; src++; pc++; break; case engine.REnotword: debug(std_regex) writefln("\tREnotword"); if (src == input.length) goto Lnomatch; if (engine.isword(input[src])) goto Lnomatch; src++; pc++; break; case engine.REbackref: { auto n = engine.program[pc + 1]; debug(std_regex) writefln("\tREbackref %d", n); auto so = pmatch[n + 1].startIdx; auto eo = pmatch[n + 1].endIdx; auto len = eo - so; debug(std_regex) writefln("len \t%d", len); if (src + len > input.length) goto Lnomatch; else if (engine.attributes & engine.REA.ignoreCase) { if (icmp(input[src .. src + len], input[so .. eo])) goto Lnomatch; } else if (memcmp(&input[src], &input[so], len * E.sizeof)) goto Lnomatch; src += len; pc += 2; break; } default: assert(0); Lnomatch: if (!backtrack()) { src = srcsave; return false; } } } assert(0); } // p is following the \ char /* ==================== replace ======================= */ /* After a match was found, this function will take the match results and, using the format string, generate and return a new string. */ private Range replace(String)(String format) { return replace3(format, input, pmatch[0 .. engine.re_nsub + 1]); } // Static version that doesn't require a Regex object to be created private static Range replace3(String)(String format, Range input, regmatch_t[] pmatch) { Range result; uint c2; sizediff_t startIdx; sizediff_t endIdx; int i; result.length = format.length; result.length = 0; for (size_t f = 0; f < format.length; f++) { char c = format[f]; L1: if (c != '$') { result ~= c; continue; } ++f; if (f == format.length) { result ~= '$'; break; } c = format[f]; switch (c) { case '&': startIdx = pmatch[0].startIdx; endIdx = pmatch[0].endIdx; goto Lstring; case '`': startIdx = 0; endIdx = pmatch[0].startIdx; goto Lstring; case '\'': startIdx = pmatch[0].endIdx; endIdx = input.length; goto Lstring; case '0': case '1': case '2': case '3': case '4': case '5': case '6': case '7': case '8': case '9': i = c - '0'; if (f + 1 == format.length) { if (i == 0) { result ~= '$'; result ~= c; continue; } } else { c2 = format[f + 1]; if (c2 >= '0' && c2 <= '9') { i = (c - '0') * 10 + (c2 - '0'); f++; } if (i == 0) { result ~= '$'; result ~= c; c = cast(char)c2; goto L1; } } if (i < pmatch.length) { startIdx = pmatch[i].startIdx; endIdx = pmatch[i].endIdx; goto Lstring; } break; Lstring: if (startIdx != endIdx) result ~= input[startIdx .. endIdx]; break; default: result ~= '$'; result ~= c; break; } } return result; } /* *********************************** * Like replace(char[] format), but uses old style formatting:
Format Description
& replace with the match
\n replace with the nth parenthesized match, n is 1..9
\c replace with char c.
*/ deprecated private string replaceOld(string format) { string result; //writefln("replace: this = %p so = %d, eo = %d", this, pmatch[0].startIdx, pmatch[0].endIdx); //writefln("3input = '%s'", input); result.length = format.length; result.length = 0; for (size_t i; i < format.length; i++) { char c = format[i]; switch (c) { case '&': result ~= to!string( input[pmatch[0].startIdx .. pmatch[0].endIdx]); break; case '\\': if (i + 1 < format.length) { c = format[++i]; if (c >= '1' && c <= '9') { uint j = c - '0'; if (j <= engine.re_nsub && pmatch[j].startIdx != pmatch[j].endIdx) result ~= to!string (input[pmatch[j].startIdx .. pmatch[j].endIdx]); break; } } result ~= c; break; default: result ~= c; break; } } return result; } } // end of class RegexMatch unittest { debug(std_regex) writefln("regex.replace.unittest()"); auto r = match("1ab2ac3", regex("a[bc]", "g")); auto result = r.replaceAll("x$&y"); auto i = std.string.cmp(result, "1xaby2xacy3"); assert(i == 0); r = match("1ab2ac3", regex("ab", "g")); result = r.replaceAll("xy"); i = std.string.cmp(result, "1xy2ac3"); assert(i == 0); r = match("wyda", regex("(giba)")); assert(r.captures.length == 0); } unittest { //@@@ assert(!match("abc", regex(".b.")).empty); assert(match("abc", regex(".b..")).empty); } //------------------------------------------------------------------------------ /** Matches a string against a regular expression. This is the main entry to the module's functionality. A call to $(D match(input, regex)) returns a $(D RegexMatch) object that can be used for direct inspection or for iterating over all matches (if the regular expression was built with the "g" option). */ RegexMatch!(Range) match(Range, Engine)(Range r, Engine engine) if (is(Unqual!Engine == Regex!(Unqual!(typeof(Range.init[0]))))) { return typeof(return)(engine, r); } RegexMatch!(Range) match(Range, E)(Range r, E[] engine, string opt = null) //if (is(Engine == Regex!(Unqual!(ElementType!(Range))))) { return typeof(return)(regex(engine, opt), r); } unittest { string abr = "abracadabra"; "abracadabra".match(regex("a[b-e]", "g")); abr.match(regex("a[b-e]", "g")); "abracadabra".match("a[b-e]", "g"); abr.match("a[b-e]", "g"); // Created and placed in public domain by Don Clugston auto re = regex(`bc\x20r[\40]s`, "i"); auto m = match("aBC r s", re); static assert(isForwardRange!(typeof(m))); static assert(isRandomAccessRange!(typeof(m.captures()))); assert(m.pre=="a"); assert(m.hit=="BC r s"); auto m2 = match("7xxyxxx", regex(`^\d([a-z]{2})\D\1`)); assert(!m2.empty); assert(m2.hit=="7xxyxx"); // Just check the parsing. auto m3 = match("dcbxx", regex(`ca|b[\d\]\D\s\S\w-\W]`)); assert(!m3.empty); auto m4 = match("xy", regex(`[^\ca-\xFa\r\n\b\f\t\v\0123]{2,485}$`)); assert(m4.empty); auto m5 = match("xxx", regex(`^^\r\n\b{13,}\f{4}\t\v\u02aF3a\w\W`)); assert(m5.empty); auto m6 = match("xxy", regex(`.*y`)); assert(!m6.empty); assert(m6.hit=="xxy"); auto m7 = match("QWDEfGH"d, regex("(ca|b|defg)+"d, "i")); assert(!m7.empty); assert(m7.hit=="DEfG"); auto m8 = match("dcbxx"w, regex(`a?\B\s\S`w)); assert(m8.empty); auto m9 = match("dcbxx"d, regex(`[-w]`d)); assert(m9.empty); auto m10 = match("dcbsfd"w, regex(`aB[c-fW]dB|\d|\D|\u012356|\w|\W|\s|\S`w, "i")); assert(!m10.empty); auto m11 = match("dcbsfd", regex(`[]a-]`)); assert(m11.empty); //m.replaceOld(`a&b\1c`); m.replace(`a$&b$'$1c`); } /****************************************************** Search string for matches with regular expression pattern with attributes. Replace the first match with string generated from $(D format). If the regular expression has the $(D "g") (global) attribute, continue and replace all matches. Params: input = Range to search. regex = Regular expression pattern. format = Replacement string format. Returns: The resulting string. Example: --- s = "ark rapacity"; assert(replace(s, regex("r"), "c") == "ack rapacity"); assert(replace(s, regex("r", "g"), "c") == "ack capacity"); --- The replacement format can reference the matches using the $&, $$, $', $`, $0 .. $99 notation: --- assert(replace("noon", regex("^n"), "[$&]") == "[n]oon"); --- */ Range replace(Range, Engine, String)(Range input, Engine regex, String format) if (is(Unqual!Engine == Regex!(Unqual!(typeof(Range.init[0]))))) { return RegexMatch!(Range)(regex, input).replaceAll(format); } unittest { debug(std_regex) writefln("regex.sub.unittest"); assert(replace("hello", regex("ll"), "ss") == "hesso"); assert(replace("barat", regex("a"), "A") == "bArat"); assert(replace("barat", regex("a", "g"), "A") == "bArAt"); auto s = "ark rapacity"; assert(replace(s, regex("r"), "c") == "ack rapacity"); assert(replace(s, regex("r", "g"), "c") == "ack capacity"); assert(replace("noon", regex("^n"), "[$&]") == "[n]oon"); } // @@@BUG@@@ workaround for bug 5003 private bool _dummyTest(Engine)(ref Engine r, size_t idx) { return r.test(idx); } // @@@BUG@@@ workaround for bug 5003 private ubyte _dummyAttributes(Engine)(ref Engine r) { return r.attributes; } /******************************************************* Search string for matches with regular expression pattern with attributes. Pass each match to function $(D fun). Replace each match with the return value from dg. Params: s = String to search. pattern = Regular expression pattern. dg = Delegate Returns: the resulting string. Example: Capitalize the letters 'a' and 'r': --- string baz(RegexMatch!(string) m) { return std.string.toupper(m.hit); } auto s = replace!(baz)("Strap a rocket engine on a chicken.", regex("[ar]", "g")); assert(s == "StRAp A Rocket engine on A chicken."); --- */ Range replace(alias fun, Range, Regex) (Range s, Regex rx) { auto r = match(s, rx); auto result = s; size_t lastindex = 0; size_t offset = 0; // @@@BUG@@@ workaround for bug 5003 while (_dummyTest(r, lastindex)) { auto so = r.pmatch[0].startIdx; auto eo = r.pmatch[0].endIdx; auto replacement = unaryFun!(fun)(r); /+ // Optimize by using std.string.replace if possible - Dave Fladebo auto slice = result[offset + so .. offset + eo]; if (rx.attributes & rx.REA.global && // global, so replace all !(rx.attributes & rx.REA.ignoreCase) && // not ignoring case !(rx.attributes & rx.REA.multiline) && // not multiline pattern == slice) // simple pattern (exact match, no // special characters) { debug(std_regex) { auto sss = result[offset + so .. offset + eo]; writefln("pattern: %s, slice: %s, replacement: %s", pattern, sss, replacement); result = std.string.replace(result, slice, replacement); break; } +/ result = replaceSlice(result, result[offset + so .. offset + eo], replacement); // @@@BUG@@@ workaround for bug 5003 if (_dummyAttributes(rx) & rx.REA.global) { offset += replacement.length - (eo - so); if (lastindex == eo) lastindex++; // always consume some source else lastindex = eo; } else break; } return result; } unittest { //debug(std_regex) writefln("regex.sub.unittest"); string foo(RegexMatch!(string) r) { return "ss"; } auto r = replace!(foo)("hello", regex("ll")); assert(r == "hesso"); string bar(RegexMatch!(string) r) { return "l"; } r = replace!(bar)("hello", regex("l", "g")); assert(r == "hello"); string baz(RegexMatch!(string) m) { return std.string.toupper(m.hit); } auto s = replace!(baz)("Strap a rocket engine on a chicken.", regex("[ar]", "g")); assert(s == "StRAp A Rocket engine on A chicken."); } //------------------------------------------------------------------------------ /** Range that splits another range using a regular expression as a separator. Example: ---- auto s1 = ", abc, de, fg, hi, "; assert(equal(splitter(s1, regex(", *")), ["", "abc", "de", "fg", "hi", ""][])); ---- */ struct Splitter(Range) { Range _input; size_t _offset; alias Regex!(Unqual!(typeof(Range.init[0]))) Rx; // Rx _rx; RegexMatch!(Range) _match; this(Range input, Rx separator) { _input = input; if (_input.empty) { // there is nothing to match at all, make _offset > 0 _offset = 1; } else { _match = match(_input, separator); } } // @@@BUG 2674 and 2675 // this(this) // { // _match.pmatch = _match.pmatch.dup; // } ref auto opSlice() { return this; } @property Range front() { //write("[");scope(success) writeln("]"); assert(!empty && _offset <= _match.pre.length && _match.pre.length <= _input.length); return _input[_offset .. min($, _match.pre.length)]; } @property bool empty() { return _offset > _input.length; } void popFront() { //write("[");scope(success) writeln("]"); assert(!empty); if (_match.empty) { // No more separators, work is done here _offset = _input.length + 1; } else { // skip past the separator _offset = _match.pre.length + _match.hit.length; _match.popFront; } } static if (isForwardRange!Range) { @property typeof(this) save() { auto ret = this; ret._input = _input.save; ret._match = _match.save; return ret; } } } /// Ditto Splitter!(Range) splitter(Range, Regex)(Range r, Regex pat) if (is(Unqual!(typeof(Range.init[0])) == char)) { static assert(is(Unqual!(typeof(Range.init[0])) == char), Unqual!(typeof(Range.init[0])).stringof); return Splitter!(Range)(r, pat); } unittest { auto s1 = ", abc, de, fg, hi, "; auto sp1 = splitter(s1, regex(", *")); auto w1 = ["", "abc", "de", "fg", "hi", ""]; assert(equal(sp1, w1[])); auto s2 = ", abc, de, fg, hi"; auto sp2 = splitter(s2, regex(", *")); auto w2 = ["", "abc", "de", "fg", "hi"]; //foreach (e; sp2) writeln(e); assert(equal(sp2, w2[])); } unittest { char[] s1 = ", abc, de, fg, hi, ".dup; auto sp2 = splitter(s1, regex(", *")); } String[] split(String)(String input, Regex!(char) rx) { auto a = appender!(String[])(); foreach (e; splitter(input, rx)) { a.put(e); } return a.data; } unittest { auto s1 = ", abc, de, fg, hi, "; auto w1 = ["", "abc", "de", "fg", "hi", ""]; assert(equal(split(s1, regex(", *")), w1[])); } /* * Copyright (C) 2000-2005 by Digital Mars, www.digitalmars.com * Written by Walter Bright and Andrei Alexandrescu * * This software is provided 'as-is', without any express or implied * warranty. In no event will the authors be held liable for any damages * arising from the use of this software. * * Permission is granted to anyone to use this software for any purpose, * including commercial applications, and to alter it and redistribute it * freely, subject to the following restrictions: * * o The origin of this software must not be misrepresented; you must not * claim that you wrote the original software. If you use this software * in a product, an acknowledgment in the product documentation would be * appreciated but is not required. * o Altered source versions must be plainly marked as such, and must not * be misrepresented as being the original software. * o This notice may not be removed or altered from any source * distribution. */ /* The test vectors in this file are altered from Henry Spencer's regexp test code. His copyright notice is: Copyright (c) 1986 by University of Toronto. Written by Henry Spencer. Not derived from licensed software. Permission is granted to anyone to use this software for any purpose on any computer system, and to redistribute it freely, subject to the following restrictions: 1. The author is not responsible for the consequences of use of this software, no matter how awful, even if they arise from defects in it. 2. The origin of this software must not be misrepresented, either by explicit claim or by omission. 3. Altered versions must be plainly marked as such, and must not be misrepresented as being the original software. */ unittest { struct TestVectors { string pattern; string input; string result; string format; string replace; }; static TestVectors tv[] = [ { "(a)\\1", "abaab","y", "&", "aa" }, { "abc", "abc", "y", "&", "abc" }, { "abc", "xbc", "n", "-", "-" }, { "abc", "axc", "n", "-", "-" }, { "abc", "abx", "n", "-", "-" }, { "abc", "xabcy","y", "&", "abc" }, { "abc", "ababc","y", "&", "abc" }, { "ab*c", "abc", "y", "&", "abc" }, { "ab*bc", "abc", "y", "&", "abc" }, { "ab*bc", "abbc", "y", "&", "abbc" }, { "ab*bc", "abbbbc","y", "&", "abbbbc" }, { "ab+bc", "abbc", "y", "&", "abbc" }, { "ab+bc", "abc", "n", "-", "-" }, { "ab+bc", "abq", "n", "-", "-" }, { "ab+bc", "abbbbc","y", "&", "abbbbc" }, { "ab?bc", "abbc", "y", "&", "abbc" }, { "ab?bc", "abc", "y", "&", "abc" }, { "ab?bc", "abbbbc","n", "-", "-" }, { "ab?c", "abc", "y", "&", "abc" }, { "^abc$", "abc", "y", "&", "abc" }, { "^abc$", "abcc", "n", "-", "-" }, { "^abc", "abcc", "y", "&", "abc" }, { "^abc$", "aabc", "n", "-", "-" }, { "abc$", "aabc", "y", "&", "abc" }, { "^", "abc", "y", "&", "" }, { "$", "abc", "y", "&", "" }, { "a.c", "abc", "y", "&", "abc" }, { "a.c", "axc", "y", "&", "axc" }, { "a.*c", "axyzc","y", "&", "axyzc" }, { "a.*c", "axyzd","n", "-", "-" }, { "a[bc]d", "abc", "n", "-", "-" }, { "a[bc]d", "abd", "y", "&", "abd" }, { "a[b-d]e", "abd", "n", "-", "-" }, { "a[b-d]e", "ace", "y", "&", "ace" }, { "a[b-d]", "aac", "y", "&", "ac" }, { "a[-b]", "a-", "y", "&", "a-" }, { "a[b-]", "a-", "y", "&", "a-" }, { "a[b-a]", "-", "c", "-", "-" }, { "a[]b", "-", "c", "-", "-" }, { "a[", "-", "c", "-", "-" }, { "a]", "a]", "y", "&", "a]" }, { "a[]]b", "a]b", "y", "&", "a]b" }, { "a[^bc]d", "aed", "y", "&", "aed" }, { "a[^bc]d", "abd", "n", "-", "-" }, { "a[^-b]c", "adc", "y", "&", "adc" }, { "a[^-b]c", "a-c", "n", "-", "-" }, { "a[^]b]c", "a]c", "n", "-", "-" }, { "a[^]b]c", "adc", "y", "&", "adc" }, { "ab|cd", "abc", "y", "&", "ab" }, { "ab|cd", "abcd", "y", "&", "ab" }, { "()ef", "def", "y", "&-\\1", "ef-" }, { "()*", "-", "y", "-", "-" }, { "*a", "-", "c", "-", "-" }, { "^*", "-", "y", "-", "-" }, { "$*", "-", "y", "-", "-" }, { "(*)b", "-", "c", "-", "-" }, { "$b", "b", "n", "-", "-" }, { "a\\", "-", "c", "-", "-" }, { "a\\(b", "a(b", "y", "&-\\1", "a(b-" }, { "a\\(*b", "ab", "y", "&", "ab" }, { "a\\(*b", "a((b", "y", "&", "a((b" }, { "a\\\\b", "a\\b", "y", "&", "a\\b" }, { "abc)", "-", "c", "-", "-" }, { "(abc", "-", "c", "-", "-" }, { "((a))", "abc", "y", "&-\\1-\\2", "a-a-a" }, { "(a)b(c)", "abc", "y", "&-\\1-\\2", "abc-a-c" }, { "a+b+c", "aabbabc","y", "&", "abc" }, { "a**", "-", "c", "-", "-" }, { "a*?a", "aa", "y", "&", "a" }, { "(a*)*", "aaa", "y", "-", "-" }, { "(a*)+", "aaa", "y", "-", "-" }, { "(a|)*", "-", "y", "-", "-" }, { "(a*|b)*", "aabb", "y", "-", "-" }, { "(a|b)*", "ab", "y", "&-\\1", "ab-b" }, { "(a+|b)*", "ab", "y", "&-\\1", "ab-b" }, { "(a+|b)+", "ab", "y", "&-\\1", "ab-b" }, { "(a+|b)?", "ab", "y", "&-\\1", "a-a" }, { "[^ab]*", "cde", "y", "&", "cde" }, { "(^)*", "-", "y", "-", "-" }, { "(ab|)*", "-", "y", "-", "-" }, { ")(", "-", "c", "-", "-" }, { "", "abc", "y", "&", "" }, { "abc", "", "n", "-", "-" }, { "a*", "", "y", "&", "" }, { "([abc])*d", "abbbcd", "y", "&-\\1", "abbbcd-c" }, { "([abc])*bcd", "abcd", "y", "&-\\1", "abcd-a" }, { "a|b|c|d|e", "e", "y", "&", "e" }, { "(a|b|c|d|e)f", "ef", "y", "&-\\1", "ef-e" }, { "((a*|b))*", "aabb", "y", "-", "-" }, { "abcd*efg", "abcdefg", "y", "&", "abcdefg" }, { "ab*", "xabyabbbz", "y", "&", "ab" }, { "ab*", "xayabbbz", "y", "&", "a" }, { "(ab|cd)e", "abcde", "y", "&-\\1", "cde-cd" }, { "[abhgefdc]ij", "hij", "y", "&", "hij" }, { "^(ab|cd)e", "abcde", "n", "x\\1y", "xy" }, { "(abc|)ef", "abcdef", "y", "&-\\1", "ef-" }, { "(a|b)c*d", "abcd", "y", "&-\\1", "bcd-b" }, { "(ab|ab*)bc", "abc", "y", "&-\\1", "abc-a" }, { "a([bc]*)c*", "abc", "y", "&-\\1", "abc-bc" }, { "a([bc]*)(c*d)", "abcd", "y", "&-\\1-\\2", "abcd-bc-d" }, { "a([bc]+)(c*d)", "abcd", "y", "&-\\1-\\2", "abcd-bc-d" }, { "a([bc]*)(c+d)", "abcd", "y", "&-\\1-\\2", "abcd-b-cd" }, { "a[bcd]*dcdcde", "adcdcde", "y", "&", "adcdcde" }, { "a[bcd]+dcdcde", "adcdcde", "n", "-", "-" }, { "(ab|a)b*c", "abc", "y", "&-\\1", "abc-ab" }, { "((a)(b)c)(d)", "abcd", "y", "\\1-\\2-\\3-\\4", "abc-a-b-d" }, { "[a-zA-Z_][a-zA-Z0-9_]*", "alpha", "y", "&", "alpha" }, { "^a(bc+|b[eh])g|.h$", "abh", "y", "&-\\1", "bh-" }, { "(bc+d$|ef*g.|h?i(j|k))", "effgz", "y", "&-\\1-\\2", "effgz-effgz-" }, { "(bc+d$|ef*g.|h?i(j|k))", "ij", "y", "&-\\1-\\2", "ij-ij-j" }, { "(bc+d$|ef*g.|h?i(j|k))", "effg", "n", "-", "-" }, { "(bc+d$|ef*g.|h?i(j|k))", "bcdd", "n", "-", "-" }, { "(bc+d$|ef*g.|h?i(j|k))", "reffgz", "y", "&-\\1-\\2", "effgz-effgz-" }, { "(((((((((a)))))))))", "a", "y", "&", "a" }, { "multiple words of text", "uh-uh", "n", "-", "-" }, { "multiple words", "multiple words, yeah", "y", "&", "multiple words" }, { "(.*)c(.*)", "abcde", "y", "&-\\1-\\2", "abcde-ab-de" }, { "\\((.*), (.*)\\)", "(a, b)", "y", "(\\2, \\1)", "(b, a)" }, { "abcd", "abcd", "y", "&-\\&-\\\\&", "abcd-&-\\abcd" }, { "a(bc)d", "abcd", "y", "\\1-\\\\1-\\\\\\1", "bc-\\1-\\bc" }, { "[k]", "ab", "n", "-", "-" }, { "[ -~]*", "abc", "y", "&", "abc" }, { "[ -~ -~]*", "abc", "y", "&", "abc" }, { "[ -~ -~ -~]*", "abc", "y", "&", "abc" }, { "[ -~ -~ -~ -~]*", "abc", "y", "&", "abc" }, { "[ -~ -~ -~ -~ -~]*", "abc", "y", "&", "abc" }, { "[ -~ -~ -~ -~ -~ -~]*", "abc", "y", "&", "abc" }, { "[ -~ -~ -~ -~ -~ -~ -~]*", "abc", "y", "&", "abc" }, { "a{2}", "candy", "n", "", "" }, { "a{2}", "caandy", "y", "&", "aa" }, { "a{2}", "caaandy", "y", "&", "aa" }, { "a{2,}", "candy", "n", "", "" }, { "a{2,}", "caandy", "y", "&", "aa" }, { "a{2,}", "caaaaaandy", "y", "&", "aaaaaa" }, { "a{1,3}", "cndy", "n", "", "" }, { "a{1,3}", "candy", "y", "&", "a" }, { "a{1,3}", "caandy", "y", "&", "aa" }, { "a{1,3}", "caaaaaandy", "y", "&", "aaa" }, { "e?le?", "angel", "y", "&", "el" }, { "e?le?", "angle", "y", "&", "le" }, { "\\bn\\w", "noonday", "y", "&", "no" }, { "\\wy\\b", "possibly yesterday", "y", "&", "ly" }, { "\\w\\Bn", "noonday", "y", "&", "on" }, { "y\\B\\w", "possibly yesterday", "y", "&", "ye" }, { "\\cJ", "abc\ndef", "y", "&", "\n" }, { "\\d", "B2 is", "y", "&", "2" }, { "\\D", "B2 is", "y", "&", "B" }, { "\\s\\w*", "foo bar", "y", "&", " bar" }, { "\\S\\w*", "foo bar", "y", "&", "foo" }, { "abc", "ababc", "y", "&", "abc" }, { "apple(,)\\sorange\\1", "apple, orange, cherry, peach", "y", "&", "apple, orange," }, { "(\\w+)\\s(\\w+)", "John Smith", "y", "\\2, \\1", "Smith, John" }, { "\\n\\f\\r\\t\\v", "abc\n\f\r\t\vdef", "y", "&", "\n\f\r\t\v" }, { ".*c", "abcde", "y", "&", "abc" }, { "^\\w+((;|=)\\w+)+$", "some=host=tld", "y", "&-\\1-\\2", "some=host=tld-=tld-=" }, { "^\\w+((\\.|-)\\w+)+$", "some.host.tld", "y", "&-\\1-\\2", "some.host.tld-.tld-." }, { "q(a|b)*q", "xxqababqyy", "y", "&-\\1", "qababq-b" }, { "^(a)(b){0,1}(c*)", "abcc", "y", "\\1 \\2 \\3", "a b cc" }, { "^(a)((b){0,1})(c*)", "abcc", "y", "\\1 \\2 \\3", "a b b" }, { "^(a)(b)?(c*)", "abcc", "y", "\\1 \\2 \\3", "a b cc" }, { "^(a)((b)?)(c*)", "abcc", "y", "\\1 \\2 \\3", "a b b" }, { "^(a)(b){0,1}(c*)", "acc", "y", "\\1 \\2 \\3", "a cc" }, { "^(a)((b){0,1})(c*)", "acc", "y", "\\1 \\2 \\3", "a " }, { "^(a)(b)?(c*)", "acc", "y", "\\1 \\2 \\3", "a cc" }, { "^(a)((b)?)(c*)", "acc", "y", "\\1 \\2 \\3", "a " }, {"(?:ab){3}", "_abababc", "y","&-\\1","ababab-" }, {"(?:a(?:x)?)+", "aaxaxx", "y","&-\\1-\\2","aaxax--" }, {"foo.(?=bar)", "foobar foodbar", "y","&-\\1", "food-" }, {"(?:(.)(?!\\1))+", "12345678990", "y", "&-\\1", "12345678-8" }, ]; int i; sizediff_t a; uint c; sizediff_t start; sizediff_t end; TestVectors tvd; foreach (Char; TypeTuple!(char, wchar, dchar)) { alias immutable(Char)[] String; String produceExpected(Range)(RegexMatch!(Range) m, String fmt) { String result; while (!fmt.empty) switch (fmt.front) { case '\\': fmt.popFront(); if (!isdigit(fmt.front) ) { result ~= fmt.front; fmt.popFront(); break; } auto nmatch = parse!uint(fmt); if (nmatch < m.captures.length) result ~= m.captures[nmatch]; break; case '&': result ~= m.hit; fmt.popFront(); break; default: result ~= fmt.front; fmt.popFront(); } return result; } Regex!(Char) r; start = 0; end = tv.length; for (a = start; a < end; a++) { // writef("width: %d tv[%d]: pattern='%s' input='%s' result=%s" // " format='%s' replace='%s'\n", // Char.sizeof, a, // tv[a].pattern, // tv[a].input, // tv[a].result, // tv[a].format, // tv[a].replace); tvd = tv[a]; c = tvd.result[0]; try { i = 1; r = regex(to!(String)(tvd.pattern)); } catch (Exception e) { i = 0; } //writefln("\tcompile() = %d", i); assert((c == 'c') ? !i : i); if (c != 'c') { auto m = match(to!(String)(tvd.input), r); i = !m.empty; //writefln("\ttest() = %d", i); //fflush(stdout); assert((c == 'y') ? i : !i, text("Match failed pattern: ", tvd.pattern)); if (c == 'y') { auto result = produceExpected(m, to!(String)(tvd.format)); assert(result == to!String(tvd.replace), text("Mismatch pattern: ", tvd.pattern," expected:", tvd.replace, " vs ", result)); } } } try { r = regex(to!(String)("a\\.b"), "i"); } catch (Exception e) { assert(0); } assert(!match(to!(String)("A.b"), r).empty); assert(!match(to!(String)("a.B"), r).empty); assert(!match(to!(String)("A.B"), r).empty); assert(!match(to!(String)("a.b"), r).empty); } } template loadFile(Types...) { Tuple!(Types)[] loadFile(Char)(string filename, Regex!(Char) rx) { auto result = appender!(typeof(return)); auto f = File(filename); scope(exit) f.close; RegexMatch!(Char[]) match; foreach (line; f.byLine()) { match = .match(line, rx); Tuple!(Types) t; foreach (i, unused; t.field) { t[i] = to!(typeof(t[i]))(match.captures[i + 1]); } result.put(t); } return result.data; } } unittest { // DAC: This doesn't create the file before running the test! pragma(msg, " --- std.regex("~ __LINE__.stringof ~") broken test --- "); /+ string tmp = "/tmp/deleteme"; std.file.write(tmp, "1 abc\n2 defg\n3 hijklm"); auto t = loadFile!(uint, string)(tmp, regex("([0-9])+ +(.+)")); //writeln(t); assert(t[0] == tuple(1, "abc")); assert(t[1] == tuple(2, "defg")); assert(t[2] == tuple(3, "hijklm")); +/ } unittest { auto str = "foo"; string[] re_strs = [ r"^(a|b|)fo[oas]$", r"^(a|o|)fo[oas]$", r"^(a|)foo$", r"^(a|)foo$", r"^(h|)foo$", r"(h|)foo", r"(h|a|)fo[oas]", r"^(a|b|)fo[o]$", r"[abf][ops](o|oo|)(h|a|)", r"(h|)[abf][ops](o|oo|)", r"(c|)[abf][ops](o|oo|)" ]; foreach (re_str; re_strs) { auto re = regex(re_str); auto matches= match(str, re); assert(!matches.empty); // writefln("'%s' matches '%s' ? %s", str, re_str, !matches.empty); // if (matches.empty) // re.printProgram(); } } //issue 5857 //matching goes out of control if ... in (...){x} has .*/.+ unittest { auto c = match("axxxzayyyyyzd",regex("(a.*z){2}d")).captures; assert(c[0] == "axxxzayyyyyzd"); assert(c[1] == "ayyyyyz"); auto c2 = match("axxxayyyyyd",regex("(a.*){2}d")).captures; assert(c2[0] == "axxxayyyyyd"); assert(c2[1] == "ayyyyy"); } //issue 2108 //greedy vs non-greedy unittest { auto nogreed = regex(""); assert(match("texttext", nogreed).hit == "text"); auto greed = regex(""); assert(match("texttext", greed).hit == "texttext"); } //issue 4574 //empty successful match still advances the input unittest { string[] pres, posts, hits; foreach(m; match("abcabc", regex(""))) { pres ~= m.pre; posts ~= m.post; assert(m.hit.empty); } auto heads = [ "abcabc", "abcab", "abca", "abc", "ab", "a", "" ]; auto tails = [ "abcabc", "bcabc", "cabc", "abc", "bc", "c", "" ]; assert(pres == array(retro(heads))); assert(posts == tails); } //issue 6076 //regression on .* unittest { auto re = regex("c.*|d"); auto m = match("mm", re); assert(m.empty); } //issue 6261 //regression: doesn't allow mutable patterns unittest{ regex("foo".dup); }