Skip to content

Commit 62f0645

Browse files
authored
Fix executemany with binary prefix (#605)
Fix #494
1 parent 3d6b8c9 commit 62f0645

File tree

3 files changed

+27
-35
lines changed

3 files changed

+27
-35
lines changed

src/MySQLdb/cursors.py

+2-32
Original file line numberDiff line numberDiff line change
@@ -110,34 +110,6 @@ def __exit__(self, *exc_info):
110110
del exc_info
111111
self.close()
112112

113-
def _escape_args(self, args, conn):
114-
encoding = conn.encoding
115-
literal = conn.literal
116-
117-
def ensure_bytes(x):
118-
if isinstance(x, str):
119-
return x.encode(encoding)
120-
elif isinstance(x, tuple):
121-
return tuple(map(ensure_bytes, x))
122-
elif isinstance(x, list):
123-
return list(map(ensure_bytes, x))
124-
return x
125-
126-
if isinstance(args, (tuple, list)):
127-
ret = tuple(literal(ensure_bytes(arg)) for arg in args)
128-
elif isinstance(args, dict):
129-
ret = {
130-
ensure_bytes(key): literal(ensure_bytes(val))
131-
for (key, val) in args.items()
132-
}
133-
else:
134-
# If it's not a dictionary let's try escaping it anyways.
135-
# Worst case it will throw a Value error
136-
ret = literal(ensure_bytes(args))
137-
138-
ensure_bytes = None # break circular reference
139-
return ret
140-
141113
def _check_executed(self):
142114
if not self._executed:
143115
raise ProgrammingError("execute() first")
@@ -279,8 +251,6 @@ def executemany(self, query, args):
279251
def _do_execute_many(
280252
self, prefix, values, postfix, args, max_stmt_length, encoding
281253
):
282-
conn = self._get_db()
283-
escape = self._escape_args
284254
if isinstance(prefix, str):
285255
prefix = prefix.encode(encoding)
286256
if isinstance(values, str):
@@ -289,11 +259,11 @@ def _do_execute_many(
289259
postfix = postfix.encode(encoding)
290260
sql = bytearray(prefix)
291261
args = iter(args)
292-
v = values % escape(next(args), conn)
262+
v = self._mogrify(values, next(args))
293263
sql += v
294264
rows = 0
295265
for arg in args:
296-
v = values % escape(arg, conn)
266+
v = self._mogrify(values, arg)
297267
if len(sql) + len(v) + len(postfix) + 1 > max_stmt_length:
298268
rows += self.execute(sql + postfix)
299269
sql = bytearray(prefix)

tests/default.cnf

+3-2
Original file line numberDiff line numberDiff line change
@@ -2,9 +2,10 @@
22
# http://dev.mysql.com/doc/refman/5.1/en/option-files.html
33
# and set TESTDB in your environment to the name of the file
44

5+
# $ docker run -e MYSQL_ALLOW_EMPTY_PASSWORD=yes -p 3306:3306 --rm --name mysqld mysql:latest
56
[MySQLdb-tests]
67
host = 127.0.0.1
7-
user = test
8+
user = root
89
database = test
910
#password =
10-
default-character-set = utf8
11+
default-character-set = utf8mb4

tests/test_cursor.py

+22-1
Original file line numberDiff line numberDiff line change
@@ -72,7 +72,7 @@ def test_executemany():
7272
# values (0),(1),(2),(3),(4),(5),(6),(7),(8),(9)
7373
# """
7474
# list args
75-
data = range(10)
75+
data = [(i,) for i in range(10)]
7676
cursor.executemany("insert into test (data) values (%s)", data)
7777
assert cursor._executed.endswith(
7878
b",(7),(8),(9)"
@@ -222,3 +222,24 @@ def test_cursor_discard_result(Cursor):
222222
"SELECT * FROM test_cursor_discard_result WHERE id BETWEEN 31 AND 40"
223223
)
224224
assert cursor.fetchone() == (31, "row 31")
225+
226+
227+
def test_binary_prefix():
228+
# https://github.com/PyMySQL/mysqlclient/issues/494
229+
conn = connect(binary_prefix=True)
230+
cursor = conn.cursor()
231+
232+
cursor.execute("DROP TABLE IF EXISTS test_binary_prefix")
233+
cursor.execute(
234+
"""\
235+
CREATE TABLE test_binary_prefix (
236+
id INTEGER NOT NULL AUTO_INCREMENT,
237+
json JSON NOT NULL,
238+
PRIMARY KEY (id)
239+
) CHARSET=utf8mb4"""
240+
)
241+
242+
cursor.executemany(
243+
"INSERT INTO test_binary_prefix (id, json) VALUES (%(id)s, %(json)s)",
244+
({"id": 1, "json": "{}"}, {"id": 2, "json": "{}"}),
245+
)

0 commit comments

Comments
 (0)