summaryrefslogtreecommitdiff
path: root/Lib/test/test_source_encoding.py
blob: 46f00401c5a6b8e86e0bbf03b8b5c8d81e846ce5 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
# -*- coding: koi8-r -*-

import unittest
from test.test_support import run_unittest, rmtree, captured_stdout
import script_helper
import os
import tempfile

class MiscSourceEncodingTest(unittest.TestCase):

    def test_pep263(self):
        self.assertEqual(
            u"ðÉÔÏÎ".encode("utf-8"),
            '\xd0\x9f\xd0\xb8\xd1\x82\xd0\xbe\xd0\xbd'
        )
        self.assertEqual(
            u"\ð".encode("utf-8"),
            '\\\xd0\x9f'
        )

    def test_compilestring(self):
        # see #1882
        c = compile("\n# coding: utf-8\nu = u'\xc3\xb3'\n", "dummy", "exec")
        d = {}
        exec c in d
        self.assertEqual(d['u'], u'\xf3')

    def test_issue3297(self):
        c = compile("a, b = '\U0001010F', '\\U0001010F'", "dummy", "exec")
        d = {}
        exec(c, d)
        self.assertEqual(d['a'], d['b'])
        self.assertEqual(len(d['a']), len(d['b']))

    def test_issue7820(self):
        # Ensure that check_bom() restores all bytes in the right order if
        # check_bom() fails in pydebug mode: a buffer starts with the first
        # byte of a valid BOM, but next bytes are different

        # one byte in common with the UTF-16-LE BOM
        self.assertRaises(SyntaxError, eval, '\xff\x20')

        # two bytes in common with the UTF-8 BOM
        self.assertRaises(SyntaxError, eval, '\xef\xbb\x20')

    def test_error_message(self):
        compile('# -*- coding: iso-8859-15 -*-\n', 'dummy', 'exec')
        compile('\xef\xbb\xbf\n', 'dummy', 'exec')
        compile('\xef\xbb\xbf# -*- coding: utf-8 -*-\n', 'dummy', 'exec')
        with self.assertRaisesRegexp(SyntaxError, 'fake'):
            compile('# -*- coding: fake -*-\n', 'dummy', 'exec')
        with self.assertRaisesRegexp(SyntaxError, 'iso-8859-15'):
            compile('\xef\xbb\xbf# -*- coding: iso-8859-15 -*-\n',
                    'dummy', 'exec')
        with self.assertRaisesRegexp(SyntaxError, 'BOM'):
            compile('\xef\xbb\xbf# -*- coding: iso-8859-15 -*-\n',
                    'dummy', 'exec')
        with self.assertRaisesRegexp(SyntaxError, 'fake'):
            compile('\xef\xbb\xbf# -*- coding: fake -*-\n', 'dummy', 'exec')
        with self.assertRaisesRegexp(SyntaxError, 'BOM'):
            compile('\xef\xbb\xbf# -*- coding: fake -*-\n', 'dummy', 'exec')

    def test_non_unicode_codec(self):
        with self.assertRaisesRegexp(SyntaxError,
                                     'codec did not return a unicode'):
            from test import bad_coding3


    def test_bad_coding(self):
        module_name = 'bad_coding'
        self.verify_bad_module(module_name)

    def test_bad_coding2(self):
        module_name = 'bad_coding2'
        self.verify_bad_module(module_name)

    def verify_bad_module(self, module_name):
        self.assertRaises(SyntaxError, __import__, 'test.' + module_name)

        path = os.path.dirname(__file__)
        filename = os.path.join(path, module_name + '.py')
        with open(filename) as fp:
            text = fp.read()
        self.assertRaises(SyntaxError, compile, text, filename, 'exec')

    def test_error_from_string(self):
        # See http://bugs.python.org/issue6289
        input = u"# coding: ascii\n\N{SNOWMAN}".encode('utf-8')
        with self.assertRaises(SyntaxError) as c:
            compile(input, "<string>", "exec")
        expected = "'ascii' codec can't decode byte 0xe2 in position 16: " \
                   "ordinal not in range(128)"
        self.assertTrue(c.exception.args[0].startswith(expected))


class AbstractSourceEncodingTest:

    def test_first_coding_line(self):
        src = ('#coding:iso8859-15\n'
               'print(repr(u"\xc3\xa4"))\n')
        self.check_script_output(src, r"u'\xc3\u20ac'")

    def test_second_coding_line(self):
        src = ('#\n'
               '#coding:iso8859-15\n'
               'print(repr(u"\xc3\xa4"))\n')
        self.check_script_output(src, r"u'\xc3\u20ac'")

    def test_double_coding_line(self):
        # If the first line matches the second line is ignored.
        src = ('#coding:iso8859-15\n'
               '#coding:latin1\n'
               'print(repr(u"\xc3\xa4"))\n')
        self.check_script_output(src, r"u'\xc3\u20ac'")

    def test_double_coding_same_line(self):
        src = ('#coding:iso8859-15 coding:latin1\n'
               'print(repr(u"\xc3\xa4"))\n')
        self.check_script_output(src, r"u'\xc3\u20ac'")

    def test_first_non_utf8_coding_line(self):
        src = ('#coding:iso-8859-15 \xa4\n'
               'print(repr(u"\xc3\xa4"))\n')
        self.check_script_output(src, r"u'\xc3\u20ac'")

    def test_second_non_utf8_coding_line(self):
        src = ('\n'
               '#coding:iso-8859-15 \xa4\n'
               'print(repr(u"\xc3\xa4"))\n')
        self.check_script_output(src, r"u'\xc3\u20ac'")

    def test_utf8_bom(self):
        src = ('\xef\xbb\xbfprint(repr(u"\xc3\xa4"))\n')
        self.check_script_output(src, r"u'\xe4'")

    def test_utf8_bom_and_utf8_coding_line(self):
        src = ('\xef\xbb\xbf#coding:utf-8\n'
               'print(repr(u"\xc3\xa4"))\n')
        self.check_script_output(src, r"u'\xe4'")


class BytesSourceEncodingTest(AbstractSourceEncodingTest, unittest.TestCase):

    def check_script_output(self, src, expected):
        with captured_stdout() as stdout:
            exec(src)
        out = stdout.getvalue().encode('latin1')
        self.assertEqual(out.rstrip(), expected)


class FileSourceEncodingTest(AbstractSourceEncodingTest, unittest.TestCase):

    def check_script_output(self, src, expected):
        tmpd = tempfile.mkdtemp()
        try:
            fn = os.path.join(tmpd, 'test.py')
            with open(fn, 'wb') as fp:
                fp.write(src)
            rc, out, err = script_helper.assert_python_ok(fn)
        finally:
            rmtree(tmpd)
        self.assertEqual(out.rstrip(), expected)


def test_main():
    run_unittest(MiscSourceEncodingTest, BytesSourceEncodingTest, FileSourceEncodingTest)

if __name__ == "__main__":
    test_main()