2013-06-14 03:56:01 +00:00
#!/usr/bin/python
# -*- coding: utf-8 -*-
# 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 = '''
---
2013-06-18 02:45:45 +00:00
module: digital_ocean
2013-06-14 03:56:01 +00:00
short_description: Create/delete a droplet/SSH_key in DigitalOcean
description:
- Create/delete a droplet in DigitalOcean and optionally waits for it to be 'running', or deploy an SSH key.
2013-06-20 02:39:08 +00:00
version_added: "1.3"
2013-06-14 03:56:01 +00:00
options:
command:
description:
- Which target you want to operate on.
2013-06-22 12:48:25 +00:00
default: droplet
2013-06-14 03:56:01 +00:00
choices: ['droplet', 'ssh']
state:
description:
- Indicate desired state of the target.
2013-06-22 12:48:25 +00:00
default: present
2013-06-14 03:56:01 +00:00
choices: ['present', 'active', 'absent', 'deleted']
2013-09-18 15:06:32 +00:00
client_id:
2013-06-20 02:33:19 +00:00
description:
2013-06-14 03:56:01 +00:00
- Digital Ocean manager id.
api_key:
description:
- Digital Ocean api key.
id:
description:
- Numeric, the droplet id you want to operate on.
name:
description:
- String, this is the name of the droplet - must be formatted by hostname rules, or the name of a SSH key.
2013-09-18 20:07:22 +00:00
unique_name:
description:
- Bool, require unique hostnames. By default, digital ocean allows multiple hosts with the same name. Setting this to "yes" allows only one host per name. Useful for idempotence.
2013-09-25 21:49:48 +00:00
version_added: "1.4"
2013-09-18 20:07:22 +00:00
default: "no"
choices: [ "yes", "no" ]
2013-06-14 03:56:01 +00:00
size_id:
description:
- Numeric, this is the id of the size you would like the droplet created at.
image_id:
description:
- Numeric, this is the id of the image you would like the droplet created with.
region_id:
description:
2013-06-20 02:37:09 +00:00
- "Numeric, this is the id of the region you would like your server"
2013-06-14 03:56:01 +00:00
ssh_key_ids:
description:
- Optional, comma separated list of ssh_key_ids that you would like to be added to the server
2013-10-02 22:06:35 +00:00
virtio:
description:
- "Bool, turn on virtio driver in droplet for improved network and storage I/O"
version_added: "1.4"
default: "yes"
choices: [ "yes", "no" ]
private_networking:
description:
- "Bool, add an additional, private network interface to droplet for inter-droplet communication"
version_added: "1.4"
default: "no"
choices: [ "yes", "no" ]
2013-06-14 03:56:01 +00:00
wait:
2013-09-18 15:06:32 +00:00
description:
2013-07-11 17:33:42 +00:00
- Wait for the droplet to be in state 'running' before returning. If wait is "no" an ip_address may not be returned.
2013-06-14 03:56:01 +00:00
default: "yes"
choices: [ "yes", "no" ]
wait_timeout:
description:
- How long before wait gives up, in seconds.
default: 300
ssh_pub_key:
description:
- The public SSH key you want to add to your account.
notes:
- Two environment variables can be used, DO_CLIENT_ID and DO_API_KEY.
'''
EXAMPLES = '''
2013-06-18 02:45:45 +00:00
# Ensure a SSH key is present
2013-06-20 01:52:59 +00:00
# If a key matches this name, will return the ssh key id and changed = False
# If no existing key matches this name, a new key is created, the ssh key id is returned and changed = False
2013-06-14 03:56:01 +00:00
2013-06-20 01:52:59 +00:00
- digital_ocean: >
2013-09-18 15:06:32 +00:00
state=present
command=ssh
name=my_ssh_key
ssh_pub_key='ssh-rsa AAAA...'
client_id=XXX
2013-06-20 01:52:59 +00:00
api_key=XXX
2013-06-18 02:45:45 +00:00
# Create a new Droplet
2013-06-20 01:52:59 +00:00
# Will return the droplet details including the droplet id (used for idempotence)
2013-06-18 02:45:45 +00:00
2013-06-20 01:52:59 +00:00
- digital_ocean: >
2013-09-18 15:06:32 +00:00
state=present
command=droplet
2013-12-25 18:51:56 +00:00
name=mydroplet
2013-09-18 15:06:32 +00:00
client_id=XXX
api_key=XXX
size_id=1
region_id=2
image_id=3
2013-06-20 01:52:59 +00:00
wait_timeout=500
2013-07-11 17:33:42 +00:00
register: my_droplet
2013-09-18 19:49:41 +00:00
- debug: msg="ID is {{ my_droplet.droplet.id }}"
- debug: msg="IP is {{ my_droplet.droplet.ip_address }}"
2013-06-18 02:45:45 +00:00
# Ensure a droplet is present
2013-06-20 01:52:59 +00:00
# If droplet id already exist, will return the droplet details and changed = False
# If no droplet matches the id, a new droplet will be created and the droplet details (including the new id) are returned, changed = True.
2013-06-18 02:45:45 +00:00
2013-06-20 01:52:59 +00:00
- digital_ocean: >
2013-09-18 15:06:32 +00:00
state=present
command=droplet
id=123
2013-12-25 18:51:56 +00:00
name=mydroplet
2013-09-18 15:06:32 +00:00
client_id=XXX
api_key=XXX
size_id=1
region_id=2
image_id=3
2013-06-20 01:52:59 +00:00
wait_timeout=500
2013-06-18 02:45:45 +00:00
# Create a droplet with ssh key
2013-09-18 15:06:32 +00:00
# The ssh key id can be passed as argument at the creation of a droplet (see ssh_key_ids).
2013-06-20 01:52:59 +00:00
# Several keys can be added to ssh_key_ids as id1,id2,id3
# The keys are used to connect as root to the droplet.
2013-06-18 02:45:45 +00:00
2013-06-20 01:52:59 +00:00
- digital_ocean: >
2013-09-18 15:06:32 +00:00
state=present
2013-06-20 01:52:59 +00:00
ssh_key_ids=id1,id2
2013-12-25 18:51:56 +00:00
name=mydroplet
2013-09-18 15:06:32 +00:00
client_id=XXX
api_key=XXX
size_id=1
region_id=2
2013-06-20 01:52:59 +00:00
image_id=3
2013-06-14 03:56:01 +00:00
'''
import sys
import os
import time
try:
2013-10-03 00:29:35 +00:00
import dopy
2013-06-14 03:56:01 +00:00
from dopy.manager import DoError, DoManager
except ImportError as e:
2013-10-03 00:29:35 +00:00
print "failed=True msg='dopy >= 0.2.2 required for this module'"
sys.exit(1)
if dopy.__version__ < '0.2.2':
print "failed=True msg='dopy >= 0.2.2 required for this module'"
2013-06-14 03:56:01 +00:00
sys.exit(1)
class TimeoutError(DoError):
def __init__(self, msg, id):
super(TimeoutError, self).__init__(msg)
self.id = id
class JsonfyMixIn(object):
def to_json(self):
return self.__dict__
class Droplet(JsonfyMixIn):
manager = None
def __init__(self, droplet_json):
self.status = 'new'
self.__dict__.update(droplet_json)
def is_powered_on(self):
return self.status == 'active'
2013-07-11 18:05:26 +00:00
def update_attr(self, attrs=None):
2013-06-14 03:56:01 +00:00
if attrs:
for k, v in attrs.iteritems():
setattr(self, k, v)
else:
json = self.manager.show_droplet(self.id)
2013-07-11 18:05:26 +00:00
if json['ip_address']:
2013-06-14 03:56:01 +00:00
self.update_attr(json)
def power_on(self):
assert self.status == 'off', 'Can only power on a closed one.'
json = self.manager.power_on_droplet(self.id)
self.update_attr(json)
def ensure_powered_on(self, wait=True, wait_timeout=300):
if self.is_powered_on():
return
if self.status == 'off': # powered off
self.power_on()
if wait:
2013-09-18 19:41:55 +00:00
end_time = time.time() + wait_timeout
2013-06-14 03:56:01 +00:00
while time.time() < end_time:
2013-09-18 19:41:55 +00:00
time.sleep(min(20, end_time - time.time()))
2013-06-14 03:56:01 +00:00
self.update_attr()
if self.is_powered_on():
2013-07-11 18:05:26 +00:00
if not self.ip_address:
raise TimeoutError('No ip is found.', self.id)
2013-06-14 03:56:01 +00:00
return
raise TimeoutError('Wait for droplet running timeout', self.id)
def destroy(self):
2013-12-31 01:21:15 +00:00
return self.manager.destroy_droplet(self.id, scrub_data=True)
2013-09-18 15:06:32 +00:00
2013-06-14 03:56:01 +00:00
@classmethod
def setup(cls, client_id, api_key):
cls.manager = DoManager(client_id, api_key)
@classmethod
2013-10-02 22:06:35 +00:00
def add(cls, name, size_id, image_id, region_id, ssh_key_ids=None, virtio=True, private_networking=False):
json = cls.manager.new_droplet(name, size_id, image_id, region_id, ssh_key_ids, virtio, private_networking)
2013-06-14 03:56:01 +00:00
droplet = cls(json)
return droplet
@classmethod
2013-09-18 20:07:22 +00:00
def find(cls, id=None, name=None):
if not id and not name:
2013-06-14 03:56:01 +00:00
return False
2013-09-18 20:07:22 +00:00
2013-06-14 03:56:01 +00:00
droplets = cls.list_all()
2013-09-18 20:07:22 +00:00
# Check first by id. digital ocean requires that it be unique
2013-06-14 03:56:01 +00:00
for droplet in droplets:
if droplet.id == id:
return droplet
2013-09-18 20:07:22 +00:00
# Failing that, check by hostname.
for droplet in droplets:
if droplet.name == name:
return droplet
2013-06-14 03:56:01 +00:00
return False
@classmethod
def list_all(cls):
json = cls.manager.all_active_droplets()
return map(cls, json)
class SSH(JsonfyMixIn):
manager = None
def __init__(self, ssh_key_json):
self.__dict__.update(ssh_key_json)
update_attr = __init__
def destroy(self):
self.manager.destroy_ssh_key(self.id)
return True
@classmethod
def setup(cls, client_id, api_key):
cls.manager = DoManager(client_id, api_key)
@classmethod
def find(cls, name):
if not name:
return False
keys = cls.list_all()
for key in keys:
if key.name == name:
return key
return False
@classmethod
def list_all(cls):
json = cls.manager.all_ssh_keys()
return map(cls, json)
@classmethod
def add(cls, name, key_pub):
json = cls.manager.new_ssh_key(name, key_pub)
return cls(json)
def core(module):
def getkeyordie(k):
v = module.params[k]
if v is None:
module.fail_json(msg='Unable to load %s' % k)
return v
try:
# params['client_id'] will be None even if client_id is not passed in
client_id = module.params['client_id'] or os.environ['DO_CLIENT_ID']
api_key = module.params['api_key'] or os.environ['DO_API_KEY']
except KeyError, e:
module.fail_json(msg='Unable to load %s' % e.message)
changed = True
command = module.params['command']
state = module.params['state']
if command == 'droplet':
Droplet.setup(client_id, api_key)
if state in ('active', 'present'):
2013-09-18 20:07:22 +00:00
# First, try to find a droplet by id.
droplet = Droplet.find(id=module.params['id'])
# If we couldn't find the droplet and the user is allowing unique
# hostnames, then check to see if a droplet with the specified
# hostname already exists.
if not droplet and module.params['unique_name']:
droplet = Droplet.find(name=getkeyordie('name'))
# If both of those attempts failed, then create a new droplet.
2013-06-14 03:56:01 +00:00
if not droplet:
droplet = Droplet.add(
2013-09-18 19:41:55 +00:00
name=getkeyordie('name'),
size_id=getkeyordie('size_id'),
image_id=getkeyordie('image_id'),
region_id=getkeyordie('region_id'),
2013-10-02 22:06:35 +00:00
ssh_key_ids=module.params['ssh_key_ids'],
virtio=module.params['virtio'],
private_networking=module.params['private_networking']
2013-09-18 19:41:55 +00:00
)
2013-09-18 20:07:22 +00:00
2013-06-14 03:56:01 +00:00
if droplet.is_powered_on():
changed = False
2013-09-18 20:07:22 +00:00
2013-06-14 03:56:01 +00:00
droplet.ensure_powered_on(
2013-09-18 19:41:55 +00:00
wait=getkeyordie('wait'),
wait_timeout=getkeyordie('wait_timeout')
)
2013-09-18 20:07:22 +00:00
2013-06-14 03:56:01 +00:00
module.exit_json(changed=changed, droplet=droplet.to_json())
2013-09-18 15:06:32 +00:00
2013-06-14 03:56:01 +00:00
elif state in ('absent', 'deleted'):
2013-09-18 20:07:22 +00:00
# First, try to find a droplet by id.
droplet = Droplet.find(id=getkeyordie('id'))
# If we couldn't find the droplet and the user is allowing unique
# hostnames, then check to see if a droplet with the specified
# hostname already exists.
if not droplet and module.params['unique_name']:
droplet = Droplet.find(name=getkeyordie('name'))
2013-06-14 03:56:01 +00:00
if not droplet:
module.exit_json(changed=False, msg='The droplet is not found.')
2013-09-18 20:07:22 +00:00
2013-06-14 03:56:01 +00:00
event_json = droplet.destroy()
module.exit_json(changed=True, event_id=event_json['event_id'])
elif command == 'ssh':
SSH.setup(client_id, api_key)
name = getkeyordie('name')
if state in ('active', 'present'):
key = SSH.find(name)
if key:
2013-06-14 06:59:52 +00:00
module.exit_json(changed=False, ssh_key=key.to_json())
2013-06-14 03:56:01 +00:00
key = SSH.add(name, getkeyordie('ssh_pub_key'))
module.exit_json(changed=True, ssh_key=key.to_json())
elif state in ('absent', 'deleted'):
key = SSH.find(name)
if not key:
module.exit_json(changed=False, msg='SSH key with the name of %s is not found.' % name)
key.destroy()
module.exit_json(changed=True)
def main():
module = AnsibleModule(
argument_spec = dict(
2013-06-22 12:48:25 +00:00
command = dict(choices=['droplet', 'ssh'], default='droplet'),
state = dict(choices=['active', 'present', 'absent', 'deleted'], default='present'),
2013-06-14 03:56:01 +00:00
client_id = dict(aliases=['CLIENT_ID'], no_log=True),
api_key = dict(aliases=['API_KEY'], no_log=True),
name = dict(type='str'),
size_id = dict(type='int'),
image_id = dict(type='int'),
region_id = dict(type='int'),
ssh_key_ids = dict(default=''),
2013-10-02 22:06:35 +00:00
virtio = dict(type='bool', choices=BOOLEANS, default='yes'),
private_networking = dict(type='bool', choices=BOOLEANS, default='no'),
2013-06-14 03:56:01 +00:00
id = dict(aliases=['droplet_id'], type='int'),
2013-10-11 12:45:13 +00:00
unique_name = dict(type='bool', default='no'),
wait = dict(type='bool', default=True),
2013-06-14 03:56:01 +00:00
wait_timeout = dict(default=300, type='int'),
ssh_pub_key = dict(type='str'),
),
required_together = (
['size_id', 'image_id', 'region_id'],
),
mutually_exclusive = (
['size_id', 'ssh_pub_key'],
['image_id', 'ssh_pub_key'],
['region_id', 'ssh_pub_key'],
),
required_one_of = (
['id', 'name'],
),
)
try:
core(module)
except TimeoutError as e:
module.fail_json(msg=str(e), id=e.id)
except (DoError, Exception) as e:
module.fail_json(msg=str(e))
2013-12-02 20:13:49 +00:00
# import module snippets
2013-12-02 20:11:23 +00:00
from ansible.module_utils.basic import *
2013-06-14 03:56:01 +00:00
main()