mirror of
https://github.com/ansible-collections/community.mysql.git
synced 2025-04-06 10:40:36 -07:00
mysql_query: fix false change reports when IF NOT EXISTS clause is used (#322)
* mysql_query: fix false change reports when IF NOT EXISTS clause is used * Fix * Fix doc, add fragment * Improve doc
This commit is contained in:
parent
ceda7662d0
commit
647461010d
3 changed files with 63 additions and 4 deletions
|
@ -0,0 +1,2 @@
|
||||||
|
bugfixes:
|
||||||
|
- mysql_query - fix false change reports when ``IF EXISTS/IF NOT EXISTS`` clause is used (https://github.com/ansible-collections/community.mysql/issues/268).
|
|
@ -22,6 +22,10 @@ options:
|
||||||
description:
|
description:
|
||||||
- SQL query to run. Multiple queries can be passed using YAML list syntax.
|
- SQL query to run. Multiple queries can be passed using YAML list syntax.
|
||||||
- Must be a string or YAML list containing strings.
|
- Must be a string or YAML list containing strings.
|
||||||
|
- Note that if you use the C(IF EXISTS/IF NOT EXISTS) clauses in your query
|
||||||
|
and C(mysqlclient) connector, the module will report that
|
||||||
|
the state has been changed even if it has not. If it is important in your
|
||||||
|
workflow, use the C(PyMySQL) connector instead.
|
||||||
type: raw
|
type: raw
|
||||||
required: yes
|
required: yes
|
||||||
positional_args:
|
positional_args:
|
||||||
|
@ -103,6 +107,8 @@ rowcount:
|
||||||
sample: [5, 1]
|
sample: [5, 1]
|
||||||
'''
|
'''
|
||||||
|
|
||||||
|
import warnings
|
||||||
|
|
||||||
from ansible.module_utils.basic import AnsibleModule
|
from ansible.module_utils.basic import AnsibleModule
|
||||||
from ansible_collections.community.mysql.plugins.module_utils.mysql import (
|
from ansible_collections.community.mysql.plugins.module_utils.mysql import (
|
||||||
mysql_connect,
|
mysql_connect,
|
||||||
|
@ -196,9 +202,22 @@ def main():
|
||||||
executed_queries = []
|
executed_queries = []
|
||||||
rowcount = []
|
rowcount = []
|
||||||
|
|
||||||
|
already_exists = False
|
||||||
for q in query:
|
for q in query:
|
||||||
try:
|
try:
|
||||||
cursor.execute(q, arguments)
|
with warnings.catch_warnings():
|
||||||
|
warnings.filterwarnings(action='error',
|
||||||
|
message='.*already exists*',
|
||||||
|
category=mysql_driver.Warning)
|
||||||
|
|
||||||
|
try:
|
||||||
|
cursor.execute(q, arguments)
|
||||||
|
except mysql_driver.Warning:
|
||||||
|
# When something is run with IF NOT EXISTS
|
||||||
|
# and there's "already exists" MySQL warning,
|
||||||
|
# set the flag as True.
|
||||||
|
# PyMySQL throws the warning, mysqlclinet does NOT.
|
||||||
|
already_exists = True
|
||||||
|
|
||||||
except Exception as e:
|
except Exception as e:
|
||||||
if not autocommit:
|
if not autocommit:
|
||||||
|
@ -208,7 +227,8 @@ def main():
|
||||||
module.fail_json(msg="Cannot execute SQL '%s' args [%s]: %s" % (q, arguments, to_native(e)))
|
module.fail_json(msg="Cannot execute SQL '%s' args [%s]: %s" % (q, arguments, to_native(e)))
|
||||||
|
|
||||||
try:
|
try:
|
||||||
query_result.append([dict(row) for row in cursor.fetchall()])
|
if not already_exists:
|
||||||
|
query_result.append([dict(row) for row in cursor.fetchall()])
|
||||||
|
|
||||||
except Exception as e:
|
except Exception as e:
|
||||||
if not autocommit:
|
if not autocommit:
|
||||||
|
@ -224,8 +244,12 @@ def main():
|
||||||
|
|
||||||
for keyword in DDL_QUERY_KEYWORDS:
|
for keyword in DDL_QUERY_KEYWORDS:
|
||||||
if keyword in q:
|
if keyword in q:
|
||||||
changed = True
|
if already_exists:
|
||||||
|
# Indicates the entity already exists
|
||||||
|
changed = False
|
||||||
|
already_exists = False # Reset flag
|
||||||
|
else:
|
||||||
|
changed = True
|
||||||
try:
|
try:
|
||||||
executed_queries.append(cursor._last_executed)
|
executed_queries.append(cursor._last_executed)
|
||||||
except AttributeError:
|
except AttributeError:
|
||||||
|
|
|
@ -321,6 +321,39 @@
|
||||||
- result is changed
|
- result is changed
|
||||||
- result.rowcount == [2]
|
- result.rowcount == [2]
|
||||||
|
|
||||||
|
# Issue https://github.com/ansible-collections/community.mysql/issues/268
|
||||||
|
- name: Create table
|
||||||
|
mysql_query:
|
||||||
|
<<: *mysql_params
|
||||||
|
login_db: '{{ test_db }}'
|
||||||
|
query: "CREATE TABLE issue268 (id int)"
|
||||||
|
single_transaction: yes
|
||||||
|
|
||||||
|
# Issue https://github.com/ansible-collections/community.mysql/issues/268
|
||||||
|
- name: Create table with IF NOT EXISTS
|
||||||
|
mysql_query:
|
||||||
|
<<: *mysql_params
|
||||||
|
login_db: '{{ test_db }}'
|
||||||
|
query: "CREATE TABLE IF NOT EXISTS issue268 (id int)"
|
||||||
|
single_transaction: yes
|
||||||
|
register: result
|
||||||
|
|
||||||
|
# Issue https://github.com/ansible-collections/community.mysql/issues/268
|
||||||
|
- assert:
|
||||||
|
that:
|
||||||
|
# PyMySQL driver throws a warning, so the following is correct
|
||||||
|
- result is not changed
|
||||||
|
when: connector.name.0 is search('pymysql')
|
||||||
|
|
||||||
|
# Issue https://github.com/ansible-collections/community.mysql/issues/268
|
||||||
|
- assert:
|
||||||
|
that:
|
||||||
|
# mysqlclient driver throws nothing, so it's impossible to figure out
|
||||||
|
# if the state was changed or not.
|
||||||
|
# We assume that it was for DDL queryes by default in the code
|
||||||
|
- result is changed
|
||||||
|
when: connector.name.0 is search('mysqlclient')
|
||||||
|
|
||||||
- name: Drop db {{ test_db }}
|
- name: Drop db {{ test_db }}
|
||||||
mysql_query:
|
mysql_query:
|
||||||
<<: *mysql_params
|
<<: *mysql_params
|
||||||
|
|
Loading…
Add table
Reference in a new issue