mirror of
				https://github.com/ansible-collections/community.mysql.git
				synced 2025-10-26 05:50:39 -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
(cherry picked from commit 647461010d)
	
	
This commit is contained in:
		
					parent
					
						
							
								e34cfba109
							
						
					
				
			
			
				commit
				
					
						2e390409a9
					
				
			
		
					 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
		Add a link
		
	
		Reference in a new issue