mirror of
				https://github.com/ansible-collections/community.general.git
				synced 2025-10-24 13:04:00 -07:00 
			
		
		
		
	* Began with filter docs.
* Add more filters.
* Add time unit filters.
* Add TOC and filters to create identifiers.
* Add more filters.
* Add documentation from ansible/ansible for json_query and random_mac.
* Update docs/docsite/rst/filter_guide.rst
Co-authored-by: Abhijeet Kasurde <akasurde@redhat.com>
Co-authored-by: Abhijeet Kasurde <akasurde@redhat.com>
(cherry picked from commit 3516acf8d4)
Co-authored-by: Felix Fontein <felix@fontein.de>
	
	
This commit is contained in:
		
					parent
					
						
							
								e74ea7c8b8
							
						
					
				
			
			
				commit
				
					
						b1a4a0ff21
					
				
			
		
					 2 changed files with 758 additions and 0 deletions
				
			
		
							
								
								
									
										5
									
								
								docs/docsite/extra-docs.yml
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										5
									
								
								docs/docsite/extra-docs.yml
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,5 @@ | |||
| --- | ||||
| sections: | ||||
|   - title: Guides | ||||
|     toctree: | ||||
|       - filter_guide | ||||
							
								
								
									
										753
									
								
								docs/docsite/rst/filter_guide.rst
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										753
									
								
								docs/docsite/rst/filter_guide.rst
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,753 @@ | |||
| .. _ansible_collections.community.general.docsite.filter_guide: | ||||
| 
 | ||||
| community.general Filter Guide | ||||
| ============================== | ||||
| 
 | ||||
| The :ref:`community.general collection <plugins_in_community.general>` offers several useful filter plugins. | ||||
| 
 | ||||
| .. contents:: Topics | ||||
| 
 | ||||
| Paths | ||||
| ----- | ||||
| 
 | ||||
| The ``path_join`` filter has been added in ansible-base 2.10. If you want to use this filter, but also need to support Ansible 2.9, you can use ``community.general``'s ``path_join`` shim, ``community.general.path_join``. This filter redirects to ``path_join`` for ansible-base 2.10 and ansible-core 2.11 or newer, and re-implements the filter for Ansible 2.9. | ||||
| 
 | ||||
| .. code-block:: yaml+jinja | ||||
| 
 | ||||
|     # ansible-base 2.10 or newer: | ||||
|     path: {{ ('/etc', path, 'subdir', file) | path_join }} | ||||
| 
 | ||||
|     # Also works with Ansible 2.9: | ||||
|     path: {{ ('/etc', path, 'subdir', file) | community.general.path_join }} | ||||
| 
 | ||||
| .. versionadded:: 3.0.0 | ||||
| 
 | ||||
| Abstract transformations | ||||
| ------------------------ | ||||
| 
 | ||||
| Dictionaries | ||||
| ^^^^^^^^^^^^ | ||||
| 
 | ||||
| You can use the ``dict_kv`` filter to create a single-entry dictionary with ``value | community.general.dict_kv(key)``: | ||||
| 
 | ||||
| .. code-block:: yaml+jinja | ||||
| 
 | ||||
|     - name: Create a single-entry dictionary | ||||
|       debug: | ||||
|         msg: "{{ myvar | community.general.dict_kv('thatsmyvar') }}" | ||||
|       vars: | ||||
|         myvar: myvalue | ||||
| 
 | ||||
|     - name: Create a list of dictionaries where the 'server' field is taken from a list | ||||
|       debug: | ||||
|         msg: >- | ||||
|           {{ myservers | map('community.general.dict_kv', 'server') | ||||
|                        | map('combine', common_config) }} | ||||
|       vars: | ||||
|         common_config: | ||||
|           type: host | ||||
|           database: all | ||||
|         myservers: | ||||
|         - server1 | ||||
|         - server2 | ||||
| 
 | ||||
| This produces: | ||||
| 
 | ||||
| .. code-block:: ansible-output | ||||
| 
 | ||||
|     TASK [Create a single-entry dictionary]  ************************************************** | ||||
|     ok: [localhost] => { | ||||
|         "msg": { | ||||
|             "thatsmyvar": "myvalue" | ||||
|         } | ||||
|     } | ||||
| 
 | ||||
|     TASK [Create a list of dictionaries where the 'server' field is taken from a list]  ******* | ||||
|     ok: [localhost] => { | ||||
|         "msg": [ | ||||
|             { | ||||
|                 "database": "all", | ||||
|                 "server": "server1", | ||||
|                 "type": "host" | ||||
|             }, | ||||
|             { | ||||
|                 "database": "all", | ||||
|                 "server": "server2", | ||||
|                 "type": "host" | ||||
|             } | ||||
|         ] | ||||
|     } | ||||
| 
 | ||||
| .. versionadded:: 2.0.0 | ||||
| 
 | ||||
| If you need to convert a list of key-value pairs to a dictionary, you can use the ``dict`` function. Unfortunately, this function cannot be used with ``map``. For this, the ``community.general.dict`` filter can be used: | ||||
| 
 | ||||
| .. code-block:: yaml+jinja | ||||
| 
 | ||||
|     - name: Create a dictionary with the dict function | ||||
|       debug: | ||||
|         msg: "{{ dict([[1, 2], ['a', 'b']]) }}" | ||||
| 
 | ||||
|     - name: Create a dictionary with the community.general.dict filter | ||||
|       debug: | ||||
|         msg: "{{ [[1, 2], ['a', 'b']] | community.general.dict }}" | ||||
| 
 | ||||
|     - name: Create a list of dictionaries with map and the community.general.dict filter | ||||
|       debug: | ||||
|         msg: >- | ||||
|           {{ values | map('zip', ['k1', 'k2', 'k3']) | ||||
|                     | map('map', 'reverse') | ||||
|                     | map('community.general.dict') }} | ||||
|       vars: | ||||
|         values: | ||||
|           - - foo | ||||
|             - 23 | ||||
|             - a | ||||
|           - - bar | ||||
|             - 42 | ||||
|             - b | ||||
| 
 | ||||
| This produces: | ||||
| 
 | ||||
| .. code-block:: ansible-output | ||||
| 
 | ||||
|     TASK [Create a dictionary with the dict function]  **************************************** | ||||
|     ok: [localhost] => { | ||||
|         "msg": { | ||||
|             "1": 2, | ||||
|             "a": "b" | ||||
|         } | ||||
|     } | ||||
| 
 | ||||
|     TASK [Create a dictionary with the community.general.dict filter]  ************************ | ||||
|     ok: [localhost] => { | ||||
|         "msg": { | ||||
|             "1": 2, | ||||
|             "a": "b" | ||||
|         } | ||||
|     } | ||||
| 
 | ||||
|     TASK [Create a list of dictionaries with map and the community.general.dict filter]  ****** | ||||
|     ok: [localhost] => { | ||||
|         "msg": [ | ||||
|             { | ||||
|                 "k1": "foo", | ||||
|                 "k2": 23, | ||||
|                 "k3": "a" | ||||
|             }, | ||||
|             { | ||||
|                 "k1": "bar", | ||||
|                 "k2": 42, | ||||
|                 "k3": "b" | ||||
|             } | ||||
|         ] | ||||
|     } | ||||
| 
 | ||||
| .. versionadded:: 3.0.0 | ||||
| 
 | ||||
| Grouping | ||||
| ^^^^^^^^ | ||||
| 
 | ||||
| If you have a list of dictionaries, the Jinja2 ``groupby`` filter allows to group the list by an attribute. This results in a list of ``(grouper, list)`` namedtuples, where ``list`` contains all dictionaries where the selected attribute equals ``grouper``. If you know that for every ``grouper``, there will be a most one entry in that list, you can use the ``community.general.groupby_as_dict`` filter to convert the original list into a dictionary which maps ``grouper`` to the corresponding dictionary. | ||||
| 
 | ||||
| One example is ``ansible_facts.mounts``, which is a list of dictionaries where each has one ``device`` element to indicate the device which is mounted. Therefore, ``ansible_facts.mounts | community.general.groupby_as_dict('device')`` is a dictionary mapping a device to the mount information: | ||||
| 
 | ||||
| .. code-block:: yaml+jinja | ||||
| 
 | ||||
|     - name: Output mount facts grouped by device name | ||||
|       debug: | ||||
|         var: ansible_facts.mounts | community.general.groupby_as_dict('device') | ||||
| 
 | ||||
|     - name: Output mount facts grouped by mount point | ||||
|       debug: | ||||
|         var: ansible_facts.mounts | community.general.groupby_as_dict('mount') | ||||
| 
 | ||||
| This produces: | ||||
| 
 | ||||
| .. code-block:: ansible-output | ||||
| 
 | ||||
|     TASK [Output mount facts grouped by device name] ****************************************** | ||||
|     ok: [localhost] => { | ||||
|         "ansible_facts.mounts | community.general.groupby_as_dict('device')": { | ||||
|             "/dev/sda1": { | ||||
|                 "block_available": 2000, | ||||
|                 "block_size": 4096, | ||||
|                 "block_total": 2345, | ||||
|                 "block_used": 345, | ||||
|                 "device": "/dev/sda1", | ||||
|                 "fstype": "ext4", | ||||
|                 "inode_available": 500, | ||||
|                 "inode_total": 512, | ||||
|                 "inode_used": 12, | ||||
|                 "mount": "/boot", | ||||
|                 "options": "rw,relatime,data=ordered", | ||||
|                 "size_available": 56821, | ||||
|                 "size_total": 543210, | ||||
|                 "uuid": "ab31cade-d9c1-484d-8482-8a4cbee5241a" | ||||
|             }, | ||||
|             "/dev/sda2": { | ||||
|                 "block_available": 1234, | ||||
|                 "block_size": 4096, | ||||
|                 "block_total": 12345, | ||||
|                 "block_used": 11111, | ||||
|                 "device": "/dev/sda2", | ||||
|                 "fstype": "ext4", | ||||
|                 "inode_available": 1111, | ||||
|                 "inode_total": 1234, | ||||
|                 "inode_used": 123, | ||||
|                 "mount": "/", | ||||
|                 "options": "rw,relatime", | ||||
|                 "size_available": 42143, | ||||
|                 "size_total": 543210, | ||||
|                 "uuid": "abcdef01-2345-6789-0abc-def012345678" | ||||
|             } | ||||
|         } | ||||
|     } | ||||
| 
 | ||||
|     TASK [Output mount facts grouped by mount point] ****************************************** | ||||
|     ok: [localhost] => { | ||||
|         "ansible_facts.mounts | community.general.groupby_as_dict('mount')": { | ||||
|             "/": { | ||||
|                 "block_available": 1234, | ||||
|                 "block_size": 4096, | ||||
|                 "block_total": 12345, | ||||
|                 "block_used": 11111, | ||||
|                 "device": "/dev/sda2", | ||||
|                 "fstype": "ext4", | ||||
|                 "inode_available": 1111, | ||||
|                 "inode_total": 1234, | ||||
|                 "inode_used": 123, | ||||
|                 "mount": "/", | ||||
|                 "options": "rw,relatime", | ||||
|                 "size_available": 42143, | ||||
|                 "size_total": 543210, | ||||
|                 "uuid": "bdf50b7d-4859-40af-8665-c637ee7a7808" | ||||
|             }, | ||||
|             "/boot": { | ||||
|                 "block_available": 2000, | ||||
|                 "block_size": 4096, | ||||
|                 "block_total": 2345, | ||||
|                 "block_used": 345, | ||||
|                 "device": "/dev/sda1", | ||||
|                 "fstype": "ext4", | ||||
|                 "inode_available": 500, | ||||
|                 "inode_total": 512, | ||||
|                 "inode_used": 12, | ||||
|                 "mount": "/boot", | ||||
|                 "options": "rw,relatime,data=ordered", | ||||
|                 "size_available": 56821, | ||||
|                 "size_total": 543210, | ||||
|                 "uuid": "ab31cade-d9c1-484d-8482-8a4cbee5241a" | ||||
|             } | ||||
|         } | ||||
|     } | ||||
| 
 | ||||
| .. versionadded: 3.0.0 | ||||
| 
 | ||||
| Merging lists of dictionaries | ||||
| ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^ | ||||
| 
 | ||||
| If you have two lists of dictionaries and want to combine them into a list of merged dictionaries, where two dictionaries are merged if they coincide in one attribute, you can use the ``lists_mergeby`` filter. | ||||
| 
 | ||||
| .. code-block:: yaml+jinja | ||||
| 
 | ||||
|     - name: Merge two lists by common attribute 'name' | ||||
|       debug: | ||||
|         var: list1 | community.general.lists_mergeby(list2, 'name') | ||||
|       vars: | ||||
|         list1: | ||||
|           - name: foo | ||||
|             extra: true | ||||
|           - name: bar | ||||
|             extra: false | ||||
|           - name: meh | ||||
|             extra: true | ||||
|         list2: | ||||
|           - name: foo | ||||
|             path: /foo | ||||
|           - name: baz | ||||
|             path: /bazzz | ||||
| 
 | ||||
| This produces: | ||||
| 
 | ||||
| .. code-block:: ansible-output | ||||
| 
 | ||||
|     TASK [Merge two lists by common attribute 'name']  **************************************** | ||||
|     ok: [localhost] => { | ||||
|         "list1 | community.general.lists_mergeby(list2, 'name')": [ | ||||
|             { | ||||
|                 "extra": false, | ||||
|                 "name": "bar" | ||||
|             }, | ||||
|             { | ||||
|                 "name": "baz", | ||||
|                 "path": "/bazzz" | ||||
|             }, | ||||
|             { | ||||
|                 "extra": true, | ||||
|                 "name": "foo", | ||||
|                 "path": "/foo" | ||||
|             }, | ||||
|             { | ||||
|                 "extra": true, | ||||
|                 "name": "meh" | ||||
|             } | ||||
|         ] | ||||
|     } | ||||
| 
 | ||||
| .. versionadded: 2.0.0 | ||||
| 
 | ||||
| Working with times | ||||
| ------------------ | ||||
| 
 | ||||
| The ``to_time_unit`` filter allows to convert times from a human-readable string to a unit. For example, ``'4h 30min 12second' | community.general.to_time_unit('hour')`` gives the number of hours that correspond to 4 hours, 30 minutes and 12 seconds. | ||||
| 
 | ||||
| There are shorthands to directly convert to various units, like ``to_hours``, ``to_minutes``, ``to_seconds``, and so on. The following table lists all units that can be used: | ||||
| 
 | ||||
| .. list-table:: Units | ||||
|    :widths: 25 25 25 25 | ||||
|    :header-rows: 1 | ||||
| 
 | ||||
|    * - Unit name | ||||
|      - Unit value in seconds | ||||
|      - Unit strings for filter | ||||
|      - Shorthand filter | ||||
|    * - Millisecond | ||||
|      - 1/1000 second | ||||
|      - ``ms``, ``millisecond``, ``milliseconds``, ``msec``, ``msecs``, ``msecond``, ``mseconds`` | ||||
|      - ``to_milliseconds`` | ||||
|    * - Second | ||||
|      - 1 second | ||||
|      - ``s``, ``sec``, ``secs``, ``second``, ``seconds`` | ||||
|      - ``to_seconds`` | ||||
|    * - Minute | ||||
|      - 60 seconds | ||||
|      - ``m``, ``min``, ``mins``, ``minute``, ``minutes`` | ||||
|      - ``to_minutes`` | ||||
|    * - Hour | ||||
|      - 60*60 seconds | ||||
|      - ``h``, ``hour``, ``hours`` | ||||
|      - ``to_hours`` | ||||
|    * - Day | ||||
|      - 24*60*60 seconds | ||||
|      - ``d``, ``day``, ``days`` | ||||
|      - ``to_days`` | ||||
|    * - Week | ||||
|      - 7*24*60*60 seconds | ||||
|      - ``w``, ``week``, ``weeks`` | ||||
|      - ``to_weeks`` | ||||
|    * - Month | ||||
|      - 30*24*60*60 seconds | ||||
|      - ``mo``, ``month``, ``months`` | ||||
|      - ``to_months`` | ||||
|    * - Year | ||||
|      - 365*24*60*60 seconds | ||||
|      - ``y``, ``year``, ``years`` | ||||
|      - ``to_years`` | ||||
| 
 | ||||
| Note that months and years are using a simplified representation: a month is 30 days, and a year is 365 days. If you need different definitions of months or years, you can pass them as keyword arguments. For example, if you want a year to be 365.25 days, and a month to be 30.5 days, you can write ``'11months 4' | community.general.to_years(year=365.25, month=30.5)``. These keyword arguments can be specified to ``to_time_unit`` and to all shorthand filters. | ||||
| 
 | ||||
| .. code-block:: yaml+jinja | ||||
| 
 | ||||
|     - name: Convert string to seconds | ||||
|       debug: | ||||
|         msg: "{{ '30h 20m 10s 123ms' | community.general.to_time_unit('seconds') }}" | ||||
| 
 | ||||
|     - name: Convert string to hours | ||||
|       debug: | ||||
|         msg: "{{ '30h 20m 10s 123ms' | community.general.to_hours }}" | ||||
| 
 | ||||
|     - name: Convert string to years (using 365.25 days == 1 year) | ||||
|       debug: | ||||
|         msg: "{{ '400d 15h' | community.general.to_years(year=365.25) }}" | ||||
| 
 | ||||
| This produces: | ||||
| 
 | ||||
| .. code-block:: ansible-output | ||||
| 
 | ||||
|     TASK [Convert string to seconds] ********************************************************** | ||||
|     ok: [localhost] => { | ||||
|         "msg": "109210.123" | ||||
|     } | ||||
| 
 | ||||
|     TASK [Convert string to hours] ************************************************************ | ||||
|     ok: [localhost] => { | ||||
|         "msg": "30.336145277778" | ||||
|     } | ||||
| 
 | ||||
|     TASK [Convert string to years (using 365.25 days == 1 year)] ****************************** | ||||
|     ok: [localhost] => { | ||||
|         "msg": "1.096851471595" | ||||
|     } | ||||
| 
 | ||||
| .. versionadded: 0.2.0 | ||||
| 
 | ||||
| Working with versions | ||||
| --------------------- | ||||
| 
 | ||||
| If you need to sort a list of version numbers, the Jinja ``sort`` filter is problematic. Since it sorts lexicographically, ``2.10`` will come before ``2.9``. To treat version numbers correctly, you can use the ``version_sort`` filter: | ||||
| 
 | ||||
| .. code-block:: yaml+jinja | ||||
| 
 | ||||
|     - name: Sort list by version number | ||||
|       debug: | ||||
|         var: ansible_versions | community.general.version_sort | ||||
|       vars: | ||||
|         ansible_versions: | ||||
|           - '2.8.0' | ||||
|           - '2.11.0' | ||||
|           - '2.7.0' | ||||
|           - '2.10.0' | ||||
|           - '2.9.0' | ||||
| 
 | ||||
| This produces: | ||||
| 
 | ||||
| .. code-block:: ansible-output | ||||
| 
 | ||||
|     TASK [Sort list by version number] ******************************************************** | ||||
|     ok: [localhost] => { | ||||
|         "ansible_versions | community.general.version_sort": [ | ||||
|             "2.7.0", | ||||
|             "2.8.0", | ||||
|             "2.9.0", | ||||
|             "2.10.0", | ||||
|             "2.11.0" | ||||
|         ] | ||||
|     } | ||||
| 
 | ||||
| .. versionadded: 2.2.0 | ||||
| 
 | ||||
| Creating identifiers | ||||
| -------------------- | ||||
| 
 | ||||
| The following filters allow to create identifiers. | ||||
| 
 | ||||
| Hashids | ||||
| ^^^^^^^ | ||||
| 
 | ||||
| `Hashids <https://hashids.org/>`_ allow to convert sequences of integers to short unique string identifiers. This filter needs the `hashids Python library <https://pypi.org/project/hashids/>`_ installed on the controller. | ||||
| 
 | ||||
| .. code-block:: yaml+jinja | ||||
| 
 | ||||
|     - name: "Create hashid" | ||||
|       debug: | ||||
|         msg: "{{ [1234, 5, 6] | community.general.hashids_encode }}" | ||||
| 
 | ||||
|     - name: "Decode hashid" | ||||
|       debug: | ||||
|         msg: "{{ 'jm2Cytn' | community.general.hashids_decode }}" | ||||
| 
 | ||||
| This produces: | ||||
| 
 | ||||
| .. code-block:: ansible-output | ||||
| 
 | ||||
|     TASK [Create hashid] ********************************************************************** | ||||
|     ok: [localhost] => { | ||||
|         "msg": "jm2Cytn" | ||||
|     } | ||||
| 
 | ||||
|     TASK [Decode hashid] ********************************************************************** | ||||
|     ok: [localhost] => { | ||||
|         "msg": [ | ||||
|             1234, | ||||
|             5, | ||||
|             6 | ||||
|         ] | ||||
|     } | ||||
| 
 | ||||
| The hashids filters accept keyword arguments to allow fine-tuning the hashids generated: | ||||
| 
 | ||||
| :salt: String to use as salt when hashing. | ||||
| :alphabet: String of 16 or more unique characters to produce a hash. | ||||
| :min_length: Minimum length of hash produced. | ||||
| 
 | ||||
| .. versionadded: 3.0.0 | ||||
| 
 | ||||
| Random MACs | ||||
| ^^^^^^^^^^^ | ||||
| 
 | ||||
| You can use the ``random_mac`` filter to complete a partial `MAC address <https://en.wikipedia.org/wiki/MAC_address>`_ to a random 6-byte MAC address. | ||||
| 
 | ||||
| .. code-block:: yaml+jinja | ||||
| 
 | ||||
|     - name: "Create a random MAC starting with ff:" | ||||
|       debug: | ||||
|         msg: "{{ 'FF' | community.general.random_mac }}" | ||||
| 
 | ||||
|     - name: "Create a random MAC starting with 00:11:22:" | ||||
|       debug: | ||||
|         msg: "{{ '00:11:22' | community.general.random_mac }}" | ||||
|    | ||||
| This produces: | ||||
| 
 | ||||
| .. code-block:: ansible-output | ||||
| 
 | ||||
|     TASK [Create a random MAC starting with ff:] ********************************************** | ||||
|     ok: [localhost] => { | ||||
|         "msg": "ff:69:d3:78:7f:b4" | ||||
|     } | ||||
| 
 | ||||
|     TASK [Create a random MAC starting with 00:11:22:] **************************************** | ||||
|     ok: [localhost] => { | ||||
|         "msg": "00:11:22:71:5d:3b" | ||||
|     } | ||||
| 
 | ||||
| You can also initialize the random number generator from a seed to create random-but-idempotent MAC addresses: | ||||
| 
 | ||||
| .. code-block:: yaml+jinja | ||||
| 
 | ||||
|     "{{ '52:54:00' | community.general.random_mac(seed=inventory_hostname) }}" | ||||
| 
 | ||||
| Conversions | ||||
| ----------- | ||||
| 
 | ||||
| Parsing CSV files | ||||
| ^^^^^^^^^^^^^^^^^ | ||||
| 
 | ||||
| Ansible offers the :ref:`community.general.read_csv module <ansible_collections.community.general.read_csv_module>` to read CSV files. Sometimes you need to convert strings to CSV files instead. For this, the ``from_csv`` filter exists. | ||||
| 
 | ||||
| .. code-block:: yaml+jinja | ||||
| 
 | ||||
|     - name: "Parse CSV from string" | ||||
|       debug: | ||||
|         msg: "{{ csv_string | community.general.from_csv }}" | ||||
|       vars: | ||||
|         csv_string: | | ||||
|           foo,bar,baz | ||||
|           1,2,3 | ||||
|           you,this,then | ||||
| 
 | ||||
| This produces: | ||||
| 
 | ||||
| .. code-block:: ansible-output | ||||
| 
 | ||||
|     TASK [Parse CSV from string] ************************************************************** | ||||
|     ok: [localhost] => { | ||||
|         "msg": [ | ||||
|             { | ||||
|                 "bar": "2", | ||||
|                 "baz": "3", | ||||
|                 "foo": "1" | ||||
|             }, | ||||
|             { | ||||
|                 "bar": "this", | ||||
|                 "baz": "then", | ||||
|                 "foo": "you" | ||||
|             } | ||||
|         ] | ||||
|     } | ||||
| 
 | ||||
| The ``from_csv`` filter has several keyword arguments to control its behavior: | ||||
| 
 | ||||
| :dialect: Dialect of the CSV file. Default is ``excel``. Other possible choices are ``excel-tab`` and ``unix``. If one of ``delimiter``, ``skipinitialspace`` or ``strict`` is specified, ``dialect`` is ignored. | ||||
| :fieldnames: A set of column names to use. If not provided, the first line of the CSV is assumed to contain the column names. | ||||
| :delimiter: Sets the delimiter to use. Default depends on the dialect used. | ||||
| :skipinitialspace: Set to ``true`` to ignore space directly after the delimiter. Default depends on the dialect used (usually ``false``). | ||||
| :strict: Set to ``true`` to error out on invalid CSV input. | ||||
| 
 | ||||
| .. versionadded: 3.0.0 | ||||
| 
 | ||||
| Converting to JSON | ||||
| ^^^^^^^^^^^^^^^^^^ | ||||
| 
 | ||||
| `JC <https://pypi.org/project/jc/>`_ is a CLI tool and Python library which allows to interpret output of various CLI programs as JSON. It is also available as a filter in community.general. This filter needs the `jc Python library <https://pypi.org/project/jc/>`_ installed on the controller. | ||||
| 
 | ||||
| .. code-block:: yaml+jinja | ||||
| 
 | ||||
|     - name: Run 'ls' to list files in / | ||||
|       command: ls / | ||||
|       register: result | ||||
| 
 | ||||
|     - name: Parse the ls output | ||||
|       debug: | ||||
|         msg: "{{ result.stdout | community.general.jc('ls') }}" | ||||
| 
 | ||||
| This produces: | ||||
| 
 | ||||
| .. code-block:: ansible-output | ||||
| 
 | ||||
|     TASK [Run 'ls' to list files in /] ******************************************************** | ||||
|     changed: [localhost] | ||||
| 
 | ||||
|     TASK [Parse the ls output] **************************************************************** | ||||
|     ok: [localhost] => { | ||||
|         "msg": [ | ||||
|             { | ||||
|                 "filename": "bin" | ||||
|             }, | ||||
|             { | ||||
|                 "filename": "boot" | ||||
|             }, | ||||
|             { | ||||
|                 "filename": "dev" | ||||
|             }, | ||||
|             { | ||||
|                 "filename": "etc" | ||||
|             }, | ||||
|             { | ||||
|                 "filename": "home" | ||||
|             }, | ||||
|             { | ||||
|                 "filename": "lib" | ||||
|             }, | ||||
|             { | ||||
|                 "filename": "proc" | ||||
|             }, | ||||
|             { | ||||
|                 "filename": "root" | ||||
|             }, | ||||
|             { | ||||
|                 "filename": "run" | ||||
|             }, | ||||
|             { | ||||
|                 "filename": "tmp" | ||||
|             } | ||||
|         ] | ||||
|     } | ||||
| 
 | ||||
| .. versionadded: 2.0.0 | ||||
| 
 | ||||
| .. _ansible_collections.community.general.docsite.json_query_filter: | ||||
| 
 | ||||
| Selecting JSON data: JSON queries | ||||
| --------------------------------- | ||||
| 
 | ||||
| To select a single element or a data subset from a complex data structure in JSON format (for example, Ansible facts), use the ``json_query`` filter.  The ``json_query`` filter lets you query a complex JSON structure and iterate over it using a loop structure. | ||||
| 
 | ||||
| .. note:: You must manually install the **jmespath** dependency on the Ansible controller before using this filter. This filter is built upon **jmespath**, and you can use the same syntax. For examples, see `jmespath examples <http://jmespath.org/examples.html>`_. | ||||
| 
 | ||||
| Consider this data structure: | ||||
| 
 | ||||
| .. code-block:: yaml+jinja | ||||
| 
 | ||||
|     { | ||||
|         "domain_definition": { | ||||
|             "domain": { | ||||
|                 "cluster": [ | ||||
|                     { | ||||
|                         "name": "cluster1" | ||||
|                     }, | ||||
|                     { | ||||
|                         "name": "cluster2" | ||||
|                     } | ||||
|                 ], | ||||
|                 "server": [ | ||||
|                     { | ||||
|                         "name": "server11", | ||||
|                         "cluster": "cluster1", | ||||
|                         "port": "8080" | ||||
|                     }, | ||||
|                     { | ||||
|                         "name": "server12", | ||||
|                         "cluster": "cluster1", | ||||
|                         "port": "8090" | ||||
|                     }, | ||||
|                     { | ||||
|                         "name": "server21", | ||||
|                         "cluster": "cluster2", | ||||
|                         "port": "9080" | ||||
|                     }, | ||||
|                     { | ||||
|                         "name": "server22", | ||||
|                         "cluster": "cluster2", | ||||
|                         "port": "9090" | ||||
|                     } | ||||
|                 ], | ||||
|                 "library": [ | ||||
|                     { | ||||
|                         "name": "lib1", | ||||
|                         "target": "cluster1" | ||||
|                     }, | ||||
|                     { | ||||
|                         "name": "lib2", | ||||
|                         "target": "cluster2" | ||||
|                     } | ||||
|                 ] | ||||
|             } | ||||
|         } | ||||
|     } | ||||
| 
 | ||||
| To extract all clusters from this structure, you can use the following query: | ||||
| 
 | ||||
| .. code-block:: yaml+jinja | ||||
| 
 | ||||
|     - name: Display all cluster names | ||||
|       ansible.builtin.debug: | ||||
|         var: item | ||||
|       loop: "{{ domain_definition | community.general.json_query('domain.cluster[*].name') }}" | ||||
| 
 | ||||
| To extract all server names: | ||||
| 
 | ||||
| .. code-block:: yaml+jinja | ||||
| 
 | ||||
|     - name: Display all server names | ||||
|       ansible.builtin.debug: | ||||
|         var: item | ||||
|       loop: "{{ domain_definition | community.general.json_query('domain.server[*].name') }}" | ||||
| 
 | ||||
| To extract ports from cluster1: | ||||
| 
 | ||||
| .. code-block:: yaml+jinja | ||||
| 
 | ||||
|     - name: Display all ports from cluster1 | ||||
|       ansible.builtin.debug: | ||||
|         var: item | ||||
|       loop: "{{ domain_definition | community.general.json_query(server_name_cluster1_query) }}" | ||||
|       vars: | ||||
|         server_name_cluster1_query: "domain.server[?cluster=='cluster1'].port" | ||||
| 
 | ||||
| .. note:: You can use a variable to make the query more readable. | ||||
| 
 | ||||
| To print out the ports from cluster1 in a comma separated string: | ||||
| 
 | ||||
| .. code-block:: yaml+jinja | ||||
| 
 | ||||
|     - name: Display all ports from cluster1 as a string | ||||
|       ansible.builtin.debug: | ||||
|         msg: "{{ domain_definition | community.general.json_query('domain.server[?cluster==`cluster1`].port') | join(', ') }}" | ||||
| 
 | ||||
| .. note:: In the example above, quoting literals using backticks avoids escaping quotes and maintains readability. | ||||
| 
 | ||||
| You can use YAML `single quote escaping <https://yaml.org/spec/current.html#id2534365>`_: | ||||
| 
 | ||||
| .. code-block:: yaml+jinja | ||||
| 
 | ||||
|     - name: Display all ports from cluster1 | ||||
|       ansible.builtin.debug: | ||||
|         var: item | ||||
|       loop: "{{ domain_definition | community.general.json_query('domain.server[?cluster==''cluster1''].port') }}" | ||||
| 
 | ||||
| .. note:: Escaping single quotes within single quotes in YAML is done by doubling the single quote. | ||||
| 
 | ||||
| To get a hash map with all ports and names of a cluster: | ||||
| 
 | ||||
| .. code-block:: yaml+jinja | ||||
| 
 | ||||
|     - name: Display all server ports and names from cluster1 | ||||
|       ansible.builtin.debug: | ||||
|         var: item | ||||
|       loop: "{{ domain_definition | community.general.json_query(server_name_cluster1_query) }}" | ||||
|       vars: | ||||
|         server_name_cluster1_query: "domain.server[?cluster=='cluster2'].{name: name, port: port}" | ||||
| 
 | ||||
| To extract ports from all clusters with name starting with 'server1': | ||||
| 
 | ||||
| .. code-block:: yaml+jinja | ||||
| 
 | ||||
|     - name: Display all ports from cluster1 | ||||
|       ansible.builtin.debug: | ||||
|         msg: "{{ domain_definition | to_json | from_json | community.general.json_query(server_name_query) }}" | ||||
|       vars: | ||||
|         server_name_query: "domain.server[?starts_with(name,'server1')].port" | ||||
| 
 | ||||
| To extract ports from all clusters with name containing 'server1': | ||||
| 
 | ||||
| .. code-block:: yaml+jinja | ||||
| 
 | ||||
|     - name: Display all ports from cluster1 | ||||
|       ansible.builtin.debug: | ||||
|         msg: "{{ domain_definition | to_json | from_json | community.general.json_query(server_name_query) }}" | ||||
|       vars: | ||||
|         server_name_query: "domain.server[?contains(name,'server1')].port" | ||||
| 
 | ||||
| .. note:: while using ``starts_with`` and ``contains``, you have to use `` to_json | from_json `` filter for correct parsing of data structure. | ||||
		Loading…
	
	Add table
		Add a link
		
	
		Reference in a new issue