2020-11-05 21:52:47 +08:00
|
|
|
# Copyright 2020, Red Hat, Inc.
|
|
|
|
#
|
|
|
|
# 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.
|
|
|
|
|
|
|
|
from __future__ import absolute_import, division, print_function
|
|
|
|
__metaclass__ = type
|
|
|
|
|
|
|
|
from ansible.module_utils.basic import AnsibleModule
|
|
|
|
try:
|
|
|
|
from ansible.module_utils.ca_common import is_containerized, \
|
2022-11-29 16:47:58 +08:00
|
|
|
generate_cmd, \
|
2020-11-05 21:52:47 +08:00
|
|
|
exec_command, \
|
|
|
|
exit_module
|
|
|
|
except ImportError:
|
|
|
|
from module_utils.ca_common import is_containerized, \
|
2022-11-29 16:47:58 +08:00
|
|
|
generate_cmd, \
|
2020-11-05 21:52:47 +08:00
|
|
|
exec_command, \
|
|
|
|
exit_module
|
|
|
|
import datetime
|
|
|
|
import json
|
|
|
|
|
|
|
|
|
|
|
|
ANSIBLE_METADATA = {
|
|
|
|
'metadata_version': '1.1',
|
|
|
|
'status': ['preview'],
|
|
|
|
'supported_by': 'community'
|
|
|
|
}
|
|
|
|
|
|
|
|
DOCUMENTATION = '''
|
|
|
|
---
|
|
|
|
module: ceph_ec_profile
|
|
|
|
|
|
|
|
short_description: Manage Ceph Erasure Code profile
|
|
|
|
|
|
|
|
version_added: "2.8"
|
|
|
|
|
|
|
|
description:
|
|
|
|
- Manage Ceph Erasure Code profile
|
|
|
|
options:
|
|
|
|
cluster:
|
|
|
|
description:
|
|
|
|
- The ceph cluster name.
|
|
|
|
required: false
|
|
|
|
default: ceph
|
|
|
|
name:
|
|
|
|
description:
|
|
|
|
- name of the profile.
|
|
|
|
required: true
|
|
|
|
state:
|
|
|
|
description:
|
|
|
|
If 'present' is used, the module creates a profile.
|
|
|
|
If 'absent' is used, the module will delete the profile.
|
|
|
|
required: false
|
|
|
|
choices: ['present', 'absent', 'info']
|
|
|
|
default: present
|
|
|
|
stripe_unit:
|
|
|
|
description:
|
|
|
|
- The amount of data in a data chunk, per stripe.
|
|
|
|
required: false
|
|
|
|
k:
|
|
|
|
description:
|
|
|
|
- Number of data-chunks the object will be split in
|
|
|
|
required: true
|
|
|
|
m:
|
|
|
|
description:
|
|
|
|
- Compute coding chunks for each object and store them on different
|
|
|
|
OSDs.
|
|
|
|
required: true
|
|
|
|
crush_root:
|
|
|
|
description:
|
|
|
|
- The name of the crush bucket used for the first step of the CRUSH
|
|
|
|
rule.
|
|
|
|
required: false
|
|
|
|
crush_device_class:
|
|
|
|
description:
|
|
|
|
- Restrict placement to devices of a specific class (hdd/ssd)
|
|
|
|
required: false
|
|
|
|
|
|
|
|
author:
|
|
|
|
- Guillaume Abrioux <gabrioux@redhat.com>
|
|
|
|
'''
|
|
|
|
|
|
|
|
EXAMPLES = '''
|
|
|
|
- name: create an erasure code profile
|
|
|
|
ceph_ec_profile:
|
|
|
|
name: foo
|
|
|
|
k: 4
|
|
|
|
m: 2
|
|
|
|
|
|
|
|
- name: delete an erassure code profile
|
|
|
|
ceph_ec_profile:
|
|
|
|
name: foo
|
|
|
|
state: absent
|
|
|
|
'''
|
|
|
|
|
|
|
|
RETURN = '''# '''
|
|
|
|
|
|
|
|
|
|
|
|
def get_profile(module, name, cluster='ceph', container_image=None):
|
|
|
|
'''
|
|
|
|
Get existing profile
|
|
|
|
'''
|
|
|
|
|
|
|
|
args = ['get', name, '--format=json']
|
|
|
|
|
2022-11-29 16:47:58 +08:00
|
|
|
cmd = generate_cmd(sub_cmd=['osd', 'erasure-code-profile'],
|
|
|
|
args=args,
|
|
|
|
cluster=cluster,
|
|
|
|
container_image=container_image)
|
2020-11-05 21:52:47 +08:00
|
|
|
|
|
|
|
return cmd
|
|
|
|
|
|
|
|
|
2021-10-26 21:49:33 +08:00
|
|
|
def create_profile(module, name, k, m, stripe_unit, crush_device_class, cluster='ceph', force=False, container_image=None): # noqa: E501
|
2020-11-05 21:52:47 +08:00
|
|
|
'''
|
|
|
|
Create a profile
|
|
|
|
'''
|
|
|
|
|
|
|
|
args = ['set', name, 'k={}'.format(k), 'm={}'.format(m)]
|
|
|
|
if stripe_unit:
|
|
|
|
args.append('stripe_unit={}'.format(stripe_unit))
|
2021-10-26 21:49:33 +08:00
|
|
|
if crush_device_class:
|
|
|
|
args.append('crush-device-class={}'.format(crush_device_class))
|
2020-11-05 21:52:47 +08:00
|
|
|
if force:
|
|
|
|
args.append('--force')
|
|
|
|
|
2022-11-29 16:47:58 +08:00
|
|
|
cmd = generate_cmd(sub_cmd=['osd', 'erasure-code-profile'],
|
|
|
|
args=args,
|
|
|
|
cluster=cluster,
|
|
|
|
container_image=container_image)
|
2020-11-05 21:52:47 +08:00
|
|
|
|
|
|
|
return cmd
|
|
|
|
|
|
|
|
|
|
|
|
def delete_profile(module, name, cluster='ceph', container_image=None):
|
|
|
|
'''
|
|
|
|
Delete a profile
|
|
|
|
'''
|
|
|
|
|
|
|
|
args = ['rm', name]
|
|
|
|
|
2022-11-29 16:47:58 +08:00
|
|
|
cmd = generate_cmd(sub_cmd=['osd', 'erasure-code-profile'],
|
|
|
|
args=args,
|
|
|
|
cluster=cluster,
|
|
|
|
container_image=container_image)
|
2020-11-05 21:52:47 +08:00
|
|
|
|
|
|
|
return cmd
|
|
|
|
|
|
|
|
|
|
|
|
def run_module():
|
|
|
|
module_args = dict(
|
|
|
|
cluster=dict(type='str', required=False, default='ceph'),
|
|
|
|
name=dict(type='str', required=True),
|
|
|
|
state=dict(type='str', required=False,
|
|
|
|
choices=['present', 'absent'], default='present'),
|
|
|
|
stripe_unit=dict(type='str', required=False),
|
|
|
|
k=dict(type='str', required=False),
|
|
|
|
m=dict(type='str', required=False),
|
2021-10-26 21:49:33 +08:00
|
|
|
crush_device_class=dict(type='str', required=False, default=''),
|
2020-11-05 21:52:47 +08:00
|
|
|
)
|
|
|
|
|
|
|
|
module = AnsibleModule(
|
|
|
|
argument_spec=module_args,
|
|
|
|
supports_check_mode=True,
|
|
|
|
required_if=[['state', 'present', ['k', 'm']]],
|
|
|
|
)
|
|
|
|
|
|
|
|
# Gather module parameters in variables
|
|
|
|
name = module.params.get('name')
|
|
|
|
cluster = module.params.get('cluster')
|
|
|
|
state = module.params.get('state')
|
|
|
|
stripe_unit = module.params.get('stripe_unit')
|
|
|
|
k = module.params.get('k')
|
|
|
|
m = module.params.get('m')
|
2021-10-26 21:49:33 +08:00
|
|
|
crush_device_class = module.params.get('crush_device_class')
|
2020-11-05 21:52:47 +08:00
|
|
|
|
|
|
|
if module.check_mode:
|
|
|
|
module.exit_json(
|
|
|
|
changed=False,
|
|
|
|
stdout='',
|
|
|
|
stderr='',
|
|
|
|
rc=0,
|
|
|
|
start='',
|
|
|
|
end='',
|
|
|
|
delta='',
|
|
|
|
)
|
|
|
|
|
|
|
|
startd = datetime.datetime.now()
|
|
|
|
changed = False
|
|
|
|
|
|
|
|
# will return either the image name or None
|
|
|
|
container_image = is_containerized()
|
|
|
|
|
|
|
|
if state == "present":
|
2021-06-18 00:18:07 +08:00
|
|
|
rc, cmd, out, err = exec_command(module, get_profile(module, name, cluster, container_image=container_image)) # noqa: E501
|
2020-11-05 21:52:47 +08:00
|
|
|
if rc == 0:
|
2021-06-18 00:18:07 +08:00
|
|
|
# the profile already exists, let's check whether we have to
|
|
|
|
# update it
|
2020-11-05 21:52:47 +08:00
|
|
|
current_profile = json.loads(out)
|
|
|
|
if current_profile['k'] != k or \
|
|
|
|
current_profile['m'] != m or \
|
2021-10-26 21:49:33 +08:00
|
|
|
current_profile.get('stripe_unit', stripe_unit) != stripe_unit or \
|
2022-11-29 16:47:58 +08:00
|
|
|
current_profile.get('crush-device-class', crush_device_class) != crush_device_class: # noqa: E501
|
2020-11-05 21:52:47 +08:00
|
|
|
rc, cmd, out, err = exec_command(module,
|
|
|
|
create_profile(module,
|
|
|
|
name,
|
|
|
|
k,
|
|
|
|
m,
|
|
|
|
stripe_unit,
|
2021-10-26 21:49:33 +08:00
|
|
|
crush_device_class, # noqa: E501
|
2020-11-05 21:52:47 +08:00
|
|
|
cluster,
|
2021-06-18 00:18:07 +08:00
|
|
|
force=True, container_image=container_image)) # noqa: E501
|
2020-11-05 21:52:47 +08:00
|
|
|
changed = True
|
|
|
|
else:
|
|
|
|
# the profile doesn't exist, it has to be created
|
|
|
|
rc, cmd, out, err = exec_command(module, create_profile(module,
|
|
|
|
name,
|
|
|
|
k,
|
|
|
|
m,
|
2021-06-18 00:18:07 +08:00
|
|
|
stripe_unit, # noqa: E501
|
2021-10-26 21:49:33 +08:00
|
|
|
crush_device_class, # noqa: E501
|
2020-11-05 21:52:47 +08:00
|
|
|
cluster,
|
2021-06-18 00:18:07 +08:00
|
|
|
container_image=container_image)) # noqa: E501
|
2020-11-05 21:52:47 +08:00
|
|
|
if rc == 0:
|
|
|
|
changed = True
|
|
|
|
|
|
|
|
elif state == "absent":
|
2021-06-18 00:18:07 +08:00
|
|
|
rc, cmd, out, err = exec_command(module, delete_profile(module, name, cluster, container_image=container_image)) # noqa: E501
|
2020-11-05 21:52:47 +08:00
|
|
|
if not err:
|
|
|
|
out = 'Profile {} removed.'.format(name)
|
|
|
|
changed = True
|
|
|
|
else:
|
|
|
|
rc = 0
|
|
|
|
out = "Skipping, the profile {} doesn't exist".format(name)
|
|
|
|
|
2021-06-18 00:18:07 +08:00
|
|
|
exit_module(module=module, out=out, rc=rc, cmd=cmd, err=err, startd=startd, changed=changed) # noqa: E501
|
2020-11-05 21:52:47 +08:00
|
|
|
|
|
|
|
|
|
|
|
def main():
|
|
|
|
run_module()
|
|
|
|
|
|
|
|
|
|
|
|
if __name__ == '__main__':
|
|
|
|
main()
|