mirror of
				https://github.com/ansible-collections/community.mysql.git
				synced 2025-10-25 05:24:01 -07:00 
			
		
		
		
	[stable-1] mysql_query: fix false change reports when IF NOT EXISTS clause is used (#375)
* 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)
* Fix tests
	
	
This commit is contained in:
		
					parent
					
						
							
								0921d4eef5
							
						
					
				
			
			
				commit
				
					
						f7677aa05b
					
				
			
		
					 4 changed files with 80 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: | ||||
|     - SQL query to run. Multiple queries can be passed using YAML list syntax. | ||||
|     - 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 | ||||
|     required: yes | ||||
|   positional_args: | ||||
|  | @ -103,6 +107,8 @@ rowcount: | |||
|     sample: [5, 1] | ||||
| ''' | ||||
| 
 | ||||
| import warnings | ||||
| 
 | ||||
| from ansible.module_utils.basic import AnsibleModule | ||||
| from ansible_collections.community.mysql.plugins.module_utils.mysql import ( | ||||
|     mysql_connect, | ||||
|  | @ -196,9 +202,22 @@ def main(): | |||
|     executed_queries = [] | ||||
|     rowcount = [] | ||||
| 
 | ||||
|     already_exists = False | ||||
|     for q in query: | ||||
|         try: | ||||
|             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: | ||||
|             if not autocommit: | ||||
|  | @ -208,6 +227,7 @@ def main(): | |||
|             module.fail_json(msg="Cannot execute SQL '%s' args [%s]: %s" % (q, arguments, to_native(e))) | ||||
| 
 | ||||
|         try: | ||||
|             if not already_exists: | ||||
|                 query_result.append([dict(row) for row in cursor.fetchall()]) | ||||
| 
 | ||||
|         except Exception as e: | ||||
|  | @ -224,8 +244,12 @@ def main(): | |||
| 
 | ||||
|         for keyword in DDL_QUERY_KEYWORDS: | ||||
|             if keyword in q: | ||||
|                 if already_exists: | ||||
|                     # Indicates the entity already exists | ||||
|                     changed = False | ||||
|                     already_exists = False  # Reset flag | ||||
|                 else: | ||||
|                     changed = True | ||||
| 
 | ||||
|         try: | ||||
|             executed_queries.append(cursor._last_executed) | ||||
|         except AttributeError: | ||||
|  |  | |||
|  | @ -28,6 +28,23 @@ | |||
| - name: "{{ role_name }} | install | install python packages" | ||||
|   pip: | ||||
|     name: "{{ python_packages }}" | ||||
|   register: connector | ||||
| 
 | ||||
| - name: Extract connector.name.0 content | ||||
|   set_fact: | ||||
|     connector_name: "{{ connector.name.0 }}" | ||||
| 
 | ||||
| - name: Debug connector_name content | ||||
|   debug: | ||||
|     msg: '{{ connector_name }}' | ||||
| 
 | ||||
| - name: Extract connector version | ||||
|   set_fact: | ||||
|     connector_ver: "{{ connector_name.split('=')[2].strip() }}" | ||||
| 
 | ||||
| - name: Debug connector_ver var content | ||||
|   debug: | ||||
|     msg: '{{ connector_ver }}' | ||||
| 
 | ||||
| - name: "{{ role_name }} | install | install packages required by mysql" | ||||
|   apt: | ||||
|  |  | |||
|  | @ -321,6 +321,39 @@ | |||
|       - result is changed | ||||
|       - 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 }} | ||||
|     mysql_query: | ||||
|       <<: *mysql_params | ||||
|  |  | |||
		Loading…
	
	Add table
		Add a link
		
	
		Reference in a new issue