blob: 4dc11a1e4983dc108f77b6f43c4adb4ae274777a [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
Martin v. Löwis3550dd32001-07-19 14:26:10 +000011import _sre,sys
Guido van Rossum7627c0d2000-03-31 14:58:54 +000012
13from sre_constants import *
14
Fredrik Lundhb35ffc02001-01-15 12:46:09 +000015assert _sre.MAGIC == MAGIC, "SRE module mismatch"
16
Fredrik Lundh3562f112000-07-02 12:00:07 +000017MAXCODE = 65535
18
Fredrik Lundh7898c3e2000-08-07 20:59:04 +000019def _compile(code, pattern, flags):
20 # internal: compile a (sub)pattern
21 emit = code.append
22 for op, av in pattern:
23 if op in (LITERAL, NOT_LITERAL):
24 if flags & SRE_FLAG_IGNORECASE:
25 emit(OPCODES[OP_IGNORE[op]])
Fredrik Lundh2e240442001-01-15 18:28:14 +000026 emit(_sre.getlower(av, flags))
Fredrik Lundh7898c3e2000-08-07 20:59:04 +000027 else:
28 emit(OPCODES[op])
Fredrik Lundh2e240442001-01-15 18:28:14 +000029 emit(av)
Fredrik Lundh7898c3e2000-08-07 20:59:04 +000030 elif op is IN:
31 if flags & SRE_FLAG_IGNORECASE:
32 emit(OPCODES[OP_IGNORE[op]])
33 def fixup(literal, flags=flags):
34 return _sre.getlower(literal, flags)
35 else:
36 emit(OPCODES[op])
37 fixup = lambda x: x
38 skip = len(code); emit(0)
39 _compile_charset(av, flags, code, fixup)
40 code[skip] = len(code) - skip
41 elif op is ANY:
42 if flags & SRE_FLAG_DOTALL:
43 emit(OPCODES[ANY_ALL])
44 else:
45 emit(OPCODES[ANY])
46 elif op in (REPEAT, MIN_REPEAT, MAX_REPEAT):
47 if flags & SRE_FLAG_TEMPLATE:
48 raise error, "internal: unsupported template operator"
49 emit(OPCODES[REPEAT])
50 skip = len(code); emit(0)
51 emit(av[0])
52 emit(av[1])
53 _compile(code, av[2], flags)
54 emit(OPCODES[SUCCESS])
55 code[skip] = len(code) - skip
56 elif _simple(av) and op == MAX_REPEAT:
57 emit(OPCODES[REPEAT_ONE])
58 skip = len(code); emit(0)
59 emit(av[0])
60 emit(av[1])
61 _compile(code, av[2], flags)
62 emit(OPCODES[SUCCESS])
63 code[skip] = len(code) - skip
64 else:
65 emit(OPCODES[REPEAT])
66 skip = len(code); emit(0)
67 emit(av[0])
68 emit(av[1])
69 _compile(code, av[2], flags)
70 code[skip] = len(code) - skip
71 if op == MAX_REPEAT:
72 emit(OPCODES[MAX_UNTIL])
73 else:
74 emit(OPCODES[MIN_UNTIL])
75 elif op is SUBPATTERN:
76 if av[0]:
77 emit(OPCODES[MARK])
78 emit((av[0]-1)*2)
79 # _compile_info(code, av[1], flags)
80 _compile(code, av[1], flags)
81 if av[0]:
82 emit(OPCODES[MARK])
83 emit((av[0]-1)*2+1)
84 elif op in (SUCCESS, FAILURE):
85 emit(OPCODES[op])
86 elif op in (ASSERT, ASSERT_NOT):
87 emit(OPCODES[op])
88 skip = len(code); emit(0)
89 if av[0] >= 0:
90 emit(0) # look ahead
91 else:
92 lo, hi = av[1].getwidth()
93 if lo != hi:
94 raise error, "look-behind requires fixed-width pattern"
95 emit(lo) # look behind
96 _compile(code, av[1], flags)
97 emit(OPCODES[SUCCESS])
98 code[skip] = len(code) - skip
99 elif op is CALL:
100 emit(OPCODES[op])
101 skip = len(code); emit(0)
102 _compile(code, av, flags)
103 emit(OPCODES[SUCCESS])
104 code[skip] = len(code) - skip
105 elif op is AT:
106 emit(OPCODES[op])
107 if flags & SRE_FLAG_MULTILINE:
Fredrik Lundhb25e1ad2001-03-22 15:50:10 +0000108 av = AT_MULTILINE.get(av, av)
109 if flags & SRE_FLAG_LOCALE:
110 av = AT_LOCALE.get(av, av)
111 elif flags & SRE_FLAG_UNICODE:
112 av = AT_UNICODE.get(av, av)
113 emit(ATCODES[av])
Fredrik Lundh7898c3e2000-08-07 20:59:04 +0000114 elif op is BRANCH:
115 emit(OPCODES[op])
116 tail = []
117 for av in av[1]:
118 skip = len(code); emit(0)
119 # _compile_info(code, av, flags)
120 _compile(code, av, flags)
121 emit(OPCODES[JUMP])
122 tail.append(len(code)); emit(0)
123 code[skip] = len(code) - skip
124 emit(0) # end of branch
125 for tail in tail:
126 code[tail] = len(code) - tail
127 elif op is CATEGORY:
128 emit(OPCODES[op])
129 if flags & SRE_FLAG_LOCALE:
Fredrik Lundhb25e1ad2001-03-22 15:50:10 +0000130 av = CH_LOCALE[av]
Fredrik Lundh7898c3e2000-08-07 20:59:04 +0000131 elif flags & SRE_FLAG_UNICODE:
Fredrik Lundhb25e1ad2001-03-22 15:50:10 +0000132 av = CH_UNICODE[av]
133 emit(CHCODES[av])
Fredrik Lundh7898c3e2000-08-07 20:59:04 +0000134 elif op is GROUPREF:
135 if flags & SRE_FLAG_IGNORECASE:
136 emit(OPCODES[OP_IGNORE[op]])
137 else:
138 emit(OPCODES[op])
139 emit(av-1)
140 else:
141 raise ValueError, ("unsupported operand type", op)
142
143def _compile_charset(charset, flags, code, fixup=None):
144 # compile charset subprogram
145 emit = code.append
Fredrik Lundh28552902000-07-05 21:14:16 +0000146 if not fixup:
147 fixup = lambda x: x
Fredrik Lundh7898c3e2000-08-07 20:59:04 +0000148 for op, av in _optimize_charset(charset, fixup):
149 emit(OPCODES[op])
150 if op is NEGATE:
151 pass
152 elif op is LITERAL:
153 emit(fixup(av))
154 elif op is RANGE:
155 emit(fixup(av[0]))
156 emit(fixup(av[1]))
157 elif op is CHARSET:
158 code.extend(av)
Fredrik Lundh19af43d2001-07-02 16:58:38 +0000159 elif op is BIGCHARSET:
160 code.extend(av)
Fredrik Lundh7898c3e2000-08-07 20:59:04 +0000161 elif op is CATEGORY:
162 if flags & SRE_FLAG_LOCALE:
163 emit(CHCODES[CH_LOCALE[av]])
164 elif flags & SRE_FLAG_UNICODE:
165 emit(CHCODES[CH_UNICODE[av]])
166 else:
167 emit(CHCODES[av])
168 else:
169 raise error, "internal: unsupported set operator"
170 emit(OPCODES[FAILURE])
171
172def _optimize_charset(charset, fixup):
173 # internal: optimize character set
Fredrik Lundh3562f112000-07-02 12:00:07 +0000174 out = []
175 charmap = [0]*256
176 try:
177 for op, av in charset:
178 if op is NEGATE:
179 out.append((op, av))
180 elif op is LITERAL:
181 charmap[fixup(av)] = 1
182 elif op is RANGE:
183 for i in range(fixup(av[0]), fixup(av[1])+1):
184 charmap[i] = 1
185 elif op is CATEGORY:
Fredrik Lundh770617b2001-01-14 15:06:11 +0000186 # XXX: could append to charmap tail
Fredrik Lundh3562f112000-07-02 12:00:07 +0000187 return charset # cannot compress
188 except IndexError:
Fredrik Lundh7898c3e2000-08-07 20:59:04 +0000189 # character set contains unicode characters
Fredrik Lundh19af43d2001-07-02 16:58:38 +0000190 return _optimize_unicode(charset, fixup)
Fredrik Lundh3562f112000-07-02 12:00:07 +0000191 # compress character map
192 i = p = n = 0
193 runs = []
194 for c in charmap:
195 if c:
196 if n == 0:
197 p = i
198 n = n + 1
199 elif n:
200 runs.append((p, n))
201 n = 0
202 i = i + 1
203 if n:
204 runs.append((p, n))
205 if len(runs) <= 2:
206 # use literal/range
207 for p, n in runs:
208 if n == 1:
209 out.append((LITERAL, p))
210 else:
211 out.append((RANGE, (p, p+n-1)))
212 if len(out) < len(charset):
213 return out
214 else:
215 # use bitmap
Fredrik Lundh19af43d2001-07-02 16:58:38 +0000216 data = _mk_bitmap(charmap)
Fredrik Lundh3562f112000-07-02 12:00:07 +0000217 out.append((CHARSET, data))
218 return out
219 return charset
220
Fredrik Lundh19af43d2001-07-02 16:58:38 +0000221def _mk_bitmap(bits):
222 data = []
223 m = 1; v = 0
224 for c in bits:
225 if c:
226 v = v + m
227 m = m << 1
228 if m > MAXCODE:
229 data.append(v)
230 m = 1; v = 0
231 return data
232
233# To represent a big charset, first a bitmap of all characters in the
234# set is constructed. Then, this bitmap is sliced into chunks of 256
235# characters, duplicate chunks are eliminitated, and each chunk is
236# given a number. In the compiled expression, the charset is
237# represented by a 16-bit word sequence, consisting of one word for
238# the number of different chunks, a sequence of 256 bytes (128 words)
239# of chunk numbers indexed by their original chunk position, and a
240# sequence of chunks (16 words each).
241
242# Compression is normally good: in a typical charset, large ranges of
243# Unicode will be either completely excluded (e.g. if only cyrillic
244# letters are to be matched), or completely included (e.g. if large
245# subranges of Kanji match). These ranges will be represented by
246# chunks of all one-bits or all zero-bits.
247
248# Matching can be also done efficiently: the more significant byte of
249# the Unicode character is an index into the chunk number, and the
250# less significant byte is a bit index in the chunk (just like the
251# CHARSET matching).
252
253def _optimize_unicode(charset, fixup):
254 charmap = [0]*65536
255 negate = 0
256 for op, av in charset:
257 if op is NEGATE:
258 negate = 1
259 elif op is LITERAL:
260 charmap[fixup(av)] = 1
261 elif op is RANGE:
262 for i in range(fixup(av[0]), fixup(av[1])+1):
263 charmap[i] = 1
264 elif op is CATEGORY:
265 # XXX: could expand category
266 return charset # cannot compress
267 if negate:
268 for i in range(65536):
269 charmap[i] = not charmap[i]
270 comps = {}
271 mapping = [0]*256
272 block = 0
273 data = []
274 for i in range(256):
275 chunk = tuple(charmap[i*256:(i+1)*256])
276 new = comps.setdefault(chunk, block)
277 mapping[i] = new
278 if new == block:
279 block += 1
280 data += _mk_bitmap(chunk)
281 header = [block]
282 assert MAXCODE == 65535
283 for i in range(128):
Martin v. Löwis3550dd32001-07-19 14:26:10 +0000284 if sys.byteorder == 'big':
285 header.append(256*mapping[2*i]+mapping[2*i+1])
286 else:
287 header.append(mapping[2*i]+256*mapping[2*i+1])
Fredrik Lundh19af43d2001-07-02 16:58:38 +0000288 data[0:0] = header
Tim Peters87cc0c32001-07-21 01:41:30 +0000289 return [(BIGCHARSET, data)]
Fredrik Lundh19af43d2001-07-02 16:58:38 +0000290
Fredrik Lundhe1869832000-08-01 22:47:49 +0000291def _simple(av):
292 # check if av is a "simple" operator
293 lo, hi = av[2].getwidth()
Fredrik Lundh13ac9922000-10-07 17:38:23 +0000294 if lo == 0 and hi == MAXREPEAT:
Fredrik Lundhe1869832000-08-01 22:47:49 +0000295 raise error, "nothing to repeat"
296 return lo == hi == 1 and av[2][0][0] != SUBPATTERN
297
Fredrik Lundh29c08be2000-06-29 23:33:12 +0000298def _compile_info(code, pattern, flags):
299 # internal: compile an info block. in the current version,
Fredrik Lundh3562f112000-07-02 12:00:07 +0000300 # this contains min/max pattern width, and an optional literal
301 # prefix or a character map
Fredrik Lundh29c08be2000-06-29 23:33:12 +0000302 lo, hi = pattern.getwidth()
303 if lo == 0:
Fredrik Lundh90a07912000-06-30 07:50:59 +0000304 return # not worth it
Fredrik Lundh29c08be2000-06-29 23:33:12 +0000305 # look for a literal prefix
306 prefix = []
Fredrik Lundh7898c3e2000-08-07 20:59:04 +0000307 prefix_skip = 0
Fredrik Lundh3562f112000-07-02 12:00:07 +0000308 charset = [] # not used
Fredrik Lundh29c08be2000-06-29 23:33:12 +0000309 if not (flags & SRE_FLAG_IGNORECASE):
Fredrik Lundh7898c3e2000-08-07 20:59:04 +0000310 # look for literal prefix
Fredrik Lundh90a07912000-06-30 07:50:59 +0000311 for op, av in pattern.data:
312 if op is LITERAL:
Fredrik Lundh7898c3e2000-08-07 20:59:04 +0000313 if len(prefix) == prefix_skip:
314 prefix_skip = prefix_skip + 1
Fredrik Lundh0640e112000-06-30 13:55:15 +0000315 prefix.append(av)
Fredrik Lundh7898c3e2000-08-07 20:59:04 +0000316 elif op is SUBPATTERN and len(av[1]) == 1:
317 op, av = av[1][0]
318 if op is LITERAL:
319 prefix.append(av)
320 else:
321 break
Fredrik Lundh90a07912000-06-30 07:50:59 +0000322 else:
323 break
Fredrik Lundh7898c3e2000-08-07 20:59:04 +0000324 # if no prefix, look for charset prefix
325 if not prefix and pattern.data:
326 op, av = pattern.data[0]
327 if op is SUBPATTERN and av[1]:
328 op, av = av[1][0]
329 if op is LITERAL:
330 charset.append((op, av))
331 elif op is BRANCH:
332 c = []
333 for p in av[1]:
334 if not p:
335 break
336 op, av = p[0]
337 if op is LITERAL:
338 c.append((op, av))
339 else:
340 break
341 else:
342 charset = c
343 elif op is BRANCH:
344 c = []
345 for p in av[1]:
346 if not p:
347 break
348 op, av = p[0]
349 if op is LITERAL:
350 c.append((op, av))
351 else:
352 break
353 else:
354 charset = c
355 elif op is IN:
356 charset = av
357## if prefix:
358## print "*** PREFIX", prefix, prefix_skip
359## if charset:
360## print "*** CHARSET", charset
Fredrik Lundh29c08be2000-06-29 23:33:12 +0000361 # add an info block
362 emit = code.append
363 emit(OPCODES[INFO])
364 skip = len(code); emit(0)
365 # literal flag
366 mask = 0
Fredrik Lundh3562f112000-07-02 12:00:07 +0000367 if prefix:
368 mask = SRE_INFO_PREFIX
Fredrik Lundh7898c3e2000-08-07 20:59:04 +0000369 if len(prefix) == prefix_skip == len(pattern.data):
Fredrik Lundh3562f112000-07-02 12:00:07 +0000370 mask = mask + SRE_INFO_LITERAL
371 elif charset:
372 mask = mask + SRE_INFO_CHARSET
Fredrik Lundh29c08be2000-06-29 23:33:12 +0000373 emit(mask)
374 # pattern length
Fredrik Lundh3562f112000-07-02 12:00:07 +0000375 if lo < MAXCODE:
376 emit(lo)
377 else:
378 emit(MAXCODE)
379 prefix = prefix[:MAXCODE]
380 if hi < MAXCODE:
Fredrik Lundh90a07912000-06-30 07:50:59 +0000381 emit(hi)
Fredrik Lundh29c08be2000-06-29 23:33:12 +0000382 else:
Fredrik Lundh90a07912000-06-30 07:50:59 +0000383 emit(0)
Fredrik Lundh29c08be2000-06-29 23:33:12 +0000384 # add literal prefix
Fredrik Lundh29c08be2000-06-29 23:33:12 +0000385 if prefix:
Fredrik Lundh7898c3e2000-08-07 20:59:04 +0000386 emit(len(prefix)) # length
387 emit(prefix_skip) # skip
388 code.extend(prefix)
389 # generate overlap table
390 table = [-1] + ([0]*len(prefix))
391 for i in range(len(prefix)):
392 table[i+1] = table[i]+1
393 while table[i+1] > 0 and prefix[i] != prefix[table[i+1]-1]:
394 table[i+1] = table[table[i+1]-1]+1
395 code.extend(table[1:]) # don't store first entry
Fredrik Lundh3562f112000-07-02 12:00:07 +0000396 elif charset:
Fredrik Lundh7898c3e2000-08-07 20:59:04 +0000397 _compile_charset(charset, 0, code)
Fredrik Lundh29c08be2000-06-29 23:33:12 +0000398 code[skip] = len(code) - skip
399
Fredrik Lundh8a3ebf82000-07-23 21:46:17 +0000400STRING_TYPES = [type("")]
401
402try:
403 STRING_TYPES.append(type(unicode("")))
404except NameError:
405 pass
406
Fredrik Lundh2f2c67d2000-08-01 21:05:41 +0000407def _code(p, flags):
Fredrik Lundh29c08be2000-06-29 23:33:12 +0000408
Jeremy Hyltonb1aa1952000-06-01 17:39:12 +0000409 flags = p.pattern.flags | flags
Fredrik Lundhbe2211e2000-06-29 16:57:40 +0000410 code = []
Fredrik Lundh29c08be2000-06-29 23:33:12 +0000411
412 # compile info block
413 _compile_info(code, p, flags)
414
415 # compile the pattern
Jeremy Hyltonb1aa1952000-06-01 17:39:12 +0000416 _compile(code, p.data, flags)
Fredrik Lundh29c08be2000-06-29 23:33:12 +0000417
Jeremy Hyltonb1aa1952000-06-01 17:39:12 +0000418 code.append(OPCODES[SUCCESS])
Fredrik Lundh29c08be2000-06-29 23:33:12 +0000419
Fredrik Lundh29c4ba92000-08-01 18:20:07 +0000420 return code
421
422def compile(p, flags=0):
423 # internal: convert pattern list to internal format
424
425 if type(p) in STRING_TYPES:
426 import sre_parse
427 pattern = p
428 p = sre_parse.parse(p, flags)
429 else:
430 pattern = None
431
Fredrik Lundh2f2c67d2000-08-01 21:05:41 +0000432 code = _code(p, flags)
Fredrik Lundh29c4ba92000-08-01 18:20:07 +0000433
Fredrik Lundh8a3ebf82000-07-23 21:46:17 +0000434 # print code
435
Fredrik Lundh770617b2001-01-14 15:06:11 +0000436 # XXX: <fl> get rid of this limitation!
Fredrik Lundhbe2211e2000-06-29 16:57:40 +0000437 assert p.pattern.groups <= 100,\
Fredrik Lundh90a07912000-06-30 07:50:59 +0000438 "sorry, but this version only supports 100 named groups"
Fredrik Lundh29c08be2000-06-29 23:33:12 +0000439
Fredrik Lundhc2301732000-07-02 22:25:39 +0000440 # map in either direction
441 groupindex = p.pattern.groupdict
442 indexgroup = [None] * p.pattern.groups
443 for k, i in groupindex.items():
444 indexgroup[i] = k
445
Jeremy Hyltonb1aa1952000-06-01 17:39:12 +0000446 return _sre.compile(
Fredrik Lundh6f013982000-07-03 18:44:21 +0000447 pattern, flags, code,
448 p.pattern.groups-1,
449 groupindex, indexgroup
Fredrik Lundh90a07912000-06-30 07:50:59 +0000450 )