blob: 00bb7d49edba355524f83d010387c649f61c26f7 [file] [log] [blame]
#!/usr/bin/python
DOCUMENTATION='''
---
module: rancher_k8s_environment
description:
- This module will create or delete Kubernetes environment.
- It will also delete other environments when variables are set accordingly.
notes:
- It identifies environment only by name. Expect problems with same named environments.
- All hosts running Kubernetes cluster should have same OS otherwise there
is possibility of misbehavement.
options:
server:
required: true
description:
- Url of rancher server i.e. "http://10.0.0.1:8080".
name:
required: true
descritpion:
- Name of the environment to create/remove.
descr:
description:
- Description of environment to create.
state:
description:
- If "present" environment will be created or setup depending if it exists.
With multiple environments with same name expect error.
If "absent" environment will be removed. If multiple environments have same
name all will be deleted.
default: present
choices: [present, absent]
delete_not_k8s:
description:
- Indicates if environments with different orchestration than Kubernetes should
be deleted.
type: bool
default: yes
delete_other_k8s:
description:
- Indicates if environments with different name than specified should
be deleted.
type: bool
default: no
force:
description:
- Indicates if environment should be deleted and recreated.
type: bool
default: yes
host_os:
required: true
description:
- OS (family from ansible_os_family variable) of the hosts running cluster. If
"RedHat" then datavolume fix will be applied.
Fix described here:
https://github.com/rancher/rancher/issues/10015
'''
import json
import time
import requests
from ansible.module_utils.basic import AnsibleModule
def get_existing_environments(rancher_address):
req = requests.get('{}/v2-beta/projects'.format(rancher_address))
envs = req.json()['data']
return envs
def not_k8s_ids(environments):
envs = filter(lambda x: x['orchestration'] != 'kubernetes', environments)
return [env['id'] for env in envs]
def other_k8s_ids(environments, name):
envs = filter(lambda x: x['orchestration'] == 'kubernetes' and x['name'] != name,
environments)
return [env['id'] for env in envs]
def env_ids_by_name(environments, name):
envs = filter(lambda x: x['name'] == name, environments)
return [env['id'] for env in envs]
def env_info_by_id(environments, env_id):
env = filter(lambda x: x['id'] == env_id, environments)
return [{'id': x['id'], 'name': x['name']} for x in env][0]
def delete_multiple_environments(rancher_address, env_ids):
deleted = []
for env_id in env_ids:
deleted.append(delete_environment(rancher_address, env_id))
return deleted
def delete_environment(rancher_address, env_id):
req = requests.delete('{}/v2-beta/projects/{}'.format(rancher_address, env_id))
deleted = req.json()['data'][0]
return {'id': deleted['id'],
'name': deleted['name'],
'orchestration': deleted['orchestration']}
def create_k8s_environment(rancher_address, name, descr):
k8s_template_id = None
for _ in range(10):
k8s_template = requests.get(
'{}/v2-beta/projecttemplates?name=Kubernetes'.format(rancher_address)).json()
if k8s_template['data']:
k8s_template_id = k8s_template['data'][0]['id']
break
time.sleep(3)
if k8s_template_id is None:
raise ValueError('Template for kubernetes not found.')
body = {
'name': name,
'description': descr,
'projectTemplateId': k8s_template_id,
'allowSystemRole': False,
'members': [],
'virtualMachine': False,
'servicesPortRange': None,
'projectLinks': []
}
body_json = json.dumps(body)
req = requests.post('{}/v2-beta/projects'.format(rancher_address), data=body_json)
created = req.json()
return {'id': created['id'], 'name': created['name']}
def get_kubelet_service(rancher_address, env_id):
for _ in range(10):
response = requests.get(
'{}/v2-beta/projects/{}/services/?name=kubelet'.format(rancher_address,
env_id))
if response.status_code >= 400:
# too early or too late for obtaining data
# small delay will improve our chances to collect it
time.sleep(1)
continue
content = response.json()
if content['data']:
return content['data'][0]
# this is unfortunate, response from service api received but data
# not available, lets try again
time.sleep(5)
return None
def fix_datavolume_rhel(rancher_address, env_id):
kubelet_svc = get_kubelet_service(rancher_address, env_id)
if kubelet_svc:
try:
data_volume_index = kubelet_svc['launchConfig']['dataVolumes'].index(
'/sys:/sys:ro,rprivate')
except ValueError:
return 'Already changed'
kubelet_svc['launchConfig']['dataVolumes'][
data_volume_index] = '/sys/fs/cgroup:/sys/fs/cgroup:ro,rprivate'
body = {
'inServiceStrategy': {
'batchSize': 1,
'intervalMillis': 2000,
'startFirst': False,
'launchConfig': kubelet_svc['launchConfig'],
'secondaryLaunchConfigs': []
}
}
body_json = json.dumps(body)
requests.post(
'{}/v2-beta/projects/{}/services/{}?action=upgrade'.format(rancher_address,
env_id,
kubelet_svc[
'id']),
data=body_json)
for _ in range(10):
req_svc = requests.get(
'{}/v2-beta/projects/{}/services/{}'.format(rancher_address, env_id,
kubelet_svc['id']))
req_svc_content = req_svc.json()
if 'finishupgrade' in req_svc_content['actions']:
req_finish = requests.post(
req_svc_content['actions']['finishupgrade'])
return {
'dataVolumes': req_finish.json()['upgrade']['inServiceStrategy'][
'launchConfig']['dataVolumes']}
time.sleep(5)
else:
raise ValueError('Could not get kubelet service')
def create_registration_tokens(rancher_address, env_id):
body = {'name': str(env_id)}
body_json = json.dumps(body)
response = requests.post(
'{}/v2-beta/projects/{}/registrationtokens'.format(rancher_address, env_id,
data=body_json))
for _ in range(10):
tokens = requests.get(response.json()['links']['self'])
tokens_content = tokens.json()
if tokens_content['image'] is not None and tokens_content[
'registrationUrl'] is not None:
return {'image': tokens_content['image'],
'reg_url': tokens_content['registrationUrl']}
time.sleep(3)
return None
def get_registration_tokens(rancher_address, env_id):
reg_tokens = requests.get(
'{}/v2-beta/projects/{}/registrationtokens'.format(rancher_address, env_id))
reg_tokens_content = reg_tokens.json()
tokens = reg_tokens_content['data']
if not tokens:
return None
return {'image': tokens[0]['image'], 'reg_url': tokens[0]['registrationUrl']}
def create_apikey(rancher_address, env_id):
body = {
'name': 'kubectl_env_{}'.format(env_id),
'description': "Provides access to kubectl"
}
body_json = json.dumps(body)
apikey_req = requests.post(
'{}/v2-beta/apikey'.format(rancher_address, env_id, data=body_json))
apikey_content = apikey_req.json()
return {'public': apikey_content['publicValue'],
'private': apikey_content['secretValue']}
def run_module():
module = AnsibleModule(
argument_spec=dict(
server=dict(type='str', required=True),
name=dict(type='str', required=True),
descr=dict(type='str'),
state=dict(type='str', choices=['present', 'absent'], default='present'),
delete_other_k8s=dict(type='bool', default=False),
delete_not_k8s=dict(type='bool', default=True),
force=dict(type='bool', default=True),
host_os=dict(type='str', required=True)
)
)
params = module.params
rancher_address = params['server']
name = params['name']
descr = params['descr']
delete_not_k8s = params['delete_not_k8s']
delete_other_k8s = params['delete_other_k8s']
force = params['force']
host_os = params['host_os']
state = params['state']
existing_envs = get_existing_environments(rancher_address)
same_name_ids = env_ids_by_name(existing_envs, name)
to_delete_ids = []
changes = {}
if delete_other_k8s:
to_delete_ids += other_k8s_ids(existing_envs, name)
if delete_not_k8s:
to_delete_ids += not_k8s_ids(existing_envs)
if force or state == 'absent':
to_delete_ids += same_name_ids
deleted = delete_multiple_environments(rancher_address, to_delete_ids)
if deleted:
changes['deleted'] = deleted
if state == 'absent':
module.exit_json(changed=True, deleted=changes['deleted'])
else:
if state == 'absent':
module.exit_json(changed=False)
if len(same_name_ids) > 1 and not force:
module.fail_json(msg='Multiple environments with same name. '
'Use "force: yes" to delete '
'all environments with same name.')
if same_name_ids and not force:
changes['environment'] = env_info_by_id(existing_envs, same_name_ids[0])
if host_os == 'RedHat':
try:
rhel_fix = fix_datavolume_rhel(rancher_address, same_name_ids[0])
changes['rhel_fix'] = rhel_fix
except ValueError as err:
module.fail_json(
msg='Error: {} Try to recreate k8s environment.'.format(err))
reg_tokens = get_registration_tokens(rancher_address, same_name_ids[0])
if not reg_tokens:
reg_tokens = create_registration_tokens(rancher_address, same_name_ids[0])
changes['registration_tokens'] = reg_tokens
apikey = create_apikey(rancher_address, same_name_ids[0])
changes['apikey'] = apikey
module.exit_json(changed=True, data=changes,
msg='New environment was not created. Only set up was done')
try:
new_env = create_k8s_environment(rancher_address, name, descr)
except ValueError as err:
module.fail_json(msg='Error: {} Try to recreate k8s environment.'.format(err))
if host_os == 'RedHat':
try:
rhel_fix = fix_datavolume_rhel(rancher_address, new_env['id'])
changes['rhel_fix'] = rhel_fix
except ValueError as err:
module.fail_json(msg='Error: {} Try to recreate k8s environment.'.format(
err))
reg_tokens = create_registration_tokens(rancher_address, new_env['id'])
apikey = create_apikey(rancher_address, new_env['id'])
changes['environment'] = new_env
changes['registration_tokens'] = reg_tokens
changes['apikey'] = apikey
module.exit_json(changed=True, data=changes)
if __name__ == '__main__':
run_module()