add k8s library
This commit is contained in:
@@ -1,2 +0,0 @@
|
|||||||
[all]
|
|
||||||
ansible_master ansible_host=<ansible host IP address> ansible_port=<SSH port on ansible host> ansible_user=root
|
|
||||||
@@ -1,18 +0,0 @@
|
|||||||
#
|
|
||||||
# More details on how to use DECORT Ansible module can be found at:
|
|
||||||
# https://github.com/rudecs/decort-ansible/wiki
|
|
||||||
#
|
|
||||||
|
|
||||||
- hosts: ansible_master
|
|
||||||
tasks:
|
|
||||||
- name: obtain JWT
|
|
||||||
decort_jwt:
|
|
||||||
oauth2_url: "{{ decort_sso }}" # "https://sso.digitalenergy.online"
|
|
||||||
validity: 1200
|
|
||||||
register: my_jwt
|
|
||||||
delegate_to: localhost
|
|
||||||
|
|
||||||
- name: print out JWT
|
|
||||||
debug:
|
|
||||||
var: my_jwt.jwt
|
|
||||||
delegate_to: localhost
|
|
||||||
@@ -1,190 +0,0 @@
|
|||||||
#
|
|
||||||
# More details on how to use DECORT Ansible module can be found at:
|
|
||||||
# https://github.com/rudecs/decort-ansible/wiki
|
|
||||||
#
|
|
||||||
|
|
||||||
- hosts: ansible_master
|
|
||||||
vars_files:
|
|
||||||
- vars.yaml
|
|
||||||
tasks:
|
|
||||||
- name: obtain JWT
|
|
||||||
decort_jwt:
|
|
||||||
oauth2_url: "{{ decort_sso }}"
|
|
||||||
validity: 1200
|
|
||||||
register: token
|
|
||||||
delegate_to: localhost
|
|
||||||
|
|
||||||
- name: obtain OS image
|
|
||||||
decort_osimage:
|
|
||||||
authenticator: jwt
|
|
||||||
jwt: "{{ token.jwt }}"
|
|
||||||
oauth2_url: "{{ decort_sso }}"
|
|
||||||
controller_url: "{{ decort_ctrl }}"
|
|
||||||
image_name: "{{ os_image_name }}"
|
|
||||||
account_name: "{{ target_account_name }}"
|
|
||||||
verify_ssl: false
|
|
||||||
register: my_image
|
|
||||||
delegate_to: localhost
|
|
||||||
|
|
||||||
- name: print out the result
|
|
||||||
debug:
|
|
||||||
var: my_image.facts
|
|
||||||
delegate_to: localhost
|
|
||||||
|
|
||||||
- name: manage RG
|
|
||||||
decort_rg:
|
|
||||||
authenticator: jwt
|
|
||||||
jwt: "{{ token.jwt }}"
|
|
||||||
oauth2_url: "{{ decort_sso }}"
|
|
||||||
controller_url: "{{ decort_ctrl }}"
|
|
||||||
account_id: 32
|
|
||||||
rg_name: "{{ target_rg_name }}"
|
|
||||||
state: present
|
|
||||||
verify_ssl: false
|
|
||||||
register: my_rg
|
|
||||||
delegate_to: localhost
|
|
||||||
|
|
||||||
- name: print out the result
|
|
||||||
debug:
|
|
||||||
var: my_rg.facts
|
|
||||||
delegate_to: localhost
|
|
||||||
|
|
||||||
- name: manage ViNS 01
|
|
||||||
decort_vins:
|
|
||||||
authenticator: jwt
|
|
||||||
jwt: "{{ token.jwt }}"
|
|
||||||
oauth2_url: "{{ decort_sso }}"
|
|
||||||
controller_url: "{{ decort_ctrl }}"
|
|
||||||
vins_name: "{{ vins01_name }}"
|
|
||||||
rg_id: "{{ my_rg.facts.id }}"
|
|
||||||
ext_net_id: "{{ target_ext_net_id }}"
|
|
||||||
state: present
|
|
||||||
verify_ssl: false
|
|
||||||
register: my_vins01
|
|
||||||
delegate_to: localhost
|
|
||||||
|
|
||||||
- name: print out the result
|
|
||||||
debug:
|
|
||||||
var: my_vins01.facts
|
|
||||||
delegate_to: localhost
|
|
||||||
|
|
||||||
- name: manage ViNS 02
|
|
||||||
decort_vins:
|
|
||||||
authenticator: jwt
|
|
||||||
jwt: "{{ token.jwt }}"
|
|
||||||
oauth2_url: "{{ decort_sso }}"
|
|
||||||
controller_url: "{{ decort_ctrl }}"
|
|
||||||
vins_name: "{{ vins02_name }}"
|
|
||||||
rg_id: "{{ my_rg.facts.id }}"
|
|
||||||
ext_net_id: -1
|
|
||||||
state: present
|
|
||||||
verify_ssl: false
|
|
||||||
register: my_vins02
|
|
||||||
delegate_to: localhost
|
|
||||||
|
|
||||||
- name: print out the result
|
|
||||||
debug:
|
|
||||||
var: my_vins02.facts
|
|
||||||
delegate_to: localhost
|
|
||||||
|
|
||||||
- name: manage data disk 01
|
|
||||||
decort_disk:
|
|
||||||
authenticator: jwt
|
|
||||||
jwt: "{{ token.jwt }}"
|
|
||||||
oauth2_url: "{{ decort_sso }}"
|
|
||||||
controller_url: "{{ decort_ctrl }}"
|
|
||||||
name: "{{ datadisk01_name }}"
|
|
||||||
size: "{{ datadisk01_size }}"
|
|
||||||
account_name: "{{ target_account_name }}"
|
|
||||||
pool: data01
|
|
||||||
place_with: "{{ my_image.facts.id }}"
|
|
||||||
state: present
|
|
||||||
verify_ssl: false
|
|
||||||
register: my_disk01
|
|
||||||
delegate_to: localhost
|
|
||||||
|
|
||||||
- name: print out the result
|
|
||||||
debug:
|
|
||||||
var: my_disk01.facts
|
|
||||||
delegate_to: localhost
|
|
||||||
|
|
||||||
- name: manage data disk 02
|
|
||||||
decort_disk:
|
|
||||||
authenticator: jwt
|
|
||||||
jwt: "{{ token.jwt }}"
|
|
||||||
oauth2_url: "{{ decort_sso }}"
|
|
||||||
controller_url: "{{ decort_ctrl }}"
|
|
||||||
name: "{{ datadisk02_name }}"
|
|
||||||
size: "{{ datadisk02_size }}"
|
|
||||||
account_name: "{{ target_account_name }}"
|
|
||||||
pool: data01
|
|
||||||
place_with: "{{ my_image.facts.id }}"
|
|
||||||
state: present
|
|
||||||
verify_ssl: false
|
|
||||||
register: my_disk02
|
|
||||||
delegate_to: localhost
|
|
||||||
|
|
||||||
- name: print out the result
|
|
||||||
debug:
|
|
||||||
var: my_disk02.facts
|
|
||||||
delegate_to: localhost
|
|
||||||
|
|
||||||
- name: manage KVM X86 VM
|
|
||||||
decort_kvmvm:
|
|
||||||
authenticator: jwt
|
|
||||||
jwt: "{{ token.jwt }}"
|
|
||||||
oauth2_url: "{{ decort_sso }}"
|
|
||||||
controller_url: "{{ decort_ctrl }}"
|
|
||||||
name: "{{ vm01_name }}"
|
|
||||||
arch: KVM_X86
|
|
||||||
ram: "{{ vm01_ram }}"
|
|
||||||
cpu: "{{ vm01_cpu }}"
|
|
||||||
image_id: "{{ my_image.facts.id }}"
|
|
||||||
boot_disk: "{{ vm01_boot_disk }}"
|
|
||||||
data_disks:
|
|
||||||
- "{{ my_disk01.facts.id }}"
|
|
||||||
- "{{ my_disk02.facts.id }}"
|
|
||||||
networks:
|
|
||||||
- type: VINS
|
|
||||||
id: "{{ my_vins01.facts.id }}"
|
|
||||||
- type: VINS
|
|
||||||
id: "{{ my_vins02.facts.id }}"
|
|
||||||
- type: EXTNET
|
|
||||||
id: "{{ target_ext_net_id }}"
|
|
||||||
rg_id: "{{ my_rg.facts.id }}"
|
|
||||||
state: present
|
|
||||||
verify_ssl: false
|
|
||||||
register: my_kvmvm
|
|
||||||
delegate_to: localhost
|
|
||||||
|
|
||||||
- name: print out the result
|
|
||||||
debug:
|
|
||||||
var: my_kvmvm.facts
|
|
||||||
delegate_to: localhost
|
|
||||||
|
|
||||||
- name: manage PFW rules on Compute
|
|
||||||
decort_pfw:
|
|
||||||
authenticator: jwt
|
|
||||||
jwt: "{{ token.jwt }}"
|
|
||||||
oauth2_url: "{{ decort_sso }}"
|
|
||||||
controller_url: "{{ decort_ctrl }}"
|
|
||||||
compute_id: "{{ my_kvmvm.facts.id }}"
|
|
||||||
vins_id: "{{ my_vins01.facts.id }}"
|
|
||||||
rules:
|
|
||||||
- public_port_start: 30022
|
|
||||||
local_port: 22
|
|
||||||
proto: tcp
|
|
||||||
- public_port_start: 30080
|
|
||||||
public_port_end: 30085
|
|
||||||
local_port: 30080
|
|
||||||
proto: tcp
|
|
||||||
state: present
|
|
||||||
verify_ssl: false
|
|
||||||
register: my_pfw
|
|
||||||
delegate_to: localhost
|
|
||||||
|
|
||||||
- name: print out the result
|
|
||||||
debug:
|
|
||||||
var: my_pfw.facts
|
|
||||||
delegate_to: localhost
|
|
||||||
|
|
||||||
@@ -1,21 +0,0 @@
|
|||||||
#!/bin/bash
|
|
||||||
|
|
||||||
# 1. As this file will contain sensitive data (application ID & Secret pair) put this file
|
|
||||||
# in a directory, where only you will have access to it, e.g. your ~/.ssh directory.
|
|
||||||
# 2. Make sure this file is not readable by anybody else (chmod 400).
|
|
||||||
# 3. Paste your Application ID (obtained from DECORT SSO application) to DECORT_APP_ID.
|
|
||||||
# 4. Paste your Application Secret (obtained from DECORT SSO application) to DECORT_APP_SECRET.
|
|
||||||
# 5. Paste DECORT SSO application URL to DECORT_OAUTH2_URL.
|
|
||||||
#
|
|
||||||
# Source this file into shell to prepare environment for running DECORT Ansible module, e.g.
|
|
||||||
# . ~/.ssh/prepenv.sh
|
|
||||||
#
|
|
||||||
# More informaiton on DECORT Ansible module can be found at:
|
|
||||||
# https://github.com/rudecs/decort-ansible/wiki
|
|
||||||
#
|
|
||||||
|
|
||||||
export DECORT_APP_ID="put your application ID here"
|
|
||||||
export DECORT_APP_SECRET="put your application secret here"
|
|
||||||
export DECORT_OAUTH2_URL="put DECORT SSO URL here" # "https://sso.digitalenergy.online"
|
|
||||||
|
|
||||||
export ANSIBLE_HOST_KEY_CHEKCING=False
|
|
||||||
@@ -1,26 +0,0 @@
|
|||||||
#
|
|
||||||
# More details on how to use DECORT Ansible module can be found at:
|
|
||||||
# https://github.com/rudecs/decort-ansible/wiki
|
|
||||||
#
|
|
||||||
|
|
||||||
decort_sso: "put DECORT SSO application URL here" # "https://sso.digitalenergy.online"
|
|
||||||
decort_ctrl: "put DECORT controller URL here" # "https://ds1.digitalenergy.online"
|
|
||||||
|
|
||||||
target_account_name: "your account name"
|
|
||||||
target_rg_name: "target resource group name"
|
|
||||||
os_image_name: "OS image name"
|
|
||||||
|
|
||||||
vins01_name: "Vins01-ansible"
|
|
||||||
vins02_name: "Vins02-ansible"
|
|
||||||
target_ext_net_id: 0
|
|
||||||
|
|
||||||
datadisk01_name: "Data01-ansible"
|
|
||||||
datadisk01_size: 5
|
|
||||||
|
|
||||||
datadisk02_name: "Data02-ansible"
|
|
||||||
datadisk02_size: 5
|
|
||||||
|
|
||||||
vm01_name: "Vm01-ansible"
|
|
||||||
vm01_cpu: 1
|
|
||||||
vm01_ram: 1024
|
|
||||||
vm01_boot_disk: 10
|
|
||||||
243
library/decort_k8s.py
Normal file
243
library/decort_k8s.py
Normal file
@@ -0,0 +1,243 @@
|
|||||||
|
#!/usr/bin/python
|
||||||
|
#
|
||||||
|
# Digital Enegry Cloud Orchestration Technology (DECORT) modules for Ansible
|
||||||
|
# Copyright: (c) 2018-2021 Digital Energy Cloud Solutions LLC
|
||||||
|
#
|
||||||
|
# Apache License 2.0 (see http://www.apache.org/licenses/LICENSE-2.0.txt)
|
||||||
|
#
|
||||||
|
|
||||||
|
#
|
||||||
|
# Author: Sergey Shubin (sergey.shubin@digitalenergy.online)
|
||||||
|
#
|
||||||
|
|
||||||
|
ANSIBLE_METADATA = {'metadata_version': '1.1',
|
||||||
|
'status': ['preview'],
|
||||||
|
'supported_by': 'community'}
|
||||||
|
|
||||||
|
|
||||||
|
from ansible.module_utils.basic import AnsibleModule
|
||||||
|
from ansible.module_utils.basic import env_fallback
|
||||||
|
|
||||||
|
from ansible.module_utils.decort_utils import *
|
||||||
|
|
||||||
|
|
||||||
|
def decort_k8s_package_facts(arg_k8s_facts, arg_check_mode=False):
|
||||||
|
"""Package a dictionary of RG facts according to the decort_rg module specification. This dictionary will
|
||||||
|
be returned to the upstream Ansible engine at the completion of the module run.
|
||||||
|
|
||||||
|
@param arg_k8s_facts: dictionary with RG facts as returned by API call to .../rg/get
|
||||||
|
@param arg_check_mode: boolean that tells if this Ansible module is run in check mode
|
||||||
|
"""
|
||||||
|
|
||||||
|
ret_dict = dict(id=0,
|
||||||
|
name="none",
|
||||||
|
state="CHECK_MODE",
|
||||||
|
)
|
||||||
|
|
||||||
|
if arg_check_mode:
|
||||||
|
# in check mode return immediately with the default values
|
||||||
|
return ret_dict
|
||||||
|
|
||||||
|
if arg_k8s_facts is None:
|
||||||
|
# if void facts provided - change state value to ABSENT and return
|
||||||
|
ret_dict['state'] = "ABSENT"
|
||||||
|
return ret_dict
|
||||||
|
|
||||||
|
ret_dict['id'] = arg_k8s_facts['id']
|
||||||
|
ret_dict['name'] = arg_k8s_facts['name']
|
||||||
|
ret_dict['techStatus'] = arg_k8s_facts['techStatus']
|
||||||
|
ret_dict['state'] = arg_k8s_facts['status']
|
||||||
|
|
||||||
|
return ret_dict
|
||||||
|
|
||||||
|
def decort_k8s_parameters():
|
||||||
|
"""Build and return a dictionary of parameters expected by decort_rg module in a form accepted
|
||||||
|
by AnsibleModule utility class."""
|
||||||
|
|
||||||
|
return dict(
|
||||||
|
account_id=dict(type='int', required=False),
|
||||||
|
account_name=dict(type='str', required=False, default=''),
|
||||||
|
annotation=dict(type='str', required=False, default=''),
|
||||||
|
app_id=dict(type='str',
|
||||||
|
required=False,
|
||||||
|
fallback=(env_fallback, ['DECORT_APP_ID'])),
|
||||||
|
app_secret=dict(type='str',
|
||||||
|
required=False,
|
||||||
|
fallback=(env_fallback, ['DECORT_APP_SECRET']),
|
||||||
|
no_log=True),
|
||||||
|
authenticator=dict(type='str',
|
||||||
|
required=True,
|
||||||
|
choices=['legacy', 'oauth2', 'jwt']),
|
||||||
|
controller_url=dict(type='str', required=True),
|
||||||
|
# datacenter=dict(type='str', required=False, default=''),
|
||||||
|
jwt=dict(type='str',
|
||||||
|
required=False,
|
||||||
|
fallback=(env_fallback, ['DECORT_JWT']),
|
||||||
|
no_log=True),
|
||||||
|
oauth2_url=dict(type='str',
|
||||||
|
required=False,
|
||||||
|
fallback=(env_fallback, ['DECORT_OAUTH2_URL'])),
|
||||||
|
password=dict(type='str',
|
||||||
|
required=False,
|
||||||
|
fallback=(env_fallback, ['DECORT_PASSWORD']),
|
||||||
|
no_log=True),
|
||||||
|
quotas=dict(type='dict', required=False),
|
||||||
|
state=dict(type='str',
|
||||||
|
default='present',
|
||||||
|
choices=['absent', 'disabled', 'enabled', 'present']),
|
||||||
|
permanent=dict(type='bool', default=False),
|
||||||
|
started=dict(type='bool', default=True),
|
||||||
|
user=dict(type='str',
|
||||||
|
required=False,
|
||||||
|
fallback=(env_fallback, ['DECORT_USER'])),
|
||||||
|
k8s_name=dict(type='str', required=True),
|
||||||
|
rg_id=dict(type='int', required=True),
|
||||||
|
k8ci_id=dict(type='int', required=True),
|
||||||
|
wg_name=dict(type='str', required=True),
|
||||||
|
master_count=dict(type='int', default=1),
|
||||||
|
master_cpu=dict(type='int', default=2),
|
||||||
|
master_ram_mb=dict(type='int', default=2048),
|
||||||
|
master_disk_gb=dict(type='int', default=10),
|
||||||
|
worker_count=dict(type='int', default=1),
|
||||||
|
worker_cpu=dict(type='int', default=1),
|
||||||
|
worker_ram_mb=dict(type='int', default=1024),
|
||||||
|
worker_disk_gb=dict(type='int', default=0),
|
||||||
|
extnet_id=dict(type='int', default=0),
|
||||||
|
description=dict(type='str', default="Created by decort ansible module"),
|
||||||
|
with_lb=dict(type='bool', default=True),
|
||||||
|
verify_ssl=dict(type='bool', required=False, default=True),
|
||||||
|
workflow_callback=dict(type='str', required=False),
|
||||||
|
workflow_context=dict(type='str', required=False),
|
||||||
|
)
|
||||||
|
|
||||||
|
|
||||||
|
def main():
|
||||||
|
module_parameters = decort_k8s_parameters()
|
||||||
|
|
||||||
|
amodule = AnsibleModule(argument_spec=module_parameters,
|
||||||
|
supports_check_mode=True,
|
||||||
|
mutually_exclusive=[
|
||||||
|
['oauth2', 'password'],
|
||||||
|
['password', 'jwt'],
|
||||||
|
['jwt', 'oauth2'],
|
||||||
|
],
|
||||||
|
required_together=[
|
||||||
|
['app_id', 'app_secret'],
|
||||||
|
['user', 'password'],
|
||||||
|
],
|
||||||
|
)
|
||||||
|
|
||||||
|
decon = DecortController(amodule)
|
||||||
|
k8s_id, k8s_facts = decon.k8s_find(arg_k8s_name=amodule.params['k8s_name'],
|
||||||
|
arg_check_state=False)
|
||||||
|
|
||||||
|
k8s_should_exist = True
|
||||||
|
|
||||||
|
#TODO check variables
|
||||||
|
|
||||||
|
#TODO check rg rss
|
||||||
|
if k8s_id:
|
||||||
|
if k8s_facts['status'] in ["MODELED", "DISABLING", "ENABLING", "DELETING", "DESTROYING", "CREATING",
|
||||||
|
"RESTORING"] and amodule.params['state'] != "present":
|
||||||
|
decon.result['failed'] = True
|
||||||
|
decon.result['changed'] = False
|
||||||
|
decon.result['msg'] = ("No change can be done for existing k8s ID {} because of its current "
|
||||||
|
"status '{}'").format(k8s_id, k8s_facts['status'])
|
||||||
|
elif k8s_facts['status'] in ["DISABLED", "ENABLED", "CREATED", "DELETED"] and amodule.params['state'] == "absent":
|
||||||
|
if amodule.params['permanent'] is True:
|
||||||
|
decon.k8s_delete(k8s_id, True)
|
||||||
|
k8s_facts['status'] = 'DESTROYED'
|
||||||
|
k8s_should_exist = False
|
||||||
|
else:
|
||||||
|
decon.k8s_delete(k8s_id)
|
||||||
|
k8s_facts['status'] = 'DELETED'
|
||||||
|
k8s_should_exist = True
|
||||||
|
elif k8s_facts['status'] == "ENABLED":
|
||||||
|
if amodule.params['started'] is True:
|
||||||
|
decon.k8s_state(k8s_facts, amodule.params['state'], amodule.params['started'])
|
||||||
|
else:
|
||||||
|
decon.k8s_state(k8s_facts, amodule.params['state'], amodule.params['started'])
|
||||||
|
elif k8s_facts['status'] == amodule.params['state'].upper():
|
||||||
|
decon.k8s_state(k8s_facts, amodule.params['state'])
|
||||||
|
elif k8s_facts['status'] in ["ENABLED", "CREATED"] and amodule.params['state'] == "disabled":
|
||||||
|
decon.k8s_state(k8s_facts, 'disabled')
|
||||||
|
elif k8s_facts['status'] in ["DISABLED", "CREATED"]:
|
||||||
|
if amodule.params['state'] == 'enabled':
|
||||||
|
decon.k8s_state(k8s_facts, 'enabled', amodule.params['started'])
|
||||||
|
elif amodule.params['state'] == "disabled":
|
||||||
|
decon.k8s_state(k8s_facts, 'disabled')
|
||||||
|
k8s_should_exist = True
|
||||||
|
elif k8s_facts['status'] == "DELETED":
|
||||||
|
if amodule.params['state'] in ('enabled', 'present'):
|
||||||
|
decon.k8s_restore(k8s_id)
|
||||||
|
k8s_should_exist = True
|
||||||
|
elif amodule.params['state'] == 'disabled':
|
||||||
|
decon.result['failed'] = True
|
||||||
|
decon.result['changed'] = False
|
||||||
|
decon.result['msg'] = ("Invalid target state '{}' requested for k8s ID {} in the "
|
||||||
|
"current status '{}'").format(k8s_id,
|
||||||
|
amodule.params['state'],
|
||||||
|
k8s_facts['status'])
|
||||||
|
k8s_should_exist = False
|
||||||
|
elif k8s_facts['status'] == "DESTROYED":
|
||||||
|
if amodule.params['state'] in ('present', 'enabled'):
|
||||||
|
k8s_should_exist = True
|
||||||
|
elif amodule.params['state'] == 'absent':
|
||||||
|
# nop
|
||||||
|
decon.result['failed'] = False
|
||||||
|
decon.result['changed'] = False
|
||||||
|
decon.result['msg'] = ("No state change required for k8s ID {} because of its "
|
||||||
|
"current status '{}'").format(k8s_id,
|
||||||
|
k8s_facts['status'])
|
||||||
|
k8s_should_exist = False
|
||||||
|
elif amodule.params['state'] == 'disabled':
|
||||||
|
# error
|
||||||
|
decon.result['failed'] = True
|
||||||
|
decon.result['changed'] = False
|
||||||
|
decon.result['msg'] = ("Invalid target state '{}' requested for k8s ID {} in the "
|
||||||
|
"current status '{}'").format(k8s_id,
|
||||||
|
amodule.params['state'],
|
||||||
|
k8s_facts['status'])
|
||||||
|
else:
|
||||||
|
k8s_should_exist = False
|
||||||
|
if amodule.params['state'] == 'absent':
|
||||||
|
decon.result['failed'] = False
|
||||||
|
decon.result['changed'] = False
|
||||||
|
decon.result['msg'] = ("Nothing to do as target state 'absent' was requested for "
|
||||||
|
"non-existent k8s name '{}'").format(amodule.params['k8s_name'])
|
||||||
|
elif amodule.params['state'] in ('present', 'enabled'):
|
||||||
|
decon.check_amodule_argument('k8s_name')
|
||||||
|
k8s_id = decon.k8s_provision(amodule.params['k8s_name'],
|
||||||
|
amodule.params['wg_name'],
|
||||||
|
amodule.params['k8ci_id'],
|
||||||
|
amodule.params['rg_id'],
|
||||||
|
amodule.params['master_count'],
|
||||||
|
amodule.params['master_cpu'],
|
||||||
|
amodule.params['master_ram_mb'],
|
||||||
|
amodule.params['master_disk_gb'],
|
||||||
|
amodule.params['worker_count'],
|
||||||
|
amodule.params['worker_cpu'],
|
||||||
|
amodule.params['worker_ram_mb'],
|
||||||
|
amodule.params['worker_disk_gb'],
|
||||||
|
amodule.params['extnet_id'],
|
||||||
|
amodule.params['with_lb'],
|
||||||
|
amodule.params['description'],
|
||||||
|
)
|
||||||
|
k8s_should_exist = True
|
||||||
|
elif amodule.params['state'] == 'disabled':
|
||||||
|
decon.result['failed'] = True
|
||||||
|
decon.result['changed'] = False
|
||||||
|
decon.result['msg'] = ("Invalid target state '{}' requested for non-existent "
|
||||||
|
"k8s name '{}' ").format(amodule.params['state'],
|
||||||
|
amodule.params['k8s_name'])
|
||||||
|
if decon.result['failed']:
|
||||||
|
amodule.fail_json(**decon.result)
|
||||||
|
else:
|
||||||
|
if k8s_should_exist:
|
||||||
|
if decon.result['changed']:
|
||||||
|
_, k8s_facts = decon.k8s_find(arg_k8s_id=k8s_id)
|
||||||
|
decon.result['facts'] = decort_k8s_package_facts(k8s_facts, amodule.check_mode)
|
||||||
|
amodule.exit_json(**decon.result)
|
||||||
|
|
||||||
|
if __name__ == "__main__":
|
||||||
|
main()
|
||||||
@@ -290,9 +290,9 @@ def decort_vins_package_facts(arg_vins_facts, arg_check_mode=False):
|
|||||||
# ??? -> ext_net_id
|
# ??? -> ext_net_id
|
||||||
# tech_status -> techStatus
|
# tech_status -> techStatus
|
||||||
|
|
||||||
|
|
||||||
return ret_dict
|
return ret_dict
|
||||||
|
|
||||||
|
|
||||||
def decort_vins_parameters():
|
def decort_vins_parameters():
|
||||||
"""Build and return a dictionary of parameters expected by decort_vins module in a form accepted
|
"""Build and return a dictionary of parameters expected by decort_vins module in a form accepted
|
||||||
by AnsibleModule utility class."""
|
by AnsibleModule utility class."""
|
||||||
@@ -342,6 +342,7 @@ def decort_vins_parameters():
|
|||||||
workflow_context=dict(type='str', required=False),
|
workflow_context=dict(type='str', required=False),
|
||||||
)
|
)
|
||||||
|
|
||||||
|
|
||||||
# Workflow digest:
|
# Workflow digest:
|
||||||
# 1) authenticate to DECORT controller & validate authentication by issuing API call - done when creating DECORTController
|
# 1) authenticate to DECORT controller & validate authentication by issuing API call - done when creating DECORTController
|
||||||
# 2) check if the ViNS with this id or name exists under specified account / resource group
|
# 2) check if the ViNS with this id or name exists under specified account / resource group
|
||||||
@@ -570,7 +571,6 @@ def main():
|
|||||||
decon.result['msg'] = ("Invalid target state '{}' requested for non-existent "
|
decon.result['msg'] = ("Invalid target state '{}' requested for non-existent "
|
||||||
"ViNS name '{}'").format(amodule.params['state'],
|
"ViNS name '{}'").format(amodule.params['state'],
|
||||||
amodule.params['vins_name'])
|
amodule.params['vins_name'])
|
||||||
|
|
||||||
#
|
#
|
||||||
# conditional switch end - complete module run
|
# conditional switch end - complete module run
|
||||||
#
|
#
|
||||||
|
|||||||
@@ -40,6 +40,7 @@ import requests
|
|||||||
|
|
||||||
from ansible.module_utils.basic import AnsibleModule
|
from ansible.module_utils.basic import AnsibleModule
|
||||||
|
|
||||||
|
|
||||||
#
|
#
|
||||||
# TODO: the following functionality to be implemented and/or tested
|
# TODO: the following functionality to be implemented and/or tested
|
||||||
# 4) workflow callbacks
|
# 4) workflow callbacks
|
||||||
@@ -388,7 +389,8 @@ class DecortController(object):
|
|||||||
return None # actually, this directive will never be executed as fail_json aborts the script
|
return None # actually, this directive will never be executed as fail_json aborts the script
|
||||||
except requests.exceptions.Timeout:
|
except requests.exceptions.Timeout:
|
||||||
self.result['failed'] = True
|
self.result['failed'] = True
|
||||||
self.result['msg'] = "Timeout when trying to connect to '{}' when calling DECORT API.".format(api_resp.url)
|
self.result['msg'] = "Timeout when trying to connect to '{}' when calling DECORT API.".format(
|
||||||
|
api_resp.url)
|
||||||
self.amodule.fail_json(**self.result)
|
self.amodule.fail_json(**self.result)
|
||||||
return None
|
return None
|
||||||
|
|
||||||
@@ -605,7 +607,6 @@ class DecortController(object):
|
|||||||
|
|
||||||
return ret_comp_id, ret_comp_dict, ret_rg_id
|
return ret_comp_id, ret_comp_dict, ret_rg_id
|
||||||
|
|
||||||
|
|
||||||
def compute_find(self, comp_id,
|
def compute_find(self, comp_id,
|
||||||
comp_name="", rg_id=0,
|
comp_name="", rg_id=0,
|
||||||
check_state=True):
|
check_state=True):
|
||||||
@@ -652,7 +653,8 @@ class DecortController(object):
|
|||||||
# Therefore, RG ID cannot be zero and compute name cannot be empty.
|
# Therefore, RG ID cannot be zero and compute name cannot be empty.
|
||||||
if not rg_id and comp_name == "":
|
if not rg_id and comp_name == "":
|
||||||
self.result['failed'] = True
|
self.result['failed'] = True
|
||||||
self.result['msg'] = "compute_find(): cannot find Compute by name when either name is empty or RG ID is zero."
|
self.result[
|
||||||
|
'msg'] = "compute_find(): cannot find Compute by name when either name is empty or RG ID is zero."
|
||||||
self.amodule.fail_json(**self.result)
|
self.amodule.fail_json(**self.result)
|
||||||
# fail the module - exit
|
# fail the module - exit
|
||||||
|
|
||||||
@@ -879,7 +881,8 @@ class DecortController(object):
|
|||||||
# return
|
# return
|
||||||
|
|
||||||
api_resp = self.decort_api_call(requests.post, "/restmachine/cloudapi/extnet/list", api_params)
|
api_resp = self.decort_api_call(requests.post, "/restmachine/cloudapi/extnet/list", api_params)
|
||||||
extnet_list = json.loads(api_resp.content.decode('utf8')) # list of dicts: "name" holds "NET_ADDR/NETMASK", "id" is ID
|
extnet_list = json.loads(
|
||||||
|
api_resp.content.decode('utf8')) # list of dicts: "name" holds "NET_ADDR/NETMASK", "id" is ID
|
||||||
#
|
#
|
||||||
# Empty extnet_list does not constitute error condition, so we should not fail the module in
|
# Empty extnet_list does not constitute error condition, so we should not fail the module in
|
||||||
# this case. Therefore the following code fragment is commented out.
|
# this case. Therefore the following code fragment is commented out.
|
||||||
@@ -1200,7 +1203,6 @@ class DecortController(object):
|
|||||||
|
|
||||||
return image_id, ret_image_dict
|
return image_id, ret_image_dict
|
||||||
|
|
||||||
|
|
||||||
def image_find(self, image_id, image_name, account_id, rg_id=0, sepid=0, pool=""):
|
def image_find(self, image_id, image_name, account_id, rg_id=0, sepid=0, pool=""):
|
||||||
"""Locates image specified by name and returns its facts as dictionary.
|
"""Locates image specified by name and returns its facts as dictionary.
|
||||||
Primary use of this function is to obtain the ID of the image identified by its name and,
|
Primary use of this function is to obtain the ID of the image identified by its name and,
|
||||||
@@ -1402,7 +1404,6 @@ class DecortController(object):
|
|||||||
|
|
||||||
return ret_rg_id, ret_rg_dict
|
return ret_rg_id, ret_rg_dict
|
||||||
|
|
||||||
|
|
||||||
def rg_provision(self, arg_account_id, arg_rg_name, arg_username, arg_quota={}, arg_location="", arg_desc=""):
|
def rg_provision(self, arg_account_id, arg_rg_name, arg_username, arg_quota={}, arg_location="", arg_desc=""):
|
||||||
"""Provision new RG according to the specified arguments.
|
"""Provision new RG according to the specified arguments.
|
||||||
If critical error occurs the embedded call to API function will abort further execution of the script
|
If critical error occurs the embedded call to API function will abort further execution of the script
|
||||||
@@ -1434,7 +1435,8 @@ class DecortController(object):
|
|||||||
target_gid = self.gid_get(arg_location)
|
target_gid = self.gid_get(arg_location)
|
||||||
if not target_gid:
|
if not target_gid:
|
||||||
self.result['failed'] = True
|
self.result['failed'] = True
|
||||||
self.result['msg'] = ("rg_provision() failed to obtain valid Grid ID for location '{}'").format(arg_location)
|
self.result['msg'] = ("rg_provision() failed to obtain valid Grid ID for location '{}'").format(
|
||||||
|
arg_location)
|
||||||
self.amodule.fail_json(**self.result)
|
self.amodule.fail_json(**self.result)
|
||||||
|
|
||||||
api_params = dict(accountId=arg_account_id,
|
api_params = dict(accountId=arg_account_id,
|
||||||
@@ -1560,7 +1562,8 @@ class DecortController(object):
|
|||||||
|
|
||||||
self.result['waypoints'] = "{} -> {}".format(self.result['waypoints'], "rg_state")
|
self.result['waypoints'] = "{} -> {}".format(self.result['waypoints'], "rg_state")
|
||||||
|
|
||||||
NOP_STATES_FOR_RG_CHANGE = ["MODELED", "DISABLING", "ENABLING", "DELETING", "DELETED", "DESTROYING", "DESTROYED"]
|
NOP_STATES_FOR_RG_CHANGE = ["MODELED", "DISABLING", "ENABLING", "DELETING", "DELETED", "DESTROYING",
|
||||||
|
"DESTROYED"]
|
||||||
VALID_TARGET_STATES = ["enabled", "disabled"]
|
VALID_TARGET_STATES = ["enabled", "disabled"]
|
||||||
|
|
||||||
if arg_rg_dict['status'] in NOP_STATES_FOR_RG_CHANGE:
|
if arg_rg_dict['status'] in NOP_STATES_FOR_RG_CHANGE:
|
||||||
@@ -2072,7 +2075,8 @@ class DecortController(object):
|
|||||||
|
|
||||||
self.result['waypoints'] = "{} -> {}".format(self.result['waypoints'], "vins_state")
|
self.result['waypoints'] = "{} -> {}".format(self.result['waypoints'], "vins_state")
|
||||||
|
|
||||||
NOP_STATES_FOR_VINS_CHANGE = ["MODELED", "DISABLING", "ENABLING", "DELETING", "DELETED", "DESTROYING", "DESTROYED"]
|
NOP_STATES_FOR_VINS_CHANGE = ["MODELED", "DISABLING", "ENABLING", "DELETING", "DELETED", "DESTROYING",
|
||||||
|
"DESTROYED"]
|
||||||
VALID_TARGET_STATES = ["enabled", "disabled"]
|
VALID_TARGET_STATES = ["enabled", "disabled"]
|
||||||
|
|
||||||
if vins_dict['status'] in NOP_STATES_FOR_VINS_CHANGE:
|
if vins_dict['status'] in NOP_STATES_FOR_VINS_CHANGE:
|
||||||
@@ -2195,7 +2199,8 @@ class DecortController(object):
|
|||||||
self.result['failed'] = False
|
self.result['failed'] = False
|
||||||
self.result['warning'] = ("vins_update(): ViNS ID {} is already connected to ext net ID {}, "
|
self.result['warning'] = ("vins_update(): ViNS ID {} is already connected to ext net ID {}, "
|
||||||
"no reconnection to default network will be done.").format(vins_dict['id'],
|
"no reconnection to default network will be done.").format(vins_dict['id'],
|
||||||
gw_config['ext_net_id'])
|
gw_config[
|
||||||
|
'ext_net_id'])
|
||||||
|
|
||||||
return
|
return
|
||||||
|
|
||||||
@@ -2285,7 +2290,8 @@ class DecortController(object):
|
|||||||
|
|
||||||
if self.amodule.check_mode:
|
if self.amodule.check_mode:
|
||||||
self.result['failed'] = False
|
self.result['failed'] = False
|
||||||
self.result['msg'] = "disk_find() in check mode: find Disk ID {} / name '{}' was requested.".format(disk_id, disk_name)
|
self.result['msg'] = "disk_find() in check mode: find Disk ID {} / name '{}' was requested.".format(disk_id,
|
||||||
|
disk_name)
|
||||||
return
|
return
|
||||||
|
|
||||||
ret_disk_id = 0
|
ret_disk_id = 0
|
||||||
@@ -2348,7 +2354,8 @@ class DecortController(object):
|
|||||||
|
|
||||||
if self.amodule.check_mode:
|
if self.amodule.check_mode:
|
||||||
self.result['failed'] = False
|
self.result['failed'] = False
|
||||||
self.result['msg'] = "disk_provision() in check mode: create Disk name '{}' was requested.".format(disk_name)
|
self.result['msg'] = "disk_provision() in check mode: create Disk name '{}' was requested.".format(
|
||||||
|
disk_name)
|
||||||
return 0
|
return 0
|
||||||
|
|
||||||
target_gid = self.gid_get(location)
|
target_gid = self.gid_get(location)
|
||||||
@@ -2400,7 +2407,8 @@ class DecortController(object):
|
|||||||
|
|
||||||
if not new_size:
|
if not new_size:
|
||||||
self.result['failed'] = False
|
self.result['failed'] = False
|
||||||
self.result['warning'] = "disk_resize(): zero size requested for Disk ID {} - ignoring.".format(disk_facts['id'])
|
self.result['warning'] = "disk_resize(): zero size requested for Disk ID {} - ignoring.".format(
|
||||||
|
disk_facts['id'])
|
||||||
return
|
return
|
||||||
|
|
||||||
if new_size < disk_facts['sizeMax']:
|
if new_size < disk_facts['sizeMax']:
|
||||||
@@ -2451,7 +2459,6 @@ class DecortController(object):
|
|||||||
self.result['changed'] = True
|
self.result['changed'] = True
|
||||||
return
|
return
|
||||||
|
|
||||||
|
|
||||||
##############################
|
##############################
|
||||||
#
|
#
|
||||||
# Port Forward rules management
|
# Port Forward rules management
|
||||||
@@ -2526,7 +2533,8 @@ class DecortController(object):
|
|||||||
break
|
break
|
||||||
else:
|
else:
|
||||||
decon.result['failed'] = True
|
decon.result['failed'] = True
|
||||||
decon.result['msg'] = "Compute ID {} is not connected to ViNS ID {}.".format(comp_facts['id'], vins_facts['id'])
|
decon.result['msg'] = "Compute ID {} is not connected to ViNS ID {}.".format(comp_facts['id'],
|
||||||
|
vins_facts['id'])
|
||||||
return ret_rules
|
return ret_rules
|
||||||
|
|
||||||
existing_rules = []
|
existing_rules = []
|
||||||
@@ -2641,3 +2649,256 @@ class DecortController(object):
|
|||||||
|
|
||||||
ret_rules = self._pfw_get(comp_facts['id'], vins_facts['id'])
|
ret_rules = self._pfw_get(comp_facts['id'], vins_facts['id'])
|
||||||
return ret_rules
|
return ret_rules
|
||||||
|
|
||||||
|
def _k8s_get_by_id(self, k8s_id):
|
||||||
|
"""Helper function that locates k8s by ID and returns k8s facts.
|
||||||
|
|
||||||
|
@param (int) k8s_id: ID of the k8s to find and return facts for.
|
||||||
|
|
||||||
|
@return: k8s ID and a dictionary of k8s facts as provided by rg/get API call. Note that if it fails
|
||||||
|
to find the k8s with the specified ID, it may return 0 for ID and empty dictionary for the facts. So
|
||||||
|
it is suggested to check the return values accordingly.
|
||||||
|
"""
|
||||||
|
ret_k8s_id = 0
|
||||||
|
ret_k8s_dict = dict()
|
||||||
|
|
||||||
|
if not k8s_id:
|
||||||
|
self.result['failed'] = True
|
||||||
|
self.result['msg'] = "k8s_get_by_id(): zero k8s ID specified."
|
||||||
|
self.amodule.fail_json(**self.result)
|
||||||
|
|
||||||
|
api_params = dict(k8sId=k8s_id, )
|
||||||
|
api_resp = self.decort_api_call(requests.post, "/restmachine/cloudapi/k8s/get", api_params)
|
||||||
|
if api_resp.status_code == 200:
|
||||||
|
ret_k8s_id = k8s_id
|
||||||
|
ret_k8s_dict = json.loads(api_resp.content.decode('utf8'))
|
||||||
|
else:
|
||||||
|
self.result['warning'] = ("k8s_get_by_id(): failed to get k8s by ID {}. HTTP code {}, "
|
||||||
|
"response {}.").format(k8s_id, api_resp.status_code, api_resp.reason)
|
||||||
|
|
||||||
|
return ret_k8s_id, ret_k8s_dict
|
||||||
|
|
||||||
|
def k8s_find(self, arg_k8s_id=0, arg_k8s_name="", arg_check_state=True):
|
||||||
|
"""Returns non zero k8s ID and a dictionary with k8s details on success, 0 and empty dictionary otherwise.
|
||||||
|
This method does not fail the run if k8s cannot be located by its name (arg_k8s_name), because this could be
|
||||||
|
an indicator of the requested k8s never existed before.
|
||||||
|
However, it does fail the run if k8s cannot be located by arg_k8s_id (if non zero specified) or if API errors
|
||||||
|
occur.
|
||||||
|
@param (int) arg_k8s_id: integer ID of the k8s to be found. If non-zero k8s ID is passed, account ID and k8s name
|
||||||
|
are ignored. However, k8s must be present in this case, as knowing its ID implies it already exists, otherwise
|
||||||
|
method will fail.
|
||||||
|
@param (string) arg_k8s_name: string that defines the name of k8s to be found. This parameter is case sensitive.
|
||||||
|
@param (bool) arg_check_state: tells the method to report RGs in valid states only.
|
||||||
|
|
||||||
|
@return: ID of the RG, if found. Zero otherwise.
|
||||||
|
@return: dictionary with k8s facts if k8s is present. Empty dictionary otherwise. None on error.
|
||||||
|
"""
|
||||||
|
|
||||||
|
# Resource group can be in one of the following states:
|
||||||
|
# MODELED, CREATED, DISABLING, DISABLED, ENABLING, DELETING, DELETED, DESTROYED, DESTROYED
|
||||||
|
#
|
||||||
|
# Transient state (ending with ING) are invalid from k8s manipulation viewpoint
|
||||||
|
#
|
||||||
|
|
||||||
|
K8S_INVALID_STATES = ["MODELED"]
|
||||||
|
|
||||||
|
self.result['waypoints'] = "{} -> {}".format(self.result['waypoints'], "k8s_find")
|
||||||
|
|
||||||
|
ret_k8s_id = 0
|
||||||
|
api_params = dict(includedeleted=True)
|
||||||
|
ret_k8s_dict = None
|
||||||
|
|
||||||
|
if arg_k8s_id > 0:
|
||||||
|
ret_k8s_id, ret_k8s_dict = self._k8s_get_by_id(arg_k8s_id)
|
||||||
|
if not ret_k8s_id:
|
||||||
|
self.result['failed'] = True
|
||||||
|
self.result['msg'] = "k8s_find(): cannot find k8s by ID {}.".format(arg_k8s_id)
|
||||||
|
self.amodule.fail_json(**self.result)
|
||||||
|
elif arg_k8s_name != "":
|
||||||
|
api_resp = self.decort_api_call(requests.post, "/restmachine/cloudapi/k8s/list", api_params)
|
||||||
|
if api_resp.status_code == 200:
|
||||||
|
account_specs = json.loads(api_resp.content.decode('utf8'))
|
||||||
|
for k8s_item in account_specs:
|
||||||
|
got_id, got_specs = self._k8s_get_by_id(k8s_item['id'])
|
||||||
|
if got_id and got_specs['name'] == arg_k8s_name:
|
||||||
|
# name matches
|
||||||
|
if not arg_check_state or got_specs['status'] not in K8S_INVALID_STATES:
|
||||||
|
ret_k8s_id = got_id
|
||||||
|
ret_k8s_dict = got_specs
|
||||||
|
break
|
||||||
|
# Note: we do not fail the run if k8s cannot be located by its name, because it could be a new k8s
|
||||||
|
# that never existed before. In this case ret_k8s_id=0 and empty ret_k8s_dict will be returned.
|
||||||
|
else:
|
||||||
|
# Both arg_k8s_id and arg_k8s_name are empty - there is no way to locate k8s in this case
|
||||||
|
self.result['failed'] = True
|
||||||
|
self.result['msg'] = "k8s_find(): either non-zero ID or a non-empty name must be specified."
|
||||||
|
self.amodule.fail_json(**self.result)
|
||||||
|
|
||||||
|
return ret_k8s_id, ret_k8s_dict
|
||||||
|
|
||||||
|
def k8s_state(self, arg_k8s_dict, arg_desired_state, arg_started=False):
|
||||||
|
"""Enable or disable k8s cluster.
|
||||||
|
|
||||||
|
@param arg_k8s_dict: dictionary with the target k8s facts as returned by k8s_find(...) method or
|
||||||
|
.../k8s/get API call.
|
||||||
|
@param arg_desired_state: the desired state for this k8s cluster. Valid states are 'enabled' and 'disabled'.
|
||||||
|
@param arg_started: the desired tech state for this k8s cluster. Valid states are 'True' and 'False'.
|
||||||
|
"""
|
||||||
|
|
||||||
|
self.result['waypoints'] = "{} -> {}".format(self.result['waypoints'], "k8s_state")
|
||||||
|
|
||||||
|
NOP_STATES_FOR_K8S_CHANGE = ["MODELED", "DISABLING",
|
||||||
|
"ENABLING", "DELETING",
|
||||||
|
"DELETED", "DESTROYING",
|
||||||
|
"DESTROYED", "CREATING",
|
||||||
|
"RESTORING"]
|
||||||
|
VALID_TARGET_STATES = ["ENABLED", "DISABLED"]
|
||||||
|
|
||||||
|
if arg_k8s_dict['status'] in NOP_STATES_FOR_K8S_CHANGE:
|
||||||
|
self.result['failed'] = False
|
||||||
|
self.result['msg'] = ("k8s_state(): no state change possible for k8s ID {} "
|
||||||
|
"in its current state '{}'.").format(arg_k8s_dict['id'], arg_k8s_dict['status'])
|
||||||
|
return
|
||||||
|
if arg_k8s_dict['status'] not in VALID_TARGET_STATES:
|
||||||
|
self.result['failed'] = False
|
||||||
|
self.result['warning'] = ("k8s_state(): unrecognized desired state '{}' requested "
|
||||||
|
"for k8s ID {}. No k8s state change will be done.").format(arg_desired_state,
|
||||||
|
arg_k8s_dict['id'])
|
||||||
|
return
|
||||||
|
|
||||||
|
if self.amodule.check_mode:
|
||||||
|
self.result['failed'] = False
|
||||||
|
self.result['msg'] = ("k8s_state() in check mode: setting state of k8s ID {}, name '{}' to "
|
||||||
|
"'{}' was requested.").format(arg_k8s_dict['id'], arg_k8s_dict['name'],
|
||||||
|
arg_desired_state)
|
||||||
|
return
|
||||||
|
|
||||||
|
k8s_state_api = "" # this string will also be used as a flag to indicate that API call is necessary
|
||||||
|
api_params = dict(k8sId=arg_k8s_dict['id'])
|
||||||
|
expected_state = ""
|
||||||
|
tech_state = ""
|
||||||
|
if arg_k8s_dict['status'] in ["CREATED", "ENABLED"] and arg_desired_state == 'disabled':
|
||||||
|
k8s_state_api = "/restmachine/cloudapi/k8s/disable"
|
||||||
|
expected_state = "DISABLED"
|
||||||
|
elif arg_k8s_dict['status'] in ["CREATED", "DISABLED"] and arg_desired_state == 'enabled':
|
||||||
|
k8s_state_api = "/restmachine/cloudapi/k8s/enable"
|
||||||
|
expected_state = "ENABLED"
|
||||||
|
elif arg_k8s_dict['status'] == "ENABLED" and arg_desired_state == 'enabled' and arg_started is True and arg_k8s_dict['techStatus'] == "STOPPED":
|
||||||
|
k8s_state_api = "/restmachine/cloudapi/k8s/start"
|
||||||
|
tech_state = "STARTED"
|
||||||
|
elif arg_k8s_dict['status'] == "ENABLED" and arg_desired_state == 'enabled' and arg_started is False and arg_k8s_dict['techStatus'] == "STARTED":
|
||||||
|
k8s_state_api = "/restmachine/cloudapi/k8s/stop"
|
||||||
|
tech_state = "STOPPED"
|
||||||
|
if k8s_state_api != "":
|
||||||
|
self.decort_api_call(requests.post, k8s_state_api, api_params)
|
||||||
|
# On success the above call will return here. On error it will abort execution by calling fail_json.
|
||||||
|
self.result['failed'] = False
|
||||||
|
self.result['changed'] = True
|
||||||
|
arg_k8s_dict['status'] = expected_state
|
||||||
|
arg_k8s_dict['started'] = tech_state
|
||||||
|
else:
|
||||||
|
self.result['failed'] = False
|
||||||
|
self.result['msg'] = ("k8s_state(): no state change required for k8s ID {} from current "
|
||||||
|
"state '{}' to desired state '{}'.").format(arg_k8s_dict['id'],
|
||||||
|
arg_k8s_dict['status'],
|
||||||
|
arg_desired_state)
|
||||||
|
return
|
||||||
|
def k8s_delete(self, k8s_id, permanently=False):
|
||||||
|
self.result['waypoints'] = "{} -> {}".format(self.result['waypoints'], "k8s_delete")
|
||||||
|
|
||||||
|
if self.amodule.check_mode:
|
||||||
|
self.result['failed'] = False
|
||||||
|
self.result['msg'] = "k8s_delete() in check mode: delete Compute ID {} was requested.".format(k8s_id)
|
||||||
|
return
|
||||||
|
|
||||||
|
api_params = dict(k8sId=k8s_id,
|
||||||
|
permanently=permanently,
|
||||||
|
)
|
||||||
|
self.decort_api_call(requests.post, "/restmachine/cloudapi/k8s/delete", api_params)
|
||||||
|
# On success the above call will return here. On error it will abort execution by calling fail_json.
|
||||||
|
self.result['failed'] = False
|
||||||
|
self.result['changed'] = True
|
||||||
|
return
|
||||||
|
def k8s_restore(self, k8s_id ):
|
||||||
|
"""Restores a deleted k8s cluster identified by ID.
|
||||||
|
|
||||||
|
@param k8s_id: ID of the k8s cluster to restore.
|
||||||
|
"""
|
||||||
|
|
||||||
|
self.result['waypoints'] = "{} -> {}".format(self.result['waypoints'], "k8s_restore")
|
||||||
|
|
||||||
|
if self.amodule.check_mode:
|
||||||
|
self.result['failed'] = False
|
||||||
|
self.result['msg'] = "k8s_restore() in check mode: restore k8s ID {} was requested.".format(k8s_id)
|
||||||
|
return
|
||||||
|
|
||||||
|
api_params = dict(k8sId=k8s_id)
|
||||||
|
self.decort_api_call(requests.post, "/restmachine/cloudapi/k8s/restore", api_params)
|
||||||
|
# On success the above call will return here. On error it will abort execution by calling fail_json.
|
||||||
|
self.result['failed'] = False
|
||||||
|
self.result['changed'] = True
|
||||||
|
return
|
||||||
|
def k8s_provision(self, k8s_name,
|
||||||
|
wg_name, k8ci_id,
|
||||||
|
rg_id, master_count,
|
||||||
|
master_cpu, master_ram,
|
||||||
|
master_disk, worker_count,
|
||||||
|
worker_cpu, worker_ram,
|
||||||
|
worker_disk, extnet_id,
|
||||||
|
with_lb, annotation, ):
|
||||||
|
|
||||||
|
self.result['waypoints'] = "{} -> {}".format(self.result['waypoints'], "k8s_provision")
|
||||||
|
|
||||||
|
if self.amodule.check_mode:
|
||||||
|
self.result['failed'] = False
|
||||||
|
self.result['msg'] = ("k8s_provision() in check mode. Provision k8s '{}' in RG ID {} "
|
||||||
|
"was requested.").format(k8s_name, rg_id)
|
||||||
|
return 0
|
||||||
|
|
||||||
|
api_url = "/restmachine/cloudapi/k8s/create"
|
||||||
|
api_params = dict(name=k8s_name,
|
||||||
|
rgId=rg_id,
|
||||||
|
k8ciId=k8ci_id,
|
||||||
|
workerGroupName=wg_name,
|
||||||
|
masterNum=master_count,
|
||||||
|
masterCpu=master_cpu,
|
||||||
|
masterRam=master_ram,
|
||||||
|
masterDisk=master_disk,
|
||||||
|
workerNum=worker_count,
|
||||||
|
workerCpu=worker_cpu,
|
||||||
|
workerRam=worker_ram,
|
||||||
|
workerDisk=worker_disk,
|
||||||
|
extnetId=extnet_id,
|
||||||
|
withLB=with_lb,
|
||||||
|
desc=annotation,
|
||||||
|
)
|
||||||
|
api_resp = self.decort_api_call(requests.post, api_url, api_params)
|
||||||
|
k8s_id = ""
|
||||||
|
if api_resp.status_code == 200:
|
||||||
|
for i in range(300):
|
||||||
|
api_get_url = "/restmachine/cloudbroker/tasks/get"
|
||||||
|
api_get_params = dict(
|
||||||
|
auditId=api_resp.content.decode('utf8').replace('"', '')
|
||||||
|
)
|
||||||
|
api_get_resp = self.decort_api_call(requests.post, api_get_url, api_get_params)
|
||||||
|
ret_info = json.loads(api_get_resp.content.decode('utf8'))
|
||||||
|
if api_get_resp.status_code == 200:
|
||||||
|
if ret_info['status'] in ["PROCESSING", "SCHEDULED"]:
|
||||||
|
self.result['failed'] = False
|
||||||
|
time.sleep(30)
|
||||||
|
elif ret_info['status'] == "ERROR":
|
||||||
|
self.result['failed'] = True
|
||||||
|
return
|
||||||
|
elif ret_info['status'] == "OK":
|
||||||
|
k8s_id = ret_info['result'][0]
|
||||||
|
self.result['changed'] = True
|
||||||
|
return k8s_id
|
||||||
|
else:
|
||||||
|
k8s_id = ret_info['status']
|
||||||
|
else:
|
||||||
|
self.result['failed'] = True
|
||||||
|
# Timeout
|
||||||
|
self.result['failed'] = True
|
||||||
|
else:
|
||||||
|
self.result['failed'] = True
|
||||||
|
return
|
||||||
|
|||||||
Reference in New Issue
Block a user