2013-10-07 23:01:37 +02:00
|
|
|
#!/usr/bin/python
|
|
|
|
# Copyright 2013 Google Inc.
|
|
|
|
#
|
|
|
|
# This file is part of Ansible
|
|
|
|
#
|
|
|
|
# Ansible is free software: you can redistribute it and/or modify
|
|
|
|
# it under the terms of the GNU General Public License as published by
|
|
|
|
# the Free Software Foundation, either version 3 of the License, or
|
|
|
|
# (at your option) any later version.
|
|
|
|
#
|
|
|
|
# Ansible is distributed in the hope that it will be useful,
|
|
|
|
# but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
# GNU General Public License for more details.
|
|
|
|
#
|
|
|
|
# You should have received a copy of the GNU General Public License
|
|
|
|
# along with Ansible. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
|
|
|
|
DOCUMENTATION = '''
|
|
|
|
---
|
|
|
|
module: gce_pd
|
2013-11-19 00:55:49 +01:00
|
|
|
version_added: "1.4"
|
2013-10-07 23:01:37 +02:00
|
|
|
short_description: utilize GCE persistent disk resources
|
|
|
|
description:
|
|
|
|
- This module can create and destroy unformatted GCE persistent disks
|
|
|
|
U(https://developers.google.com/compute/docs/disks#persistentdisks).
|
2014-04-12 00:45:56 +02:00
|
|
|
It also supports attaching and detaching disks from running instances.
|
2013-10-07 23:01:37 +02:00
|
|
|
Full install/configuration instructions for the gce* modules can
|
|
|
|
be found in the comments of ansible/test/gce_tests.py.
|
|
|
|
options:
|
|
|
|
detach_only:
|
|
|
|
description:
|
|
|
|
- do not destroy the disk, merely detach it from an instance
|
|
|
|
required: false
|
|
|
|
default: "no"
|
|
|
|
choices: ["yes", "no"]
|
|
|
|
aliases: []
|
|
|
|
instance_name:
|
|
|
|
description:
|
|
|
|
- instance name if you wish to attach or detach the disk
|
|
|
|
required: false
|
|
|
|
default: null
|
|
|
|
aliases: []
|
|
|
|
mode:
|
|
|
|
description:
|
|
|
|
- GCE mount mode of disk, READ_ONLY (default) or READ_WRITE
|
|
|
|
required: false
|
|
|
|
default: "READ_ONLY"
|
|
|
|
choices: ["READ_WRITE", "READ_ONLY"]
|
|
|
|
aliases: []
|
|
|
|
name:
|
|
|
|
description:
|
|
|
|
- name of the disk
|
|
|
|
required: true
|
|
|
|
default: null
|
|
|
|
aliases: []
|
|
|
|
size_gb:
|
|
|
|
description:
|
|
|
|
- whole integer size of disk (in GB) to create, default is 10 GB
|
|
|
|
required: false
|
|
|
|
default: 10
|
|
|
|
aliases: []
|
2014-04-11 23:40:52 +02:00
|
|
|
image:
|
|
|
|
description:
|
|
|
|
- the source image to use for the disk
|
|
|
|
required: false
|
|
|
|
default: null
|
|
|
|
aliases: []
|
2014-04-14 17:33:45 +02:00
|
|
|
version_added: "1.6"
|
2014-04-12 00:45:56 +02:00
|
|
|
snapshot:
|
|
|
|
description:
|
|
|
|
- the source snapshot to use for the disk
|
|
|
|
required: false
|
|
|
|
default: null
|
|
|
|
aliases: []
|
2014-04-14 17:33:45 +02:00
|
|
|
version_added: "1.6"
|
2013-10-07 23:01:37 +02:00
|
|
|
state:
|
|
|
|
description:
|
|
|
|
- desired state of the persistent disk
|
|
|
|
required: false
|
|
|
|
default: "present"
|
|
|
|
choices: ["active", "present", "absent", "deleted"]
|
|
|
|
aliases: []
|
|
|
|
zone:
|
|
|
|
description:
|
|
|
|
- zone in which to create the disk
|
|
|
|
required: false
|
|
|
|
default: "us-central1-b"
|
|
|
|
aliases: []
|
2014-03-01 21:56:15 +01:00
|
|
|
service_account_email:
|
2014-04-08 16:05:57 +02:00
|
|
|
version_added: "1.6"
|
2014-03-01 21:56:15 +01:00
|
|
|
description:
|
|
|
|
- service account email
|
|
|
|
required: false
|
|
|
|
default: null
|
|
|
|
aliases: []
|
|
|
|
pem_file:
|
2014-04-08 16:05:57 +02:00
|
|
|
version_added: "1.6"
|
2014-03-01 21:56:15 +01:00
|
|
|
description:
|
|
|
|
- path to the pem file associated with the service account email
|
|
|
|
required: false
|
|
|
|
default: null
|
|
|
|
aliases: []
|
|
|
|
project_id:
|
2014-04-08 16:05:57 +02:00
|
|
|
version_added: "1.6"
|
2014-03-01 21:56:15 +01:00
|
|
|
description:
|
|
|
|
- your GCE project ID
|
|
|
|
required: false
|
|
|
|
default: null
|
|
|
|
aliases: []
|
2013-10-07 23:01:37 +02:00
|
|
|
|
|
|
|
requirements: [ "libcloud" ]
|
|
|
|
author: Eric Johnson <erjohnso@google.com>
|
|
|
|
'''
|
|
|
|
|
|
|
|
EXAMPLES = '''
|
|
|
|
# Simple attachment action to an existing instance
|
2014-03-01 21:56:15 +01:00
|
|
|
- local_action:
|
|
|
|
module: gce_pd
|
2013-10-07 23:01:37 +02:00
|
|
|
instance_name: notlocalhost
|
2014-03-01 21:56:15 +01:00
|
|
|
size_gb: 5
|
2013-10-07 23:01:37 +02:00
|
|
|
name: pd
|
|
|
|
'''
|
|
|
|
|
|
|
|
import sys
|
|
|
|
|
2014-03-01 21:56:15 +01:00
|
|
|
try:
|
|
|
|
from libcloud.compute.types import Provider
|
|
|
|
from libcloud.compute.providers import get_driver
|
2013-10-07 23:01:37 +02:00
|
|
|
from libcloud.common.google import GoogleBaseError, QuotaExceededError, \
|
|
|
|
ResourceExistsError, ResourceNotFoundError, ResourceInUseError
|
2014-03-01 21:56:15 +01:00
|
|
|
_ = Provider.GCE
|
|
|
|
except ImportError:
|
2013-10-07 23:01:37 +02:00
|
|
|
print("failed=True " + \
|
|
|
|
"msg='libcloud with GCE support is required for this module.'")
|
|
|
|
sys.exit(1)
|
|
|
|
|
|
|
|
|
|
|
|
def main():
|
|
|
|
module = AnsibleModule(
|
|
|
|
argument_spec = dict(
|
2014-03-28 16:41:35 +01:00
|
|
|
detach_only = dict(type='bool'),
|
2013-10-07 23:01:37 +02:00
|
|
|
instance_name = dict(),
|
2014-03-28 16:41:35 +01:00
|
|
|
mode = dict(default='READ_ONLY', choices=['READ_WRITE', 'READ_ONLY']),
|
2013-10-07 23:01:37 +02:00
|
|
|
name = dict(required=True),
|
|
|
|
size_gb = dict(default=10),
|
2014-04-11 23:40:52 +02:00
|
|
|
image = dict(),
|
2014-04-12 00:45:56 +02:00
|
|
|
snapshot = dict(),
|
2013-10-07 23:01:37 +02:00
|
|
|
state = dict(default='present'),
|
|
|
|
zone = dict(default='us-central1-b'),
|
2014-03-01 21:56:15 +01:00
|
|
|
service_account_email = dict(),
|
|
|
|
pem_file = dict(),
|
|
|
|
project_id = dict(),
|
2013-10-07 23:01:37 +02:00
|
|
|
)
|
|
|
|
)
|
|
|
|
|
2014-03-01 21:56:15 +01:00
|
|
|
gce = gce_connect(module)
|
|
|
|
|
2013-10-07 23:01:37 +02:00
|
|
|
detach_only = module.params.get('detach_only')
|
|
|
|
instance_name = module.params.get('instance_name')
|
|
|
|
mode = module.params.get('mode')
|
|
|
|
name = module.params.get('name')
|
|
|
|
size_gb = module.params.get('size_gb')
|
2014-04-11 23:40:52 +02:00
|
|
|
image = module.params.get('image')
|
2014-04-12 00:45:56 +02:00
|
|
|
snapshot = module.params.get('snapshot')
|
2013-10-07 23:01:37 +02:00
|
|
|
state = module.params.get('state')
|
|
|
|
zone = module.params.get('zone')
|
|
|
|
|
|
|
|
if detach_only and not instance_name:
|
|
|
|
module.fail_json(
|
|
|
|
msg='Must specify an instance name when detaching a disk',
|
|
|
|
changed=False)
|
|
|
|
|
|
|
|
disk = inst = None
|
|
|
|
changed = is_attached = False
|
|
|
|
|
|
|
|
json_output = { 'name': name, 'zone': zone, 'state': state }
|
|
|
|
if detach_only:
|
|
|
|
json_output['detach_only'] = True
|
|
|
|
json_output['detached_from_instance'] = instance_name
|
|
|
|
|
|
|
|
if instance_name:
|
|
|
|
# user wants to attach/detach from an existing instance
|
|
|
|
try:
|
|
|
|
inst = gce.ex_get_node(instance_name, zone)
|
|
|
|
# is the disk attached?
|
|
|
|
for d in inst.extra['disks']:
|
|
|
|
if d['deviceName'] == name:
|
|
|
|
is_attached = True
|
|
|
|
json_output['attached_mode'] = d['mode']
|
|
|
|
json_output['attached_to_instance'] = inst.name
|
|
|
|
except:
|
|
|
|
pass
|
|
|
|
|
|
|
|
# find disk if it already exists
|
|
|
|
try:
|
|
|
|
disk = gce.ex_get_volume(name)
|
|
|
|
json_output['size_gb'] = int(disk.size)
|
|
|
|
except ResourceNotFoundError:
|
|
|
|
pass
|
2014-02-02 18:33:27 +01:00
|
|
|
except Exception, e:
|
2013-10-07 23:01:37 +02:00
|
|
|
module.fail_json(msg=unexpected_error_msg(e), changed=False)
|
|
|
|
|
|
|
|
# user wants a disk to exist. If "instance_name" is supplied the user
|
|
|
|
# also wants it attached
|
|
|
|
if state in ['active', 'present']:
|
|
|
|
|
|
|
|
if not size_gb:
|
|
|
|
module.fail_json(msg="Must supply a size_gb", changed=False)
|
|
|
|
try:
|
|
|
|
size_gb = int(round(float(size_gb)))
|
2013-10-19 18:57:47 +02:00
|
|
|
if size_gb < 1:
|
|
|
|
raise Exception
|
2013-10-07 23:01:37 +02:00
|
|
|
except:
|
|
|
|
module.fail_json(msg="Must supply a size_gb larger than 1 GB",
|
|
|
|
changed=False)
|
|
|
|
|
|
|
|
if instance_name and inst is None:
|
|
|
|
module.fail_json(msg='Instance %s does not exist in zone %s' % (
|
|
|
|
instance_name, zone), changed=False)
|
|
|
|
|
|
|
|
if not disk:
|
2014-04-12 00:45:56 +02:00
|
|
|
if image is not None and snapshot is not None:
|
|
|
|
module.fail_json(
|
|
|
|
msg='Cannot give both image (%s) and snapshot (%s)' % (
|
|
|
|
image, snapshot), changed=False)
|
2014-04-11 23:40:52 +02:00
|
|
|
lc_image = None
|
2014-04-12 00:45:56 +02:00
|
|
|
lc_snapshot = None
|
2014-04-11 23:40:52 +02:00
|
|
|
if image is not None:
|
2014-04-12 00:45:56 +02:00
|
|
|
lc_image = gce.ex_get_image(image)
|
|
|
|
elif snapshot is not None:
|
|
|
|
lc_snapshot = gce.ex_get_snapshot(snapshot)
|
2013-10-07 23:01:37 +02:00
|
|
|
try:
|
2014-04-12 00:45:56 +02:00
|
|
|
disk = gce.create_volume(
|
|
|
|
size_gb, name, location=zone, image=lc_image,
|
|
|
|
snapshot=lc_snapshot)
|
2013-10-07 23:01:37 +02:00
|
|
|
except ResourceExistsError:
|
|
|
|
pass
|
|
|
|
except QuotaExceededError:
|
|
|
|
module.fail_json(msg='Requested disk size exceeds quota',
|
|
|
|
changed=False)
|
2014-02-02 18:33:27 +01:00
|
|
|
except Exception, e:
|
2013-10-07 23:01:37 +02:00
|
|
|
module.fail_json(msg=unexpected_error_msg(e), changed=False)
|
|
|
|
json_output['size_gb'] = size_gb
|
2014-04-12 00:45:56 +02:00
|
|
|
if image is not None:
|
|
|
|
json_output['image'] = image
|
|
|
|
if snapshot is not None:
|
|
|
|
json_output['snapshot'] = snapshot
|
2013-10-07 23:01:37 +02:00
|
|
|
changed = True
|
|
|
|
if inst and not is_attached:
|
|
|
|
try:
|
|
|
|
gce.attach_volume(inst, disk, device=name, ex_mode=mode)
|
2014-02-02 18:33:27 +01:00
|
|
|
except Exception, e:
|
2013-10-07 23:01:37 +02:00
|
|
|
module.fail_json(msg=unexpected_error_msg(e), changed=False)
|
|
|
|
json_output['attached_to_instance'] = inst.name
|
|
|
|
json_output['attached_mode'] = mode
|
|
|
|
changed = True
|
|
|
|
|
|
|
|
# user wants to delete a disk (or perhaps just detach it).
|
|
|
|
if state in ['absent', 'deleted'] and disk:
|
|
|
|
|
|
|
|
if inst and is_attached:
|
|
|
|
try:
|
|
|
|
gce.detach_volume(disk, ex_node=inst)
|
2014-02-02 18:33:27 +01:00
|
|
|
except Exception, e:
|
2013-10-07 23:01:37 +02:00
|
|
|
module.fail_json(msg=unexpected_error_msg(e), changed=False)
|
|
|
|
changed = True
|
|
|
|
if not detach_only:
|
|
|
|
try:
|
|
|
|
gce.destroy_volume(disk)
|
2014-02-02 18:33:27 +01:00
|
|
|
except ResourceInUseError, e:
|
2013-10-07 23:01:37 +02:00
|
|
|
module.fail_json(msg=str(e.value), changed=False)
|
2014-02-02 18:33:27 +01:00
|
|
|
except Exception, e:
|
2013-10-07 23:01:37 +02:00
|
|
|
module.fail_json(msg=unexpected_error_msg(e), changed=False)
|
|
|
|
changed = True
|
|
|
|
|
|
|
|
json_output['changed'] = changed
|
|
|
|
print json.dumps(json_output)
|
|
|
|
sys.exit(0)
|
|
|
|
|
2013-12-02 21:13:49 +01:00
|
|
|
# import module snippets
|
2013-12-02 21:11:23 +01:00
|
|
|
from ansible.module_utils.basic import *
|
2014-03-01 21:56:15 +01:00
|
|
|
from ansible.module_utils.gce import *
|
2013-10-07 23:01:37 +02:00
|
|
|
|
|
|
|
main()
|