-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathcleanup_deployment.py
executable file
·133 lines (104 loc) · 4.08 KB
/
cleanup_deployment.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
#!/usr/bin/env python
import botocore
from boto3.session import Session
def get_stack_outputvars(stack, ec2):
return {v['OutputKey']: v['OutputValue'] for v in stack.outputs}
def get_stack(stackName, cff):
try:
stt = list(cff.stacks.filter(StackName=stackName))
return stt[0]
except botocore.exceptions.ClientError as e:
if "{} does not exist".format(stackName) in\
e.response['Error']['Message']:
return None
else:
raise
def get_args():
import argparse
argp = argparse.ArgumentParser()
argp.add_argument('--profile')
argp.add_argument('--yes', default=False, action='store_true')
argp.add_argument('--remove-stack', default=False, action='store_true')
argp.add_argument('--stack-name', required=True)
argp.add_argument('--region', default='us-east-1')
return argp
def main(argv):
args = get_args().parse_args(argv)
session = Session(profile_name=args.profile, region_name=args.region)
ec2 = session.resource("ec2")
s3 = session.resource("s3")
cff = session.resource("cloudformation")
try:
stack = get_stack(args.stack_name, cff)
if stack is None:
print "Stack in no longer active"
return 0
stack_vars = get_stack_outputvars(stack, ec2)
cleanup_ec2(stack, stack_vars, ec2, args.yes, exclude_nat=True)
cleanup_s3(stack, stack_vars, s3, args.yes)
if args.remove_stack is True:
print "Removing stack"
if confirm_oprn("Stack "+args.stack_name, 1, args.yes) is False:
return -1
cleanup_ec2(stack, stack_vars, ec2, args.yes, exclude_nat=False)
stack.delete()
else:
print "Stack was not deleted."
print "Re-run the same command with --remove-stack"
except botocore.exceptions.NoCredentialsError as ex:
print ex
print "Missing ~/.aws/credentials directory?"
print "http://boto3.readthedocs.org/en/latest/guide/configuration.html"
return -1
return 0
def cleanup_ec2(stack, stack_vars, ec2, yes, exclude_nat=True):
def getname(iii):
return next((v['Value'] for v in iii.tags if v['Key'] == 'Name'), "")
selector = lambda x: True
if exclude_nat is True:
selector = lambda x: getname(x) != 'NAT Instance'
insts = [ii for ii in ec2.instances.filter(
Filters=[{'Name': 'vpc-id', 'Values': [stack_vars['PcfVpc']]},
{'Name': 'instance-state-name',
'Values': ['running', 'stopping', 'stopped']}])
if selector(ii)]
if confirm_oprn("ec2 instances", len(insts), yes) is False:
return -1
ec2.meta.client.terminate_instances(
InstanceIds=[ii.id for ii in insts])
for ii in insts:
print "Waiting for instance {} to terminate".format(getname(ii))
ii.wait_until_terminated()
return 0
def confirm_oprn(resource_type, resource_number, yes):
print "Removing {} {}".format(resource_number, resource_type)
if resource_number == 0:
return False
if yes is True:
return True
confirm = raw_input("This cannot be recovered. Proceed? YES/NO ")
if confirm.lower() != 'yes':
print "Skipping"
return False
return True
def cleanup_s3(stack, stack_vars, s3, yes):
delkeys = [
'PcfOpsManagerS3Bucket', 'PcfElasticRuntimeS3BuildpacksBucket',
'PcfElasticRuntimeS3DropletsBucket',
'PcfElasticRuntimeS3PackagesBucket',
'PcfElasticRuntimeS3ResourcesBucket']
bset = set([stack_vars[k] for k in delkeys if k in stack_vars])
stackbuckets = [bk for bk in s3.buckets.all() if bk.name in bset]
if confirm_oprn("s3 buckets", len(stackbuckets), yes) is False:
return -1
for bk in stackbuckets:
print "Emptying {}".format(bk.name)
try:
bk.objects.delete()
except botocore.exceptions.ClientError as ce:
if ce.response['Error']['Code'] != '404':
print ce
return 0
if __name__ == "__main__":
import sys
sys.exit(main(sys.argv[1:]))