This repository was archived by the owner on May 13, 2024. It is now read-only.
-
Notifications
You must be signed in to change notification settings - Fork 1.4k
/
Copy pathcompute_page.py
145 lines (108 loc) · 4.66 KB
/
compute_page.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
# Copyright 2015 Google Inc. All Rights Reserved.
"""Compute page for handling tasks related to compute engine."""
import logging
import apiauth
import webapp2
from google.appengine.api import app_identity
from google.appengine.api import taskqueue
# Page actions
# Get the status of an instance.
ACTION_STATUS = 'status'
# Start the compute instance if it is not already started. When an
# instance is in the TERMINATED state, it will be started. If the
# instance is already RUNNING do nothing. If the instance is in an
# intermediate state--PROVISIONING, STAGING, STOPPING--the task is
# requeued.
ACTION_START = 'start'
# Stop the compute instance if it is RUNNING. Then queue a task to start it. Do
# nothing if the instance is not RUNNING.
ACTION_RESTART = 'restart'
# Constants for the Compute Engine API
COMPUTE_STATUS = 'status'
COMPUTE_STATUS_TERMINATED = 'TERMINATED'
COMPUTE_STATUS_RUNNING = 'RUNNING'
# Seconds between API retries.
START_TASK_MIN_WAIT_S = 3
COMPUTE_API_URL = 'https://www.googleapis.com/auth/compute'
def enqueue_start_task(instance, zone):
taskqueue.add(url='/compute/%s/%s/%s' % (ACTION_START, instance, zone),
countdown=START_TASK_MIN_WAIT_S)
def enqueue_restart_task(instance, zone):
taskqueue.add(url='/compute/%s/%s/%s' % (ACTION_RESTART, instance, zone),
countdown=START_TASK_MIN_WAIT_S)
class ComputePage(webapp2.RequestHandler):
"""Page to handle requests against GCE."""
def __init__(self, request, response):
# Call initialize rather than the parent constructor fun. See:
# https://webapp-improved.appspot.com/guide/handlers.html#overriding-init
self.initialize(request, response)
self.compute_service = self._build_compute_service()
if self.compute_service is None:
logging.warning('Unable to create Compute service object.')
def _build_compute_service(self):
return apiauth.build(scope=COMPUTE_API_URL,
service_name='compute',
version='v1')
def _maybe_restart_instance(self, instance, zone):
"""Implementation for restart action.
Args:
instance: Name of the instance to restart.
zone: Name of the zone the instance belongs to.
"""
if self.compute_service is None:
logging.warning('Compute service unavailable.')
return
status = self._compute_status(instance, zone)
logging.info('GCE VM \'%s (%s)\' status: \'%s\'.',
instance, zone, status)
# Do nothing if the status is not RUNNING to avoid race. This will cover
# most of the cases.
if status == COMPUTE_STATUS_RUNNING:
logging.info('Stopping GCE VM: %s (%s)', instance, zone)
self.compute_service.instances().stop(
project=app_identity.get_application_id(),
instance=instance,
zone=zone).execute()
enqueue_start_task(instance, zone)
def _maybe_start_instance(self, instance, zone):
"""Implementation for start action.
Args:
instance: Name of the instance to start.
zone: Name of the zone the instance belongs to.
"""
if self.compute_service is None:
logging.warning('Unable to start Compute instance, service unavailable.')
return
status = self._compute_status(instance, zone)
logging.info('GCE VM \'%s (%s)\' status: \'%s\'.',
instance, zone, status)
if status == COMPUTE_STATUS_TERMINATED:
logging.info('Starting GCE VM: %s (%s)', instance, zone)
self.compute_service.instances().start(
project=app_identity.get_application_id(),
instance=instance,
zone=zone).execute()
if status != COMPUTE_STATUS_RUNNING:
# If in an intermediate state: PROVISIONING, STAGING, STOPPING, requeue
# the task to check back later. If in TERMINATED state, also requeue the
# task since the start attempt may fail and we should retry.
enqueue_start_task(instance, zone)
def _compute_status(self, instance, zone):
"""Return the status of the compute instance."""
if self.compute_service is None:
logging.warning('Service unavailable: unable to start GCE VM: %s (%s)',
instance, zone)
return
info = self.compute_service.instances().get(
project=app_identity.get_application_id(),
instance=instance,
zone=zone).execute()
return info[COMPUTE_STATUS]
def get(self, action, instance, zone):
if action == ACTION_STATUS:
self.response.write(self._compute_status(instance, zone))
def post(self, action, instance, zone):
if action == ACTION_START:
self._maybe_start_instance(instance, zone)
elif action == ACTION_RESTART:
self._maybe_restart_instance(instance, zone)