123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248 |
- #!/usr/bin/env python
- # pylint: disable=missing-docstring
- #
- # Copyright 2017, 2018 Red Hat, Inc. and/or its affiliates
- # and other contributors as indicated by the @author tags.
- #
- # Licensed under the Apache License, Version 2.0 (the "License");
- # you may not use this file except in compliance with the License.
- # You may obtain a copy of the License at
- #
- # http://www.apache.org/licenses/LICENSE-2.0
- #
- # Unless required by applicable law or agreed to in writing, software
- # distributed under the License is distributed on an "AS IS" BASIS,
- # WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
- # See the License for the specific language governing permissions and
- # limitations under the License.
- import base64
- import json
- import os
- from ansible.module_utils.basic import AnsibleModule
- DOCUMENTATION = '''
- ---
- module: docker_creds
- short_description: Creates/updates a 'docker login' file in place of using 'docker login'
- version_added: "2.4"
- description:
- - This module creates a docker config.json file in the directory provided by 'path'
- on hosts that do not support 'docker login' but need the file present for
- registry authentication purposes of various other services.
- options:
- path:
- description:
- - This is the message to send to the sample module
- required: true
- registry:
- description:
- - This is the registry the credentials are for.
- required: true
- username:
- description:
- - This is the username to authenticate to the registry with.
- required: true
- password:
- description:
- - This is the password to authenticate to the registry with.
- required: true
- test_login:
- description:
- - Attempt to connect to registry with username + password provided.
- default: true
- required: false
- author:
- - "Michael Gugino <mgugino@redhat.com>"
- '''
- EXAMPLES = '''
- # Pass in a message
- - name: Place credentials in file
- docker_creds:
- path: /root/.docker
- registry: registry.example.com:443
- username: myuser
- password: mypassword
- test_login: True
- '''
- def check_dest_dir_exists(module, dest):
- '''Check if dest dir is present and is a directory'''
- dir_exists = os.path.exists(dest)
- if dir_exists:
- if not os.path.isdir(dest):
- msg = "{} exists but is not a directory".format(dest)
- result = {'failed': True,
- 'changed': False,
- 'msg': msg,
- 'state': 'unknown'}
- module.fail_json(**result)
- else:
- return 1
- else:
- return 0
- def create_dest_dir(module, dest):
- try:
- os.makedirs(dest, mode=0o700)
- except OSError as oserror:
- result = {'failed': True,
- 'changed': False,
- 'msg': str(oserror),
- 'state': 'unknown'}
- module.fail_json(**result)
- def load_config_file(module, dest):
- '''load the config.json in directory dest'''
- conf_file_path = os.path.join(dest, 'config.json')
- if os.path.exists(conf_file_path):
- # Try to open the file and load json data
- try:
- with open(conf_file_path) as conf_file:
- data = conf_file.read()
- jdata = json.loads(data)
- except IOError as ioerror:
- result = {'failed': True,
- 'changed': False,
- 'msg': str(ioerror),
- 'state': 'unknown'}
- module.fail_json(**result)
- except ValueError as jsonerror:
- result = {'failed': True,
- 'changed': False,
- 'msg': str(jsonerror),
- 'state': 'unknown'}
- module.fail_json(**result)
- return jdata
- else:
- # File doesn't exist, we just return an empty dictionary.
- return {}
- def gen_skopeo_cmd(registry, username, password, proxy_vars, image_name):
- '''Generate skopeo command to run'''
- skopeo_temp = ("{proxy_vars} timeout 10 skopeo inspect"
- " {creds} docker://{registry}/{image_name}")
- creds = '--creds {}:{}'.format(username, password)
- skopeo_args = {'proxy_vars': proxy_vars, 'creds': creds,
- 'registry': registry, 'image_name': image_name}
- return skopeo_temp.format(**skopeo_args).strip()
- def validate_registry_login(module, skopeo_command):
- '''Attempt to use credentials to log into registry'''
- # skopeo doesn't honor docker config file proxy settings; need to specify
- # proxy vars on the cli.
- rtnc, _, err = module.run_command(skopeo_command, use_unsafe_shell=True)
- if rtnc:
- result = {'failed': True,
- 'changed': False,
- 'msg': str(err),
- 'state': 'unknown'}
- module.fail_json(**result)
- def update_config(docker_config, registry, encoded_auth):
- '''Add our registry auth credentials into docker_config dict'''
- # Add anything that might be missing in our dictionary
- if 'auths' not in docker_config:
- docker_config['auths'] = {}
- if registry not in docker_config['auths']:
- docker_config['auths'][registry] = {}
- # check if the same value is already present for idempotency.
- if 'auth' in docker_config['auths'][registry]:
- if docker_config['auths'][registry]['auth'] == encoded_auth:
- # No need to go further, everything is already set in file.
- return False
- docker_config['auths'][registry]['auth'] = encoded_auth
- return True
- def write_config(module, docker_config, dest):
- '''Write updated credentials into dest/config.json'''
- if not isinstance(docker_config, dict):
- docker_config = docker_config.decode()
- conf_file_path = os.path.join(dest, 'config.json')
- try:
- with open(conf_file_path, 'w') as conf_file:
- json.dump(docker_config, conf_file, indent=8)
- except IOError as ioerror:
- result = {'failed': True,
- 'changed': False,
- 'msg': str(ioerror),
- 'state': 'unknown'}
- module.fail_json(**result)
- def run_module():
- '''Run this module'''
- module_args = dict(
- path=dict(aliases=['dest', 'name'], required=True, type='path'),
- registry=dict(type='str', required=True),
- username=dict(type='str', required=True),
- password=dict(type='str', required=True, no_log=True),
- test_login=dict(type='str', required=False, default=True),
- proxy_vars=dict(type='str', required=False, default=''),
- image_name=dict(type='str', required=True),
- )
- module = AnsibleModule(
- argument_spec=module_args,
- supports_check_mode=False
- )
- # First, create our dest dir if necessary
- dest = module.params['path']
- registry = module.params['registry']
- username = module.params['username']
- password = module.params['password']
- test_login = module.params['test_login']
- proxy_vars = module.params['proxy_vars']
- image_name = module.params['image_name']
- if not check_dest_dir_exists(module, dest):
- create_dest_dir(module, dest)
- docker_config = {}
- else:
- # We want to scrape the contents of dest/config.json
- # in case there are other registries/settings already present.
- docker_config = load_config_file(module, dest)
- # Test the credentials
- if test_login:
- skopeo_command = gen_skopeo_cmd(registry, username, password,
- proxy_vars, image_name)
- validate_registry_login(module, skopeo_command)
- # base64 encode our username:password string
- encoded_auth = base64.b64encode('{}:{}'.format(username, password).encode())
- # Put the registry auth info into the config dict.
- changed = update_config(docker_config, registry, encoded_auth)
- if changed:
- write_config(module, docker_config, dest)
- result = {'changed': changed, 'rc': 0}
- module.exit_json(**result)
- def main():
- run_module()
- if __name__ == '__main__':
- main()
|