blob: c2996fc3a25b340f3dc77f1a1645d1186de28ecc [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
Guido van Rossum7627c0d2000-03-31 14:58:54 +000011import _sre
12
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]])
26 else:
27 emit(OPCODES[op])
28 emit(av)
29 elif op is IN:
30 if flags & SRE_FLAG_IGNORECASE:
31 emit(OPCODES[OP_IGNORE[op]])
32 def fixup(literal, flags=flags):
33 return _sre.getlower(literal, flags)
34 else:
35 emit(OPCODES[op])
36 fixup = lambda x: x
37 skip = len(code); emit(0)
38 _compile_charset(av, flags, code, fixup)
39 code[skip] = len(code) - skip
40 elif op is ANY:
41 if flags & SRE_FLAG_DOTALL:
42 emit(OPCODES[ANY_ALL])
43 else:
44 emit(OPCODES[ANY])
45 elif op in (REPEAT, MIN_REPEAT, MAX_REPEAT):
46 if flags & SRE_FLAG_TEMPLATE:
47 raise error, "internal: unsupported template operator"
48 emit(OPCODES[REPEAT])
49 skip = len(code); emit(0)
50 emit(av[0])
51 emit(av[1])
52 _compile(code, av[2], flags)
53 emit(OPCODES[SUCCESS])
54 code[skip] = len(code) - skip
55 elif _simple(av) and op == MAX_REPEAT:
56 emit(OPCODES[REPEAT_ONE])
57 skip = len(code); emit(0)
58 emit(av[0])
59 emit(av[1])
60 _compile(code, av[2], flags)
61 emit(OPCODES[SUCCESS])
62 code[skip] = len(code) - skip
63 else:
64 emit(OPCODES[REPEAT])
65 skip = len(code); emit(0)
66 emit(av[0])
67 emit(av[1])
68 _compile(code, av[2], flags)
69 code[skip] = len(code) - skip
70 if op == MAX_REPEAT:
71 emit(OPCODES[MAX_UNTIL])
72 else:
73 emit(OPCODES[MIN_UNTIL])
74 elif op is SUBPATTERN:
75 if av[0]:
76 emit(OPCODES[MARK])
77 emit((av[0]-1)*2)
78 # _compile_info(code, av[1], flags)
79 _compile(code, av[1], flags)
80 if av[0]:
81 emit(OPCODES[MARK])
82 emit((av[0]-1)*2+1)
83 elif op in (SUCCESS, FAILURE):
84 emit(OPCODES[op])
85 elif op in (ASSERT, ASSERT_NOT):
86 emit(OPCODES[op])
87 skip = len(code); emit(0)
88 if av[0] >= 0:
89 emit(0) # look ahead
90 else:
91 lo, hi = av[1].getwidth()
92 if lo != hi:
93 raise error, "look-behind requires fixed-width pattern"
94 emit(lo) # look behind
95 _compile(code, av[1], flags)
96 emit(OPCODES[SUCCESS])
97 code[skip] = len(code) - skip
98 elif op is CALL:
99 emit(OPCODES[op])
100 skip = len(code); emit(0)
101 _compile(code, av, flags)
102 emit(OPCODES[SUCCESS])
103 code[skip] = len(code) - skip
104 elif op is AT:
105 emit(OPCODES[op])
106 if flags & SRE_FLAG_MULTILINE:
107 emit(ATCODES[AT_MULTILINE.get(av, av)])
108 else:
109 emit(ATCODES[av])
110 elif op is BRANCH:
111 emit(OPCODES[op])
112 tail = []
113 for av in av[1]:
114 skip = len(code); emit(0)
115 # _compile_info(code, av, flags)
116 _compile(code, av, flags)
117 emit(OPCODES[JUMP])
118 tail.append(len(code)); emit(0)
119 code[skip] = len(code) - skip
120 emit(0) # end of branch
121 for tail in tail:
122 code[tail] = len(code) - tail
123 elif op is CATEGORY:
124 emit(OPCODES[op])
125 if flags & SRE_FLAG_LOCALE:
126 emit(CHCODES[CH_LOCALE[av]])
127 elif flags & SRE_FLAG_UNICODE:
128 emit(CHCODES[CH_UNICODE[av]])
129 else:
130 emit(CHCODES[av])
131 elif op is GROUPREF:
132 if flags & SRE_FLAG_IGNORECASE:
133 emit(OPCODES[OP_IGNORE[op]])
134 else:
135 emit(OPCODES[op])
136 emit(av-1)
137 else:
138 raise ValueError, ("unsupported operand type", op)
139
140def _compile_charset(charset, flags, code, fixup=None):
141 # compile charset subprogram
142 emit = code.append
Fredrik Lundh28552902000-07-05 21:14:16 +0000143 if not fixup:
144 fixup = lambda x: x
Fredrik Lundh7898c3e2000-08-07 20:59:04 +0000145 for op, av in _optimize_charset(charset, fixup):
146 emit(OPCODES[op])
147 if op is NEGATE:
148 pass
149 elif op is LITERAL:
150 emit(fixup(av))
151 elif op is RANGE:
152 emit(fixup(av[0]))
153 emit(fixup(av[1]))
154 elif op is CHARSET:
155 code.extend(av)
156 elif op is CATEGORY:
157 if flags & SRE_FLAG_LOCALE:
158 emit(CHCODES[CH_LOCALE[av]])
159 elif flags & SRE_FLAG_UNICODE:
160 emit(CHCODES[CH_UNICODE[av]])
161 else:
162 emit(CHCODES[av])
163 else:
164 raise error, "internal: unsupported set operator"
165 emit(OPCODES[FAILURE])
166
167def _optimize_charset(charset, fixup):
168 # internal: optimize character set
Fredrik Lundh3562f112000-07-02 12:00:07 +0000169 out = []
170 charmap = [0]*256
171 try:
172 for op, av in charset:
173 if op is NEGATE:
174 out.append((op, av))
175 elif op is LITERAL:
176 charmap[fixup(av)] = 1
177 elif op is RANGE:
178 for i in range(fixup(av[0]), fixup(av[1])+1):
179 charmap[i] = 1
180 elif op is CATEGORY:
Fredrik Lundh770617b2001-01-14 15:06:11 +0000181 # XXX: could append to charmap tail
Fredrik Lundh3562f112000-07-02 12:00:07 +0000182 return charset # cannot compress
183 except IndexError:
Fredrik Lundh7898c3e2000-08-07 20:59:04 +0000184 # character set contains unicode characters
Fredrik Lundh3562f112000-07-02 12:00:07 +0000185 return charset
186 # compress character map
187 i = p = n = 0
188 runs = []
189 for c in charmap:
190 if c:
191 if n == 0:
192 p = i
193 n = n + 1
194 elif n:
195 runs.append((p, n))
196 n = 0
197 i = i + 1
198 if n:
199 runs.append((p, n))
200 if len(runs) <= 2:
201 # use literal/range
202 for p, n in runs:
203 if n == 1:
204 out.append((LITERAL, p))
205 else:
206 out.append((RANGE, (p, p+n-1)))
207 if len(out) < len(charset):
208 return out
209 else:
210 # use bitmap
211 data = []
212 m = 1; v = 0
213 for c in charmap:
214 if c:
215 v = v + m
216 m = m << 1
217 if m > MAXCODE:
218 data.append(v)
219 m = 1; v = 0
220 out.append((CHARSET, data))
221 return out
222 return charset
223
Fredrik Lundhe1869832000-08-01 22:47:49 +0000224def _simple(av):
225 # check if av is a "simple" operator
226 lo, hi = av[2].getwidth()
Fredrik Lundh13ac9922000-10-07 17:38:23 +0000227 if lo == 0 and hi == MAXREPEAT:
Fredrik Lundhe1869832000-08-01 22:47:49 +0000228 raise error, "nothing to repeat"
229 return lo == hi == 1 and av[2][0][0] != SUBPATTERN
230
Fredrik Lundh29c08be2000-06-29 23:33:12 +0000231def _compile_info(code, pattern, flags):
232 # internal: compile an info block. in the current version,
Fredrik Lundh3562f112000-07-02 12:00:07 +0000233 # this contains min/max pattern width, and an optional literal
234 # prefix or a character map
Fredrik Lundh29c08be2000-06-29 23:33:12 +0000235 lo, hi = pattern.getwidth()
236 if lo == 0:
Fredrik Lundh90a07912000-06-30 07:50:59 +0000237 return # not worth it
Fredrik Lundh29c08be2000-06-29 23:33:12 +0000238 # look for a literal prefix
239 prefix = []
Fredrik Lundh7898c3e2000-08-07 20:59:04 +0000240 prefix_skip = 0
Fredrik Lundh3562f112000-07-02 12:00:07 +0000241 charset = [] # not used
Fredrik Lundh29c08be2000-06-29 23:33:12 +0000242 if not (flags & SRE_FLAG_IGNORECASE):
Fredrik Lundh7898c3e2000-08-07 20:59:04 +0000243 # look for literal prefix
Fredrik Lundh90a07912000-06-30 07:50:59 +0000244 for op, av in pattern.data:
245 if op is LITERAL:
Fredrik Lundh7898c3e2000-08-07 20:59:04 +0000246 if len(prefix) == prefix_skip:
247 prefix_skip = prefix_skip + 1
Fredrik Lundh0640e112000-06-30 13:55:15 +0000248 prefix.append(av)
Fredrik Lundh7898c3e2000-08-07 20:59:04 +0000249 elif op is SUBPATTERN and len(av[1]) == 1:
250 op, av = av[1][0]
251 if op is LITERAL:
252 prefix.append(av)
253 else:
254 break
Fredrik Lundh90a07912000-06-30 07:50:59 +0000255 else:
256 break
Fredrik Lundh7898c3e2000-08-07 20:59:04 +0000257 # if no prefix, look for charset prefix
258 if not prefix and pattern.data:
259 op, av = pattern.data[0]
260 if op is SUBPATTERN and av[1]:
261 op, av = av[1][0]
262 if op is LITERAL:
263 charset.append((op, av))
264 elif op is BRANCH:
265 c = []
266 for p in av[1]:
267 if not p:
268 break
269 op, av = p[0]
270 if op is LITERAL:
271 c.append((op, av))
272 else:
273 break
274 else:
275 charset = c
276 elif op is BRANCH:
277 c = []
278 for p in av[1]:
279 if not p:
280 break
281 op, av = p[0]
282 if op is LITERAL:
283 c.append((op, av))
284 else:
285 break
286 else:
287 charset = c
288 elif op is IN:
289 charset = av
290## if prefix:
291## print "*** PREFIX", prefix, prefix_skip
292## if charset:
293## print "*** CHARSET", charset
Fredrik Lundh29c08be2000-06-29 23:33:12 +0000294 # add an info block
295 emit = code.append
296 emit(OPCODES[INFO])
297 skip = len(code); emit(0)
298 # literal flag
299 mask = 0
Fredrik Lundh3562f112000-07-02 12:00:07 +0000300 if prefix:
301 mask = SRE_INFO_PREFIX
Fredrik Lundh7898c3e2000-08-07 20:59:04 +0000302 if len(prefix) == prefix_skip == len(pattern.data):
Fredrik Lundh3562f112000-07-02 12:00:07 +0000303 mask = mask + SRE_INFO_LITERAL
304 elif charset:
305 mask = mask + SRE_INFO_CHARSET
Fredrik Lundh29c08be2000-06-29 23:33:12 +0000306 emit(mask)
307 # pattern length
Fredrik Lundh3562f112000-07-02 12:00:07 +0000308 if lo < MAXCODE:
309 emit(lo)
310 else:
311 emit(MAXCODE)
312 prefix = prefix[:MAXCODE]
313 if hi < MAXCODE:
Fredrik Lundh90a07912000-06-30 07:50:59 +0000314 emit(hi)
Fredrik Lundh29c08be2000-06-29 23:33:12 +0000315 else:
Fredrik Lundh90a07912000-06-30 07:50:59 +0000316 emit(0)
Fredrik Lundh29c08be2000-06-29 23:33:12 +0000317 # add literal prefix
Fredrik Lundh29c08be2000-06-29 23:33:12 +0000318 if prefix:
Fredrik Lundh7898c3e2000-08-07 20:59:04 +0000319 emit(len(prefix)) # length
320 emit(prefix_skip) # skip
321 code.extend(prefix)
322 # generate overlap table
323 table = [-1] + ([0]*len(prefix))
324 for i in range(len(prefix)):
325 table[i+1] = table[i]+1
326 while table[i+1] > 0 and prefix[i] != prefix[table[i+1]-1]:
327 table[i+1] = table[table[i+1]-1]+1
328 code.extend(table[1:]) # don't store first entry
Fredrik Lundh3562f112000-07-02 12:00:07 +0000329 elif charset:
Fredrik Lundh7898c3e2000-08-07 20:59:04 +0000330 _compile_charset(charset, 0, code)
Fredrik Lundh29c08be2000-06-29 23:33:12 +0000331 code[skip] = len(code) - skip
332
Fredrik Lundh8a3ebf82000-07-23 21:46:17 +0000333STRING_TYPES = [type("")]
334
335try:
336 STRING_TYPES.append(type(unicode("")))
337except NameError:
338 pass
339
Fredrik Lundh2f2c67d2000-08-01 21:05:41 +0000340def _code(p, flags):
Fredrik Lundh29c08be2000-06-29 23:33:12 +0000341
Jeremy Hyltonb1aa1952000-06-01 17:39:12 +0000342 flags = p.pattern.flags | flags
Fredrik Lundhbe2211e2000-06-29 16:57:40 +0000343 code = []
Fredrik Lundh29c08be2000-06-29 23:33:12 +0000344
345 # compile info block
346 _compile_info(code, p, flags)
347
348 # compile the pattern
Jeremy Hyltonb1aa1952000-06-01 17:39:12 +0000349 _compile(code, p.data, flags)
Fredrik Lundh29c08be2000-06-29 23:33:12 +0000350
Jeremy Hyltonb1aa1952000-06-01 17:39:12 +0000351 code.append(OPCODES[SUCCESS])
Fredrik Lundh29c08be2000-06-29 23:33:12 +0000352
Fredrik Lundh29c4ba92000-08-01 18:20:07 +0000353 return code
354
355def compile(p, flags=0):
356 # internal: convert pattern list to internal format
357
358 if type(p) in STRING_TYPES:
359 import sre_parse
360 pattern = p
361 p = sre_parse.parse(p, flags)
362 else:
363 pattern = None
364
Fredrik Lundh2f2c67d2000-08-01 21:05:41 +0000365 code = _code(p, flags)
Fredrik Lundh29c4ba92000-08-01 18:20:07 +0000366
Fredrik Lundh8a3ebf82000-07-23 21:46:17 +0000367 # print code
368
Fredrik Lundh770617b2001-01-14 15:06:11 +0000369 # XXX: <fl> get rid of this limitation!
Fredrik Lundhbe2211e2000-06-29 16:57:40 +0000370 assert p.pattern.groups <= 100,\
Fredrik Lundh90a07912000-06-30 07:50:59 +0000371 "sorry, but this version only supports 100 named groups"
Fredrik Lundh29c08be2000-06-29 23:33:12 +0000372
Fredrik Lundhc2301732000-07-02 22:25:39 +0000373 # map in either direction
374 groupindex = p.pattern.groupdict
375 indexgroup = [None] * p.pattern.groups
376 for k, i in groupindex.items():
377 indexgroup[i] = k
378
Jeremy Hyltonb1aa1952000-06-01 17:39:12 +0000379 return _sre.compile(
Fredrik Lundh6f013982000-07-03 18:44:21 +0000380 pattern, flags, code,
381 p.pattern.groups-1,
382 groupindex, indexgroup
Fredrik Lundh90a07912000-06-30 07:50:59 +0000383 )