blob: 3e54819bc9d5f9cb671ef945281a840bdf9df524 [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
Guido van Rossum7627c0d2000-03-31 14:58:54 +000014
15from sre_constants import *
16
Fredrik Lundhb35ffc02001-01-15 12:46:09 +000017assert _sre.MAGIC == MAGIC, "SRE module mismatch"
18
Fredrik Lundh3562f112000-07-02 12:00:07 +000019MAXCODE = 65535
20
Fredrik Lundh7898c3e2000-08-07 20:59:04 +000021def _compile(code, pattern, flags):
22 # internal: compile a (sub)pattern
23 emit = code.append
24 for op, av in pattern:
25 if op in (LITERAL, NOT_LITERAL):
26 if flags & SRE_FLAG_IGNORECASE:
27 emit(OPCODES[OP_IGNORE[op]])
Fredrik Lundh2e240442001-01-15 18:28:14 +000028 emit(_sre.getlower(av, flags))
Fredrik Lundh7898c3e2000-08-07 20:59:04 +000029 else:
30 emit(OPCODES[op])
Fredrik Lundh2e240442001-01-15 18:28:14 +000031 emit(av)
Fredrik Lundh7898c3e2000-08-07 20:59:04 +000032 elif op is IN:
33 if flags & SRE_FLAG_IGNORECASE:
34 emit(OPCODES[OP_IGNORE[op]])
35 def fixup(literal, flags=flags):
36 return _sre.getlower(literal, flags)
37 else:
38 emit(OPCODES[op])
39 fixup = lambda x: x
40 skip = len(code); emit(0)
41 _compile_charset(av, flags, code, fixup)
42 code[skip] = len(code) - skip
43 elif op is ANY:
44 if flags & SRE_FLAG_DOTALL:
45 emit(OPCODES[ANY_ALL])
46 else:
47 emit(OPCODES[ANY])
48 elif op in (REPEAT, MIN_REPEAT, MAX_REPEAT):
49 if flags & SRE_FLAG_TEMPLATE:
50 raise error, "internal: unsupported template operator"
51 emit(OPCODES[REPEAT])
52 skip = len(code); emit(0)
53 emit(av[0])
54 emit(av[1])
55 _compile(code, av[2], flags)
56 emit(OPCODES[SUCCESS])
57 code[skip] = len(code) - skip
Guido van Rossum41c99e72003-04-14 17:59:34 +000058 elif _simple(av) and op != REPEAT:
59 if op == MAX_REPEAT:
60 emit(OPCODES[REPEAT_ONE])
61 else:
62 emit(OPCODES[MIN_REPEAT_ONE])
Fredrik Lundh7898c3e2000-08-07 20:59:04 +000063 skip = len(code); emit(0)
64 emit(av[0])
65 emit(av[1])
66 _compile(code, av[2], flags)
67 emit(OPCODES[SUCCESS])
68 code[skip] = len(code) - skip
69 else:
70 emit(OPCODES[REPEAT])
71 skip = len(code); emit(0)
72 emit(av[0])
73 emit(av[1])
74 _compile(code, av[2], flags)
75 code[skip] = len(code) - skip
76 if op == MAX_REPEAT:
77 emit(OPCODES[MAX_UNTIL])
78 else:
79 emit(OPCODES[MIN_UNTIL])
80 elif op is SUBPATTERN:
81 if av[0]:
82 emit(OPCODES[MARK])
83 emit((av[0]-1)*2)
84 # _compile_info(code, av[1], flags)
85 _compile(code, av[1], flags)
86 if av[0]:
87 emit(OPCODES[MARK])
88 emit((av[0]-1)*2+1)
89 elif op in (SUCCESS, FAILURE):
90 emit(OPCODES[op])
91 elif op in (ASSERT, ASSERT_NOT):
92 emit(OPCODES[op])
93 skip = len(code); emit(0)
94 if av[0] >= 0:
95 emit(0) # look ahead
96 else:
97 lo, hi = av[1].getwidth()
98 if lo != hi:
99 raise error, "look-behind requires fixed-width pattern"
100 emit(lo) # look behind
101 _compile(code, av[1], flags)
102 emit(OPCODES[SUCCESS])
103 code[skip] = len(code) - skip
104 elif op is CALL:
105 emit(OPCODES[op])
106 skip = len(code); emit(0)
107 _compile(code, av, flags)
108 emit(OPCODES[SUCCESS])
109 code[skip] = len(code) - skip
110 elif op is AT:
111 emit(OPCODES[op])
112 if flags & SRE_FLAG_MULTILINE:
Fredrik Lundhb25e1ad2001-03-22 15:50:10 +0000113 av = AT_MULTILINE.get(av, av)
114 if flags & SRE_FLAG_LOCALE:
115 av = AT_LOCALE.get(av, av)
116 elif flags & SRE_FLAG_UNICODE:
117 av = AT_UNICODE.get(av, av)
118 emit(ATCODES[av])
Fredrik Lundh7898c3e2000-08-07 20:59:04 +0000119 elif op is BRANCH:
120 emit(OPCODES[op])
121 tail = []
122 for av in av[1]:
123 skip = len(code); emit(0)
124 # _compile_info(code, av, flags)
125 _compile(code, av, flags)
126 emit(OPCODES[JUMP])
127 tail.append(len(code)); emit(0)
128 code[skip] = len(code) - skip
129 emit(0) # end of branch
130 for tail in tail:
131 code[tail] = len(code) - tail
132 elif op is CATEGORY:
133 emit(OPCODES[op])
134 if flags & SRE_FLAG_LOCALE:
Fredrik Lundhb25e1ad2001-03-22 15:50:10 +0000135 av = CH_LOCALE[av]
Fredrik Lundh7898c3e2000-08-07 20:59:04 +0000136 elif flags & SRE_FLAG_UNICODE:
Fredrik Lundhb25e1ad2001-03-22 15:50:10 +0000137 av = CH_UNICODE[av]
138 emit(CHCODES[av])
Fredrik Lundh7898c3e2000-08-07 20:59:04 +0000139 elif op is GROUPREF:
140 if flags & SRE_FLAG_IGNORECASE:
141 emit(OPCODES[OP_IGNORE[op]])
142 else:
143 emit(OPCODES[op])
144 emit(av-1)
145 else:
146 raise ValueError, ("unsupported operand type", op)
147
148def _compile_charset(charset, flags, code, fixup=None):
149 # compile charset subprogram
150 emit = code.append
Raymond Hettingerf13eb552002-06-02 00:40:05 +0000151 if fixup is None:
Fredrik Lundh28552902000-07-05 21:14:16 +0000152 fixup = lambda x: x
Fredrik Lundh7898c3e2000-08-07 20:59:04 +0000153 for op, av in _optimize_charset(charset, fixup):
154 emit(OPCODES[op])
155 if op is NEGATE:
156 pass
157 elif op is LITERAL:
158 emit(fixup(av))
159 elif op is RANGE:
160 emit(fixup(av[0]))
161 emit(fixup(av[1]))
162 elif op is CHARSET:
163 code.extend(av)
Fredrik Lundh19af43d2001-07-02 16:58:38 +0000164 elif op is BIGCHARSET:
165 code.extend(av)
Fredrik Lundh7898c3e2000-08-07 20:59:04 +0000166 elif op is CATEGORY:
167 if flags & SRE_FLAG_LOCALE:
168 emit(CHCODES[CH_LOCALE[av]])
169 elif flags & SRE_FLAG_UNICODE:
170 emit(CHCODES[CH_UNICODE[av]])
171 else:
172 emit(CHCODES[av])
173 else:
174 raise error, "internal: unsupported set operator"
175 emit(OPCODES[FAILURE])
176
177def _optimize_charset(charset, fixup):
178 # internal: optimize character set
Fredrik Lundh3562f112000-07-02 12:00:07 +0000179 out = []
180 charmap = [0]*256
181 try:
182 for op, av in charset:
183 if op is NEGATE:
184 out.append((op, av))
185 elif op is LITERAL:
186 charmap[fixup(av)] = 1
187 elif op is RANGE:
188 for i in range(fixup(av[0]), fixup(av[1])+1):
189 charmap[i] = 1
190 elif op is CATEGORY:
Fredrik Lundh770617b2001-01-14 15:06:11 +0000191 # XXX: could append to charmap tail
Fredrik Lundh3562f112000-07-02 12:00:07 +0000192 return charset # cannot compress
193 except IndexError:
Martin v. Löwis67c4cb12002-09-26 16:39:20 +0000194 if sys.maxunicode != 65535:
195 # XXX: big charsets don't work in UCS-4 builds
196 return charset
Fredrik Lundh7898c3e2000-08-07 20:59:04 +0000197 # character set contains unicode characters
Fredrik Lundh19af43d2001-07-02 16:58:38 +0000198 return _optimize_unicode(charset, fixup)
Fredrik Lundh3562f112000-07-02 12:00:07 +0000199 # compress character map
200 i = p = n = 0
201 runs = []
202 for c in charmap:
203 if c:
204 if n == 0:
205 p = i
206 n = n + 1
207 elif n:
208 runs.append((p, n))
209 n = 0
210 i = i + 1
211 if n:
212 runs.append((p, n))
213 if len(runs) <= 2:
214 # use literal/range
215 for p, n in runs:
216 if n == 1:
217 out.append((LITERAL, p))
218 else:
219 out.append((RANGE, (p, p+n-1)))
220 if len(out) < len(charset):
221 return out
222 else:
223 # use bitmap
Fredrik Lundh19af43d2001-07-02 16:58:38 +0000224 data = _mk_bitmap(charmap)
Fredrik Lundh3562f112000-07-02 12:00:07 +0000225 out.append((CHARSET, data))
226 return out
227 return charset
228
Fredrik Lundh19af43d2001-07-02 16:58:38 +0000229def _mk_bitmap(bits):
230 data = []
231 m = 1; v = 0
232 for c in bits:
233 if c:
234 v = v + m
235 m = m << 1
236 if m > MAXCODE:
237 data.append(v)
238 m = 1; v = 0
239 return data
240
241# To represent a big charset, first a bitmap of all characters in the
242# set is constructed. Then, this bitmap is sliced into chunks of 256
243# characters, duplicate chunks are eliminitated, and each chunk is
244# given a number. In the compiled expression, the charset is
245# represented by a 16-bit word sequence, consisting of one word for
246# the number of different chunks, a sequence of 256 bytes (128 words)
247# of chunk numbers indexed by their original chunk position, and a
248# sequence of chunks (16 words each).
249
250# Compression is normally good: in a typical charset, large ranges of
251# Unicode will be either completely excluded (e.g. if only cyrillic
252# letters are to be matched), or completely included (e.g. if large
253# subranges of Kanji match). These ranges will be represented by
254# chunks of all one-bits or all zero-bits.
255
256# Matching can be also done efficiently: the more significant byte of
257# the Unicode character is an index into the chunk number, and the
258# less significant byte is a bit index in the chunk (just like the
259# CHARSET matching).
260
261def _optimize_unicode(charset, fixup):
262 charmap = [0]*65536
263 negate = 0
264 for op, av in charset:
265 if op is NEGATE:
266 negate = 1
267 elif op is LITERAL:
268 charmap[fixup(av)] = 1
269 elif op is RANGE:
270 for i in range(fixup(av[0]), fixup(av[1])+1):
271 charmap[i] = 1
272 elif op is CATEGORY:
273 # XXX: could expand category
274 return charset # cannot compress
275 if negate:
276 for i in range(65536):
277 charmap[i] = not charmap[i]
278 comps = {}
279 mapping = [0]*256
280 block = 0
281 data = []
282 for i in range(256):
283 chunk = tuple(charmap[i*256:(i+1)*256])
284 new = comps.setdefault(chunk, block)
285 mapping[i] = new
286 if new == block:
Fredrik Lundh4fb70272002-06-27 20:08:25 +0000287 block = block + 1
288 data = data + _mk_bitmap(chunk)
Fredrik Lundh19af43d2001-07-02 16:58:38 +0000289 header = [block]
290 assert MAXCODE == 65535
291 for i in range(128):
Martin v. Löwis3550dd32001-07-19 14:26:10 +0000292 if sys.byteorder == 'big':
293 header.append(256*mapping[2*i]+mapping[2*i+1])
294 else:
295 header.append(mapping[2*i]+256*mapping[2*i+1])
Fredrik Lundh19af43d2001-07-02 16:58:38 +0000296 data[0:0] = header
Tim Peters87cc0c32001-07-21 01:41:30 +0000297 return [(BIGCHARSET, data)]
Fredrik Lundh19af43d2001-07-02 16:58:38 +0000298
Fredrik Lundhe1869832000-08-01 22:47:49 +0000299def _simple(av):
300 # check if av is a "simple" operator
301 lo, hi = av[2].getwidth()
Fredrik Lundh13ac9922000-10-07 17:38:23 +0000302 if lo == 0 and hi == MAXREPEAT:
Fredrik Lundhe1869832000-08-01 22:47:49 +0000303 raise error, "nothing to repeat"
304 return lo == hi == 1 and av[2][0][0] != SUBPATTERN
305
Fredrik Lundh29c08be2000-06-29 23:33:12 +0000306def _compile_info(code, pattern, flags):
307 # internal: compile an info block. in the current version,
Fredrik Lundh3562f112000-07-02 12:00:07 +0000308 # this contains min/max pattern width, and an optional literal
309 # prefix or a character map
Fredrik Lundh29c08be2000-06-29 23:33:12 +0000310 lo, hi = pattern.getwidth()
311 if lo == 0:
Fredrik Lundh90a07912000-06-30 07:50:59 +0000312 return # not worth it
Fredrik Lundh29c08be2000-06-29 23:33:12 +0000313 # look for a literal prefix
314 prefix = []
Fredrik Lundh7898c3e2000-08-07 20:59:04 +0000315 prefix_skip = 0
Fredrik Lundh3562f112000-07-02 12:00:07 +0000316 charset = [] # not used
Fredrik Lundh29c08be2000-06-29 23:33:12 +0000317 if not (flags & SRE_FLAG_IGNORECASE):
Fredrik Lundh7898c3e2000-08-07 20:59:04 +0000318 # look for literal prefix
Fredrik Lundh90a07912000-06-30 07:50:59 +0000319 for op, av in pattern.data:
320 if op is LITERAL:
Fredrik Lundh7898c3e2000-08-07 20:59:04 +0000321 if len(prefix) == prefix_skip:
322 prefix_skip = prefix_skip + 1
Fredrik Lundh0640e112000-06-30 13:55:15 +0000323 prefix.append(av)
Fredrik Lundh7898c3e2000-08-07 20:59:04 +0000324 elif op is SUBPATTERN and len(av[1]) == 1:
325 op, av = av[1][0]
326 if op is LITERAL:
327 prefix.append(av)
328 else:
329 break
Fredrik Lundh90a07912000-06-30 07:50:59 +0000330 else:
331 break
Fredrik Lundh7898c3e2000-08-07 20:59:04 +0000332 # if no prefix, look for charset prefix
333 if not prefix and pattern.data:
334 op, av = pattern.data[0]
335 if op is SUBPATTERN and av[1]:
336 op, av = av[1][0]
337 if op is LITERAL:
338 charset.append((op, av))
339 elif op is BRANCH:
340 c = []
341 for p in av[1]:
342 if not p:
343 break
344 op, av = p[0]
345 if op is LITERAL:
346 c.append((op, av))
347 else:
348 break
349 else:
350 charset = c
351 elif op is BRANCH:
352 c = []
353 for p in av[1]:
354 if not p:
355 break
356 op, av = p[0]
357 if op is LITERAL:
358 c.append((op, av))
359 else:
360 break
361 else:
362 charset = c
363 elif op is IN:
364 charset = av
365## if prefix:
366## print "*** PREFIX", prefix, prefix_skip
367## if charset:
368## print "*** CHARSET", charset
Fredrik Lundh29c08be2000-06-29 23:33:12 +0000369 # add an info block
370 emit = code.append
371 emit(OPCODES[INFO])
372 skip = len(code); emit(0)
373 # literal flag
374 mask = 0
Fredrik Lundh3562f112000-07-02 12:00:07 +0000375 if prefix:
376 mask = SRE_INFO_PREFIX
Fredrik Lundh7898c3e2000-08-07 20:59:04 +0000377 if len(prefix) == prefix_skip == len(pattern.data):
Fredrik Lundh3562f112000-07-02 12:00:07 +0000378 mask = mask + SRE_INFO_LITERAL
379 elif charset:
380 mask = mask + SRE_INFO_CHARSET
Fredrik Lundh29c08be2000-06-29 23:33:12 +0000381 emit(mask)
382 # pattern length
Fredrik Lundh3562f112000-07-02 12:00:07 +0000383 if lo < MAXCODE:
384 emit(lo)
385 else:
386 emit(MAXCODE)
387 prefix = prefix[:MAXCODE]
388 if hi < MAXCODE:
Fredrik Lundh90a07912000-06-30 07:50:59 +0000389 emit(hi)
Fredrik Lundh29c08be2000-06-29 23:33:12 +0000390 else:
Fredrik Lundh90a07912000-06-30 07:50:59 +0000391 emit(0)
Fredrik Lundh29c08be2000-06-29 23:33:12 +0000392 # add literal prefix
Fredrik Lundh29c08be2000-06-29 23:33:12 +0000393 if prefix:
Fredrik Lundh7898c3e2000-08-07 20:59:04 +0000394 emit(len(prefix)) # length
395 emit(prefix_skip) # skip
396 code.extend(prefix)
397 # generate overlap table
398 table = [-1] + ([0]*len(prefix))
399 for i in range(len(prefix)):
400 table[i+1] = table[i]+1
401 while table[i+1] > 0 and prefix[i] != prefix[table[i+1]-1]:
402 table[i+1] = table[table[i+1]-1]+1
403 code.extend(table[1:]) # don't store first entry
Fredrik Lundh3562f112000-07-02 12:00:07 +0000404 elif charset:
Guido van Rossum577fb5a2003-02-24 01:18:35 +0000405 _compile_charset(charset, flags, code)
Fredrik Lundh29c08be2000-06-29 23:33:12 +0000406 code[skip] = len(code) - skip
407
Fredrik Lundh8a3ebf82000-07-23 21:46:17 +0000408STRING_TYPES = [type("")]
409
410try:
411 STRING_TYPES.append(type(unicode("")))
412except NameError:
413 pass
414
Fredrik Lundh2f2c67d2000-08-01 21:05:41 +0000415def _code(p, flags):
Fredrik Lundh29c08be2000-06-29 23:33:12 +0000416
Jeremy Hyltonb1aa1952000-06-01 17:39:12 +0000417 flags = p.pattern.flags | flags
Fredrik Lundhbe2211e2000-06-29 16:57:40 +0000418 code = []
Fredrik Lundh29c08be2000-06-29 23:33:12 +0000419
420 # compile info block
421 _compile_info(code, p, flags)
422
423 # compile the pattern
Jeremy Hyltonb1aa1952000-06-01 17:39:12 +0000424 _compile(code, p.data, flags)
Fredrik Lundh29c08be2000-06-29 23:33:12 +0000425
Jeremy Hyltonb1aa1952000-06-01 17:39:12 +0000426 code.append(OPCODES[SUCCESS])
Fredrik Lundh29c08be2000-06-29 23:33:12 +0000427
Fredrik Lundh29c4ba92000-08-01 18:20:07 +0000428 return code
429
430def compile(p, flags=0):
431 # internal: convert pattern list to internal format
432
433 if type(p) in STRING_TYPES:
434 import sre_parse
435 pattern = p
436 p = sre_parse.parse(p, flags)
437 else:
438 pattern = None
439
Fredrik Lundh2f2c67d2000-08-01 21:05:41 +0000440 code = _code(p, flags)
Fredrik Lundh29c4ba92000-08-01 18:20:07 +0000441
Fredrik Lundh8a3ebf82000-07-23 21:46:17 +0000442 # print code
443
Fredrik Lundh770617b2001-01-14 15:06:11 +0000444 # XXX: <fl> get rid of this limitation!
Fredrik Lundhbe2211e2000-06-29 16:57:40 +0000445 assert p.pattern.groups <= 100,\
Fredrik Lundh90a07912000-06-30 07:50:59 +0000446 "sorry, but this version only supports 100 named groups"
Fredrik Lundh29c08be2000-06-29 23:33:12 +0000447
Fredrik Lundhc2301732000-07-02 22:25:39 +0000448 # map in either direction
449 groupindex = p.pattern.groupdict
450 indexgroup = [None] * p.pattern.groups
451 for k, i in groupindex.items():
452 indexgroup[i] = k
453
Jeremy Hyltonb1aa1952000-06-01 17:39:12 +0000454 return _sre.compile(
Fredrik Lundh6f013982000-07-03 18:44:21 +0000455 pattern, flags, code,
456 p.pattern.groups-1,
457 groupindex, indexgroup
Fredrik Lundh90a07912000-06-30 07:50:59 +0000458 )