home *** CD-ROM | disk | FTP | other *** search
/ Freelog 125 / Freelog_MarsAvril2015_No125.iso / Bureautique / LibreOffice / LibreOffice_4.3.5_Win_x86.msi / sre_compile.py < prev    next >
Text File  |  2014-12-12  |  16KB  |  516 lines

  1. #
  2. # Secret Labs' Regular Expression Engine
  3. #
  4. # convert template to internal format
  5. #
  6. # Copyright (c) 1997-2001 by Secret Labs AB.  All rights reserved.
  7. #
  8. # See the sre.py file for information on usage and redistribution.
  9. #
  10.  
  11. """Internal support module for sre"""
  12.  
  13. import _sre, sys
  14. import sre_parse
  15. from sre_constants import *
  16.  
  17. assert _sre.MAGIC == MAGIC, "SRE module mismatch"
  18.  
  19. if _sre.CODESIZE == 2:
  20.     MAXCODE = 65535
  21. else:
  22.     MAXCODE = 0xFFFFFFFF
  23.  
  24. def _identityfunction(x):
  25.     return x
  26.  
  27. _LITERAL_CODES = set([LITERAL, NOT_LITERAL])
  28. _REPEATING_CODES = set([REPEAT, MIN_REPEAT, MAX_REPEAT])
  29. _SUCCESS_CODES = set([SUCCESS, FAILURE])
  30. _ASSERT_CODES = set([ASSERT, ASSERT_NOT])
  31.  
  32. def _compile(code, pattern, flags):
  33.     # internal: compile a (sub)pattern
  34.     emit = code.append
  35.     _len = len
  36.     LITERAL_CODES = _LITERAL_CODES
  37.     REPEATING_CODES = _REPEATING_CODES
  38.     SUCCESS_CODES = _SUCCESS_CODES
  39.     ASSERT_CODES = _ASSERT_CODES
  40.     for op, av in pattern:
  41.         if op in LITERAL_CODES:
  42.             if flags & SRE_FLAG_IGNORECASE:
  43.                 emit(OPCODES[OP_IGNORE[op]])
  44.                 emit(_sre.getlower(av, flags))
  45.             else:
  46.                 emit(OPCODES[op])
  47.                 emit(av)
  48.         elif op is IN:
  49.             if flags & SRE_FLAG_IGNORECASE:
  50.                 emit(OPCODES[OP_IGNORE[op]])
  51.                 def fixup(literal, flags=flags):
  52.                     return _sre.getlower(literal, flags)
  53.             else:
  54.                 emit(OPCODES[op])
  55.                 fixup = _identityfunction
  56.             skip = _len(code); emit(0)
  57.             _compile_charset(av, flags, code, fixup)
  58.             code[skip] = _len(code) - skip
  59.         elif op is ANY:
  60.             if flags & SRE_FLAG_DOTALL:
  61.                 emit(OPCODES[ANY_ALL])
  62.             else:
  63.                 emit(OPCODES[ANY])
  64.         elif op in REPEATING_CODES:
  65.             if flags & SRE_FLAG_TEMPLATE:
  66.                 raise error("internal: unsupported template operator")
  67.                 emit(OPCODES[REPEAT])
  68.                 skip = _len(code); emit(0)
  69.                 emit(av[0])
  70.                 emit(av[1])
  71.                 _compile(code, av[2], flags)
  72.                 emit(OPCODES[SUCCESS])
  73.                 code[skip] = _len(code) - skip
  74.             elif _simple(av) and op is not REPEAT:
  75.                 if op is MAX_REPEAT:
  76.                     emit(OPCODES[REPEAT_ONE])
  77.                 else:
  78.                     emit(OPCODES[MIN_REPEAT_ONE])
  79.                 skip = _len(code); emit(0)
  80.                 emit(av[0])
  81.                 emit(av[1])
  82.                 _compile(code, av[2], flags)
  83.                 emit(OPCODES[SUCCESS])
  84.                 code[skip] = _len(code) - skip
  85.             else:
  86.                 emit(OPCODES[REPEAT])
  87.                 skip = _len(code); emit(0)
  88.                 emit(av[0])
  89.                 emit(av[1])
  90.                 _compile(code, av[2], flags)
  91.                 code[skip] = _len(code) - skip
  92.                 if op is MAX_REPEAT:
  93.                     emit(OPCODES[MAX_UNTIL])
  94.                 else:
  95.                     emit(OPCODES[MIN_UNTIL])
  96.         elif op is SUBPATTERN:
  97.             if av[0]:
  98.                 emit(OPCODES[MARK])
  99.                 emit((av[0]-1)*2)
  100.             # _compile_info(code, av[1], flags)
  101.             _compile(code, av[1], flags)
  102.             if av[0]:
  103.                 emit(OPCODES[MARK])
  104.                 emit((av[0]-1)*2+1)
  105.         elif op in SUCCESS_CODES:
  106.             emit(OPCODES[op])
  107.         elif op in ASSERT_CODES:
  108.             emit(OPCODES[op])
  109.             skip = _len(code); emit(0)
  110.             if av[0] >= 0:
  111.                 emit(0) # look ahead
  112.             else:
  113.                 lo, hi = av[1].getwidth()
  114.                 if lo != hi:
  115.                     raise error("look-behind requires fixed-width pattern")
  116.                 emit(lo) # look behind
  117.             _compile(code, av[1], flags)
  118.             emit(OPCODES[SUCCESS])
  119.             code[skip] = _len(code) - skip
  120.         elif op is CALL:
  121.             emit(OPCODES[op])
  122.             skip = _len(code); emit(0)
  123.             _compile(code, av, flags)
  124.             emit(OPCODES[SUCCESS])
  125.             code[skip] = _len(code) - skip
  126.         elif op is AT:
  127.             emit(OPCODES[op])
  128.             if flags & SRE_FLAG_MULTILINE:
  129.                 av = AT_MULTILINE.get(av, av)
  130.             if flags & SRE_FLAG_LOCALE:
  131.                 av = AT_LOCALE.get(av, av)
  132.             elif flags & SRE_FLAG_UNICODE:
  133.                 av = AT_UNICODE.get(av, av)
  134.             emit(ATCODES[av])
  135.         elif op is BRANCH:
  136.             emit(OPCODES[op])
  137.             tail = []
  138.             tailappend = tail.append
  139.             for av in av[1]:
  140.                 skip = _len(code); emit(0)
  141.                 # _compile_info(code, av, flags)
  142.                 _compile(code, av, flags)
  143.                 emit(OPCODES[JUMP])
  144.                 tailappend(_len(code)); emit(0)
  145.                 code[skip] = _len(code) - skip
  146.             emit(0) # end of branch
  147.             for tail in tail:
  148.                 code[tail] = _len(code) - tail
  149.         elif op is CATEGORY:
  150.             emit(OPCODES[op])
  151.             if flags & SRE_FLAG_LOCALE:
  152.                 av = CH_LOCALE[av]
  153.             elif flags & SRE_FLAG_UNICODE:
  154.                 av = CH_UNICODE[av]
  155.             emit(CHCODES[av])
  156.         elif op is GROUPREF:
  157.             if flags & SRE_FLAG_IGNORECASE:
  158.                 emit(OPCODES[OP_IGNORE[op]])
  159.             else:
  160.                 emit(OPCODES[op])
  161.             emit(av-1)
  162.         elif op is GROUPREF_EXISTS:
  163.             emit(OPCODES[op])
  164.             emit(av[0]-1)
  165.             skipyes = _len(code); emit(0)
  166.             _compile(code, av[1], flags)
  167.             if av[2]:
  168.                 emit(OPCODES[JUMP])
  169.                 skipno = _len(code); emit(0)
  170.                 code[skipyes] = _len(code) - skipyes + 1
  171.                 _compile(code, av[2], flags)
  172.                 code[skipno] = _len(code) - skipno
  173.             else:
  174.                 code[skipyes] = _len(code) - skipyes + 1
  175.         else:
  176.             raise ValueError("unsupported operand type", op)
  177.  
  178. def _compile_charset(charset, flags, code, fixup=None):
  179.     # compile charset subprogram
  180.     emit = code.append
  181.     if fixup is None:
  182.         fixup = _identityfunction
  183.     for op, av in _optimize_charset(charset, fixup):
  184.         emit(OPCODES[op])
  185.         if op is NEGATE:
  186.             pass
  187.         elif op is LITERAL:
  188.             emit(fixup(av))
  189.         elif op is RANGE:
  190.             emit(fixup(av[0]))
  191.             emit(fixup(av[1]))
  192.         elif op is CHARSET:
  193.             code.extend(av)
  194.         elif op is BIGCHARSET:
  195.             code.extend(av)
  196.         elif op is CATEGORY:
  197.             if flags & SRE_FLAG_LOCALE:
  198.                 emit(CHCODES[CH_LOCALE[av]])
  199.             elif flags & SRE_FLAG_UNICODE:
  200.                 emit(CHCODES[CH_UNICODE[av]])
  201.             else:
  202.                 emit(CHCODES[av])
  203.         else:
  204.             raise error("internal: unsupported set operator")
  205.     emit(OPCODES[FAILURE])
  206.  
  207. def _optimize_charset(charset, fixup):
  208.     # internal: optimize character set
  209.     out = []
  210.     outappend = out.append
  211.     charmap = [0]*256
  212.     try:
  213.         for op, av in charset:
  214.             if op is NEGATE:
  215.                 outappend((op, av))
  216.             elif op is LITERAL:
  217.                 charmap[fixup(av)] = 1
  218.             elif op is RANGE:
  219.                 for i in range(fixup(av[0]), fixup(av[1])+1):
  220.                     charmap[i] = 1
  221.             elif op is CATEGORY:
  222.                 # XXX: could append to charmap tail
  223.                 return charset # cannot compress
  224.     except IndexError:
  225.         # character set contains unicode characters
  226.         return _optimize_unicode(charset, fixup)
  227.     # compress character map
  228.     i = p = n = 0
  229.     runs = []
  230.     runsappend = runs.append
  231.     for c in charmap:
  232.         if c:
  233.             if n == 0:
  234.                 p = i
  235.             n = n + 1
  236.         elif n:
  237.             runsappend((p, n))
  238.             n = 0
  239.         i = i + 1
  240.     if n:
  241.         runsappend((p, n))
  242.     if len(runs) <= 2:
  243.         # use literal/range
  244.         for p, n in runs:
  245.             if n == 1:
  246.                 outappend((LITERAL, p))
  247.             else:
  248.                 outappend((RANGE, (p, p+n-1)))
  249.         if len(out) < len(charset):
  250.             return out
  251.     else:
  252.         # use bitmap
  253.         data = _mk_bitmap(charmap)
  254.         outappend((CHARSET, data))
  255.         return out
  256.     return charset
  257.  
  258. def _mk_bitmap(bits):
  259.     data = []
  260.     dataappend = data.append
  261.     if _sre.CODESIZE == 2:
  262.         start = (1, 0)
  263.     else:
  264.         start = (1, 0)
  265.     m, v = start
  266.     for c in bits:
  267.         if c:
  268.             v = v + m
  269.         m = m + m
  270.         if m > MAXCODE:
  271.             dataappend(v)
  272.             m, v = start
  273.     return data
  274.  
  275. # To represent a big charset, first a bitmap of all characters in the
  276. # set is constructed. Then, this bitmap is sliced into chunks of 256
  277. # characters, duplicate chunks are eliminated, and each chunk is
  278. # given a number. In the compiled expression, the charset is
  279. # represented by a 32-bit word sequence, consisting of one word for
  280. # the number of different chunks, a sequence of 256 bytes (64 words)
  281. # of chunk numbers indexed by their original chunk position, and a
  282. # sequence of 256-bit chunks (8 words each).
  283.  
  284. # Compression is normally good: in a typical charset, large ranges of
  285. # Unicode will be either completely excluded (e.g. if only cyrillic
  286. # letters are to be matched), or completely included (e.g. if large
  287. # subranges of Kanji match). These ranges will be represented by
  288. # chunks of all one-bits or all zero-bits.
  289.  
  290. # Matching can be also done efficiently: the more significant byte of
  291. # the Unicode character is an index into the chunk number, and the
  292. # less significant byte is a bit index in the chunk (just like the
  293. # CHARSET matching).
  294.  
  295. # The BIGCHARSET opcode still supports only subsets
  296. # of the basic multilingual plane; an efficient representation
  297. # for all of Unicode has not yet been developed. This means,
  298. # in particular, that negated charsets cannot be represented as
  299. # bigcharsets.
  300.  
  301. def _optimize_unicode(charset, fixup):
  302.     try:
  303.         import array
  304.     except ImportError:
  305.         return charset
  306.     charmap = [0]*65536
  307.     negate = 0
  308.     try:
  309.         for op, av in charset:
  310.             if op is NEGATE:
  311.                 negate = 1
  312.             elif op is LITERAL:
  313.                 charmap[fixup(av)] = 1
  314.             elif op is RANGE:
  315.                 for i in range(fixup(av[0]), fixup(av[1])+1):
  316.                     charmap[i] = 1
  317.             elif op is CATEGORY:
  318.                 # XXX: could expand category
  319.                 return charset # cannot compress
  320.     except IndexError:
  321.         # non-BMP characters; XXX now they should work
  322.         return charset
  323.     if negate:
  324.         if sys.maxunicode != 65535:
  325.             # XXX: negation does not work with big charsets
  326.             # XXX2: now they should work, but removing this will make the
  327.             # charmap 17 times bigger
  328.             return charset
  329.         for i in range(65536):
  330.             charmap[i] = not charmap[i]
  331.     comps = {}
  332.     mapping = [0]*256
  333.     block = 0
  334.     data = []
  335.     for i in range(256):
  336.         chunk = tuple(charmap[i*256:(i+1)*256])
  337.         new = comps.setdefault(chunk, block)
  338.         mapping[i] = new
  339.         if new == block:
  340.             block = block + 1
  341.             data = data + _mk_bitmap(chunk)
  342.     header = [block]
  343.     if _sre.CODESIZE == 2:
  344.         code = 'H'
  345.     else:
  346.         code = 'I'
  347.     # Convert block indices to byte array of 256 bytes
  348.     mapping = array.array('B', mapping).tobytes()
  349.     # Convert byte array to word array
  350.     mapping = array.array(code, mapping)
  351.     assert mapping.itemsize == _sre.CODESIZE
  352.     assert len(mapping) * mapping.itemsize == 256
  353.     header = header + mapping.tolist()
  354.     data[0:0] = header
  355.     return [(BIGCHARSET, data)]
  356.  
  357. def _simple(av):
  358.     # check if av is a "simple" operator
  359.     lo, hi = av[2].getwidth()
  360.     return lo == hi == 1 and av[2][0][0] != SUBPATTERN
  361.  
  362. def _compile_info(code, pattern, flags):
  363.     # internal: compile an info block.  in the current version,
  364.     # this contains min/max pattern width, and an optional literal
  365.     # prefix or a character map
  366.     lo, hi = pattern.getwidth()
  367.     if lo == 0:
  368.         return # not worth it
  369.     # look for a literal prefix
  370.     prefix = []
  371.     prefixappend = prefix.append
  372.     prefix_skip = 0
  373.     charset = [] # not used
  374.     charsetappend = charset.append
  375.     if not (flags & SRE_FLAG_IGNORECASE):
  376.         # look for literal prefix
  377.         for op, av in pattern.data:
  378.             if op is LITERAL:
  379.                 if len(prefix) == prefix_skip:
  380.                     prefix_skip = prefix_skip + 1
  381.                 prefixappend(av)
  382.             elif op is SUBPATTERN and len(av[1]) == 1:
  383.                 op, av = av[1][0]
  384.                 if op is LITERAL:
  385.                     prefixappend(av)
  386.                 else:
  387.                     break
  388.             else:
  389.                 break
  390.         # if no prefix, look for charset prefix
  391.         if not prefix and pattern.data:
  392.             op, av = pattern.data[0]
  393.             if op is SUBPATTERN and av[1]:
  394.                 op, av = av[1][0]
  395.                 if op is LITERAL:
  396.                     charsetappend((op, av))
  397.                 elif op is BRANCH:
  398.                     c = []
  399.                     cappend = c.append
  400.                     for p in av[1]:
  401.                         if not p:
  402.                             break
  403.                         op, av = p[0]
  404.                         if op is LITERAL:
  405.                             cappend((op, av))
  406.                         else:
  407.                             break
  408.                     else:
  409.                         charset = c
  410.             elif op is BRANCH:
  411.                 c = []
  412.                 cappend = c.append
  413.                 for p in av[1]:
  414.                     if not p:
  415.                         break
  416.                     op, av = p[0]
  417.                     if op is LITERAL:
  418.                         cappend((op, av))
  419.                     else:
  420.                         break
  421.                 else:
  422.                     charset = c
  423.             elif op is IN:
  424.                 charset = av
  425. ##     if prefix:
  426. ##         print "*** PREFIX", prefix, prefix_skip
  427. ##     if charset:
  428. ##         print "*** CHARSET", charset
  429.     # add an info block
  430.     emit = code.append
  431.     emit(OPCODES[INFO])
  432.     skip = len(code); emit(0)
  433.     # literal flag
  434.     mask = 0
  435.     if prefix:
  436.         mask = SRE_INFO_PREFIX
  437.         if len(prefix) == prefix_skip == len(pattern.data):
  438.             mask = mask + SRE_INFO_LITERAL
  439.     elif charset:
  440.         mask = mask + SRE_INFO_CHARSET
  441.     emit(mask)
  442.     # pattern length
  443.     if lo < MAXCODE:
  444.         emit(lo)
  445.     else:
  446.         emit(MAXCODE)
  447.         prefix = prefix[:MAXCODE]
  448.     if hi < MAXCODE:
  449.         emit(hi)
  450.     else:
  451.         emit(0)
  452.     # add literal prefix
  453.     if prefix:
  454.         emit(len(prefix)) # length
  455.         emit(prefix_skip) # skip
  456.         code.extend(prefix)
  457.         # generate overlap table
  458.         table = [-1] + ([0]*len(prefix))
  459.         for i in range(len(prefix)):
  460.             table[i+1] = table[i]+1
  461.             while table[i+1] > 0 and prefix[i] != prefix[table[i+1]-1]:
  462.                 table[i+1] = table[table[i+1]-1]+1
  463.         code.extend(table[1:]) # don't store first entry
  464.     elif charset:
  465.         _compile_charset(charset, flags, code)
  466.     code[skip] = len(code) - skip
  467.  
  468. def isstring(obj):
  469.     return isinstance(obj, (str, bytes))
  470.  
  471. def _code(p, flags):
  472.  
  473.     flags = p.pattern.flags | flags
  474.     code = []
  475.  
  476.     # compile info block
  477.     _compile_info(code, p, flags)
  478.  
  479.     # compile the pattern
  480.     _compile(code, p.data, flags)
  481.  
  482.     code.append(OPCODES[SUCCESS])
  483.  
  484.     return code
  485.  
  486. def compile(p, flags=0):
  487.     # internal: convert pattern list to internal format
  488.  
  489.     if isstring(p):
  490.         pattern = p
  491.         p = sre_parse.parse(p, flags)
  492.     else:
  493.         pattern = None
  494.  
  495.     code = _code(p, flags)
  496.  
  497.     # print code
  498.  
  499.     # XXX: <fl> get rid of this limitation!
  500.     if p.pattern.groups > 100:
  501.         raise AssertionError(
  502.             "sorry, but this version only supports 100 named groups"
  503.             )
  504.  
  505.     # map in either direction
  506.     groupindex = p.pattern.groupdict
  507.     indexgroup = [None] * p.pattern.groups
  508.     for k, i in groupindex.items():
  509.         indexgroup[i] = k
  510.  
  511.     return _sre.compile(
  512.         pattern, flags | p.pattern.flags, code,
  513.         p.pattern.groups-1,
  514.         groupindex, indexgroup
  515.         )
  516.