mirror of
https://github.com/morpheus65535/bazarr.git
synced 2025-01-08 15:57:36 +08:00
214 lines
7.7 KiB
Python
214 lines
7.7 KiB
Python
# -*- coding: utf-8 -*-
|
||
"""
|
||
This gives other modules access to the gritty details about characters and the
|
||
encodings that use them.
|
||
"""
|
||
|
||
import re
|
||
import zlib
|
||
import unicodedata
|
||
import itertools
|
||
from pkg_resources import resource_string
|
||
from ftfy.compatibility import unichr
|
||
|
||
# These are the encodings we will try to fix in ftfy, in the
|
||
# order that they should be tried.
|
||
CHARMAP_ENCODINGS = [
|
||
u'latin-1',
|
||
u'sloppy-windows-1252',
|
||
u'sloppy-windows-1250',
|
||
u'iso-8859-2',
|
||
u'sloppy-windows-1251',
|
||
u'macroman',
|
||
u'cp437',
|
||
]
|
||
|
||
|
||
def _build_regexes():
|
||
"""
|
||
ENCODING_REGEXES contain reasonably fast ways to detect if we
|
||
could represent a given string in a given encoding. The simplest one is
|
||
the u'ascii' detector, which of course just determines if all characters
|
||
are between U+0000 and U+007F.
|
||
"""
|
||
# Define a regex that matches ASCII text.
|
||
encoding_regexes = {u'ascii': re.compile('^[\x00-\x7f]*$')}
|
||
|
||
for encoding in CHARMAP_ENCODINGS:
|
||
# Make a sequence of characters that bytes \x80 to \xFF decode to
|
||
# in each encoding, as well as byte \x1A, which is used to represent
|
||
# the replacement character <20> in the sloppy-* encodings.
|
||
latin1table = u''.join(unichr(i) for i in range(128, 256)) + '\x1a'
|
||
charlist = latin1table.encode(u'latin-1').decode(encoding)
|
||
|
||
# The rest of the ASCII bytes -- bytes \x00 to \x19 and \x1B
|
||
# to \x7F -- will decode as those ASCII characters in any encoding we
|
||
# support, so we can just include them as ranges. This also lets us
|
||
# not worry about escaping regex special characters, because all of
|
||
# them are in the \x1B to \x7F range.
|
||
regex = u'^[\x00-\x19\x1b-\x7f{0}]*$'.format(charlist)
|
||
encoding_regexes[encoding] = re.compile(regex)
|
||
return encoding_regexes
|
||
ENCODING_REGEXES = _build_regexes()
|
||
|
||
|
||
def _build_utf8_punct_regex():
|
||
"""
|
||
Recognize UTF-8 mojibake that's so blatant that we can fix it even when the
|
||
rest of the string doesn't decode as UTF-8 -- namely, UTF-8 sequences for
|
||
the u'General Punctuation' characters U+2000 to U+2040, re-encoded in
|
||
Windows-1252.
|
||
|
||
These are recognizable by the distinctiveu'â€u' ('\xe2\x80') sequence they
|
||
all begin with when decoded as Windows-1252.
|
||
"""
|
||
# We're making a regex that has all the literal bytes from 0x80 to 0xbf in
|
||
# a range. "Couldn't this have just said [\x80-\xbf]?", you might ask.
|
||
# However, when we decode the regex as Windows-1252, the resulting
|
||
# characters won't even be remotely contiguous.
|
||
#
|
||
# Unrelatedly, the expression that generates these bytes will be so much
|
||
# prettier when we deprecate Python 2.
|
||
continuation_char_list = ''.join(
|
||
unichr(i) for i in range(0x80, 0xc0)
|
||
).encode(u'latin-1')
|
||
obvious_utf8 = (u'â€['
|
||
+ continuation_char_list.decode(u'sloppy-windows-1252')
|
||
+ u']')
|
||
return re.compile(obvious_utf8)
|
||
PARTIAL_UTF8_PUNCT_RE = _build_utf8_punct_regex()
|
||
|
||
|
||
# Recognize UTF-8 sequences that would be valid if it weren't for a b'\xa0'
|
||
# that some Windows-1252 program converted to a plain space.
|
||
#
|
||
# The smaller values are included on a case-by-case basis, because we don't want
|
||
# to decode likely input sequences to unlikely characters. These are the ones
|
||
# that *do* form likely characters before 0xa0:
|
||
#
|
||
# 0xc2 -> U+A0 NO-BREAK SPACE
|
||
# 0xc3 -> U+E0 LATIN SMALL LETTER A WITH GRAVE
|
||
# 0xc5 -> U+160 LATIN CAPITAL LETTER S WITH CARON
|
||
# 0xce -> U+3A0 GREEK CAPITAL LETTER PI
|
||
# 0xd0 -> U+420 CYRILLIC CAPITAL LETTER ER
|
||
#
|
||
# These still need to come with a cost, so that they only get converted when
|
||
# there's evidence that it fixes other things. Any of these could represent
|
||
# characters that legitimately appear surrounded by spaces, particularly U+C5
|
||
# (Å), which is a word in multiple languages!
|
||
#
|
||
# We should consider checking for b'\x85' being converted to ... in the future.
|
||
# I've seen it once, but the text still wasn't recoverable.
|
||
|
||
ALTERED_UTF8_RE = re.compile(b'[\xc2\xc3\xc5\xce\xd0][ ]'
|
||
b'|[\xe0-\xef][ ][\x80-\xbf]'
|
||
b'|[\xe0-\xef][\x80-\xbf][ ]'
|
||
b'|[\xf0-\xf4][ ][\x80-\xbf][\x80-\xbf]'
|
||
b'|[\xf0-\xf4][\x80-\xbf][ ][\x80-\xbf]'
|
||
b'|[\xf0-\xf4][\x80-\xbf][\x80-\xbf][ ]')
|
||
|
||
# This expression matches UTF-8 and CESU-8 sequences where some of the
|
||
# continuation bytes have been lost. The byte 0x1a (sometimes written as ^Z) is
|
||
# used within ftfy to represent a byte that produced the replacement character
|
||
# \ufffd. We don't know which byte it was, but we can at least decode the UTF-8
|
||
# sequence as \ufffd instead of failing to re-decode it at all.
|
||
LOSSY_UTF8_RE = re.compile(
|
||
b'[\xc2-\xdf][\x1a]'
|
||
b'|\xed[\xa0-\xaf][\x1a]\xed[\xb0-\xbf][\x1a\x80-\xbf]'
|
||
b'|\xed[\xa0-\xaf][\x1a\x80-\xbf]\xed[\xb0-\xbf][\x1a]'
|
||
b'|[\xe0-\xef][\x1a][\x1a\x80-\xbf]'
|
||
b'|[\xe0-\xef][\x1a\x80-\xbf][\x1a]'
|
||
b'|[\xf0-\xf4][\x1a][\x1a\x80-\xbf][\x1a\x80-\xbf]'
|
||
b'|[\xf0-\xf4][\x1a\x80-\xbf][\x1a][\x1a\x80-\xbf]'
|
||
b'|[\xf0-\xf4][\x1a\x80-\xbf][\x1a\x80-\xbf][\x1a]'
|
||
b'|\x1a'
|
||
)
|
||
|
||
# These regexes match various Unicode variations on single and double quotes.
|
||
SINGLE_QUOTE_RE = re.compile(u'[\u2018-\u201b]')
|
||
DOUBLE_QUOTE_RE = re.compile(u'[\u201c-\u201f]')
|
||
|
||
|
||
def possible_encoding(text, encoding):
|
||
"""
|
||
Given text and a single-byte encoding, check whether that text could have
|
||
been decoded from that single-byte encoding.
|
||
|
||
In other words, check whether it can be encoded in that encoding, possibly
|
||
sloppily.
|
||
"""
|
||
return bool(ENCODING_REGEXES[encoding].match(text))
|
||
|
||
|
||
CHAR_CLASS_STRING = zlib.decompress(
|
||
resource_string(__name__, 'char_classes.dat')
|
||
).decode(u'ascii')
|
||
|
||
def chars_to_classes(string):
|
||
"""
|
||
Convert each Unicode character to a letter indicating which of many
|
||
classes it's in.
|
||
|
||
See build_data.py for where this data comes from and what it means.
|
||
"""
|
||
return string.translate(CHAR_CLASS_STRING)
|
||
|
||
|
||
def _build_control_char_mapping():
|
||
"""
|
||
Build a translate mapping that strips likely-unintended control characters.
|
||
See :func:`ftfy.fixes.remove_control_chars` for a description of these
|
||
codepoint ranges and why they should be removed.
|
||
"""
|
||
control_chars = {}
|
||
|
||
for i in itertools.chain(
|
||
range(0x00, 0x09), [0x0b],
|
||
range(0x0e, 0x20), [0x7f],
|
||
range(0x206a, 0x2070),
|
||
[0xfeff],
|
||
range(0xfff9, 0xfffd),
|
||
range(0x1d173, 0x1d17b),
|
||
range(0xe0000, 0xe0080)
|
||
):
|
||
control_chars[i] = None
|
||
|
||
return control_chars
|
||
CONTROL_CHARS = _build_control_char_mapping()
|
||
|
||
|
||
# A translate mapping that breaks ligatures made of Latin letters. While
|
||
# ligatures may be important to the representation of other languages, in
|
||
# Latin letters they tend to represent a copy/paste error.
|
||
#
|
||
# Ligatures may also be separated by NFKC normalization, but that is sometimes
|
||
# more normalization than you want.
|
||
LIGATURES = {
|
||
ord(u'IJ'): u'IJ',
|
||
ord(u'ij'): u'ij',
|
||
ord(u'ff'): u'ff',
|
||
ord(u'fi'): u'fi',
|
||
ord(u'fl'): u'fl',
|
||
ord(u'ffi'): u'ffi',
|
||
ord(u'ffl'): u'ffl',
|
||
ord(u'ſt'): u'ſt',
|
||
ord(u'st'): u'st'
|
||
}
|
||
|
||
|
||
def _build_width_map():
|
||
"""
|
||
Build a translate mapping that replaces halfwidth and fullwidth forms
|
||
with their standard-width forms.
|
||
"""
|
||
# Though it's not listed as a fullwidth character, we'll want to convert
|
||
# U+3000 IDEOGRAPHIC SPACE to U+20 SPACE on the same principle, so start
|
||
# with that in the dictionary.
|
||
width_map = {0x3000: u' '}
|
||
for i in range(0xff01, 0xfff0):
|
||
char = unichr(i)
|
||
alternate = unicodedata.normalize(u'NFKC', char)
|
||
if alternate != char:
|
||
width_map[i] = alternate
|
||
return width_map
|
||
WIDTH_MAP = _build_width_map()
|