| #!/usr/bin/env python |
| """ |
| Management of debmirror |
| ======================= |
| |
| Create debian mirrors, using debmirror |
| -------------------------------------- |
| |
| .. code-block:: yaml |
| debmirror_test1_present: |
| debmirror.mirror_present: |
| - name: test1 |
| |
| """ |
| import logging |
| import os |
| from functools import wraps |
| from salt.exceptions import CommandExecutionError, SaltInvocationError |
| log = logging.getLogger(__name__) |
| |
| |
| def __virtual__(): |
| ''' |
| Only load this module if debmirror is installed with deps. |
| ''' |
| return 'debmirror' |
| |
| |
| def _test_call(method): |
| (resource, functionality) = method.func_name.split('_') |
| if functionality == 'present': |
| functionality = 'updated' |
| else: |
| functionality = 'removed' |
| |
| @wraps(method) |
| def check_for_testing(name, *args, **kwargs): |
| if __opts__.get('test', None): |
| return _no_change(name, resource, test=functionality) |
| return method(name, *args, **kwargs) |
| |
| return check_for_testing |
| |
| |
| def _created(name, resource, resource_definition={}): |
| changes_dict = {'name': name, |
| 'changes': resource_definition, |
| 'result': True, |
| 'comment': '{0} {1} created'.format(resource, name)} |
| return changes_dict |
| |
| |
| def _failed(name, resource, resource_definition={}): |
| changes_dict = {'name': name, |
| 'changes': resource_definition, |
| 'result': False, |
| 'comment': '{0} {1} failed to create'.format(resource, |
| name)} |
| return changes_dict |
| |
| |
| def _no_change(name, resource, test=False): |
| changes_dict = {'name': name, |
| 'changes': {}, |
| 'result': True} |
| if test: |
| changes_dict['comment'] = \ |
| '{0} {1} will be {2}'.format(resource, name, test) |
| else: |
| changes_dict['comment'] = \ |
| '{0} {1} is in correct state'.format(resource, name) |
| return changes_dict |
| |
| |
| def _check_state(name, tgt): |
| lock_file = _get_target_path(name, tgt)['lock_file'] |
| if os.path.isfile(lock_file) and not tgt.get('force', False): |
| return _no_change(name, '{} exist=>repo locked.'.format(lock_file)) |
| return False |
| |
| |
| def _get_target_path(name, tgt): |
| if not tgt.get('target_dir', False): |
| raise SaltInvocationError('Argument "target_dir" is mandatory! ') |
| |
| out = {'target_dir': tgt.get('target_dir'), |
| 'lock_file': tgt.get('lock_file', |
| os.path.join(tgt.get('target_dir'), |
| '.lockmirror')), |
| 'log_file': tgt.get('log_file', '/var/log/debmirror_'.join(name))} |
| |
| if tgt.get('cache_dir', False): |
| out['cache_dir'] = tgt.get('cache_dir') |
| out['cache_ready_file'] = os.path.join(out['cache_dir'], '.ready') |
| |
| return out |
| |
| |
| def _get_env(tgt): |
| env = {} |
| for k in ['http_proxy', 'https_proxy', 'ftp_proxy', 'rsync_proxy']: |
| if tgt.get(k, False): |
| env[k] = tgt[k] |
| env[k.upper()] = env[k] |
| if tgt.get('no_proxy', False): |
| env['no_proxy'] = ','.join(str(x) for x in tgt['no_proxy']) |
| env['no_proxy'.upper()] = env['no_proxy'] |
| return env |
| |
| |
| def _check_mirror_cache(file_name): |
| return os.path.isfile(file_name) |
| |
| |
| def _cache_copied(file_name, target_dir, action='check'): |
| pattern = '%s - copied\n' % target_dir |
| if action == 'add': |
| with open(file_name, 'a+') as f: |
| f.write(pattern) |
| return True |
| else: |
| with open(file_name, 'r') as f: |
| cache_copied = f.readlines() |
| if pattern in cache_copied: |
| return True |
| else: |
| return False |
| |
| |
| def _get_cmdline(name, tgt): |
| cmdline = " debmirror " |
| if tgt.get('extra_flags'): |
| cmdline += ' '.join(tgt['extra_flags']) |
| if tgt.get('rsync_options'): |
| rsync_options = tgt.get('rsync_options') |
| if tgt.get('rsync_options_defaults', True): |
| rsync_options = ['-aL', '--partial'] + rsync_options |
| cmdline += " --rsync-options='" + ' '.join(rsync_options) + "'" |
| if tgt.get('dist'): |
| cmdline += ' --dist=' + ",".join(tgt['dist']) |
| if tgt.get('section'): |
| cmdline += ' --section=' + ",".join(tgt['section']) |
| if tgt.get('method'): |
| cmdline += ' --method=' + tgt.get('method') |
| if tgt.get('mirror_host'): |
| cmdline += ' --host="{}"'.format(tgt.get('mirror_host')) |
| if tgt.get('mirror_root'): |
| cmdline += ' --root="{}"'.format(tgt.get('mirror_root')) |
| if tgt.get('arch', 'amd64'): |
| cmdline += ' --arch=' + ','.join(tgt.get('arch')) |
| if tgt.get('exclude_deb_section'): |
| for section in tgt['exclude_deb_section']: |
| cmdline += " --exclude-deb-section='" + section + "'" |
| if tgt.get('filter'): |
| for key, value in enumerate(sorted(tgt['filter'])): |
| cmdline += " " + tgt['filter'][value] |
| if tgt.get('target_dir', False): |
| targets = _get_target_path(name, tgt) |
| target_dir = targets['target_dir'] |
| if tgt.get('cache_dir'): |
| cache_dir = targets['cache_dir'] |
| if _check_mirror_cache(targets['cache_ready_file']): |
| cmdline += ' ' + target_dir |
| else: |
| cmdline += ' ' + cache_dir |
| else: |
| cmdline += ' ' + target_dir |
| return cmdline |
| |
| |
| def _update_mirror(name, tgt): |
| targetPaths = _get_target_path(name, tgt) |
| # Remove old lock file, is was. |
| lock_file = targetPaths['lock_file'] |
| if os.path.isfile(lock_file): |
| log.debug("Removing lockfile: {} for mirror {}".format( |
| lock_file, name)) |
| __states__['file.absent'](lock_file) |
| cmdline = _get_cmdline(name, tgt) |
| # fetch ENV params for proxy |
| env_vars = _get_env(tgt) |
| # init file logger |
| l_dir = os.path.dirname(tgt['log_file']) |
| if not os.path.isdir(l_dir): |
| __salt__['file.makedirs'](l_dir + '/') |
| fh = logging.FileHandler( |
| "{0}".format(targetPaths['log_file'])) |
| fh.setLevel(logging.DEBUG) |
| fh_format = logging.Formatter( |
| '%(asctime)s - %(lineno)d - %(levelname)-8s - %(message)s') |
| fh.setFormatter(fh_format) |
| log2file = logging.getLogger("debmirror") |
| log2file.addHandler(fh) |
| # check cache usage |
| if tgt.get('cache_dir'): |
| cache_dir = targetPaths['cache_dir'] |
| cache_ready = targetPaths['cache_ready_file'] |
| target_dir = targetPaths['target_dir'] |
| cpline = "find . -name *.deb -exec cp -rlf --parent {} %s \\;" % ( |
| target_dir) |
| |
| # if cache exists and not copied - copy and run sync over cache |
| # if cache exists and copied - do nothing |
| if _check_mirror_cache(cache_ready): |
| if not _cache_copied(cache_ready, target_dir): |
| result = __salt__['cmd.run_all'](cpline, redirect_stderr=True, |
| cwd=cache_dir, env=env_vars) |
| log2file.debug(result['stdout']) |
| |
| if result['retcode'] != 0: |
| return _failed(name, |
| "debmirror failed to copy cache. Reason {0}".format( |
| result['stderr'])) |
| |
| _cache_copied(cache_ready, target_dir, 'add') |
| # if cache does not exist - create cache and copy |
| else: |
| result = __salt__['cmd.run_all'](cmdline, redirect_stderr=True, |
| env=env_vars) |
| log2file.debug(result['stdout']) |
| |
| if result['retcode'] != 0: |
| return _failed(name, |
| "debmirror failed. Reason {0}".format( |
| result['stderr'])) |
| |
| __states__['file.touch'](cache_ready) |
| result = __salt__['cmd.run_all'](cpline, redirect_stderr=True, |
| cwd=cache_dir, env=env_vars) |
| log2file.debug(result['stdout']) |
| |
| if result['retcode'] != 0: |
| return _failed(name, |
| "debmirror failed to copy cache. Reason {0}".format( |
| result['stderr'])) |
| |
| _cache_copied(cache_ready, target_dir, 'add') |
| cmdline = _get_cmdline(name, tgt) |
| |
| result = __salt__['cmd.run_all'](cmdline, redirect_stderr=True, |
| env=env_vars) |
| log2file.debug(result['stdout']) |
| # destroy file logger |
| for i in list(log2file.handlers): |
| log2file.removeHandler(i) |
| i.flush() |
| i.close() |
| if result['retcode'] != 0: |
| # raise CommandExecutionError(result['stderr']) |
| return _failed(name, |
| "debmirror failed.Reason {0}".format(result['stderr'])) |
| if tgt.get('lock_target', None): |
| __states__['file.touch'](lock_file) |
| return _created(name, "Mirror {0} created.".format(name)) |
| |
| |
| @_test_call |
| def mirror_present(name, **kwargs): |
| ''' |
| |
| :param name: mirror key name |
| ''' |
| try: |
| tgt = __salt__['config.get']('debmirror')['client']['mirrors'][name] |
| except KeyError: |
| comment = 'Mirror "{0}" not exist in configurathion,skipping..'.format( |
| name) |
| return _no_change(name, comment) |
| |
| current_state = _check_state(name, tgt) |
| if not current_state: |
| return _update_mirror(name, tgt) |
| return current_state |