mirror of
				https://github.com/ansible-collections/community.general.git
				synced 2025-10-25 05:23:58 -07:00 
			
		
		
		
	
		
			
				
	
	
		
			303 lines
		
	
	
	
		
			11 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
			
		
		
	
	
			303 lines
		
	
	
	
		
			11 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
| #!/usr/bin/python
 | |
| # -*- coding: utf-8 -*-
 | |
| 
 | |
| # Copyright (c) 2020, Ansible Project
 | |
| # Copyright (c) 2020, VMware, Inc. All Rights Reserved.
 | |
| # GNU General Public License v3.0+ (see LICENSES/GPL-3.0-or-later.txt or https://www.gnu.org/licenses/gpl-3.0.txt)
 | |
| # SPDX-License-Identifier: GPL-3.0-or-later
 | |
| 
 | |
| from __future__ import absolute_import, division, print_function
 | |
| __metaclass__ = type
 | |
| 
 | |
| DOCUMENTATION = r"""
 | |
| module: iso_create
 | |
| short_description: Generate ISO file with specified files or folders
 | |
| description:
 | |
|   - This module is used to generate ISO file with specified path of files.
 | |
| author:
 | |
|   - Diane Wang (@Tomorrow9) <dianew@vmware.com>
 | |
| requirements:
 | |
|   - "pycdlib"
 | |
| version_added: '0.2.0'
 | |
| 
 | |
| extends_documentation_fragment:
 | |
|   - community.general.attributes
 | |
| 
 | |
| attributes:
 | |
|   check_mode:
 | |
|     support: full
 | |
|   diff_mode:
 | |
|     support: none
 | |
| 
 | |
| options:
 | |
|   src_files:
 | |
|     description:
 | |
|       - This is a list of absolute paths of source files or folders to be contained in the new generated ISO file.
 | |
|       - The module fails if specified file or folder in O(src_files) does not exist on local machine.
 | |
|       - 'Note: With all ISO9660 levels from 1 to 3, all file names are restricted to uppercase letters, numbers and underscores
 | |
|         (_). File names are limited to 31 characters, directory nesting is limited to 8 levels, and path names are limited
 | |
|         to 255 characters.'
 | |
|     type: list
 | |
|     required: true
 | |
|     elements: path
 | |
|   dest_iso:
 | |
|     description:
 | |
|       - The absolute path with file name of the new generated ISO file on local machine.
 | |
|       - It creates intermediate folders when they do not exist.
 | |
|     type: path
 | |
|     required: true
 | |
|   interchange_level:
 | |
|     description:
 | |
|       - The ISO9660 interchange level to use, it dictates the rules on the names of files.
 | |
|       - Levels and valid values V(1), V(2), V(3), V(4) are supported.
 | |
|       - The default value is level V(1), which is the most conservative, level V(3) is recommended.
 | |
|       - ISO9660 file names at interchange level V(1) cannot have more than 8 characters or 3 characters in the extension.
 | |
|     type: int
 | |
|     default: 1
 | |
|     choices: [1, 2, 3, 4]
 | |
|   vol_ident:
 | |
|     description:
 | |
|       - The volume identification string to use on the new generated ISO image.
 | |
|     type: str
 | |
|   rock_ridge:
 | |
|     description:
 | |
|       - Whether to make this ISO have the Rock Ridge extensions or not.
 | |
|       - Valid values are V(1.09), V(1.10) or V(1.12), means adding the specified Rock Ridge version to the ISO.
 | |
|       - If unsure, set V(1.09) to ensure maximum compatibility.
 | |
|       - If not specified, then not add Rock Ridge extension to the ISO.
 | |
|     type: str
 | |
|     choices: ['1.09', '1.10', '1.12']
 | |
|   joliet:
 | |
|     description:
 | |
|       - Support levels and valid values are V(1), V(2), or V(3).
 | |
|       - Level V(3) is by far the most common.
 | |
|       - If not specified, then no Joliet support is added.
 | |
|     type: int
 | |
|     choices: [1, 2, 3]
 | |
|   udf:
 | |
|     description:
 | |
|       - Whether to add UDF support to this ISO.
 | |
|       - If set to V(true), then version 2.60 of the UDF spec is used.
 | |
|       - If not specified or set to V(false), then no UDF support is added.
 | |
|     type: bool
 | |
|     default: false
 | |
| """
 | |
| 
 | |
| EXAMPLES = r"""
 | |
| - name: Create an ISO file
 | |
|   community.general.iso_create:
 | |
|     src_files:
 | |
|       - /root/testfile.yml
 | |
|       - /root/testfolder
 | |
|     dest_iso: /tmp/test.iso
 | |
|     interchange_level: 3
 | |
| 
 | |
| - name: Create an ISO file with Rock Ridge extension
 | |
|   community.general.iso_create:
 | |
|     src_files:
 | |
|       - /root/testfile.yml
 | |
|       - /root/testfolder
 | |
|     dest_iso: /tmp/test.iso
 | |
|     rock_ridge: 1.09
 | |
| 
 | |
| - name: Create an ISO file with Joliet support
 | |
|   community.general.iso_create:
 | |
|     src_files:
 | |
|       - ./windows_config/Autounattend.xml
 | |
|     dest_iso: ./test.iso
 | |
|     interchange_level: 3
 | |
|     joliet: 3
 | |
|     vol_ident: WIN_AUTOINSTALL
 | |
| """
 | |
| 
 | |
| RETURN = r"""
 | |
| source_file:
 | |
|   description: Configured source files or directories list.
 | |
|   returned: on success
 | |
|   type: list
 | |
|   elements: path
 | |
|   sample: ["/path/to/file.txt", "/path/to/folder"]
 | |
| created_iso:
 | |
|   description: Created iso file path.
 | |
|   returned: on success
 | |
|   type: str
 | |
|   sample: "/path/to/test.iso"
 | |
| interchange_level:
 | |
|   description: Configured interchange level.
 | |
|   returned: on success
 | |
|   type: int
 | |
|   sample: 3
 | |
| vol_ident:
 | |
|   description: Configured volume identification string.
 | |
|   returned: on success
 | |
|   type: str
 | |
|   sample: "OEMDRV"
 | |
| joliet:
 | |
|   description: Configured Joliet support level.
 | |
|   returned: on success
 | |
|   type: int
 | |
|   sample: 3
 | |
| rock_ridge:
 | |
|   description: Configured Rock Ridge version.
 | |
|   returned: on success
 | |
|   type: str
 | |
|   sample: "1.09"
 | |
| udf:
 | |
|   description: Configured UDF support.
 | |
|   returned: on success
 | |
|   type: bool
 | |
|   sample: false
 | |
| """
 | |
| 
 | |
| import os
 | |
| import traceback
 | |
| 
 | |
| PYCDLIB_IMP_ERR = None
 | |
| try:
 | |
|     import pycdlib
 | |
|     HAS_PYCDLIB = True
 | |
| except ImportError:
 | |
|     PYCDLIB_IMP_ERR = traceback.format_exc()
 | |
|     HAS_PYCDLIB = False
 | |
| 
 | |
| from ansible.module_utils.basic import AnsibleModule, missing_required_lib
 | |
| from ansible.module_utils.common.text.converters import to_native
 | |
| 
 | |
| 
 | |
| def add_file(module, iso_file=None, src_file=None, file_path=None, rock_ridge=None, use_joliet=None, use_udf=None):
 | |
|     rr_name = None
 | |
|     joliet_path = None
 | |
|     udf_path = None
 | |
|     # In standard ISO interchange level 1, file names have a maximum of 8 characters, followed by a required dot,
 | |
|     # followed by a maximum 3 character extension, followed by a semicolon and a version
 | |
|     file_name = os.path.basename(file_path)
 | |
|     if '.' not in file_name:
 | |
|         file_in_iso_path = file_path.upper() + '.;1'
 | |
|     else:
 | |
|         file_in_iso_path = file_path.upper() + ';1'
 | |
|     if rock_ridge:
 | |
|         rr_name = file_name
 | |
|     if use_joliet:
 | |
|         joliet_path = file_path
 | |
|     if use_udf:
 | |
|         udf_path = file_path
 | |
|     try:
 | |
|         iso_file.add_file(src_file, iso_path=file_in_iso_path, rr_name=rr_name, joliet_path=joliet_path, udf_path=udf_path)
 | |
|     except Exception as err:
 | |
|         module.fail_json(msg="Failed to add file %s to ISO file due to %s" % (src_file, to_native(err)))
 | |
| 
 | |
| 
 | |
| def add_directory(module, iso_file=None, dir_path=None, rock_ridge=None, use_joliet=None, use_udf=None):
 | |
|     rr_name = None
 | |
|     joliet_path = None
 | |
|     udf_path = None
 | |
|     iso_dir_path = dir_path.upper()
 | |
|     if rock_ridge:
 | |
|         rr_name = os.path.basename(dir_path)
 | |
|     if use_joliet:
 | |
|         joliet_path = dir_path
 | |
|     if use_udf:
 | |
|         udf_path = dir_path
 | |
|     try:
 | |
|         iso_file.add_directory(iso_path=iso_dir_path, rr_name=rr_name, joliet_path=joliet_path, udf_path=udf_path)
 | |
|     except Exception as err:
 | |
|         module.fail_json(msg="Failed to directory %s to ISO file due to %s" % (dir_path, to_native(err)))
 | |
| 
 | |
| 
 | |
| def main():
 | |
|     argument_spec = dict(
 | |
|         src_files=dict(type='list', required=True, elements='path'),
 | |
|         dest_iso=dict(type='path', required=True),
 | |
|         interchange_level=dict(type='int', choices=[1, 2, 3, 4], default=1),
 | |
|         vol_ident=dict(type='str'),
 | |
|         rock_ridge=dict(type='str', choices=['1.09', '1.10', '1.12']),
 | |
|         joliet=dict(type='int', choices=[1, 2, 3]),
 | |
|         udf=dict(type='bool', default=False),
 | |
|     )
 | |
|     module = AnsibleModule(
 | |
|         argument_spec=argument_spec,
 | |
|         supports_check_mode=True,
 | |
|     )
 | |
|     if not HAS_PYCDLIB:
 | |
|         module.fail_json(missing_required_lib('pycdlib'), exception=PYCDLIB_IMP_ERR)
 | |
| 
 | |
|     src_file_list = module.params.get('src_files')
 | |
|     if src_file_list and len(src_file_list) == 0:
 | |
|         module.fail_json(msg='Please specify source file and/or directory list using src_files parameter.')
 | |
|     for src_file in src_file_list:
 | |
|         if not os.path.exists(src_file):
 | |
|             module.fail_json(msg="Specified source file/directory path does not exist on local machine, %s" % src_file)
 | |
| 
 | |
|     dest_iso = module.params.get('dest_iso')
 | |
|     if dest_iso and len(dest_iso) == 0:
 | |
|         module.fail_json(msg='Please specify the absolute path of the new created ISO file using dest_iso parameter.')
 | |
| 
 | |
|     dest_iso_dir = os.path.dirname(dest_iso)
 | |
|     if dest_iso_dir and not os.path.exists(dest_iso_dir):
 | |
|         # will create intermediate dir for new ISO file
 | |
|         try:
 | |
|             os.makedirs(dest_iso_dir)
 | |
|         except OSError as err:
 | |
|             module.fail_json(msg='Exception caught when creating folder %s, with error %s' % (dest_iso_dir, to_native(err)))
 | |
| 
 | |
|     volume_id = module.params.get('vol_ident')
 | |
|     if volume_id is None:
 | |
|         volume_id = ''
 | |
|     inter_level = module.params.get('interchange_level')
 | |
|     rock_ridge = module.params.get('rock_ridge')
 | |
|     use_joliet = module.params.get('joliet')
 | |
|     use_udf = None
 | |
|     if module.params['udf']:
 | |
|         use_udf = '2.60'
 | |
| 
 | |
|     result = dict(
 | |
|         changed=False,
 | |
|         source_file=src_file_list,
 | |
|         created_iso=dest_iso,
 | |
|         interchange_level=inter_level,
 | |
|         vol_ident=volume_id,
 | |
|         rock_ridge=rock_ridge,
 | |
|         joliet=use_joliet,
 | |
|         udf=use_udf
 | |
|     )
 | |
|     if not module.check_mode:
 | |
|         iso_file = pycdlib.PyCdlib(always_consistent=True)
 | |
|         iso_file.new(interchange_level=inter_level, vol_ident=volume_id, rock_ridge=rock_ridge, joliet=use_joliet, udf=use_udf)
 | |
| 
 | |
|         for src_file in src_file_list:
 | |
|             # if specify a dir then go through the dir to add files and dirs
 | |
|             if os.path.isdir(src_file):
 | |
|                 dir_list = []
 | |
|                 file_list = []
 | |
|                 src_file = src_file.rstrip('/')
 | |
|                 dir_name = os.path.basename(src_file)
 | |
|                 add_directory(module, iso_file=iso_file, dir_path='/' + dir_name, rock_ridge=rock_ridge,
 | |
|                               use_joliet=use_joliet, use_udf=use_udf)
 | |
| 
 | |
|                 # get dir list and file list
 | |
|                 for path, dirs, files in os.walk(src_file):
 | |
|                     for filename in files:
 | |
|                         file_list.append(os.path.join(path, filename))
 | |
|                     for dir in dirs:
 | |
|                         dir_list.append(os.path.join(path, dir))
 | |
|                 for new_dir in dir_list:
 | |
|                     add_directory(module, iso_file=iso_file, dir_path=new_dir.split(os.path.dirname(src_file))[1],
 | |
|                                   rock_ridge=rock_ridge, use_joliet=use_joliet, use_udf=use_udf)
 | |
|                 for new_file in file_list:
 | |
|                     add_file(module, iso_file=iso_file, src_file=new_file,
 | |
|                              file_path=new_file.split(os.path.dirname(src_file))[1], rock_ridge=rock_ridge,
 | |
|                              use_joliet=use_joliet, use_udf=use_udf)
 | |
|             # if specify a file then add this file directly to the '/' path in ISO
 | |
|             else:
 | |
|                 add_file(module, iso_file=iso_file, src_file=src_file, file_path='/' + os.path.basename(src_file),
 | |
|                          rock_ridge=rock_ridge, use_joliet=use_joliet, use_udf=use_udf)
 | |
| 
 | |
|         iso_file.write(dest_iso)
 | |
|         iso_file.close()
 | |
| 
 | |
|     result['changed'] = True
 | |
|     module.exit_json(**result)
 | |
| 
 | |
| 
 | |
| if __name__ == '__main__':
 | |
|     main()
 |