-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy path__init__.py
executable file
·179 lines (146 loc) · 6.07 KB
/
__init__.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
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
#!flaskapi/bin/python
#importing the basic libraries
from flask import Flask, jsonify, make_response, request, abort
from flask_basicauth import BasicAuth
import boto3
from botocore.exceptions import ClientError
# Creating an app for flask
app = Flask(__name__)
app.config['BASIC_AUTH_USERNAME'] = 'user'
app.config['BASIC_AUTH_PASSWORD'] = 'test'
# Configuring session settings
session = boto3.setup_default_session(region_name='replaceRegion',aws_access_key_id='replaceKey',aws_secret_access_key='replaceSecret')
elb_client = boto3.client('elb')
ec2_client = boto3.client('ec2')
basic_auth = BasicAuth(app)
# Begin custom error messages functions
# Return wrong page or no access error
@app.errorhandler(404)
def wrongEnd(error):
return make_response(jsonify({'error': 'Wrong page or access denied'}), 404)
# Return 500 in case instanceId parameter is not written in camelCase
@app.errorhandler(500)
def wrongEnd(error):
return make_response(jsonify({'error': 'Wrong data format - please use camelCase for instanceId'}), 500)
# Return ELB does not exist error
def wrongEnd(error):
return make_response(jsonify({'error': 'The ELB does not exist'}), 404)
# Return error on malformed json request on attaching and dettaching instances
@app.errorhandler(400)
def wrongData(error):
return make_response(jsonify({'error': 'Wrong data format'}), 400)
# Return instance already attached to loadbalancer error
def instanceAttached(error):
return make_response(jsonify({'error': 'Instance is already on LoadBalancer'}), 409)
# Return instance not attached to loadbalancer error
def instanceNotAttached(error):
return make_response(jsonify({'error': 'Instance is not on LoadBalancer'}), 409)
# End custom error messages functions
# API health check function
@app.route('/healthcheck', methods=['GET'])
@basic_auth.required
#this function would be executed whenever the /healthcheck page would be access on the browser
def get_health():
return jsonify({'Status': 'The service is up!'})
# Get MachineInfo from EC2 instances
def getInstanceData(instanceid):
instance_data = []
reservations = ec2_client.describe_instances(Filters=[{'Name' : 'instance-id', 'Values' : [instanceid]}])
for reservation in reservations['Reservations']:
for instance in reservation['Instances']:
instance_data.append(
{
'instanceId': instance['InstanceId'],
'instanceType': instance['InstanceType'],
'launchDate': instance['LaunchTime'].strftime('%Y-%m-%dT%H:%M:%S.%f')
}
)
return instance_data
# Get all instances in Reservation
def getAllInstances(reservations):
all_instances = []
for reservation in reservations:
for instance in reservation['Instances']:
all_instances.append(
{
'instanceId': instance['InstanceId'],
'instanceType': instance['InstanceType'],
'launchDate': instance['LaunchTime'].strftime('%Y-%m-%dT%H:%M:%S.%f')
}
)
return all_instances
# Get the all the ids of instances attached to elb
def getAllInstanceIDs(elbs):
elb_instances_ids = []
for elb in elbs:
instances = elb['Instances']
for instance in instances:
elb_instances_ids.append(instance['InstanceId'])
return elb_instances_ids
# Get elb instances IDS from the load balancer name
def getelbInstanceIDs(elb_name):
elbs = elb_client.describe_load_balancers(LoadBalancerNames=[elb_name])['LoadBalancerDescriptions']
return sum(list(map(lambda elb: list(map(lambda i: i['InstanceId'], elb['Instances'])), elbs)), [])
# Get HTTP method
def httpGETmethod(elb_name):
try:
elbs = elb_client.describe_load_balancers(
LoadBalancerNames=[
elb_name
]
)['LoadBalancerDescriptions']
except ClientError as e:
if e.response['Error']['Code'] == 'LoadBalancerNotFound':
abort(wrongEnd(404))
else:
elb_instances_ids = getAllInstanceIDs(elbs)
reservations = ec2_client.describe_instances(
InstanceIds=elb_instances_ids
)['Reservations']
all_instances = getAllInstances(reservations)
return jsonify({'Machines' : all_instances})
# Post HTTP method
def httpPOSTmethod(elb_name, instanceid):
try:
elb_instances_ids = getelbInstanceIDs(elb_name)
if instanceid in elb_instances_ids:
abort(instanceAttached(409))
else:
response = elb_client.register_instances_with_load_balancer(
Instances=[{'InstanceId': instanceid,},],
LoadBalancerName= elb_name,
)
return jsonify({'instance added' : getInstanceData(instanceid)})
except ClientError as e:
if e.response ['Error']['Code'] in 'InvalidInstanceID':
abort(400)
def httpDELETEmethod(elb_name, instanceid):
try:
elb_instances_ids = getelbInstanceIDs(elb_name)
if instanceid not in elb_instances_ids:
abort(instanceNotAttached(409))
else:
response = elb_client.deregister_instances_from_load_balancer(
Instances=[{'InstanceId': instanceid,},],
LoadBalancerName= elb_name,
)
return jsonify({'instance removed' : getInstanceData(instanceid)})
except ClientError as e:
if e.response ['Error']['Code'] in 'InvalidInstanceID':
abort(400)
# This method is executed whenever /elb/{elb_name} is executed.
@app.route('/elb/<elb_name>', methods=['GET', 'POST', 'DELETE'])
@basic_auth.required
def elb_methods(elb_name):
assert elb_name == request.view_args['elb_name']
if request.method == 'GET':
return httpGETmethod(elb_name)
else:
instanceid = request.json['instanceId']
if request.method == 'POST':
return httpPOSTmethod(elb_name, instanceid)
if request.method == 'DELETE':
return httpDELETEmethod(elb_name, instanceid)
# Main body that is executed when this function is called
if __name__ == '__main__':
app.run(debug=True)