Skip to content

Commit

Permalink
[3.6] bpo-32922: dbm.open() now encodes filename with the filesystem …
Browse files Browse the repository at this point in the history
…encoding. (GH-5832). (GH-5906)

(cherry picked from commit 6f600ff)

Co-authored-by: Serhiy Storchaka <storchaka@gmail.com>
  • Loading branch information
serhiy-storchaka committed Feb 27, 2018
1 parent 3f6c172 commit 5bf7485
Show file tree
Hide file tree
Showing 8 changed files with 134 additions and 19 deletions.
15 changes: 15 additions & 0 deletions Lib/test/test_dbm_dumb.py
Expand Up @@ -275,6 +275,21 @@ def test_readonly_files(self):
self.assertEqual(sorted(f.keys()), sorted(self._dict))
f.close() # don't write

@unittest.skipUnless(support.TESTFN_NONASCII,
'requires OS support of non-ASCII encodings')
def test_nonascii_filename(self):
filename = support.TESTFN_NONASCII
for suffix in ['.dir', '.dat', '.bak']:
self.addCleanup(support.unlink, filename + suffix)
with dumbdbm.open(filename, 'c') as db:
db[b'key'] = b'value'
self.assertTrue(os.path.exists(filename + '.dat'))
self.assertTrue(os.path.exists(filename + '.dir'))
with dumbdbm.open(filename, 'r') as db:
self.assertEqual(list(db.keys()), [b'key'])
self.assertTrue(b'key' in db)
self.assertEqual(db[b'key'], b'value')

def tearDown(self):
_delete_files()

Expand Down
36 changes: 35 additions & 1 deletion Lib/test/test_dbm_gnu.py
Expand Up @@ -2,7 +2,7 @@
gdbm = support.import_module("dbm.gnu") #skip if not supported
import unittest
import os
from test.support import TESTFN, unlink
from test.support import TESTFN, TESTFN_NONASCII, unlink


filename = TESTFN
Expand Down Expand Up @@ -93,5 +93,39 @@ def test_context_manager(self):
self.assertEqual(str(cm.exception),
"GDBM object has already been closed")

def test_bytes(self):
with gdbm.open(filename, 'c') as db:
db[b'bytes key \xbd'] = b'bytes value \xbd'
with gdbm.open(filename, 'r') as db:
self.assertEqual(list(db.keys()), [b'bytes key \xbd'])
self.assertTrue(b'bytes key \xbd' in db)
self.assertEqual(db[b'bytes key \xbd'], b'bytes value \xbd')

def test_unicode(self):
with gdbm.open(filename, 'c') as db:
db['Unicode key \U0001f40d'] = 'Unicode value \U0001f40d'
with gdbm.open(filename, 'r') as db:
self.assertEqual(list(db.keys()), ['Unicode key \U0001f40d'.encode()])
self.assertTrue('Unicode key \U0001f40d'.encode() in db)
self.assertTrue('Unicode key \U0001f40d' in db)
self.assertEqual(db['Unicode key \U0001f40d'.encode()],
'Unicode value \U0001f40d'.encode())
self.assertEqual(db['Unicode key \U0001f40d'],
'Unicode value \U0001f40d'.encode())

@unittest.skipUnless(TESTFN_NONASCII,
'requires OS support of non-ASCII encodings')
def test_nonascii_filename(self):
filename = TESTFN_NONASCII
self.addCleanup(unlink, filename)
with gdbm.open(filename, 'c') as db:
db[b'key'] = b'value'
self.assertTrue(os.path.exists(filename))
with gdbm.open(filename, 'r') as db:
self.assertEqual(list(db.keys()), [b'key'])
self.assertTrue(b'key' in db)
self.assertEqual(db[b'key'], b'value')


if __name__ == '__main__':
unittest.main()
37 changes: 37 additions & 0 deletions Lib/test/test_dbm_ndbm.py
@@ -1,5 +1,6 @@
from test import support
support.import_module("dbm.ndbm") #skip if not supported
import os
import unittest
import dbm.ndbm
from dbm.ndbm import error
Expand Down Expand Up @@ -47,6 +48,42 @@ def test_context_manager(self):
self.assertEqual(str(cm.exception),
"DBM object has already been closed")

def test_bytes(self):
with dbm.ndbm.open(self.filename, 'c') as db:
db[b'bytes key \xbd'] = b'bytes value \xbd'
with dbm.ndbm.open(self.filename, 'r') as db:
self.assertEqual(list(db.keys()), [b'bytes key \xbd'])
self.assertTrue(b'bytes key \xbd' in db)
self.assertEqual(db[b'bytes key \xbd'], b'bytes value \xbd')

def test_unicode(self):
with dbm.ndbm.open(self.filename, 'c') as db:
db['Unicode key \U0001f40d'] = 'Unicode value \U0001f40d'
with dbm.ndbm.open(self.filename, 'r') as db:
self.assertEqual(list(db.keys()), ['Unicode key \U0001f40d'.encode()])
self.assertTrue('Unicode key \U0001f40d'.encode() in db)
self.assertTrue('Unicode key \U0001f40d' in db)
self.assertEqual(db['Unicode key \U0001f40d'.encode()],
'Unicode value \U0001f40d'.encode())
self.assertEqual(db['Unicode key \U0001f40d'],
'Unicode value \U0001f40d'.encode())

@unittest.skipUnless(support.TESTFN_NONASCII,
'requires OS support of non-ASCII encodings')
def test_nonascii_filename(self):
filename = support.TESTFN_NONASCII
for suffix in ['', '.pag', '.dir', '.db']:
self.addCleanup(support.unlink, filename + suffix)
with dbm.ndbm.open(filename, 'c') as db:
db[b'key'] = b'value'
self.assertTrue(any(os.path.exists(filename + suffix)
for suffix in ['', '.pag', '.dir', '.db']))
with dbm.ndbm.open(filename, 'r') as db:
self.assertEqual(list(db.keys()), [b'key'])
self.assertTrue(b'key' in db)
self.assertEqual(db[b'key'], b'value')



if __name__ == '__main__':
unittest.main()
@@ -0,0 +1,2 @@
dbm.open() now encodes filename with the filesystem encoding rather than
default encoding.
21 changes: 17 additions & 4 deletions Modules/_dbmmodule.c
Expand Up @@ -413,7 +413,7 @@ static PyTypeObject Dbmtype = {
_dbm.open as dbmopen
filename: str
filename: unicode
The filename to open.
flags: str="r"
Expand All @@ -430,9 +430,9 @@ Return a database object.
[clinic start generated code]*/

static PyObject *
dbmopen_impl(PyObject *module, const char *filename, const char *flags,
dbmopen_impl(PyObject *module, PyObject *filename, const char *flags,
int mode)
/*[clinic end generated code: output=5fade8cf16e0755f input=226334bade5764e6]*/
/*[clinic end generated code: output=9527750f5df90764 input=376a9d903a50df59]*/
{
int iflags;

Expand All @@ -451,7 +451,20 @@ dbmopen_impl(PyObject *module, const char *filename, const char *flags,
"arg 2 to open should be 'r', 'w', 'c', or 'n'");
return NULL;
}
return newdbmobject(filename, iflags, mode);

PyObject *filenamebytes = PyUnicode_EncodeFSDefault(filename);
if (filenamebytes == NULL) {
return NULL;
}
const char *name = PyBytes_AS_STRING(filenamebytes);
if (strlen(name) != (size_t)PyBytes_GET_SIZE(filenamebytes)) {
Py_DECREF(filenamebytes);
PyErr_SetString(PyExc_ValueError, "embedded null character");
return NULL;
}
PyObject *self = newdbmobject(name, iflags, mode);
Py_DECREF(filenamebytes);
return self;
}

static PyMethodDef dbmmodule_methods[] = {
Expand Down
21 changes: 17 additions & 4 deletions Modules/_gdbmmodule.c
Expand Up @@ -532,7 +532,7 @@ static PyTypeObject Dbmtype = {

/*[clinic input]
_gdbm.open as dbmopen
filename as name: str
filename: unicode
flags: str="r"
mode: int(py_default="0o666") = 0o666
/
Expand Down Expand Up @@ -562,8 +562,9 @@ when the database has to be created. It defaults to octal 0o666.
[clinic start generated code]*/

static PyObject *
dbmopen_impl(PyObject *module, const char *name, const char *flags, int mode)
/*[clinic end generated code: output=31aa1bafdf5da688 input=55563cd60e51984a]*/
dbmopen_impl(PyObject *module, PyObject *filename, const char *flags,
int mode)
/*[clinic end generated code: output=9527750f5df90764 input=3be0b0875974b928]*/
{
int iflags;

Expand Down Expand Up @@ -611,7 +612,19 @@ dbmopen_impl(PyObject *module, const char *name, const char *flags, int mode)
}
}

return newdbmobject(name, iflags, mode);
PyObject *filenamebytes = PyUnicode_EncodeFSDefault(filename);
if (filenamebytes == NULL) {
return NULL;
}
const char *name = PyBytes_AS_STRING(filenamebytes);
if (strlen(name) != (size_t)PyBytes_GET_SIZE(filenamebytes)) {
Py_DECREF(filenamebytes);
PyErr_SetString(PyExc_ValueError, "embedded null character");
return NULL;
}
PyObject *self = newdbmobject(name, iflags, mode);
Py_DECREF(filenamebytes);
return self;
}

static const char dbmmodule_open_flags[] = "rwcn"
Expand Down
8 changes: 4 additions & 4 deletions Modules/clinic/_dbmmodule.c.h
Expand Up @@ -121,18 +121,18 @@ PyDoc_STRVAR(dbmopen__doc__,
{"open", (PyCFunction)dbmopen, METH_VARARGS, dbmopen__doc__},

static PyObject *
dbmopen_impl(PyObject *module, const char *filename, const char *flags,
dbmopen_impl(PyObject *module, PyObject *filename, const char *flags,
int mode);

static PyObject *
dbmopen(PyObject *module, PyObject *args)
{
PyObject *return_value = NULL;
const char *filename;
PyObject *filename;
const char *flags = "r";
int mode = 438;

if (!PyArg_ParseTuple(args, "s|si:open",
if (!PyArg_ParseTuple(args, "U|si:open",
&filename, &flags, &mode)) {
goto exit;
}
Expand All @@ -141,4 +141,4 @@ dbmopen(PyObject *module, PyObject *args)
exit:
return return_value;
}
/*[clinic end generated code: output=99adf966ef0475ff input=a9049054013a1b77]*/
/*[clinic end generated code: output=001fabffcecb99f1 input=a9049054013a1b77]*/
13 changes: 7 additions & 6 deletions Modules/clinic/_gdbmmodule.c.h
Expand Up @@ -234,23 +234,24 @@ PyDoc_STRVAR(dbmopen__doc__,
{"open", (PyCFunction)dbmopen, METH_VARARGS, dbmopen__doc__},

static PyObject *
dbmopen_impl(PyObject *module, const char *name, const char *flags, int mode);
dbmopen_impl(PyObject *module, PyObject *filename, const char *flags,
int mode);

static PyObject *
dbmopen(PyObject *module, PyObject *args)
{
PyObject *return_value = NULL;
const char *name;
PyObject *filename;
const char *flags = "r";
int mode = 438;

if (!PyArg_ParseTuple(args, "s|si:open",
&name, &flags, &mode)) {
if (!PyArg_ParseTuple(args, "U|si:open",
&filename, &flags, &mode)) {
goto exit;
}
return_value = dbmopen_impl(module, name, flags, mode);
return_value = dbmopen_impl(module, filename, flags, mode);

exit:
return return_value;
}
/*[clinic end generated code: output=ed0f5d4e3d79b80c input=a9049054013a1b77]*/
/*[clinic end generated code: output=afb99364ac420d10 input=a9049054013a1b77]*/

0 comments on commit 5bf7485

Please sign in to comment.