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.
11 """Internal support module for sre"""
15 from sre_constants
import *
17 assert _sre
.MAGIC
== MAGIC
, "SRE module mismatch"
21 def _compile(code
, pattern
, flags
):
22 # internal: compile a (sub)pattern
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
]])
28 emit(_sre
.getlower(av
, flags
))
33 if flags
& SRE_FLAG_IGNORECASE
:
34 emit(OPCODES
[OP_IGNORE
[op
]])
35 def fixup(literal
, flags
=flags
):
36 return _sre
.getlower(literal
, flags
)
40 skip
= len(code
); emit(0)
41 _compile_charset(av
, flags
, code
, fixup
)
42 code
[skip
] = len(code
) - skip
44 if flags
& SRE_FLAG_DOTALL
:
45 emit(OPCODES
[ANY_ALL
])
48 elif op
in (REPEAT
, MIN_REPEAT
, MAX_REPEAT
):
49 if flags
& SRE_FLAG_TEMPLATE
:
50 raise error
, "internal: unsupported template operator"
52 skip
= len(code
); emit(0)
55 _compile(code
, av
[2], flags
)
56 emit(OPCODES
[SUCCESS
])
57 code
[skip
] = len(code
) - skip
58 elif _simple(av
) and op
== MAX_REPEAT
:
59 emit(OPCODES
[REPEAT_ONE
])
60 skip
= len(code
); emit(0)
63 _compile(code
, av
[2], flags
)
64 emit(OPCODES
[SUCCESS
])
65 code
[skip
] = len(code
) - skip
68 skip
= len(code
); emit(0)
71 _compile(code
, av
[2], flags
)
72 code
[skip
] = len(code
) - skip
74 emit(OPCODES
[MAX_UNTIL
])
76 emit(OPCODES
[MIN_UNTIL
])
77 elif op
is SUBPATTERN
:
81 # _compile_info(code, av[1], flags)
82 _compile(code
, av
[1], flags
)
86 elif op
in (SUCCESS
, FAILURE
):
88 elif op
in (ASSERT
, ASSERT_NOT
):
90 skip
= len(code
); emit(0)
94 lo
, hi
= av
[1].getwidth()
96 raise error
, "look-behind requires fixed-width pattern"
97 emit(lo
) # look behind
98 _compile(code
, av
[1], flags
)
99 emit(OPCODES
[SUCCESS
])
100 code
[skip
] = len(code
) - skip
103 skip
= len(code
); emit(0)
104 _compile(code
, av
, flags
)
105 emit(OPCODES
[SUCCESS
])
106 code
[skip
] = len(code
) - skip
109 if flags
& SRE_FLAG_MULTILINE
:
110 av
= AT_MULTILINE
.get(av
, av
)
111 if flags
& SRE_FLAG_LOCALE
:
112 av
= AT_LOCALE
.get(av
, av
)
113 elif flags
& SRE_FLAG_UNICODE
:
114 av
= AT_UNICODE
.get(av
, av
)
120 skip
= len(code
); emit(0)
121 # _compile_info(code, av, flags)
122 _compile(code
, av
, flags
)
124 tail
.append(len(code
)); emit(0)
125 code
[skip
] = len(code
) - skip
126 emit(0) # end of branch
128 code
[tail
] = len(code
) - tail
131 if flags
& SRE_FLAG_LOCALE
:
133 elif flags
& SRE_FLAG_UNICODE
:
137 if flags
& SRE_FLAG_IGNORECASE
:
138 emit(OPCODES
[OP_IGNORE
[op
]])
143 raise ValueError, ("unsupported operand type", op
)
145 def _compile_charset(charset
, flags
, code
, fixup
=None):
146 # compile charset subprogram
150 for op
, av
in _optimize_charset(charset
, fixup
):
161 elif op
is BIGCHARSET
:
164 if flags
& SRE_FLAG_LOCALE
:
165 emit(CHCODES
[CH_LOCALE
[av
]])
166 elif flags
& SRE_FLAG_UNICODE
:
167 emit(CHCODES
[CH_UNICODE
[av
]])
171 raise error
, "internal: unsupported set operator"
172 emit(OPCODES
[FAILURE
])
174 def _optimize_charset(charset
, fixup
):
175 # internal: optimize character set
179 for op
, av
in charset
:
183 charmap
[fixup(av
)] = 1
185 for i
in range(fixup(av
[0]), fixup(av
[1])+1):
188 # XXX: could append to charmap tail
189 return charset
# cannot compress
191 if sys
.maxunicode
!= 65535:
192 # XXX: big charsets don't work in UCS-4 builds
194 # character set contains unicode characters
195 return _optimize_unicode(charset
, fixup
)
196 # compress character map
214 out
.append((LITERAL
, p
))
216 out
.append((RANGE
, (p
, p
+n
-1)))
217 if len(out
) < len(charset
):
221 data
= _mk_bitmap(charmap
)
222 out
.append((CHARSET
, data
))
226 def _mk_bitmap(bits
):
238 # To represent a big charset, first a bitmap of all characters in the
239 # set is constructed. Then, this bitmap is sliced into chunks of 256
240 # characters, duplicate chunks are eliminitated, and each chunk is
241 # given a number. In the compiled expression, the charset is
242 # represented by a 16-bit word sequence, consisting of one word for
243 # the number of different chunks, a sequence of 256 bytes (128 words)
244 # of chunk numbers indexed by their original chunk position, and a
245 # sequence of chunks (16 words each).
247 # Compression is normally good: in a typical charset, large ranges of
248 # Unicode will be either completely excluded (e.g. if only cyrillic
249 # letters are to be matched), or completely included (e.g. if large
250 # subranges of Kanji match). These ranges will be represented by
251 # chunks of all one-bits or all zero-bits.
253 # Matching can be also done efficiently: the more significant byte of
254 # the Unicode character is an index into the chunk number, and the
255 # less significant byte is a bit index in the chunk (just like the
258 def _optimize_unicode(charset
, fixup
):
261 for op
, av
in charset
:
265 charmap
[fixup(av
)] = 1
267 for i
in range(fixup(av
[0]), fixup(av
[1])+1):
270 # XXX: could expand category
271 return charset
# cannot compress
273 for i
in range(65536):
274 charmap
[i
] = not charmap
[i
]
280 chunk
= tuple(charmap
[i
*256:(i
+1)*256])
281 new
= comps
.setdefault(chunk
, block
)
285 data
= data
+ _mk_bitmap(chunk
)
287 assert MAXCODE
== 65535
289 if sys
.byteorder
== 'big':
290 header
.append(256*mapping
[2*i
]+mapping
[2*i
+1])
292 header
.append(mapping
[2*i
]+256*mapping
[2*i
+1])
294 return [(BIGCHARSET
, data
)]
297 # check if av is a "simple" operator
298 lo
, hi
= av
[2].getwidth()
299 if lo
== 0 and hi
== MAXREPEAT
:
300 raise error
, "nothing to repeat"
301 return lo
== hi
== 1 and av
[2][0][0] != SUBPATTERN
303 def _compile_info(code
, pattern
, flags
):
304 # internal: compile an info block. in the current version,
305 # this contains min/max pattern width, and an optional literal
306 # prefix or a character map
307 lo
, hi
= pattern
.getwidth()
309 return # not worth it
310 # look for a literal prefix
313 charset
= [] # not used
314 if not (flags
& SRE_FLAG_IGNORECASE
):
315 # look for literal prefix
316 for op
, av
in pattern
.data
:
318 if len(prefix
) == prefix_skip
:
319 prefix_skip
= prefix_skip
+ 1
321 elif op
is SUBPATTERN
and len(av
[1]) == 1:
329 # if no prefix, look for charset prefix
330 if not prefix
and pattern
.data
:
331 op
, av
= pattern
.data
[0]
332 if op
is SUBPATTERN
and av
[1]:
335 charset
.append((op
, av
))
363 ## print "*** PREFIX", prefix, prefix_skip
365 ## print "*** CHARSET", charset
369 skip
= len(code
); emit(0)
373 mask
= SRE_INFO_PREFIX
374 if len(prefix
) == prefix_skip
== len(pattern
.data
):
375 mask
= mask
+ SRE_INFO_LITERAL
377 mask
= mask
+ SRE_INFO_CHARSET
384 prefix
= prefix
[:MAXCODE
]
391 emit(len(prefix
)) # length
392 emit(prefix_skip
) # skip
394 # generate overlap table
395 table
= [-1] + ([0]*len(prefix
))
396 for i
in range(len(prefix
)):
397 table
[i
+1] = table
[i
]+1
398 while table
[i
+1] > 0 and prefix
[i
] != prefix
[table
[i
+1]-1]:
399 table
[i
+1] = table
[table
[i
+1]-1]+1
400 code
.extend(table
[1:]) # don't store first entry
402 _compile_charset(charset
, 0, code
)
403 code
[skip
] = len(code
) - skip
405 STRING_TYPES
= [type("")]
408 STRING_TYPES
.append(type(unicode("")))
414 flags
= p
.pattern
.flags | flags
418 _compile_info(code
, p
, flags
)
420 # compile the pattern
421 _compile(code
, p
.data
, flags
)
423 code
.append(OPCODES
[SUCCESS
])
427 def compile(p
, flags
=0):
428 # internal: convert pattern list to internal format
430 if type(p
) in STRING_TYPES
:
433 p
= sre_parse
.parse(p
, flags
)
437 code
= _code(p
, flags
)
441 # XXX: <fl> get rid of this limitation!
442 assert p
.pattern
.groups
<= 100,\
443 "sorry, but this version only supports 100 named groups"
445 # map in either direction
446 groupindex
= p
.pattern
.groupdict
447 indexgroup
= [None] * p
.pattern
.groups
448 for k
, i
in groupindex
.items():
452 pattern
, flags
, code
,
454 groupindex
, indexgroup