2 # Secret Labs' Regular Expression Engine
4 # convert template to internal format
6 # Copyright (c) 1997-2001 by Secret Labs AB. All rights reserved.
8 # See the sre.py file for information on usage and redistribution.
13 from sre_constants
import *
15 assert _sre
.MAGIC
== MAGIC
, "SRE module mismatch"
19 def _compile(code
, pattern
, flags
):
20 # internal: compile a (sub)pattern
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 emit(_sre
.getlower(av
, flags
))
31 if flags
& SRE_FLAG_IGNORECASE
:
32 emit(OPCODES
[OP_IGNORE
[op
]])
33 def fixup(literal
, flags
=flags
):
34 return _sre
.getlower(literal
, flags
)
38 skip
= len(code
); emit(0)
39 _compile_charset(av
, flags
, code
, fixup
)
40 code
[skip
] = len(code
) - skip
42 if flags
& SRE_FLAG_DOTALL
:
43 emit(OPCODES
[ANY_ALL
])
46 elif op
in (REPEAT
, MIN_REPEAT
, MAX_REPEAT
):
47 if flags
& SRE_FLAG_TEMPLATE
:
48 raise error
, "internal: unsupported template operator"
50 skip
= len(code
); emit(0)
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)
61 _compile(code
, av
[2], flags
)
62 emit(OPCODES
[SUCCESS
])
63 code
[skip
] = len(code
) - skip
66 skip
= len(code
); emit(0)
69 _compile(code
, av
[2], flags
)
70 code
[skip
] = len(code
) - skip
72 emit(OPCODES
[MAX_UNTIL
])
74 emit(OPCODES
[MIN_UNTIL
])
75 elif op
is SUBPATTERN
:
79 # _compile_info(code, av[1], flags)
80 _compile(code
, av
[1], flags
)
84 elif op
in (SUCCESS
, FAILURE
):
86 elif op
in (ASSERT
, ASSERT_NOT
):
88 skip
= len(code
); emit(0)
92 lo
, hi
= av
[1].getwidth()
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
101 skip
= len(code
); emit(0)
102 _compile(code
, av
, flags
)
103 emit(OPCODES
[SUCCESS
])
104 code
[skip
] = len(code
) - skip
107 if flags
& SRE_FLAG_MULTILINE
:
108 emit(ATCODES
[AT_MULTILINE
.get(av
, av
)])
115 skip
= len(code
); emit(0)
116 # _compile_info(code, av, flags)
117 _compile(code
, av
, flags
)
119 tail
.append(len(code
)); emit(0)
120 code
[skip
] = len(code
) - skip
121 emit(0) # end of branch
123 code
[tail
] = len(code
) - tail
126 if flags
& SRE_FLAG_LOCALE
:
127 emit(CHCODES
[CH_LOCALE
[av
]])
128 elif flags
& SRE_FLAG_UNICODE
:
129 emit(CHCODES
[CH_UNICODE
[av
]])
133 if flags
& SRE_FLAG_IGNORECASE
:
134 emit(OPCODES
[OP_IGNORE
[op
]])
139 raise ValueError, ("unsupported operand type", op
)
141 def _compile_charset(charset
, flags
, code
, fixup
=None):
142 # compile charset subprogram
146 for op
, av
in _optimize_charset(charset
, fixup
):
158 if flags
& SRE_FLAG_LOCALE
:
159 emit(CHCODES
[CH_LOCALE
[av
]])
160 elif flags
& SRE_FLAG_UNICODE
:
161 emit(CHCODES
[CH_UNICODE
[av
]])
165 raise error
, "internal: unsupported set operator"
166 emit(OPCODES
[FAILURE
])
168 def _optimize_charset(charset
, fixup
):
169 # internal: optimize character set
173 for op
, av
in charset
:
177 charmap
[fixup(av
)] = 1
179 for i
in range(fixup(av
[0]), fixup(av
[1])+1):
182 # XXX: could append to charmap tail
183 return charset
# cannot compress
185 # character set contains unicode characters
187 # compress character map
205 out
.append((LITERAL
, p
))
207 out
.append((RANGE
, (p
, p
+n
-1)))
208 if len(out
) < len(charset
):
221 out
.append((CHARSET
, data
))
226 # check if av is a "simple" operator
227 lo
, hi
= av
[2].getwidth()
228 if lo
== 0 and hi
== MAXREPEAT
:
229 raise error
, "nothing to repeat"
230 return lo
== hi
== 1 and av
[2][0][0] != SUBPATTERN
232 def _compile_info(code
, pattern
, flags
):
233 # internal: compile an info block. in the current version,
234 # this contains min/max pattern width, and an optional literal
235 # prefix or a character map
236 lo
, hi
= pattern
.getwidth()
238 return # not worth it
239 # look for a literal prefix
242 charset
= [] # not used
243 if not (flags
& SRE_FLAG_IGNORECASE
):
244 # look for literal prefix
245 for op
, av
in pattern
.data
:
247 if len(prefix
) == prefix_skip
:
248 prefix_skip
= prefix_skip
+ 1
250 elif op
is SUBPATTERN
and len(av
[1]) == 1:
258 # if no prefix, look for charset prefix
259 if not prefix
and pattern
.data
:
260 op
, av
= pattern
.data
[0]
261 if op
is SUBPATTERN
and av
[1]:
264 charset
.append((op
, av
))
292 ## print "*** PREFIX", prefix, prefix_skip
294 ## print "*** CHARSET", charset
298 skip
= len(code
); emit(0)
302 mask
= SRE_INFO_PREFIX
303 if len(prefix
) == prefix_skip
== len(pattern
.data
):
304 mask
= mask
+ SRE_INFO_LITERAL
306 mask
= mask
+ SRE_INFO_CHARSET
313 prefix
= prefix
[:MAXCODE
]
320 emit(len(prefix
)) # length
321 emit(prefix_skip
) # skip
323 # generate overlap table
324 table
= [-1] + ([0]*len(prefix
))
325 for i
in range(len(prefix
)):
326 table
[i
+1] = table
[i
]+1
327 while table
[i
+1] > 0 and prefix
[i
] != prefix
[table
[i
+1]-1]:
328 table
[i
+1] = table
[table
[i
+1]-1]+1
329 code
.extend(table
[1:]) # don't store first entry
331 _compile_charset(charset
, 0, code
)
332 code
[skip
] = len(code
) - skip
334 STRING_TYPES
= [type("")]
337 STRING_TYPES
.append(type(unicode("")))
343 flags
= p
.pattern
.flags | flags
347 _compile_info(code
, p
, flags
)
349 # compile the pattern
350 _compile(code
, p
.data
, flags
)
352 code
.append(OPCODES
[SUCCESS
])
356 def compile(p
, flags
=0):
357 # internal: convert pattern list to internal format
359 if type(p
) in STRING_TYPES
:
362 p
= sre_parse
.parse(p
, flags
)
366 code
= _code(p
, flags
)
370 # XXX: <fl> get rid of this limitation!
371 assert p
.pattern
.groups
<= 100,\
372 "sorry, but this version only supports 100 named groups"
374 # map in either direction
375 groupindex
= p
.pattern
.groupdict
376 indexgroup
= [None] * p
.pattern
.groups
377 for k
, i
in groupindex
.items():
381 pattern
, flags
, code
,
383 groupindex
, indexgroup