self.assertEqual(unicode_asutf8andsize(nonbmp), (b'\xf4\x8f\xbf\xbf', 4))
self.assertRaises(UnicodeEncodeError, unicode_asutf8andsize, 'a\ud800b\udfffc')
+ @support.cpython_only
+ def test_decodeutf8(self):
+ """Test PyUnicode_DecodeUTF8()"""
+ import _testcapi
+ decodeutf8 = _testcapi.unicode_decodeutf8
+
+ for s in ['abc', '\xa1\xa2', '\u4f60\u597d', 'a\U0001f600']:
+ b = s.encode('utf-8')
+ self.assertEqual(decodeutf8(b), s)
+ self.assertEqual(decodeutf8(b, 'strict'), s)
+
+ self.assertRaises(UnicodeDecodeError, decodeutf8, b'\x80')
+ self.assertRaises(UnicodeDecodeError, decodeutf8, b'\xc0')
+ self.assertRaises(UnicodeDecodeError, decodeutf8, b'\xff')
+ self.assertRaises(UnicodeDecodeError, decodeutf8, b'a\xf0\x9f')
+ self.assertEqual(decodeutf8(b'a\xf0\x9f', 'replace'), 'a\ufffd')
+ self.assertEqual(decodeutf8(b'a\xf0\x9fb', 'replace'), 'a\ufffdb')
+
+ self.assertRaises(LookupError, decodeutf8, b'a\x80', 'foo')
+ # TODO: Test PyUnicode_DecodeUTF8() with NULL as data and
+ # negative size.
+
+ @support.cpython_only
+ def test_decodeutf8stateful(self):
+ """Test PyUnicode_DecodeUTF8Stateful()"""
+ import _testcapi
+ decodeutf8stateful = _testcapi.unicode_decodeutf8stateful
+
+ for s in ['abc', '\xa1\xa2', '\u4f60\u597d', 'a\U0001f600']:
+ b = s.encode('utf-8')
+ self.assertEqual(decodeutf8stateful(b), (s, len(b)))
+ self.assertEqual(decodeutf8stateful(b, 'strict'), (s, len(b)))
+
+ self.assertRaises(UnicodeDecodeError, decodeutf8stateful, b'\x80')
+ self.assertRaises(UnicodeDecodeError, decodeutf8stateful, b'\xc0')
+ self.assertRaises(UnicodeDecodeError, decodeutf8stateful, b'\xff')
+ self.assertEqual(decodeutf8stateful(b'a\xf0\x9f'), ('a', 1))
+ self.assertEqual(decodeutf8stateful(b'a\xf0\x9f', 'replace'), ('a', 1))
+ self.assertRaises(UnicodeDecodeError, decodeutf8stateful, b'a\xf0\x9fb')
+ self.assertEqual(decodeutf8stateful(b'a\xf0\x9fb', 'replace'), ('a\ufffdb', 4))
+
+ self.assertRaises(LookupError, decodeutf8stateful, b'a\x80', 'foo')
+ # TODO: Test PyUnicode_DecodeUTF8Stateful() with NULL as data and
+ # negative size.
+ # TODO: Test PyUnicode_DecodeUTF8Stateful() with NULL as the address of
+ # "consumed".
+
# Test PyUnicode_FindChar()
@support.cpython_only
def test_findchar(self):
return Py_BuildValue("(Nn)", result, utf8_len);
}
+/* Test PyUnicode_DecodeUTF8() */
+static PyObject *
+unicode_decodeutf8(PyObject *self, PyObject *args)
+{
+ const char *data;
+ Py_ssize_t size;
+ const char *errors = NULL;
+
+ if (!PyArg_ParseTuple(args, "y#|z", &data, &size, &errors))
+ return NULL;
+
+ return PyUnicode_DecodeUTF8(data, size, errors);
+}
+
+/* Test PyUnicode_DecodeUTF8Stateful() */
+static PyObject *
+unicode_decodeutf8stateful(PyObject *self, PyObject *args)
+{
+ const char *data;
+ Py_ssize_t size;
+ const char *errors = NULL;
+ Py_ssize_t consumed = 123456789;
+ PyObject *result;
+
+ if (!PyArg_ParseTuple(args, "y#|z", &data, &size, &errors))
+ return NULL;
+
+ result = PyUnicode_DecodeUTF8Stateful(data, size, errors, &consumed);
+ if (!result) {
+ return NULL;
+ }
+ return Py_BuildValue("(Nn)", result, consumed);
+}
+
static PyObject *
unicode_findchar(PyObject *self, PyObject *args)
{
{"unicode_asucs4", unicode_asucs4, METH_VARARGS},
{"unicode_asutf8", unicode_asutf8, METH_VARARGS},
{"unicode_asutf8andsize", unicode_asutf8andsize, METH_VARARGS},
- {"unicode_findchar", unicode_findchar, METH_VARARGS},
+ {"unicode_decodeutf8", unicode_decodeutf8, METH_VARARGS},
+ {"unicode_decodeutf8stateful",unicode_decodeutf8stateful, METH_VARARGS}, {"unicode_findchar", unicode_findchar, METH_VARARGS},
{"unicode_copycharacters", unicode_copycharacters, METH_VARARGS},
{"unicode_encodedecimal", unicode_encodedecimal, METH_VARARGS},
{"unicode_transformdecimaltoascii", unicode_transformdecimaltoascii, METH_VARARGS},