_mysql - Commands out of sync; you can't run this command now
다음의 코드를 수행하면,
# MySQLdb._mysql
# https://mysqlclient.readthedocs.io/user_guide.html#some-mysql-examples
def mysqlclient_raw2(request):
from MySQLdb import _mysql
mysql_ip = util.get_mysql_ip()
con = _mysql.connect(mysql_ip, "...", "...", "...")
con.query("SELECT SLEEP(0)")
con.commit()
con.close()
return render(request, 'bbs/mysqlclient_raw.html', {'result_set': None})
(필요없는 호출이긴 하지만) con.commit()에서 이런 예외가 발생합니다.
Traceback (most recent call last):
File "/home/kevin/.local/lib/python3.8/site-packages/django/core/handlers/exception.py", line 47, in inner
response = get_response(request)
File "/home/kevin/.local/lib/python3.8/site-packages/django/core/handlers/base.py", line 181, in _get_response
response = wrapped_callback(request, *callback_args, **callback_kwargs)
File "/mnt/d/workshop2/python-agent/testprj/myapp/./bbs/views.py", line 346, in mysqlclient_raw
con.commit()
MySQLdb.ProgrammingError: (2014, "Commands out of sync; you can't run this command now")
관련해서 검색해 보면,
B.3.2.12 Commands out of sync
; https://dev.mysql.com/doc/refman/8.0/en/commands-out-of-sync.html
This can happen, for example, if you are using mysql_use_result() and try to execute a new query before you have called mysql_free_result(). It can also happen if you try to execute two queries that return data without calling mysql_use_result() or mysql_store_result() in betwee
con.query("SELECT SLEEP(0)") 코드 수행으로 resultset이 열려 있게 되고, 그런 와중에 con.commit()으로 인해 수행되는 또 다른 MySql 쿼리로 인해 문제가 되는 것 같습니다.
따라서, 이 문제를 해결하려면 commit 전에 ("SELECT"를 동반한 쿼리인 경우라면) 반드시 store_result를 호출해야 합니다. (혹은, commit을 호출하지 않으면 됩니다.)
con = _mysql.connect(mysql_ip, "...", "...", "...")
con.query("SELECT SLEEP(0)")
con.store_result()
con.commit()
반면에 use_result를 쓰면,
con.query("SELECT SLEEP(0)")
r = con.use_result()
con.commit() # 예외 발생: MySQLdb.ProgrammingError: (2014, "Commands out of sync; you can't run this command now")
다시 예외가 발생하는데, "
B.3.2.12 Commands out of sync" 문서에서 설명했듯이 use_result/free_result를 쌍으로 호출해야 하지만 웬일인지 MySQLdb._mysql에는 해당 함수가 제공되지 않습니다. (MySQL측의
connector에는 free_result 함수가 제공됩니다.)
그렇다면 use_result와 store_result 간에는 어떤 차이점이 있을까요?
mysqli_store_result() vs. mysqli_use_result()
; https://stackoverflow.com/questions/9876730/mysqli-store-result-vs-mysqli-use-result
mysqli::store_result() will fetch the whole resultset from the MySQL server while mysqli::use_result() will fetch the rows one by one.
설명에 따르면 store_result의 경우 resultset을 전부 순회한 것이 되고, use_result의 경우에는 그렇지 않은 유형입니다. 따라서, 다음과 같이 직접 순회를 마치면,
con.query("SELECT SLEEP(0)")
r = con.use_result()
r.fetch_row() # 2번을 호출
r.fetch_row()
con.commit() # 정상 동작
또는, while 문으로 돌려주면,
con.query("SELECT SLEEP(0)")
r = con.use_result()
while r.fetch_row():
pass
con.commit() # 정상 동작
commit 호출 시 예외가 발생하지 않습니다. 참고로, 다른 의견을 보면 그 둘 간의 차이점에는 buffer가 관여한다고 합니다.
use_result returns an unbuffered result.
store_result returns a buffered result.
그렇다면 store_result의 경우에도 버퍼를 초과하는 select인 경우라면 문제가 된다는 것일까요? 딱히 버퍼의 용량을 알아내는 것도 없으니 테스트가 애매합니다.
정리해 보면, 결국 제가 _mysql의 사용법을 제대로 익히지 않았다고 볼 수 있습니다. ^^
[이 글에 대해서 여러분들과 의견을 공유하고 싶습니다. 틀리거나 미흡한 부분 또는 의문 사항이 있으시면 언제든 댓글 남겨주십시오.]