summaryrefslogtreecommitdiffstats
path: root/laas-fog/pharoslaas/sensors
diff options
context:
space:
mode:
authorParker Berberian <pberberian@iol.unh.edu>2017-12-20 12:55:28 -0500
committerParker Berberian <pberberian@iol.unh.edu>2017-12-20 12:55:28 -0500
commitfc52e89492c20ca7a93d4e48f0cfa01ddacf7fa8 (patch)
tree1102bbb4e7e7a40711f9e2600516fe4b97e773a0 /laas-fog/pharoslaas/sensors
parent30f389c70e8a0a8bd2ef27be09839eef243ab7f5 (diff)
Added sensor and rules
JIRA: PHAROS-318 Adds pharos.py which talks with the pharos dashboard to detect when bookings are created and stores the needed information in the stackstorm datastore. By default, the sensor will poll the dashboard every 30 seconds to check for changes. The dashboard sensor will dispatch the triggers defined in dashboard_listener.yaml. The rules in rules/* links the triggers thrown by the api sensor to workflows which will do the work of deploying / cleaning the hosts. Change-Id: I7411a16ebbb48739a8f1f5b924dea6493c400071 Signed-off-by: Parker Berberian <pberberian@iol.unh.edu>
Diffstat (limited to 'laas-fog/pharoslaas/sensors')
-rw-r--r--laas-fog/pharoslaas/sensors/dashboard_listener.yaml45
-rwxr-xr-xlaas-fog/pharoslaas/sensors/pharos.py238
2 files changed, 283 insertions, 0 deletions
diff --git a/laas-fog/pharoslaas/sensors/dashboard_listener.yaml b/laas-fog/pharoslaas/sensors/dashboard_listener.yaml
new file mode 100644
index 0000000..31c215f
--- /dev/null
+++ b/laas-fog/pharoslaas/sensors/dashboard_listener.yaml
@@ -0,0 +1,45 @@
+---
+##############################################################################
+# Copyright 2017 Parker Berberian and Others #
+# #
+# Licensed under the Apache License, Version 2.0 (the "License"); #
+# you may not use this file except in compliance with the License. #
+# You may obtain a copy of the License at #
+# #
+# http://www.apache.org/licenses/LICENSE-2.0 #
+# #
+# Unless required by applicable law or agreed to in writing, software #
+# distributed under the License is distributed on an "AS IS" BASIS, #
+# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. #
+# See the License for the specific language governing permissions and #
+# limitations under the License. #
+##############################################################################
+
+class_name: "Pharos_api"
+entry_point: "pharos.py"
+description: "polls the dashboard api for deployments"
+poll_interval: 30
+trigger_types:
+ -
+ name: "start_deployment_trigger"
+ descrition: "a simple deployment trigger"
+ payload_schema:
+ type: "object"
+ properties:
+ host:
+ type: "string"
+ installer:
+ type: "string"
+ scenario:
+ type: "string"
+ booking:
+ type: "string"
+
+ -
+ name: "end_deployment_trigger"
+ description: "marks the end of a booking"
+ payload_schema:
+ host:
+ type: "string"
+ key:
+ type: "string"
diff --git a/laas-fog/pharoslaas/sensors/pharos.py b/laas-fog/pharoslaas/sensors/pharos.py
new file mode 100755
index 0000000..a11160c
--- /dev/null
+++ b/laas-fog/pharoslaas/sensors/pharos.py
@@ -0,0 +1,238 @@
+##############################################################################
+# Copyright 2017 Parker Berberian and Others #
+# #
+# Licensed under the Apache License, Version 2.0 (the "License"); #
+# you may not use this file except in compliance with the License. #
+# You may obtain a copy of the License at #
+# #
+# http://www.apache.org/licenses/LICENSE-2.0 #
+# #
+# Unless required by applicable law or agreed to in writing, software #
+# distributed under the License is distributed on an "AS IS" BASIS, #
+# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. #
+# See the License for the specific language governing permissions and #
+# limitations under the License. #
+##############################################################################
+
+import requests
+import time
+import calendar
+import json
+from st2reactor.sensor.base import PollingSensor
+
+
+class Pharos_api(PollingSensor):
+ """
+ This class listens to the dashboard and starts/stops bookings accordingly.
+ """
+
+ def getBookingList(self):
+ return json.loads(
+ self.sensor_service.get_value(name='bookings', local=False)
+ )
+
+ def updateBookingList(self, blist):
+ self.sensor_service.set_value(
+ name='bookings',
+ value=json.dumps(blist),
+ local=False
+ )
+
+ def AddBooking(self, new_booking):
+ """
+ checks if booking is in the database, and adds it if it isnt
+ """
+ # first, check if booking is already expired
+ if time.time() > new_booking['end']:
+ return
+ # check if booking already in db
+ booking_list = self.getBookingList()
+ if new_booking['id'] in booking_list:
+ return
+ new_booking['status'] = 0 # add status code
+ booking_list.append(new_booking['id'])
+ name = "booking_" + str(new_booking['id'])
+ self.sensor_service.set_value(
+ name=name,
+ value=json.dumps(new_booking),
+ local=False
+ )
+ self.updateBookingList(booking_list)
+
+ def convertTimes(self, booking):
+ """
+ this method will take the time reported by Pharos in the
+ format yyyy-mm-ddThh:mm:ssZ
+ and convert it into seconds since the epoch,
+ for easier management
+ """
+ booking['start'] = self.pharosToEpoch(booking['start'])
+ booking['end'] = self.pharosToEpoch(booking['end'])
+
+ def pharosToEpoch(self, timeStr):
+ """
+ Converts the dates from the dashboard to epoch time.
+ """
+ time_struct = time.strptime(timeStr, '%Y-%m-%dT%H:%M:%SZ')
+ epoch_time = calendar.timegm(time_struct)
+ return epoch_time
+
+ def checkBookings(self):
+ """
+ This method checks all the bookings in our database to see if any
+ action is required.
+ """
+
+ # get all active bookings from database into a usable form
+ booking_list = self.getBookingList()
+ for booking_id in booking_list:
+ booking = self.getBooking(booking_id)
+ # first, check if booking is over
+ if time.time() > booking['end']:
+ self.log.info("ending the booking with id %i", booking_id)
+ self.endBooking(booking)
+ # Then check if booking has begun and the host is still idle
+ elif time.time() > booking['start'] and booking['status'] < 1:
+ self.log.info("starting the booking with id %i", booking['id'])
+ self.startBooking(booking)
+
+ def startBooking(self, booking):
+ """
+ Starts the scheduled booking on the requested host with
+ the correct config file.
+ The provisioning process gets spun up in a subproccess,
+ so the api listener is not interupted.
+ """
+ host = self.getServer(pharos_id=booking['resource_id'])['hostname']
+ self.log.info("Detected a new booking started for host %s", host)
+ self.setBookingStatus(booking['id'], 1) # mark booking started
+ # dispatch trigger into system
+ trigger = "pharoslaas.start_deployment_trigger"
+ payload = {"host": host, "installer": booking['installer_name']}
+ payload['scenario'] = booking['scenario_name']
+ payload['booking'] = booking['id']
+ self.sensor_service.dispatch(trigger=trigger, payload=payload)
+
+ def endBooking(self, booking):
+ """
+ Resets a host once its booking has ended.
+ """
+ host = self.getServer(pharos_id=booking['resource_id'])['hostname']
+ self.log.info('Lease expired. Resetting host %s', host)
+ self.setBookingStatus(booking['id'], 3)
+ self.removeBooking(booking['id'])
+ # dispatch trigger to clean
+ host = self.getServer(pharos_id=booking['resource_id'])['hostname']
+ trigger = "pharoslaas.end_deployment_trigger"
+ payload = {"host": host, "booking": booking['id']}
+ if 'vpn_key' in booking.keys():
+ payload['key'] = booking['vpn_key']
+ else:
+ payload['key'] = ''
+ self.sensor_service.dispatch(trigger=trigger, payload=payload)
+
+ def getServer(self, fog_name=None, hostname=None, pharos_id=None):
+ key = ""
+ value = ""
+ if fog_name is not None:
+ key = "fog_name"
+ value = fog_name
+ elif hostname is not None:
+ key = "hostname"
+ value = hostname
+ elif pharos_id is not None:
+ key = "pharos_id"
+ value = pharos_id
+ for server in self.servers:
+ if server[key] == value:
+ return server
+
+ def getBooking(self, booking_id):
+ name = "booking_" + str(booking_id)
+ return json.loads(
+ self.sensor_service.get_value(
+ name=name,
+ local=False
+ )
+ )
+
+ def setBookingStatus(self, booking_id, status):
+ booking = self.getBooking(booking_id)
+ booking['status'] = status
+ name = "booking_" + str(booking_id)
+ self.sensor_service.set_value(
+ name=name,
+ value=json.dumps(booking),
+ local=False
+ )
+
+ def removeBooking(self, booking_id):
+ blist = self.getBookingList()
+ blist.remove(booking_id)
+ self.updateBookingList(blist)
+ name = "booking_" + str(booking_id)
+ self.sensor_service.delete_value(name=name, local=False)
+
+ # Sensor Interface Methods #
+
+ def setup(self):
+ """
+ This method is called by stackstorm once to setup this polling sensor.
+ Basically __init__, assigns instance variables, etc
+ """
+ server_names = json.loads(
+ self.sensor_service.get_value('hosts', local=False)
+ )
+ self.servers = []
+ for server in server_names:
+ self.servers.append(
+ json.loads(
+ self.sensor_service.get_value(server, local=False)
+ )
+ )
+ self.resource_ids = []
+ for host in self.servers:
+ self.resource_ids.append(host['pharos_id'])
+ self.log = self.sensor_service.get_logger(name=self.__class__.__name__)
+ self.dashboard = self.sensor_service.get_value(
+ name='dashboard_url',
+ local=False
+ )
+ self.log.info("connecting to dashboard at %s", self.dashboard)
+ # get token here for when dashboard supports it
+
+ def poll(self):
+ """
+ this method will continuously poll the pharos dashboard.
+ If a booking is found on our server,
+ we will start a deployment in the background with the
+ proper config file for the requested
+ installer and scenario.
+ """
+ self.log.debug("%s", "Beginning polling of dashboard")
+ try:
+ url = self.dashboard+"/api/bookings/"
+ bookings = requests.get(url).json()
+ for booking in bookings:
+ if booking['resource_id'] in self.resource_ids:
+ self.convertTimes(booking)
+ self.AddBooking(booking)
+ self.checkBookings()
+ except Exception:
+ self.log.exception('%s', "failed to connect to dashboard")
+
+ def cleanup(self):
+ # called when st2 goes down
+ pass
+
+ def add_trigger(self, trigger):
+ # called when trigger is created
+ pass
+
+ def update_trigger(self, trigger):
+ # called when trigger is updated
+ pass
+
+ def remove_trigger(self, trigger):
+ # called when trigger is deleted
+ pass