1""" Standard "encodings" Package
2
3    Standard Python encoding modules are stored in this package
4    directory.
5
6    Codec modules must have names corresponding to normalized encoding
7    names as defined in the normalize_encoding() function below, e.g.
8    'utf-8' must be implemented by the module 'utf_8.py'.
9
10    Each codec module must export the following interface:
11
12    * getregentry() -> codecs.CodecInfo object
13    The getregentry() API must return a CodecInfo object with encoder, decoder,
14    incrementalencoder, incrementaldecoder, streamwriter and streamreader
15    atttributes which adhere to the Python Codec Interface Standard.
16
17    In addition, a module may optionally also define the following
18    APIs which are then used by the package's codec search function:
19
20    * getaliases() -> sequence of encoding name strings to use as aliases
21
22    Alias names returned by getaliases() must be normalized encoding
23    names as defined by normalize_encoding().
24
25Written by Marc-Andre Lemburg (mal@lemburg.com).
26
27(c) Copyright CNRI, All Rights Reserved. NO WARRANTY.
28
29"""#"
30
31import codecs
32import sys
33from . import aliases
34
35_cache = {}
36_unknown = '--unknown--'
37_import_tail = ['*']
38_aliases = aliases.aliases
39
40class CodecRegistryError(LookupError, SystemError):
41    pass
42
43def normalize_encoding(encoding):
44
45    """ Normalize an encoding name.
46
47        Normalization works as follows: all non-alphanumeric
48        characters except the dot used for Python package names are
49        collapsed and replaced with a single underscore, e.g. '  -;#'
50        becomes '_'. Leading and trailing underscores are removed.
51
52        Note that encoding names should be ASCII only; if they do use
53        non-ASCII characters, these must be Latin-1 compatible.
54
55    """
56    if isinstance(encoding, bytes):
57        encoding = str(encoding, "ascii")
58
59    chars = []
60    punct = False
61    for c in encoding:
62        if c.isalnum() or c == '.':
63            if punct and chars:
64                chars.append('_')
65            chars.append(c)
66            punct = False
67        else:
68            punct = True
69    return ''.join(chars)
70
71def search_function(encoding):
72
73    # Cache lookup
74    entry = _cache.get(encoding, _unknown)
75    if entry is not _unknown:
76        return entry
77
78    # Import the module:
79    #
80    # First try to find an alias for the normalized encoding
81    # name and lookup the module using the aliased name, then try to
82    # lookup the module using the standard import scheme, i.e. first
83    # try in the encodings package, then at top-level.
84    #
85    norm_encoding = normalize_encoding(encoding)
86    aliased_encoding = _aliases.get(norm_encoding) or \
87                       _aliases.get(norm_encoding.replace('.', '_'))
88    if aliased_encoding is not None:
89        modnames = [aliased_encoding,
90                    norm_encoding]
91    else:
92        modnames = [norm_encoding]
93    for modname in modnames:
94        if not modname or '.' in modname:
95            continue
96        try:
97            # Import is absolute to prevent the possibly malicious import of a
98            # module with side-effects that is not in the 'encodings' package.
99            mod = __import__('encodings.' + modname, fromlist=_import_tail,
100                             level=0)
101        except ImportError:
102            # ImportError may occur because 'encodings.(modname)' does not exist,
103            # or because it imports a name that does not exist (see mbcs and oem)
104            pass
105        else:
106            break
107    else:
108        mod = None
109
110    try:
111        getregentry = mod.getregentry
112    except AttributeError:
113        # Not a codec module
114        mod = None
115
116    if mod is None:
117        # Cache misses
118        _cache[encoding] = None
119        return None
120
121    # Now ask the module for the registry entry
122    entry = getregentry()
123    if not isinstance(entry, codecs.CodecInfo):
124        if not 4 <= len(entry) <= 7:
125            raise CodecRegistryError('module "%s" (%s) failed to register'
126                                     % (mod.__name__, mod.__file__))
127        if not callable(entry[0]) or not callable(entry[1]) or \
128           (entry[2] is not None and not callable(entry[2])) or \
129           (entry[3] is not None and not callable(entry[3])) or \
130           (len(entry) > 4 and entry[4] is not None and not callable(entry[4])) or \
131           (len(entry) > 5 and entry[5] is not None and not callable(entry[5])):
132            raise CodecRegistryError('incompatible codecs in module "%s" (%s)'
133                                     % (mod.__name__, mod.__file__))
134        if len(entry)<7 or entry[6] is None:
135            entry += (None,)*(6-len(entry)) + (mod.__name__.split(".", 1)[1],)
136        entry = codecs.CodecInfo(*entry)
137
138    # Cache the codec registry entry
139    _cache[encoding] = entry
140
141    # Register its aliases (without overwriting previously registered
142    # aliases)
143    try:
144        codecaliases = mod.getaliases()
145    except AttributeError:
146        pass
147    else:
148        for alias in codecaliases:
149            if alias not in _aliases:
150                _aliases[alias] = modname
151
152    # Return the registry entry
153    return entry
154
155# Register the search_function in the Python codec registry
156codecs.register(search_function)
157
158if sys.platform == 'win32':
159    def _alias_mbcs(encoding):
160        try:
161            import _bootlocale
162            if encoding == _bootlocale.getpreferredencoding(False):
163                import encodings.mbcs
164                return encodings.mbcs.getregentry()
165        except ImportError:
166            # Imports may fail while we are shutting down
167            pass
168
169    codecs.register(_alias_mbcs)
170