blob: ee0882debf9d05f965af0bfd405988fd5ca7bfe8 [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
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
Fredrik Lundh7898c3e2000-08-07 20:59:04 +000024def _compile(code, pattern, flags):
25 # internal: compile a (sub)pattern
26 emit = code.append
27 for op, av in pattern:
28 if op in (LITERAL, NOT_LITERAL):
29 if flags & SRE_FLAG_IGNORECASE:
30 emit(OPCODES[OP_IGNORE[op]])
Fredrik Lundh2e240442001-01-15 18:28:14 +000031 emit(_sre.getlower(av, flags))
Fredrik Lundh7898c3e2000-08-07 20:59:04 +000032 else:
33 emit(OPCODES[op])
Fredrik Lundh2e240442001-01-15 18:28:14 +000034 emit(av)
Fredrik Lundh7898c3e2000-08-07 20:59:04 +000035 elif op is IN:
36 if flags & SRE_FLAG_IGNORECASE:
37 emit(OPCODES[OP_IGNORE[op]])
38 def fixup(literal, flags=flags):
39 return _sre.getlower(literal, flags)
40 else:
41 emit(OPCODES[op])
42 fixup = lambda x: x
43 skip = len(code); emit(0)
44 _compile_charset(av, flags, code, fixup)
45 code[skip] = len(code) - skip
46 elif op is ANY:
47 if flags & SRE_FLAG_DOTALL:
48 emit(OPCODES[ANY_ALL])
49 else:
50 emit(OPCODES[ANY])
51 elif op in (REPEAT, MIN_REPEAT, MAX_REPEAT):
52 if flags & SRE_FLAG_TEMPLATE:
53 raise error, "internal: unsupported template operator"
54 emit(OPCODES[REPEAT])
55 skip = len(code); emit(0)
56 emit(av[0])
57 emit(av[1])
58 _compile(code, av[2], flags)
59 emit(OPCODES[SUCCESS])
60 code[skip] = len(code) - skip
Guido van Rossum41c99e72003-04-14 17:59:34 +000061 elif _simple(av) and op != REPEAT:
62 if op == MAX_REPEAT:
63 emit(OPCODES[REPEAT_ONE])
64 else:
65 emit(OPCODES[MIN_REPEAT_ONE])
Fredrik Lundh7898c3e2000-08-07 20:59:04 +000066 skip = len(code); emit(0)
67 emit(av[0])
68 emit(av[1])
69 _compile(code, av[2], flags)
70 emit(OPCODES[SUCCESS])
71 code[skip] = len(code) - skip
72 else:
73 emit(OPCODES[REPEAT])
74 skip = len(code); emit(0)
75 emit(av[0])
76 emit(av[1])
77 _compile(code, av[2], flags)
78 code[skip] = len(code) - skip
79 if op == MAX_REPEAT:
80 emit(OPCODES[MAX_UNTIL])
81 else:
82 emit(OPCODES[MIN_UNTIL])
83 elif op is SUBPATTERN:
84 if av[0]:
85 emit(OPCODES[MARK])
86 emit((av[0]-1)*2)
87 # _compile_info(code, av[1], flags)
88 _compile(code, av[1], flags)
89 if av[0]:
90 emit(OPCODES[MARK])
91 emit((av[0]-1)*2+1)
92 elif op in (SUCCESS, FAILURE):
93 emit(OPCODES[op])
94 elif op in (ASSERT, ASSERT_NOT):
95 emit(OPCODES[op])
96 skip = len(code); emit(0)
97 if av[0] >= 0:
98 emit(0) # look ahead
99 else:
100 lo, hi = av[1].getwidth()
101 if lo != hi:
102 raise error, "look-behind requires fixed-width pattern"
103 emit(lo) # look behind
104 _compile(code, av[1], flags)
105 emit(OPCODES[SUCCESS])
106 code[skip] = len(code) - skip
107 elif op is CALL:
108 emit(OPCODES[op])
109 skip = len(code); emit(0)
110 _compile(code, av, flags)
111 emit(OPCODES[SUCCESS])
112 code[skip] = len(code) - skip
113 elif op is AT:
114 emit(OPCODES[op])
115 if flags & SRE_FLAG_MULTILINE:
Fredrik Lundhb25e1ad2001-03-22 15:50:10 +0000116 av = AT_MULTILINE.get(av, av)
117 if flags & SRE_FLAG_LOCALE:
118 av = AT_LOCALE.get(av, av)
119 elif flags & SRE_FLAG_UNICODE:
120 av = AT_UNICODE.get(av, av)
121 emit(ATCODES[av])
Fredrik Lundh7898c3e2000-08-07 20:59:04 +0000122 elif op is BRANCH:
123 emit(OPCODES[op])
124 tail = []
125 for av in av[1]:
126 skip = len(code); emit(0)
127 # _compile_info(code, av, flags)
128 _compile(code, av, flags)
129 emit(OPCODES[JUMP])
130 tail.append(len(code)); emit(0)
131 code[skip] = len(code) - skip
132 emit(0) # end of branch
133 for tail in tail:
134 code[tail] = len(code) - tail
135 elif op is CATEGORY:
136 emit(OPCODES[op])
137 if flags & SRE_FLAG_LOCALE:
Fredrik Lundhb25e1ad2001-03-22 15:50:10 +0000138 av = CH_LOCALE[av]
Fredrik Lundh7898c3e2000-08-07 20:59:04 +0000139 elif flags & SRE_FLAG_UNICODE:
Fredrik Lundhb25e1ad2001-03-22 15:50:10 +0000140 av = CH_UNICODE[av]
141 emit(CHCODES[av])
Fredrik Lundh7898c3e2000-08-07 20:59:04 +0000142 elif op is GROUPREF:
143 if flags & SRE_FLAG_IGNORECASE:
144 emit(OPCODES[OP_IGNORE[op]])
145 else:
146 emit(OPCODES[op])
147 emit(av-1)
Gustavo Niemeyerad3fc442003-10-17 22:13:16 +0000148 elif op is GROUPREF_EXISTS:
149 emit(OPCODES[op])
150 emit((av[0]-1)*2)
151 skipyes = len(code); emit(0)
152 _compile(code, av[1], flags)
153 if av[2]:
154 emit(OPCODES[JUMP])
155 skipno = len(code); emit(0)
156 code[skipyes] = len(code) - skipyes + 1
157 _compile(code, av[2], flags)
158 code[skipno] = len(code) - skipno
159 else:
160 code[skipyes] = len(code) - skipyes + 1
Fredrik Lundh7898c3e2000-08-07 20:59:04 +0000161 else:
162 raise ValueError, ("unsupported operand type", op)
163
164def _compile_charset(charset, flags, code, fixup=None):
165 # compile charset subprogram
166 emit = code.append
Raymond Hettingerf13eb552002-06-02 00:40:05 +0000167 if fixup is None:
Fredrik Lundh28552902000-07-05 21:14:16 +0000168 fixup = lambda x: x
Fredrik Lundh7898c3e2000-08-07 20:59:04 +0000169 for op, av in _optimize_charset(charset, fixup):
170 emit(OPCODES[op])
171 if op is NEGATE:
172 pass
173 elif op is LITERAL:
174 emit(fixup(av))
175 elif op is RANGE:
176 emit(fixup(av[0]))
177 emit(fixup(av[1]))
178 elif op is CHARSET:
179 code.extend(av)
Fredrik Lundh19af43d2001-07-02 16:58:38 +0000180 elif op is BIGCHARSET:
181 code.extend(av)
Fredrik Lundh7898c3e2000-08-07 20:59:04 +0000182 elif op is CATEGORY:
183 if flags & SRE_FLAG_LOCALE:
184 emit(CHCODES[CH_LOCALE[av]])
185 elif flags & SRE_FLAG_UNICODE:
186 emit(CHCODES[CH_UNICODE[av]])
187 else:
188 emit(CHCODES[av])
189 else:
190 raise error, "internal: unsupported set operator"
191 emit(OPCODES[FAILURE])
192
193def _optimize_charset(charset, fixup):
194 # internal: optimize character set
Fredrik Lundh3562f112000-07-02 12:00:07 +0000195 out = []
196 charmap = [0]*256
197 try:
198 for op, av in charset:
199 if op is NEGATE:
200 out.append((op, av))
201 elif op is LITERAL:
202 charmap[fixup(av)] = 1
203 elif op is RANGE:
204 for i in range(fixup(av[0]), fixup(av[1])+1):
205 charmap[i] = 1
206 elif op is CATEGORY:
Fredrik Lundh770617b2001-01-14 15:06:11 +0000207 # XXX: could append to charmap tail
Fredrik Lundh3562f112000-07-02 12:00:07 +0000208 return charset # cannot compress
209 except IndexError:
Fredrik Lundh7898c3e2000-08-07 20:59:04 +0000210 # character set contains unicode characters
Fredrik Lundh19af43d2001-07-02 16:58:38 +0000211 return _optimize_unicode(charset, fixup)
Fredrik Lundh3562f112000-07-02 12:00:07 +0000212 # compress character map
213 i = p = n = 0
214 runs = []
215 for c in charmap:
216 if c:
217 if n == 0:
218 p = i
219 n = n + 1
220 elif n:
221 runs.append((p, n))
222 n = 0
223 i = i + 1
224 if n:
225 runs.append((p, n))
226 if len(runs) <= 2:
227 # use literal/range
228 for p, n in runs:
229 if n == 1:
230 out.append((LITERAL, p))
231 else:
232 out.append((RANGE, (p, p+n-1)))
233 if len(out) < len(charset):
234 return out
235 else:
236 # use bitmap
Fredrik Lundh19af43d2001-07-02 16:58:38 +0000237 data = _mk_bitmap(charmap)
Fredrik Lundh3562f112000-07-02 12:00:07 +0000238 out.append((CHARSET, data))
239 return out
240 return charset
241
Fredrik Lundh19af43d2001-07-02 16:58:38 +0000242def _mk_bitmap(bits):
243 data = []
Martin v. Löwis78e2f062003-04-19 12:56:08 +0000244 if _sre.CODESIZE == 2:
245 start = (1, 0)
246 else:
247 start = (1L, 0L)
248 m, v = start
Fredrik Lundh19af43d2001-07-02 16:58:38 +0000249 for c in bits:
250 if c:
251 v = v + m
252 m = m << 1
253 if m > MAXCODE:
254 data.append(v)
Martin v. Löwis78e2f062003-04-19 12:56:08 +0000255 m, v = start
Fredrik Lundh19af43d2001-07-02 16:58:38 +0000256 return data
257
258# To represent a big charset, first a bitmap of all characters in the
259# set is constructed. Then, this bitmap is sliced into chunks of 256
260# characters, duplicate chunks are eliminitated, and each chunk is
261# given a number. In the compiled expression, the charset is
262# represented by a 16-bit word sequence, consisting of one word for
263# the number of different chunks, a sequence of 256 bytes (128 words)
264# of chunk numbers indexed by their original chunk position, and a
265# sequence of chunks (16 words each).
266
267# Compression is normally good: in a typical charset, large ranges of
268# Unicode will be either completely excluded (e.g. if only cyrillic
269# letters are to be matched), or completely included (e.g. if large
270# subranges of Kanji match). These ranges will be represented by
271# chunks of all one-bits or all zero-bits.
272
273# Matching can be also done efficiently: the more significant byte of
274# the Unicode character is an index into the chunk number, and the
275# less significant byte is a bit index in the chunk (just like the
276# CHARSET matching).
277
Martin v. Löwis78e2f062003-04-19 12:56:08 +0000278# In UCS-4 mode, the BIGCHARSET opcode still supports only subsets
279# of the basic multilingual plane; an efficient representation
280# for all of UTF-16 has not yet been developed. This means,
281# in particular, that negated charsets cannot be represented as
282# bigcharsets.
283
Fredrik Lundh19af43d2001-07-02 16:58:38 +0000284def _optimize_unicode(charset, fixup):
Martin v. Löwis78e2f062003-04-19 12:56:08 +0000285 try:
286 import array
287 except ImportError:
288 return charset
Fredrik Lundh19af43d2001-07-02 16:58:38 +0000289 charmap = [0]*65536
290 negate = 0
Martin v. Löwis78e2f062003-04-19 12:56:08 +0000291 try:
292 for op, av in charset:
293 if op is NEGATE:
294 negate = 1
295 elif op is LITERAL:
296 charmap[fixup(av)] = 1
297 elif op is RANGE:
298 for i in range(fixup(av[0]), fixup(av[1])+1):
299 charmap[i] = 1
300 elif op is CATEGORY:
301 # XXX: could expand category
302 return charset # cannot compress
303 except IndexError:
304 # non-BMP characters
305 return charset
Fredrik Lundh19af43d2001-07-02 16:58:38 +0000306 if negate:
Martin v. Löwis78e2f062003-04-19 12:56:08 +0000307 if sys.maxunicode != 65535:
308 # XXX: negation does not work with big charsets
309 return charset
Fredrik Lundh19af43d2001-07-02 16:58:38 +0000310 for i in range(65536):
311 charmap[i] = not charmap[i]
312 comps = {}
313 mapping = [0]*256
314 block = 0
315 data = []
316 for i in range(256):
317 chunk = tuple(charmap[i*256:(i+1)*256])
318 new = comps.setdefault(chunk, block)
319 mapping[i] = new
320 if new == block:
Fredrik Lundh4fb70272002-06-27 20:08:25 +0000321 block = block + 1
322 data = data + _mk_bitmap(chunk)
Fredrik Lundh19af43d2001-07-02 16:58:38 +0000323 header = [block]
Martin v. Löwis78e2f062003-04-19 12:56:08 +0000324 if MAXCODE == 65535:
325 code = 'H'
326 else:
327 code = 'L'
328 # Convert block indices to byte array of 256 bytes
329 mapping = array.array('b', mapping).tostring()
330 # Convert byte array to word array
331 header = header + array.array(code, mapping).tolist()
Fredrik Lundh19af43d2001-07-02 16:58:38 +0000332 data[0:0] = header
Tim Peters87cc0c32001-07-21 01:41:30 +0000333 return [(BIGCHARSET, data)]
Fredrik Lundh19af43d2001-07-02 16:58:38 +0000334
Fredrik Lundhe1869832000-08-01 22:47:49 +0000335def _simple(av):
336 # check if av is a "simple" operator
337 lo, hi = av[2].getwidth()
Fredrik Lundh13ac9922000-10-07 17:38:23 +0000338 if lo == 0 and hi == MAXREPEAT:
Fredrik Lundhe1869832000-08-01 22:47:49 +0000339 raise error, "nothing to repeat"
340 return lo == hi == 1 and av[2][0][0] != SUBPATTERN
341
Fredrik Lundh29c08be2000-06-29 23:33:12 +0000342def _compile_info(code, pattern, flags):
343 # internal: compile an info block. in the current version,
Fredrik Lundh3562f112000-07-02 12:00:07 +0000344 # this contains min/max pattern width, and an optional literal
345 # prefix or a character map
Fredrik Lundh29c08be2000-06-29 23:33:12 +0000346 lo, hi = pattern.getwidth()
347 if lo == 0:
Fredrik Lundh90a07912000-06-30 07:50:59 +0000348 return # not worth it
Fredrik Lundh29c08be2000-06-29 23:33:12 +0000349 # look for a literal prefix
350 prefix = []
Fredrik Lundh7898c3e2000-08-07 20:59:04 +0000351 prefix_skip = 0
Fredrik Lundh3562f112000-07-02 12:00:07 +0000352 charset = [] # not used
Fredrik Lundh29c08be2000-06-29 23:33:12 +0000353 if not (flags & SRE_FLAG_IGNORECASE):
Fredrik Lundh7898c3e2000-08-07 20:59:04 +0000354 # look for literal prefix
Fredrik Lundh90a07912000-06-30 07:50:59 +0000355 for op, av in pattern.data:
356 if op is LITERAL:
Fredrik Lundh7898c3e2000-08-07 20:59:04 +0000357 if len(prefix) == prefix_skip:
358 prefix_skip = prefix_skip + 1
Fredrik Lundh0640e112000-06-30 13:55:15 +0000359 prefix.append(av)
Fredrik Lundh7898c3e2000-08-07 20:59:04 +0000360 elif op is SUBPATTERN and len(av[1]) == 1:
361 op, av = av[1][0]
362 if op is LITERAL:
363 prefix.append(av)
364 else:
365 break
Fredrik Lundh90a07912000-06-30 07:50:59 +0000366 else:
367 break
Fredrik Lundh7898c3e2000-08-07 20:59:04 +0000368 # if no prefix, look for charset prefix
369 if not prefix and pattern.data:
370 op, av = pattern.data[0]
371 if op is SUBPATTERN and av[1]:
372 op, av = av[1][0]
373 if op is LITERAL:
374 charset.append((op, av))
375 elif op is BRANCH:
376 c = []
377 for p in av[1]:
378 if not p:
379 break
380 op, av = p[0]
381 if op is LITERAL:
382 c.append((op, av))
383 else:
384 break
385 else:
386 charset = c
387 elif op is BRANCH:
388 c = []
389 for p in av[1]:
390 if not p:
391 break
392 op, av = p[0]
393 if op is LITERAL:
394 c.append((op, av))
395 else:
396 break
397 else:
398 charset = c
399 elif op is IN:
400 charset = av
401## if prefix:
402## print "*** PREFIX", prefix, prefix_skip
403## if charset:
404## print "*** CHARSET", charset
Fredrik Lundh29c08be2000-06-29 23:33:12 +0000405 # add an info block
406 emit = code.append
407 emit(OPCODES[INFO])
408 skip = len(code); emit(0)
409 # literal flag
410 mask = 0
Fredrik Lundh3562f112000-07-02 12:00:07 +0000411 if prefix:
412 mask = SRE_INFO_PREFIX
Fredrik Lundh7898c3e2000-08-07 20:59:04 +0000413 if len(prefix) == prefix_skip == len(pattern.data):
Fredrik Lundh3562f112000-07-02 12:00:07 +0000414 mask = mask + SRE_INFO_LITERAL
415 elif charset:
416 mask = mask + SRE_INFO_CHARSET
Fredrik Lundh29c08be2000-06-29 23:33:12 +0000417 emit(mask)
418 # pattern length
Fredrik Lundh3562f112000-07-02 12:00:07 +0000419 if lo < MAXCODE:
420 emit(lo)
421 else:
422 emit(MAXCODE)
423 prefix = prefix[:MAXCODE]
424 if hi < MAXCODE:
Fredrik Lundh90a07912000-06-30 07:50:59 +0000425 emit(hi)
Fredrik Lundh29c08be2000-06-29 23:33:12 +0000426 else:
Fredrik Lundh90a07912000-06-30 07:50:59 +0000427 emit(0)
Fredrik Lundh29c08be2000-06-29 23:33:12 +0000428 # add literal prefix
Fredrik Lundh29c08be2000-06-29 23:33:12 +0000429 if prefix:
Fredrik Lundh7898c3e2000-08-07 20:59:04 +0000430 emit(len(prefix)) # length
431 emit(prefix_skip) # skip
432 code.extend(prefix)
433 # generate overlap table
434 table = [-1] + ([0]*len(prefix))
435 for i in range(len(prefix)):
436 table[i+1] = table[i]+1
437 while table[i+1] > 0 and prefix[i] != prefix[table[i+1]-1]:
438 table[i+1] = table[table[i+1]-1]+1
439 code.extend(table[1:]) # don't store first entry
Fredrik Lundh3562f112000-07-02 12:00:07 +0000440 elif charset:
Guido van Rossum577fb5a2003-02-24 01:18:35 +0000441 _compile_charset(charset, flags, code)
Fredrik Lundh29c08be2000-06-29 23:33:12 +0000442 code[skip] = len(code) - skip
443
Fredrik Lundh8a3ebf82000-07-23 21:46:17 +0000444try:
Just van Rossum12723ba2003-07-02 20:03:04 +0000445 unicode
Fredrik Lundh8a3ebf82000-07-23 21:46:17 +0000446except NameError:
Just van Rossum74902502003-07-02 21:37:16 +0000447 STRING_TYPES = (type(""),)
Just van Rossum12723ba2003-07-02 20:03:04 +0000448else:
449 STRING_TYPES = (type(""), type(unicode("")))
Fredrik Lundh8a3ebf82000-07-23 21:46:17 +0000450
Just van Rossum74902502003-07-02 21:37:16 +0000451def isstring(obj):
452 for tp in STRING_TYPES:
453 if isinstance(obj, tp):
454 return 1
455 return 0
456
Fredrik Lundh2f2c67d2000-08-01 21:05:41 +0000457def _code(p, flags):
Fredrik Lundh29c08be2000-06-29 23:33:12 +0000458
Jeremy Hyltonb1aa1952000-06-01 17:39:12 +0000459 flags = p.pattern.flags | flags
Fredrik Lundhbe2211e2000-06-29 16:57:40 +0000460 code = []
Fredrik Lundh29c08be2000-06-29 23:33:12 +0000461
462 # compile info block
463 _compile_info(code, p, flags)
464
465 # compile the pattern
Jeremy Hyltonb1aa1952000-06-01 17:39:12 +0000466 _compile(code, p.data, flags)
Fredrik Lundh29c08be2000-06-29 23:33:12 +0000467
Jeremy Hyltonb1aa1952000-06-01 17:39:12 +0000468 code.append(OPCODES[SUCCESS])
Fredrik Lundh29c08be2000-06-29 23:33:12 +0000469
Fredrik Lundh29c4ba92000-08-01 18:20:07 +0000470 return code
471
472def compile(p, flags=0):
473 # internal: convert pattern list to internal format
474
Just van Rossum74902502003-07-02 21:37:16 +0000475 if isstring(p):
Fredrik Lundh29c4ba92000-08-01 18:20:07 +0000476 import sre_parse
477 pattern = p
478 p = sre_parse.parse(p, flags)
479 else:
480 pattern = None
481
Fredrik Lundh2f2c67d2000-08-01 21:05:41 +0000482 code = _code(p, flags)
Fredrik Lundh29c4ba92000-08-01 18:20:07 +0000483
Fredrik Lundh8a3ebf82000-07-23 21:46:17 +0000484 # print code
485
Fredrik Lundh770617b2001-01-14 15:06:11 +0000486 # XXX: <fl> get rid of this limitation!
Fredrik Lundhbe2211e2000-06-29 16:57:40 +0000487 assert p.pattern.groups <= 100,\
Fredrik Lundh90a07912000-06-30 07:50:59 +0000488 "sorry, but this version only supports 100 named groups"
Fredrik Lundh29c08be2000-06-29 23:33:12 +0000489
Fredrik Lundhc2301732000-07-02 22:25:39 +0000490 # map in either direction
491 groupindex = p.pattern.groupdict
492 indexgroup = [None] * p.pattern.groups
493 for k, i in groupindex.items():
494 indexgroup[i] = k
495
Jeremy Hyltonb1aa1952000-06-01 17:39:12 +0000496 return _sre.compile(
Fredrik Lundh6f013982000-07-03 18:44:21 +0000497 pattern, flags, code,
498 p.pattern.groups-1,
499 groupindex, indexgroup
Fredrik Lundh90a07912000-06-30 07:50:59 +0000500 )