blob: febbde1c5e5eb2a1b283227003d5a70089598cdb [file] [log] [blame]
Guido van Rossum7627c0d2000-03-31 14:58:54 +00001#
2# Secret Labs' Regular Expression Engine
Guido van Rossum7627c0d2000-03-31 14:58:54 +00003#
4# convert template to internal format
5#
Fredrik Lundh770617b2001-01-14 15:06:11 +00006# Copyright (c) 1997-2001 by Secret Labs AB. All rights reserved.
Guido van Rossum7627c0d2000-03-31 14:58:54 +00007#
Fredrik Lundh29c4ba92000-08-01 18:20:07 +00008# See the sre.py file for information on usage and redistribution.
Guido van Rossum7627c0d2000-03-31 14:58:54 +00009#
10
Fred Drakeb8f22742001-09-04 19:10:20 +000011"""Internal support module for sre"""
12
Fredrik Lundh4fb70272002-06-27 20:08:25 +000013import _sre, sys
Amaury Forgeot d'Arc4b798bd2008-04-08 21:27:42 +000014import sre_parse
Guido van Rossum7627c0d2000-03-31 14:58:54 +000015from sre_constants import *
16
Fredrik Lundhb35ffc02001-01-15 12:46:09 +000017assert _sre.MAGIC == MAGIC, "SRE module mismatch"
18
Martin v. Löwis78e2f062003-04-19 12:56:08 +000019if _sre.CODESIZE == 2:
20 MAXCODE = 65535
21else:
22 MAXCODE = 0xFFFFFFFFL
Fredrik Lundh3562f112000-07-02 12:00:07 +000023
Raymond Hettinger049ade22005-02-28 19:27:52 +000024_LITERAL_CODES = set([LITERAL, NOT_LITERAL])
25_REPEATING_CODES = set([REPEAT, MIN_REPEAT, MAX_REPEAT])
26_SUCCESS_CODES = set([SUCCESS, FAILURE])
27_ASSERT_CODES = set([ASSERT, ASSERT_NOT])
28
Fredrik Lundh7898c3e2000-08-07 20:59:04 +000029def _compile(code, pattern, flags):
30 # internal: compile a (sub)pattern
31 emit = code.append
Raymond Hettinger01c9f8c2004-03-26 11:16:55 +000032 _len = len
Raymond Hettinger049ade22005-02-28 19:27:52 +000033 LITERAL_CODES = _LITERAL_CODES
34 REPEATING_CODES = _REPEATING_CODES
35 SUCCESS_CODES = _SUCCESS_CODES
36 ASSERT_CODES = _ASSERT_CODES
Fredrik Lundh7898c3e2000-08-07 20:59:04 +000037 for op, av in pattern:
Raymond Hettinger01c9f8c2004-03-26 11:16:55 +000038 if op in LITERAL_CODES:
Fredrik Lundh7898c3e2000-08-07 20:59:04 +000039 if flags & SRE_FLAG_IGNORECASE:
40 emit(OPCODES[OP_IGNORE[op]])
Fredrik Lundh2e240442001-01-15 18:28:14 +000041 emit(_sre.getlower(av, flags))
Fredrik Lundh7898c3e2000-08-07 20:59:04 +000042 else:
43 emit(OPCODES[op])
Fredrik Lundh2e240442001-01-15 18:28:14 +000044 emit(av)
Fredrik Lundh7898c3e2000-08-07 20:59:04 +000045 elif op is IN:
46 if flags & SRE_FLAG_IGNORECASE:
47 emit(OPCODES[OP_IGNORE[op]])
48 def fixup(literal, flags=flags):
49 return _sre.getlower(literal, flags)
50 else:
51 emit(OPCODES[op])
Serhiy Storchakae9e54ae2014-10-31 13:53:21 +020052 fixup = None
Raymond Hettinger01c9f8c2004-03-26 11:16:55 +000053 skip = _len(code); emit(0)
Fredrik Lundh7898c3e2000-08-07 20:59:04 +000054 _compile_charset(av, flags, code, fixup)
Raymond Hettinger01c9f8c2004-03-26 11:16:55 +000055 code[skip] = _len(code) - skip
Fredrik Lundh7898c3e2000-08-07 20:59:04 +000056 elif op is ANY:
57 if flags & SRE_FLAG_DOTALL:
58 emit(OPCODES[ANY_ALL])
59 else:
60 emit(OPCODES[ANY])
Raymond Hettinger01c9f8c2004-03-26 11:16:55 +000061 elif op in REPEATING_CODES:
Fredrik Lundh7898c3e2000-08-07 20:59:04 +000062 if flags & SRE_FLAG_TEMPLATE:
63 raise error, "internal: unsupported template operator"
64 emit(OPCODES[REPEAT])
Raymond Hettinger01c9f8c2004-03-26 11:16:55 +000065 skip = _len(code); emit(0)
Fredrik Lundh7898c3e2000-08-07 20:59:04 +000066 emit(av[0])
67 emit(av[1])
68 _compile(code, av[2], flags)
69 emit(OPCODES[SUCCESS])
Raymond Hettinger01c9f8c2004-03-26 11:16:55 +000070 code[skip] = _len(code) - skip
71 elif _simple(av) and op is not REPEAT:
72 if op is MAX_REPEAT:
Guido van Rossum41c99e72003-04-14 17:59:34 +000073 emit(OPCODES[REPEAT_ONE])
74 else:
75 emit(OPCODES[MIN_REPEAT_ONE])
Raymond Hettinger01c9f8c2004-03-26 11:16:55 +000076 skip = _len(code); emit(0)
Fredrik Lundh7898c3e2000-08-07 20:59:04 +000077 emit(av[0])
78 emit(av[1])
79 _compile(code, av[2], flags)
80 emit(OPCODES[SUCCESS])
Raymond Hettinger01c9f8c2004-03-26 11:16:55 +000081 code[skip] = _len(code) - skip
Fredrik Lundh7898c3e2000-08-07 20:59:04 +000082 else:
83 emit(OPCODES[REPEAT])
Raymond Hettinger01c9f8c2004-03-26 11:16:55 +000084 skip = _len(code); emit(0)
Fredrik Lundh7898c3e2000-08-07 20:59:04 +000085 emit(av[0])
86 emit(av[1])
87 _compile(code, av[2], flags)
Raymond Hettinger01c9f8c2004-03-26 11:16:55 +000088 code[skip] = _len(code) - skip
89 if op is MAX_REPEAT:
Fredrik Lundh7898c3e2000-08-07 20:59:04 +000090 emit(OPCODES[MAX_UNTIL])
91 else:
92 emit(OPCODES[MIN_UNTIL])
93 elif op is SUBPATTERN:
94 if av[0]:
95 emit(OPCODES[MARK])
96 emit((av[0]-1)*2)
97 # _compile_info(code, av[1], flags)
98 _compile(code, av[1], flags)
99 if av[0]:
100 emit(OPCODES[MARK])
101 emit((av[0]-1)*2+1)
Raymond Hettinger01c9f8c2004-03-26 11:16:55 +0000102 elif op in SUCCESS_CODES:
Fredrik Lundh7898c3e2000-08-07 20:59:04 +0000103 emit(OPCODES[op])
Raymond Hettinger01c9f8c2004-03-26 11:16:55 +0000104 elif op in ASSERT_CODES:
Fredrik Lundh7898c3e2000-08-07 20:59:04 +0000105 emit(OPCODES[op])
Raymond Hettinger01c9f8c2004-03-26 11:16:55 +0000106 skip = _len(code); emit(0)
Fredrik Lundh7898c3e2000-08-07 20:59:04 +0000107 if av[0] >= 0:
108 emit(0) # look ahead
109 else:
110 lo, hi = av[1].getwidth()
111 if lo != hi:
112 raise error, "look-behind requires fixed-width pattern"
113 emit(lo) # look behind
114 _compile(code, av[1], flags)
115 emit(OPCODES[SUCCESS])
Raymond Hettinger01c9f8c2004-03-26 11:16:55 +0000116 code[skip] = _len(code) - skip
Fredrik Lundh7898c3e2000-08-07 20:59:04 +0000117 elif op is CALL:
118 emit(OPCODES[op])
Raymond Hettinger01c9f8c2004-03-26 11:16:55 +0000119 skip = _len(code); emit(0)
Fredrik Lundh7898c3e2000-08-07 20:59:04 +0000120 _compile(code, av, flags)
121 emit(OPCODES[SUCCESS])
Raymond Hettinger01c9f8c2004-03-26 11:16:55 +0000122 code[skip] = _len(code) - skip
Fredrik Lundh7898c3e2000-08-07 20:59:04 +0000123 elif op is AT:
124 emit(OPCODES[op])
125 if flags & SRE_FLAG_MULTILINE:
Fredrik Lundhb25e1ad2001-03-22 15:50:10 +0000126 av = AT_MULTILINE.get(av, av)
127 if flags & SRE_FLAG_LOCALE:
128 av = AT_LOCALE.get(av, av)
129 elif flags & SRE_FLAG_UNICODE:
130 av = AT_UNICODE.get(av, av)
131 emit(ATCODES[av])
Fredrik Lundh7898c3e2000-08-07 20:59:04 +0000132 elif op is BRANCH:
133 emit(OPCODES[op])
134 tail = []
Raymond Hettinger01c9f8c2004-03-26 11:16:55 +0000135 tailappend = tail.append
Fredrik Lundh7898c3e2000-08-07 20:59:04 +0000136 for av in av[1]:
Raymond Hettinger01c9f8c2004-03-26 11:16:55 +0000137 skip = _len(code); emit(0)
Fredrik Lundh7898c3e2000-08-07 20:59:04 +0000138 # _compile_info(code, av, flags)
139 _compile(code, av, flags)
140 emit(OPCODES[JUMP])
Raymond Hettinger01c9f8c2004-03-26 11:16:55 +0000141 tailappend(_len(code)); emit(0)
142 code[skip] = _len(code) - skip
Fredrik Lundh7898c3e2000-08-07 20:59:04 +0000143 emit(0) # end of branch
144 for tail in tail:
Raymond Hettinger01c9f8c2004-03-26 11:16:55 +0000145 code[tail] = _len(code) - tail
Fredrik Lundh7898c3e2000-08-07 20:59:04 +0000146 elif op is CATEGORY:
147 emit(OPCODES[op])
148 if flags & SRE_FLAG_LOCALE:
Fredrik Lundhb25e1ad2001-03-22 15:50:10 +0000149 av = CH_LOCALE[av]
Fredrik Lundh7898c3e2000-08-07 20:59:04 +0000150 elif flags & SRE_FLAG_UNICODE:
Fredrik Lundhb25e1ad2001-03-22 15:50:10 +0000151 av = CH_UNICODE[av]
152 emit(CHCODES[av])
Fredrik Lundh7898c3e2000-08-07 20:59:04 +0000153 elif op is GROUPREF:
154 if flags & SRE_FLAG_IGNORECASE:
155 emit(OPCODES[OP_IGNORE[op]])
156 else:
157 emit(OPCODES[op])
158 emit(av-1)
Gustavo Niemeyerad3fc442003-10-17 22:13:16 +0000159 elif op is GROUPREF_EXISTS:
160 emit(OPCODES[op])
Andrew M. Kuchlingc30faa82005-06-02 13:35:52 +0000161 emit(av[0]-1)
Raymond Hettinger01c9f8c2004-03-26 11:16:55 +0000162 skipyes = _len(code); emit(0)
Gustavo Niemeyerad3fc442003-10-17 22:13:16 +0000163 _compile(code, av[1], flags)
164 if av[2]:
165 emit(OPCODES[JUMP])
Raymond Hettinger01c9f8c2004-03-26 11:16:55 +0000166 skipno = _len(code); emit(0)
167 code[skipyes] = _len(code) - skipyes + 1
Gustavo Niemeyerad3fc442003-10-17 22:13:16 +0000168 _compile(code, av[2], flags)
Raymond Hettinger01c9f8c2004-03-26 11:16:55 +0000169 code[skipno] = _len(code) - skipno
Gustavo Niemeyerad3fc442003-10-17 22:13:16 +0000170 else:
Raymond Hettinger01c9f8c2004-03-26 11:16:55 +0000171 code[skipyes] = _len(code) - skipyes + 1
Fredrik Lundh7898c3e2000-08-07 20:59:04 +0000172 else:
173 raise ValueError, ("unsupported operand type", op)
174
175def _compile_charset(charset, flags, code, fixup=None):
176 # compile charset subprogram
177 emit = code.append
Serhiy Storchakae9e54ae2014-10-31 13:53:21 +0200178 for op, av in _optimize_charset(charset, fixup, flags & SRE_FLAG_UNICODE):
Fredrik Lundh7898c3e2000-08-07 20:59:04 +0000179 emit(OPCODES[op])
180 if op is NEGATE:
181 pass
182 elif op is LITERAL:
Serhiy Storchakae9e54ae2014-10-31 13:53:21 +0200183 emit(av)
Fredrik Lundh7898c3e2000-08-07 20:59:04 +0000184 elif op is RANGE:
Serhiy Storchakae9e54ae2014-10-31 13:53:21 +0200185 emit(av[0])
186 emit(av[1])
Fredrik Lundh7898c3e2000-08-07 20:59:04 +0000187 elif op is CHARSET:
188 code.extend(av)
Fredrik Lundh19af43d2001-07-02 16:58:38 +0000189 elif op is BIGCHARSET:
190 code.extend(av)
Fredrik Lundh7898c3e2000-08-07 20:59:04 +0000191 elif op is CATEGORY:
192 if flags & SRE_FLAG_LOCALE:
193 emit(CHCODES[CH_LOCALE[av]])
194 elif flags & SRE_FLAG_UNICODE:
195 emit(CHCODES[CH_UNICODE[av]])
196 else:
197 emit(CHCODES[av])
198 else:
199 raise error, "internal: unsupported set operator"
200 emit(OPCODES[FAILURE])
201
Serhiy Storchakae9e54ae2014-10-31 13:53:21 +0200202def _optimize_charset(charset, fixup, isunicode):
Fredrik Lundh7898c3e2000-08-07 20:59:04 +0000203 # internal: optimize character set
Fredrik Lundh3562f112000-07-02 12:00:07 +0000204 out = []
Serhiy Storchakac04fcd42014-10-31 13:34:06 +0200205 tail = []
206 charmap = bytearray(256)
207 for op, av in charset:
208 while True:
209 try:
210 if op is LITERAL:
Serhiy Storchakae9e54ae2014-10-31 13:53:21 +0200211 i = av
212 if fixup:
213 i = fixup(i)
214 charmap[i] = 1
Serhiy Storchakac04fcd42014-10-31 13:34:06 +0200215 elif op is RANGE:
Serhiy Storchakae9e54ae2014-10-31 13:53:21 +0200216 r = range(av[0], av[1]+1)
217 if fixup:
218 r = map(fixup, r)
219 for i in r:
Serhiy Storchakac04fcd42014-10-31 13:34:06 +0200220 charmap[i] = 1
221 elif op is NEGATE:
222 out.append((op, av))
223 else:
224 tail.append((op, av))
225 except IndexError:
226 if len(charmap) == 256:
227 # character set contains non-UCS1 character codes
228 charmap += b'\0' * 0xff00
229 continue
230 # character set contains non-BMP character codes
Serhiy Storchakae9e54ae2014-10-31 13:53:21 +0200231 if fixup and isunicode and op is RANGE:
232 lo, hi = av
233 ranges = [av]
234 # There are only two ranges of cased astral characters:
235 # 10400-1044F (Deseret) and 118A0-118DF (Warang Citi).
236 _fixup_range(max(0x10000, lo), min(0x11fff, hi),
237 ranges, fixup)
238 for lo, hi in ranges:
239 if lo == hi:
240 tail.append((LITERAL, hi))
241 else:
242 tail.append((RANGE, (lo, hi)))
243 else:
244 tail.append((op, av))
Serhiy Storchakac04fcd42014-10-31 13:34:06 +0200245 break
246
Fredrik Lundh3562f112000-07-02 12:00:07 +0000247 # compress character map
Fredrik Lundh3562f112000-07-02 12:00:07 +0000248 runs = []
Serhiy Storchakac04fcd42014-10-31 13:34:06 +0200249 q = 0
250 while True:
251 p = charmap.find(b'\1', q)
252 if p < 0:
253 break
254 if len(runs) >= 2:
255 runs = None
256 break
257 q = charmap.find(b'\0', p)
258 if q < 0:
259 runs.append((p, len(charmap)))
260 break
261 runs.append((p, q))
262 if runs is not None:
Fredrik Lundh3562f112000-07-02 12:00:07 +0000263 # use literal/range
Serhiy Storchakac04fcd42014-10-31 13:34:06 +0200264 for p, q in runs:
265 if q - p == 1:
266 out.append((LITERAL, p))
Fredrik Lundh3562f112000-07-02 12:00:07 +0000267 else:
Serhiy Storchakac04fcd42014-10-31 13:34:06 +0200268 out.append((RANGE, (p, q - 1)))
269 out += tail
Serhiy Storchakae9e54ae2014-10-31 13:53:21 +0200270 # if the case was changed or new representation is more compact
271 if fixup or len(out) < len(charset):
Fredrik Lundh3562f112000-07-02 12:00:07 +0000272 return out
Serhiy Storchakae9e54ae2014-10-31 13:53:21 +0200273 # else original character set is good enough
Serhiy Storchakac04fcd42014-10-31 13:34:06 +0200274 return charset
275
276 # use bitmap
277 if len(charmap) == 256:
Fredrik Lundh19af43d2001-07-02 16:58:38 +0000278 data = _mk_bitmap(charmap)
Serhiy Storchakac04fcd42014-10-31 13:34:06 +0200279 out.append((CHARSET, data))
280 out += tail
Fredrik Lundh3562f112000-07-02 12:00:07 +0000281 return out
Fredrik Lundh3562f112000-07-02 12:00:07 +0000282
Serhiy Storchakac04fcd42014-10-31 13:34:06 +0200283 # To represent a big charset, first a bitmap of all characters in the
284 # set is constructed. Then, this bitmap is sliced into chunks of 256
285 # characters, duplicate chunks are eliminated, and each chunk is
286 # given a number. In the compiled expression, the charset is
287 # represented by a 32-bit word sequence, consisting of one word for
288 # the number of different chunks, a sequence of 256 bytes (64 words)
289 # of chunk numbers indexed by their original chunk position, and a
290 # sequence of 256-bit chunks (8 words each).
Fredrik Lundh19af43d2001-07-02 16:58:38 +0000291
Serhiy Storchakac04fcd42014-10-31 13:34:06 +0200292 # Compression is normally good: in a typical charset, large ranges of
293 # Unicode will be either completely excluded (e.g. if only cyrillic
294 # letters are to be matched), or completely included (e.g. if large
295 # subranges of Kanji match). These ranges will be represented by
296 # chunks of all one-bits or all zero-bits.
Fredrik Lundh19af43d2001-07-02 16:58:38 +0000297
Serhiy Storchakac04fcd42014-10-31 13:34:06 +0200298 # Matching can be also done efficiently: the more significant byte of
299 # the Unicode character is an index into the chunk number, and the
300 # less significant byte is a bit index in the chunk (just like the
301 # CHARSET matching).
Fredrik Lundh19af43d2001-07-02 16:58:38 +0000302
Serhiy Storchakac04fcd42014-10-31 13:34:06 +0200303 # In UCS-4 mode, the BIGCHARSET opcode still supports only subsets
304 # of the basic multilingual plane; an efficient representation
305 # for all of Unicode has not yet been developed.
Fredrik Lundh19af43d2001-07-02 16:58:38 +0000306
Serhiy Storchakac04fcd42014-10-31 13:34:06 +0200307 charmap = bytes(charmap) # should be hashable
Fredrik Lundh19af43d2001-07-02 16:58:38 +0000308 comps = {}
Serhiy Storchakac04fcd42014-10-31 13:34:06 +0200309 mapping = bytearray(256)
Fredrik Lundh19af43d2001-07-02 16:58:38 +0000310 block = 0
Serhiy Storchakac04fcd42014-10-31 13:34:06 +0200311 data = bytearray()
312 for i in range(0, 65536, 256):
313 chunk = charmap[i: i + 256]
314 if chunk in comps:
315 mapping[i // 256] = comps[chunk]
316 else:
317 mapping[i // 256] = comps[chunk] = block
318 block += 1
319 data += chunk
320 data = _mk_bitmap(data)
321 data[0:0] = [block] + _bytes_to_codes(mapping)
322 out.append((BIGCHARSET, data))
323 out += tail
324 return out
325
Serhiy Storchakae9e54ae2014-10-31 13:53:21 +0200326def _fixup_range(lo, hi, ranges, fixup):
327 for i in map(fixup, range(lo, hi+1)):
328 for k, (lo, hi) in enumerate(ranges):
329 if i < lo:
330 if l == lo - 1:
331 ranges[k] = (i, hi)
332 else:
333 ranges.insert(k, (i, i))
334 break
335 elif i > hi:
336 if i == hi + 1:
337 ranges[k] = (lo, i)
338 break
339 else:
340 break
341 else:
342 ranges.append((i, i))
343
Serhiy Storchakac04fcd42014-10-31 13:34:06 +0200344_CODEBITS = _sre.CODESIZE * 8
345_BITS_TRANS = b'0' + b'1' * 255
346def _mk_bitmap(bits, _CODEBITS=_CODEBITS, _int=int):
347 s = bytes(bits).translate(_BITS_TRANS)[::-1]
348 return [_int(s[i - _CODEBITS: i], 2)
349 for i in range(len(s), 0, -_CODEBITS)]
350
351def _bytes_to_codes(b):
352 # Convert block indices to word array
353 import array
Martin v. Löwis7d9c6c72004-05-07 07:18:13 +0000354 if _sre.CODESIZE == 2:
Martin v. Löwis78e2f062003-04-19 12:56:08 +0000355 code = 'H'
356 else:
Martin v. Löwis7d9c6c72004-05-07 07:18:13 +0000357 code = 'I'
Serhiy Storchakac04fcd42014-10-31 13:34:06 +0200358 a = array.array(code, bytes(b))
359 assert a.itemsize == _sre.CODESIZE
360 assert len(a) * a.itemsize == len(b)
361 return a.tolist()
Fredrik Lundh19af43d2001-07-02 16:58:38 +0000362
Fredrik Lundhe1869832000-08-01 22:47:49 +0000363def _simple(av):
364 # check if av is a "simple" operator
365 lo, hi = av[2].getwidth()
Fredrik Lundhe1869832000-08-01 22:47:49 +0000366 return lo == hi == 1 and av[2][0][0] != SUBPATTERN
367
Fredrik Lundh29c08be2000-06-29 23:33:12 +0000368def _compile_info(code, pattern, flags):
369 # internal: compile an info block. in the current version,
Fredrik Lundh3562f112000-07-02 12:00:07 +0000370 # this contains min/max pattern width, and an optional literal
371 # prefix or a character map
Fredrik Lundh29c08be2000-06-29 23:33:12 +0000372 lo, hi = pattern.getwidth()
373 if lo == 0:
Fredrik Lundh90a07912000-06-30 07:50:59 +0000374 return # not worth it
Fredrik Lundh29c08be2000-06-29 23:33:12 +0000375 # look for a literal prefix
376 prefix = []
Raymond Hettinger01c9f8c2004-03-26 11:16:55 +0000377 prefixappend = prefix.append
Fredrik Lundh7898c3e2000-08-07 20:59:04 +0000378 prefix_skip = 0
Fredrik Lundh3562f112000-07-02 12:00:07 +0000379 charset = [] # not used
Raymond Hettinger01c9f8c2004-03-26 11:16:55 +0000380 charsetappend = charset.append
Fredrik Lundh29c08be2000-06-29 23:33:12 +0000381 if not (flags & SRE_FLAG_IGNORECASE):
Fredrik Lundh7898c3e2000-08-07 20:59:04 +0000382 # look for literal prefix
Fredrik Lundh90a07912000-06-30 07:50:59 +0000383 for op, av in pattern.data:
384 if op is LITERAL:
Fredrik Lundh7898c3e2000-08-07 20:59:04 +0000385 if len(prefix) == prefix_skip:
386 prefix_skip = prefix_skip + 1
Raymond Hettinger01c9f8c2004-03-26 11:16:55 +0000387 prefixappend(av)
Fredrik Lundh7898c3e2000-08-07 20:59:04 +0000388 elif op is SUBPATTERN and len(av[1]) == 1:
389 op, av = av[1][0]
390 if op is LITERAL:
Raymond Hettinger01c9f8c2004-03-26 11:16:55 +0000391 prefixappend(av)
Fredrik Lundh7898c3e2000-08-07 20:59:04 +0000392 else:
393 break
Fredrik Lundh90a07912000-06-30 07:50:59 +0000394 else:
395 break
Fredrik Lundh7898c3e2000-08-07 20:59:04 +0000396 # if no prefix, look for charset prefix
397 if not prefix and pattern.data:
398 op, av = pattern.data[0]
399 if op is SUBPATTERN and av[1]:
400 op, av = av[1][0]
401 if op is LITERAL:
Raymond Hettinger01c9f8c2004-03-26 11:16:55 +0000402 charsetappend((op, av))
Fredrik Lundh7898c3e2000-08-07 20:59:04 +0000403 elif op is BRANCH:
404 c = []
Raymond Hettinger01c9f8c2004-03-26 11:16:55 +0000405 cappend = c.append
Fredrik Lundh7898c3e2000-08-07 20:59:04 +0000406 for p in av[1]:
407 if not p:
408 break
409 op, av = p[0]
410 if op is LITERAL:
Raymond Hettinger01c9f8c2004-03-26 11:16:55 +0000411 cappend((op, av))
Fredrik Lundh7898c3e2000-08-07 20:59:04 +0000412 else:
413 break
414 else:
415 charset = c
416 elif op is BRANCH:
417 c = []
Raymond Hettinger01c9f8c2004-03-26 11:16:55 +0000418 cappend = c.append
Fredrik Lundh7898c3e2000-08-07 20:59:04 +0000419 for p in av[1]:
420 if not p:
421 break
422 op, av = p[0]
423 if op is LITERAL:
Raymond Hettinger01c9f8c2004-03-26 11:16:55 +0000424 cappend((op, av))
Fredrik Lundh7898c3e2000-08-07 20:59:04 +0000425 else:
426 break
427 else:
428 charset = c
429 elif op is IN:
430 charset = av
431## if prefix:
432## print "*** PREFIX", prefix, prefix_skip
433## if charset:
434## print "*** CHARSET", charset
Fredrik Lundh29c08be2000-06-29 23:33:12 +0000435 # add an info block
436 emit = code.append
437 emit(OPCODES[INFO])
438 skip = len(code); emit(0)
439 # literal flag
440 mask = 0
Fredrik Lundh3562f112000-07-02 12:00:07 +0000441 if prefix:
442 mask = SRE_INFO_PREFIX
Fredrik Lundh7898c3e2000-08-07 20:59:04 +0000443 if len(prefix) == prefix_skip == len(pattern.data):
Fredrik Lundh3562f112000-07-02 12:00:07 +0000444 mask = mask + SRE_INFO_LITERAL
445 elif charset:
446 mask = mask + SRE_INFO_CHARSET
Fredrik Lundh29c08be2000-06-29 23:33:12 +0000447 emit(mask)
448 # pattern length
Fredrik Lundh3562f112000-07-02 12:00:07 +0000449 if lo < MAXCODE:
450 emit(lo)
451 else:
452 emit(MAXCODE)
453 prefix = prefix[:MAXCODE]
454 if hi < MAXCODE:
Fredrik Lundh90a07912000-06-30 07:50:59 +0000455 emit(hi)
Fredrik Lundh29c08be2000-06-29 23:33:12 +0000456 else:
Fredrik Lundh90a07912000-06-30 07:50:59 +0000457 emit(0)
Fredrik Lundh29c08be2000-06-29 23:33:12 +0000458 # add literal prefix
Fredrik Lundh29c08be2000-06-29 23:33:12 +0000459 if prefix:
Fredrik Lundh7898c3e2000-08-07 20:59:04 +0000460 emit(len(prefix)) # length
461 emit(prefix_skip) # skip
462 code.extend(prefix)
463 # generate overlap table
464 table = [-1] + ([0]*len(prefix))
Raymond Hettinger01c9f8c2004-03-26 11:16:55 +0000465 for i in xrange(len(prefix)):
Fredrik Lundh7898c3e2000-08-07 20:59:04 +0000466 table[i+1] = table[i]+1
467 while table[i+1] > 0 and prefix[i] != prefix[table[i+1]-1]:
468 table[i+1] = table[table[i+1]-1]+1
469 code.extend(table[1:]) # don't store first entry
Fredrik Lundh3562f112000-07-02 12:00:07 +0000470 elif charset:
Guido van Rossum577fb5a2003-02-24 01:18:35 +0000471 _compile_charset(charset, flags, code)
Fredrik Lundh29c08be2000-06-29 23:33:12 +0000472 code[skip] = len(code) - skip
473
Fredrik Lundh8a3ebf82000-07-23 21:46:17 +0000474try:
Just van Rossum12723ba2003-07-02 20:03:04 +0000475 unicode
Fredrik Lundh8a3ebf82000-07-23 21:46:17 +0000476except NameError:
Just van Rossum74902502003-07-02 21:37:16 +0000477 STRING_TYPES = (type(""),)
Just van Rossum12723ba2003-07-02 20:03:04 +0000478else:
479 STRING_TYPES = (type(""), type(unicode("")))
Fredrik Lundh8a3ebf82000-07-23 21:46:17 +0000480
Just van Rossum74902502003-07-02 21:37:16 +0000481def isstring(obj):
482 for tp in STRING_TYPES:
483 if isinstance(obj, tp):
484 return 1
485 return 0
486
Fredrik Lundh2f2c67d2000-08-01 21:05:41 +0000487def _code(p, flags):
Fredrik Lundh29c08be2000-06-29 23:33:12 +0000488
Jeremy Hyltonb1aa1952000-06-01 17:39:12 +0000489 flags = p.pattern.flags | flags
Fredrik Lundhbe2211e2000-06-29 16:57:40 +0000490 code = []
Fredrik Lundh29c08be2000-06-29 23:33:12 +0000491
492 # compile info block
493 _compile_info(code, p, flags)
494
495 # compile the pattern
Jeremy Hyltonb1aa1952000-06-01 17:39:12 +0000496 _compile(code, p.data, flags)
Fredrik Lundh29c08be2000-06-29 23:33:12 +0000497
Jeremy Hyltonb1aa1952000-06-01 17:39:12 +0000498 code.append(OPCODES[SUCCESS])
Fredrik Lundh29c08be2000-06-29 23:33:12 +0000499
Fredrik Lundh29c4ba92000-08-01 18:20:07 +0000500 return code
501
502def compile(p, flags=0):
503 # internal: convert pattern list to internal format
504
Just van Rossum74902502003-07-02 21:37:16 +0000505 if isstring(p):
Fredrik Lundh29c4ba92000-08-01 18:20:07 +0000506 pattern = p
507 p = sre_parse.parse(p, flags)
508 else:
509 pattern = None
510
Fredrik Lundh2f2c67d2000-08-01 21:05:41 +0000511 code = _code(p, flags)
Fredrik Lundh29c4ba92000-08-01 18:20:07 +0000512
Fredrik Lundh8a3ebf82000-07-23 21:46:17 +0000513 # print code
514
Fredrik Lundh770617b2001-01-14 15:06:11 +0000515 # XXX: <fl> get rid of this limitation!
Fredrik Lundh5e7d51b2004-10-15 06:15:08 +0000516 if p.pattern.groups > 100:
517 raise AssertionError(
518 "sorry, but this version only supports 100 named groups"
519 )
Fredrik Lundh29c08be2000-06-29 23:33:12 +0000520
Fredrik Lundhc2301732000-07-02 22:25:39 +0000521 # map in either direction
522 groupindex = p.pattern.groupdict
523 indexgroup = [None] * p.pattern.groups
524 for k, i in groupindex.items():
525 indexgroup[i] = k
526
Jeremy Hyltonb1aa1952000-06-01 17:39:12 +0000527 return _sre.compile(
Guido van Rossumae04c332008-01-03 19:12:44 +0000528 pattern, flags | p.pattern.flags, code,
Fredrik Lundh6f013982000-07-03 18:44:21 +0000529 p.pattern.groups-1,
530 groupindex, indexgroup
Fredrik Lundh90a07912000-06-30 07:50:59 +0000531 )