Skip to content
Projects
Groups
Snippets
Help
This project
Loading...
Sign in / Register
Toggle navigation
C
configuration
Overview
Overview
Details
Activity
Cycle Analytics
Repository
Repository
Files
Commits
Branches
Tags
Contributors
Graph
Compare
Charts
Issues
0
Issues
0
List
Board
Labels
Milestones
Merge Requests
0
Merge Requests
0
CI / CD
CI / CD
Pipelines
Jobs
Schedules
Charts
Wiki
Wiki
Snippets
Snippets
Members
Members
Collapse sidebar
Close sidebar
Activity
Graph
Charts
Create a new issue
Jobs
Commits
Issue Boards
Open sidebar
edx
configuration
Commits
706e436c
Commit
706e436c
authored
Jan 28, 2014
by
John Jarvis
Browse files
Options
Browse Files
Download
Email Patches
Plain Diff
adding future version of ec2_elb and using it for sandbox reg
parent
a9f60293
Hide whitespace changes
Inline
Side-by-side
Showing
2 changed files
with
340 additions
and
2 deletions
+340
-2
playbooks/edx-east/edx_provision.yml
+2
-2
playbooks/library/ec2_elb_local_1.5
+338
-0
No files found.
playbooks/edx-east/edx_provision.yml
View file @
706e436c
...
@@ -32,7 +32,7 @@
...
@@ -32,7 +32,7 @@
-
roles/ora/defaults/main.yml
-
roles/ora/defaults/main.yml
-
roles/xqueue/defaults/main.yml
-
roles/xqueue/defaults/main.yml
-
roles/xserver/defaults/main.yml
-
roles/xserver/defaults/main.yml
-
roles/forum/defaults/main.yml
-
roles/forum/defaults/main.yml
roles
:
roles
:
# rerun common to set the hostname, nginx to set basic auth
# rerun common to set the hostname, nginx to set basic auth
-
common
-
common
...
@@ -61,7 +61,7 @@
...
@@ -61,7 +61,7 @@
sudo
:
False
sudo
:
False
-
name
:
register instance into an elb if one was provided
-
name
:
register instance into an elb if one was provided
local_action
:
local_action
:
module
:
ec2_elb
module
:
ec2_elb
_local_1.5
region
:
"
{{
region
}}"
region
:
"
{{
region
}}"
instance_id
:
"
{{
ec2_info.instance_ids[0]
}}"
instance_id
:
"
{{
ec2_info.instance_ids[0]
}}"
state
:
present
state
:
present
...
...
playbooks/library/ec2_elb_local_1.5
0 → 100644
View file @
706e436c
#!/usr/bin/python
# 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: ec2_elb
short_description: De-registers or registers instances from EC2 ELBs
description:
- This module de-registers or registers an AWS EC2 instance from the ELBs
that it belongs to.
- Returns fact "ec2_elbs" which is a list of elbs attached to the instance
if state=absent is passed as an argument.
- Will be marked changed when called only if there are ELBs found to operate on.
version_added: "1.2"
requirements: [ "boto" ]
author: John Jarvis
options:
state:
description:
- register or deregister the instance
required: true
choices: ['present', 'absent']
instance_id:
description:
- EC2 Instance ID
required: true
ec2_elbs:
description:
- List of ELB names, required for registration. The ec2_elbs fact should be used if there was a previous de-register.
required: false
default: None
aws_secret_key:
description:
- AWS secret key. If not set then the value of the AWS_SECRET_KEY environment variable is used.
required: false
default: None
aliases: ['ec2_secret_key', 'secret_key' ]
aws_access_key:
description:
- AWS access key. If not set then the value of the AWS_ACCESS_KEY environment variable is used.
required: false
default: None
aliases: ['ec2_access_key', 'access_key' ]
region:
description:
- The AWS region to use. If not specified then the value of the EC2_REGION environment variable, if any, is used.
required: false
aliases: ['aws_region', 'ec2_region']
enable_availability_zone:
description:
- Whether to enable the availability zone of the instance on the target ELB if the availability zone has not already
been enabled. If set to no, the task will fail if the availability zone is not enabled on the ELB.
required: false
default: yes
choices: [ "yes", "no" ]
wait:
description:
- Wait for instance registration or deregistration to complete successfully before returning.
required: false
default: yes
choices: [ "yes", "no" ]
"""
EXAMPLES
=
"""
# basic pre_task and post_task example
pre_tasks:
- name: Gathering ec2 facts
ec2_facts:
- name: Instance De-register
local_action: ec2_elb
args:
instance_id: "{{ ansible_ec2_instance_id }}"
state: 'absent'
roles:
- myrole
post_tasks:
- name: Instance Register
local_action: ec2_elb
args:
instance_id: "{{ ansible_ec2_instance_id }}"
ec2_elbs: "{{ item }}"
state: 'present'
with_items: ec2_elbs
"""
import
time
import
sys
import
os
try
:
import
boto
import
boto.ec2
import
boto.ec2.elb
from
boto.regioninfo
import
RegionInfo
except
ImportError
:
print
"failed=True msg='boto required for this module'"
sys
.
exit
(
1
)
class
ElbManager
:
"""Handles EC2 instance ELB registration and de-registration"""
def
__init__
(
self
,
module
,
instance_id
=
None
,
ec2_elbs
=
None
,
aws_access_key
=
None
,
aws_secret_key
=
None
,
region
=
None
):
self
.
aws_access_key
=
aws_access_key
self
.
aws_secret_key
=
aws_secret_key
self
.
module
=
module
self
.
instance_id
=
instance_id
self
.
region
=
region
self
.
lbs
=
self
.
_get_instance_lbs
(
ec2_elbs
)
self
.
changed
=
False
def
deregister
(
self
,
wait
):
"""De-register the instance from all ELBs and wait for the ELB
to report it out-of-service"""
for
lb
in
self
.
lbs
:
initial_state
=
self
.
_get_instance_health
(
lb
)
if
wait
else
None
if
initial_state
and
initial_state
.
state
==
'InService'
:
lb
.
deregister_instances
([
self
.
instance_id
])
else
:
return
if
wait
:
self
.
_await_elb_instance_state
(
lb
,
'OutOfService'
,
initial_state
)
else
:
# We cannot assume no change was made if we don't wait
# to find out
self
.
changed
=
True
def
register
(
self
,
wait
,
enable_availability_zone
):
"""Register the instance for all ELBs and wait for the ELB
to report the instance in-service"""
for
lb
in
self
.
lbs
:
if
wait
:
initial_state
=
self
.
_get_instance_health
(
lb
)
if
enable_availability_zone
:
self
.
_enable_availailability_zone
(
lb
)
lb
.
register_instances
([
self
.
instance_id
])
if
wait
:
self
.
_await_elb_instance_state
(
lb
,
'InService'
,
initial_state
)
else
:
# We cannot assume no change was made if we don't wait
# to find out
self
.
changed
=
True
def
exists
(
self
,
lbtest
):
""" Verify that the named ELB actually exists """
found
=
False
for
lb
in
self
.
lbs
:
if
lb
.
name
==
lbtest
:
found
=
True
break
return
found
def
_enable_availailability_zone
(
self
,
lb
):
"""Enable the current instance's availability zone in the provided lb.
Returns True if the zone was enabled or False if no change was made.
lb: load balancer"""
instance
=
self
.
_get_instance
()
if
instance
.
placement
in
lb
.
availability_zones
:
return
False
lb
.
enable_zones
(
zones
=
instance
.
placement
)
# If successful, the new zone will have been added to
# lb.availability_zones
return
instance
.
placement
in
lb
.
availability_zones
def
_await_elb_instance_state
(
self
,
lb
,
awaited_state
,
initial_state
):
"""Wait for an ELB to change state
lb: load balancer
awaited_state : state to poll for (string)"""
while
True
:
instance_state
=
self
.
_get_instance_health
(
lb
)
if
not
instance_state
:
msg
=
(
"The instance
%
s could not be put in service on
%
s."
" Reason: Invalid Instance"
)
self
.
module
.
fail_json
(
msg
=
msg
%
(
self
.
instance_id
,
lb
))
if
instance_state
.
state
==
awaited_state
:
# Check the current state agains the initial state, and only set
# changed if they are different.
if
(
initial_state
is
None
)
or
(
instance_state
.
state
!=
initial_state
.
state
):
self
.
changed
=
True
break
elif
self
.
_is_instance_state_pending
(
instance_state
):
# If it's pending, we'll skip further checks andd continue waiting
pass
elif
(
awaited_state
==
'InService'
and
instance_state
.
reason_code
==
"Instance"
):
# If the reason_code for the instance being out of service is
# "Instance" this indicates a failure state, e.g. the instance
# has failed a health check or the ELB does not have the
# instance's availabilty zone enabled. The exact reason why is
# described in InstantState.description.
msg
=
(
"The instance
%
s could not be put in service on
%
s."
" Reason:
%
s"
)
self
.
module
.
fail_json
(
msg
=
msg
%
(
self
.
instance_id
,
lb
,
instance_state
.
description
))
time
.
sleep
(
1
)
def
_is_instance_state_pending
(
self
,
instance_state
):
"""
Determines whether the instance_state is "pending", meaning there is
an operation under way to bring it in service.
"""
# This is messy, because AWS provides no way to distinguish between
# an instance that is is OutOfService because it's pending vs. OutOfService
# because it's failing health checks. So we're forced to analyze the
# description, which is likely to be brittle.
return
(
instance_state
and
'pending'
in
instance_state
.
description
)
def
_get_instance_health
(
self
,
lb
):
"""
Check instance health, should return status object or None under
certain error conditions.
"""
try
:
status
=
lb
.
get_instance_health
([
self
.
instance_id
])[
0
]
except
boto
.
exception
.
BotoServerError
,
e
:
if
e
.
error_code
==
'InvalidInstance'
:
return
None
else
:
raise
return
status
def
_get_instance_lbs
(
self
,
ec2_elbs
=
None
):
"""Returns a list of ELBs attached to self.instance_id
ec2_elbs: an optional list of elb names that will be used
for elb lookup instead of returning what elbs
are attached to self.instance_id"""
try
:
endpoint
=
"elasticloadbalancing.
%
s.amazonaws.com"
%
self
.
region
connect_region
=
RegionInfo
(
name
=
self
.
region
,
endpoint
=
endpoint
)
elb
=
boto
.
ec2
.
elb
.
ELBConnection
(
self
.
aws_access_key
,
self
.
aws_secret_key
,
region
=
connect_region
)
except
boto
.
exception
.
NoAuthHandlerFound
,
e
:
self
.
module
.
fail_json
(
msg
=
str
(
e
))
elbs
=
elb
.
get_all_load_balancers
()
if
ec2_elbs
:
lbs
=
sorted
(
lb
for
lb
in
elbs
if
lb
.
name
in
ec2_elbs
)
else
:
lbs
=
[]
for
lb
in
elbs
:
for
info
in
lb
.
instances
:
if
self
.
instance_id
==
info
.
id
:
lbs
.
append
(
lb
)
return
lbs
def
_get_instance
(
self
):
"""Returns a boto.ec2.InstanceObject for self.instance_id"""
try
:
endpoint
=
"ec2.
%
s.amazonaws.com"
%
self
.
region
connect_region
=
RegionInfo
(
name
=
self
.
region
,
endpoint
=
endpoint
)
ec2_conn
=
boto
.
ec2
.
EC2Connection
(
self
.
aws_access_key
,
self
.
aws_secret_key
,
region
=
connect_region
)
except
boto
.
exception
.
NoAuthHandlerFound
,
e
:
self
.
module
.
fail_json
(
msg
=
str
(
e
))
return
ec2_conn
.
get_only_instances
(
instance_ids
=
[
self
.
instance_id
])[
0
]
def
main
():
module
=
AnsibleModule
(
argument_spec
=
dict
(
state
=
{
'required'
:
True
,
'choices'
:
[
'present'
,
'absent'
]},
instance_id
=
{
'required'
:
True
},
ec2_elbs
=
{
'default'
:
None
,
'required'
:
False
,
'type'
:
'list'
},
ec2_secret_key
=
{
'default'
:
None
,
'aliases'
:
[
'aws_secret_key'
,
'secret_key'
],
'no_log'
:
True
},
ec2_access_key
=
{
'default'
:
None
,
'aliases'
:
[
'aws_access_key'
,
'access_key'
]},
region
=
{
'default'
:
None
,
'required'
:
False
,
'aliases'
:[
'aws_region'
,
'ec2_region'
]},
enable_availability_zone
=
{
'default'
:
True
,
'required'
:
False
,
'choices'
:
BOOLEANS
,
'type'
:
'bool'
},
wait
=
{
'required'
:
False
,
'choices'
:
BOOLEANS
,
'default'
:
True
,
'type'
:
'bool'
}
)
)
# def get_ec2_creds(module):
# return ec2_url, ec2_access_key, ec2_secret_key, region
ec2_url
,
aws_access_key
,
aws_secret_key
,
region
=
get_ec2_creds
(
module
)
ec2_elbs
=
module
.
params
[
'ec2_elbs'
]
region
=
module
.
params
[
'region'
]
wait
=
module
.
params
[
'wait'
]
enable_availability_zone
=
module
.
params
[
'enable_availability_zone'
]
if
module
.
params
[
'state'
]
==
'present'
and
'ec2_elbs'
not
in
module
.
params
:
module
.
fail_json
(
msg
=
"ELBs are required for registration"
)
instance_id
=
module
.
params
[
'instance_id'
]
elb_man
=
ElbManager
(
module
,
instance_id
,
ec2_elbs
,
aws_access_key
,
aws_secret_key
,
region
=
region
)
if
ec2_elbs
is
not
None
:
for
elb
in
ec2_elbs
:
if
not
elb_man
.
exists
(
elb
):
msg
=
"ELB
%
s does not exist"
%
elb
module
.
fail_json
(
msg
=
msg
)
if
module
.
params
[
'state'
]
==
'present'
:
elb_man
.
register
(
wait
,
enable_availability_zone
)
elif
module
.
params
[
'state'
]
==
'absent'
:
elb_man
.
deregister
(
wait
)
ansible_facts
=
{
'ec2_elbs'
:
[
lb
.
name
for
lb
in
elb_man
.
lbs
]}
ec2_facts_result
=
dict
(
changed
=
elb_man
.
changed
,
ansible_facts
=
ansible_facts
)
module
.
exit_json
(
**
ec2_facts_result
)
# import module snippets
from
ansible.module_utils.basic
import
*
from
ansible.module_utils.ec2
import
*
main
()
Write
Preview
Markdown
is supported
0%
Try again
or
attach a new file
Attach a file
Cancel
You are about to add
0
people
to the discussion. Proceed with caution.
Finish editing this message first!
Cancel
Please
register
or
sign in
to comment