|
|
6a3e49 |
From 59ae9d00060da5329d7ca538974498292bbe1d91 Mon Sep 17 00:00:00 2001
|
|
|
6a3e49 |
From: Helen Koike <helen.koike@collabora.com>
|
|
|
6a3e49 |
Date: Tue, 26 Jun 2018 10:18:29 -0300
|
|
|
6a3e49 |
Subject: [PATCH 1/7] fence_gce: add support for stackdriver logging
|
|
|
6a3e49 |
|
|
|
6a3e49 |
Add --logging option to enable sending logs to google stackdriver
|
|
|
6a3e49 |
---
|
|
|
6a3e49 |
agents/gce/fence_gce.py | 65 +++++++++++++++++++++++++++++++++++++--
|
|
|
6a3e49 |
tests/data/metadata/fence_gce.xml | 5 +++
|
|
|
6a3e49 |
2 files changed, 67 insertions(+), 3 deletions(-)
|
|
|
6a3e49 |
|
|
|
6a3e49 |
diff --git a/agents/gce/fence_gce.py b/agents/gce/fence_gce.py
|
|
|
6a3e49 |
index 3abb5207..3af5bfc8 100644
|
|
|
6a3e49 |
--- a/agents/gce/fence_gce.py
|
|
|
6a3e49 |
+++ b/agents/gce/fence_gce.py
|
|
|
6a3e49 |
@@ -1,12 +1,19 @@
|
|
|
6a3e49 |
#!@PYTHON@ -tt
|
|
|
6a3e49 |
|
|
|
6a3e49 |
import atexit
|
|
|
6a3e49 |
+import logging
|
|
|
6a3e49 |
+import platform
|
|
|
6a3e49 |
import sys
|
|
|
6a3e49 |
+import time
|
|
|
6a3e49 |
sys.path.append("@FENCEAGENTSLIBDIR@")
|
|
|
6a3e49 |
|
|
|
6a3e49 |
import googleapiclient.discovery
|
|
|
6a3e49 |
from fencing import fail_usage, run_delay, all_opt, atexit_handler, check_input, process_input, show_docs, fence_action
|
|
|
6a3e49 |
|
|
|
6a3e49 |
+
|
|
|
6a3e49 |
+LOGGER = logging
|
|
|
6a3e49 |
+
|
|
|
6a3e49 |
+
|
|
|
6a3e49 |
def translate_status(instance_status):
|
|
|
6a3e49 |
"Returns on | off | unknown."
|
|
|
6a3e49 |
if instance_status == "RUNNING":
|
|
|
6a3e49 |
@@ -27,6 +34,7 @@ def get_nodes_list(conn, options):
|
|
|
6a3e49 |
|
|
|
6a3e49 |
return result
|
|
|
6a3e49 |
|
|
|
6a3e49 |
+
|
|
|
6a3e49 |
def get_power_status(conn, options):
|
|
|
6a3e49 |
try:
|
|
|
6a3e49 |
instance = conn.instances().get(
|
|
|
6a3e49 |
@@ -38,18 +46,37 @@ def get_power_status(conn, options):
|
|
|
6a3e49 |
fail_usage("Failed: get_power_status: {}".format(str(err)))
|
|
|
6a3e49 |
|
|
|
6a3e49 |
|
|
|
6a3e49 |
+def wait_for_operation(conn, project, zone, operation):
|
|
|
6a3e49 |
+ while True:
|
|
|
6a3e49 |
+ result = conn.zoneOperations().get(
|
|
|
6a3e49 |
+ project=project,
|
|
|
6a3e49 |
+ zone=zone,
|
|
|
6a3e49 |
+ operation=operation['name']).execute()
|
|
|
6a3e49 |
+ if result['status'] == 'DONE':
|
|
|
6a3e49 |
+ if 'error' in result:
|
|
|
6a3e49 |
+ raise Exception(result['error'])
|
|
|
6a3e49 |
+ return
|
|
|
6a3e49 |
+ time.sleep(1)
|
|
|
6a3e49 |
+
|
|
|
6a3e49 |
+
|
|
|
6a3e49 |
def set_power_status(conn, options):
|
|
|
6a3e49 |
try:
|
|
|
6a3e49 |
if options["--action"] == "off":
|
|
|
6a3e49 |
- conn.instances().stop(
|
|
|
6a3e49 |
+ LOGGER.info("Issuing poweroff of %s in zone %s" % (options["--plug"], options["--zone"]))
|
|
|
6a3e49 |
+ operation = conn.instances().stop(
|
|
|
6a3e49 |
project=options["--project"],
|
|
|
6a3e49 |
zone=options["--zone"],
|
|
|
6a3e49 |
instance=options["--plug"]).execute()
|
|
|
6a3e49 |
+ wait_for_operation(conn, options["--project"], options["--zone"], operation)
|
|
|
6a3e49 |
+ LOGGER.info("Poweroff of %s in zone %s complete" % (options["--plug"], options["--zone"]))
|
|
|
6a3e49 |
elif options["--action"] == "on":
|
|
|
6a3e49 |
- conn.instances().start(
|
|
|
6a3e49 |
+ LOGGER.info("Issuing poweron of %s in zone %s" % (options["--plug"], options["--zone"]))
|
|
|
6a3e49 |
+ operation = conn.instances().start(
|
|
|
6a3e49 |
project=options["--project"],
|
|
|
6a3e49 |
zone=options["--zone"],
|
|
|
6a3e49 |
instance=options["--plug"]).execute()
|
|
|
6a3e49 |
+ wait_for_operation(conn, options["--project"], options["--zone"], operation)
|
|
|
6a3e49 |
+ LOGGER.info("Poweron of %s in zone %s complete" % (options["--plug"], options["--zone"]))
|
|
|
6a3e49 |
except Exception as err:
|
|
|
6a3e49 |
fail_usage("Failed: set_power_status: {}".format(str(err)))
|
|
|
6a3e49 |
|
|
|
6a3e49 |
@@ -71,11 +98,24 @@ def define_new_opts():
|
|
|
6a3e49 |
"required" : "1",
|
|
|
6a3e49 |
"order" : 3
|
|
|
6a3e49 |
}
|
|
|
6a3e49 |
+ all_opt["logging"] = {
|
|
|
6a3e49 |
+ "getopt" : ":",
|
|
|
6a3e49 |
+ "longopt" : "logging",
|
|
|
6a3e49 |
+ "help" : "--logging=[bool] Logging, true/false",
|
|
|
6a3e49 |
+ "shortdesc" : "Stackdriver-logging support.",
|
|
|
6a3e49 |
+ "longdesc" : "If enabled (set to true), IP failover logs will be posted to stackdriver logging.",
|
|
|
6a3e49 |
+ "required" : "0",
|
|
|
6a3e49 |
+ "default" : "false",
|
|
|
6a3e49 |
+ "order" : 4
|
|
|
6a3e49 |
+ }
|
|
|
6a3e49 |
|
|
|
6a3e49 |
def main():
|
|
|
6a3e49 |
conn = None
|
|
|
6a3e49 |
+ global LOGGER
|
|
|
6a3e49 |
+
|
|
|
6a3e49 |
+ hostname = platform.node()
|
|
|
6a3e49 |
|
|
|
6a3e49 |
- device_opt = ["port", "no_password", "zone", "project"]
|
|
|
6a3e49 |
+ device_opt = ["port", "no_password", "zone", "project", "logging"]
|
|
|
6a3e49 |
|
|
|
6a3e49 |
atexit.register(atexit_handler)
|
|
|
6a3e49 |
|
|
|
6a3e49 |
@@ -97,6 +137,25 @@ def main():
|
|
|
6a3e49 |
|
|
|
6a3e49 |
run_delay(options)
|
|
|
6a3e49 |
|
|
|
6a3e49 |
+ # Prepare logging
|
|
|
6a3e49 |
+ logging_env = options.get('--logging')
|
|
|
6a3e49 |
+ if logging_env:
|
|
|
6a3e49 |
+ logging_env = logging_env.lower()
|
|
|
6a3e49 |
+ if any(x in logging_env for x in ['yes', 'true', 'enabled']):
|
|
|
6a3e49 |
+ try:
|
|
|
6a3e49 |
+ import google.cloud.logging.handlers
|
|
|
6a3e49 |
+ client = google.cloud.logging.Client()
|
|
|
6a3e49 |
+ handler = google.cloud.logging.handlers.CloudLoggingHandler(client, name=hostname)
|
|
|
6a3e49 |
+ formatter = logging.Formatter('gcp:stonish "%(message)s"')
|
|
|
6a3e49 |
+ LOGGER = logging.getLogger(hostname)
|
|
|
6a3e49 |
+ handler.setFormatter(formatter)
|
|
|
6a3e49 |
+ LOGGER.addHandler(handler)
|
|
|
6a3e49 |
+ LOGGER.setLevel(logging.INFO)
|
|
|
6a3e49 |
+ except ImportError:
|
|
|
6a3e49 |
+ LOGGER.error('Couldn\'t import google.cloud.logging, '
|
|
|
6a3e49 |
+ 'disabling Stackdriver-logging support')
|
|
|
6a3e49 |
+
|
|
|
6a3e49 |
+ # Prepare cli
|
|
|
6a3e49 |
try:
|
|
|
6a3e49 |
credentials = None
|
|
|
6a3e49 |
if tuple(googleapiclient.__version__) < tuple("1.6.0"):
|
|
|
6a3e49 |
diff --git a/tests/data/metadata/fence_gce.xml b/tests/data/metadata/fence_gce.xml
|
|
|
6a3e49 |
index 2a147f21..805ecc6b 100644
|
|
|
6a3e49 |
--- a/tests/data/metadata/fence_gce.xml
|
|
|
6a3e49 |
+++ b/tests/data/metadata/fence_gce.xml
|
|
|
6a3e49 |
@@ -30,6 +30,11 @@ For instructions see: https://cloud.google.com/compute/docs/tutorials/python-gui
|
|
|
6a3e49 |
<content type="string" />
|
|
|
6a3e49 |
<shortdesc lang="en">Project ID.</shortdesc>
|
|
|
6a3e49 |
</parameter>
|
|
|
6a3e49 |
+ <parameter name="logging" unique="0" required="0">
|
|
|
6a3e49 |
+ <getopt mixed="--logging=[bool]" />
|
|
|
6a3e49 |
+ <content type="string" default="false" />
|
|
|
6a3e49 |
+ <shortdesc lang="en">Stackdriver-logging support.</shortdesc>
|
|
|
6a3e49 |
+ </parameter>
|
|
|
6a3e49 |
<parameter name="quiet" unique="0" required="0">
|
|
|
6a3e49 |
<getopt mixed="-q, --quiet" />
|
|
|
6a3e49 |
<content type="boolean" />
|
|
|
6a3e49 |
|
|
|
6a3e49 |
From bb34acd8b0b150599c393d56dd81a7d8185b27d3 Mon Sep 17 00:00:00 2001
|
|
|
6a3e49 |
From: Helen Koike <helen.koike@collabora.com>
|
|
|
6a3e49 |
Date: Tue, 26 Jun 2018 10:44:41 -0300
|
|
|
6a3e49 |
Subject: [PATCH 2/7] fence_gce: set project and zone as not required
|
|
|
6a3e49 |
|
|
|
6a3e49 |
Try to retrieve the GCE project if the script is being executed inside a
|
|
|
6a3e49 |
GCE machine if --project is not provided.
|
|
|
6a3e49 |
Try to retrieve the zone automatically from GCE if --zone is not
|
|
|
6a3e49 |
provided.
|
|
|
6a3e49 |
---
|
|
|
6a3e49 |
agents/gce/fence_gce.py | 63 +++++++++++++++++++++++++++++++++++++--
|
|
|
6a3e49 |
tests/data/metadata/fence_gce.xml | 4 +--
|
|
|
6a3e49 |
2 files changed, 63 insertions(+), 4 deletions(-)
|
|
|
6a3e49 |
|
|
|
6a3e49 |
diff --git a/agents/gce/fence_gce.py b/agents/gce/fence_gce.py
|
|
|
6a3e49 |
index 3af5bfc8..e53dc5a6 100644
|
|
|
6a3e49 |
--- a/agents/gce/fence_gce.py
|
|
|
6a3e49 |
+++ b/agents/gce/fence_gce.py
|
|
|
6a3e49 |
@@ -12,6 +12,8 @@
|
|
|
6a3e49 |
|
|
|
6a3e49 |
|
|
|
6a3e49 |
LOGGER = logging
|
|
|
6a3e49 |
+METADATA_SERVER = 'http://metadata.google.internal/computeMetadata/v1/'
|
|
|
6a3e49 |
+METADATA_HEADERS = {'Metadata-Flavor': 'Google'}
|
|
|
6a3e49 |
|
|
|
6a3e49 |
|
|
|
6a3e49 |
def translate_status(instance_status):
|
|
|
6a3e49 |
@@ -81,13 +83,56 @@ def set_power_status(conn, options):
|
|
|
6a3e49 |
fail_usage("Failed: set_power_status: {}".format(str(err)))
|
|
|
6a3e49 |
|
|
|
6a3e49 |
|
|
|
6a3e49 |
+def get_instance(conn, project, zone, instance):
|
|
|
6a3e49 |
+ request = conn.instances().get(
|
|
|
6a3e49 |
+ project=project, zone=zone, instance=instance)
|
|
|
6a3e49 |
+ return request.execute()
|
|
|
6a3e49 |
+
|
|
|
6a3e49 |
+
|
|
|
6a3e49 |
+def get_zone(conn, project, instance):
|
|
|
6a3e49 |
+ request = conn.instances().aggregatedList(project=project)
|
|
|
6a3e49 |
+ while request is not None:
|
|
|
6a3e49 |
+ response = request.execute()
|
|
|
6a3e49 |
+ zones = response.get('items', {})
|
|
|
6a3e49 |
+ for zone in zones.values():
|
|
|
6a3e49 |
+ for inst in zone.get('instances', []):
|
|
|
6a3e49 |
+ if inst['name'] == instance:
|
|
|
6a3e49 |
+ return inst['zone'].split("/")[-1]
|
|
|
6a3e49 |
+ request = conn.instances().aggregatedList_next(
|
|
|
6a3e49 |
+ previous_request=request, previous_response=response)
|
|
|
6a3e49 |
+ raise Exception("Unable to find instance %s" % (instance))
|
|
|
6a3e49 |
+
|
|
|
6a3e49 |
+
|
|
|
6a3e49 |
+def get_metadata(metadata_key, params=None, timeout=None):
|
|
|
6a3e49 |
+ """Performs a GET request with the metadata headers.
|
|
|
6a3e49 |
+
|
|
|
6a3e49 |
+ Args:
|
|
|
6a3e49 |
+ metadata_key: string, the metadata to perform a GET request on.
|
|
|
6a3e49 |
+ params: dictionary, the query parameters in the GET request.
|
|
|
6a3e49 |
+ timeout: int, timeout in seconds for metadata requests.
|
|
|
6a3e49 |
+
|
|
|
6a3e49 |
+ Returns:
|
|
|
6a3e49 |
+ HTTP response from the GET request.
|
|
|
6a3e49 |
+
|
|
|
6a3e49 |
+ Raises:
|
|
|
6a3e49 |
+ urlerror.HTTPError: raises when the GET request fails.
|
|
|
6a3e49 |
+ """
|
|
|
6a3e49 |
+ timeout = timeout or 60
|
|
|
6a3e49 |
+ metadata_url = os.path.join(METADATA_SERVER, metadata_key)
|
|
|
6a3e49 |
+ params = urlparse.urlencode(params or {})
|
|
|
6a3e49 |
+ url = '%s?%s' % (metadata_url, params)
|
|
|
6a3e49 |
+ request = urlrequest.Request(url, headers=METADATA_HEADERS)
|
|
|
6a3e49 |
+ request_opener = urlrequest.build_opener(urlrequest.ProxyHandler({}))
|
|
|
6a3e49 |
+ return request_opener.open(request, timeout=timeout * 1.1).read()
|
|
|
6a3e49 |
+
|
|
|
6a3e49 |
+
|
|
|
6a3e49 |
def define_new_opts():
|
|
|
6a3e49 |
all_opt["zone"] = {
|
|
|
6a3e49 |
"getopt" : ":",
|
|
|
6a3e49 |
"longopt" : "zone",
|
|
|
6a3e49 |
"help" : "--zone=[name] Zone, e.g. us-central1-b",
|
|
|
6a3e49 |
"shortdesc" : "Zone.",
|
|
|
6a3e49 |
- "required" : "1",
|
|
|
6a3e49 |
+ "required" : "0",
|
|
|
6a3e49 |
"order" : 2
|
|
|
6a3e49 |
}
|
|
|
6a3e49 |
all_opt["project"] = {
|
|
|
6a3e49 |
@@ -95,7 +140,7 @@ def define_new_opts():
|
|
|
6a3e49 |
"longopt" : "project",
|
|
|
6a3e49 |
"help" : "--project=[name] Project ID",
|
|
|
6a3e49 |
"shortdesc" : "Project ID.",
|
|
|
6a3e49 |
- "required" : "1",
|
|
|
6a3e49 |
+ "required" : "0",
|
|
|
6a3e49 |
"order" : 3
|
|
|
6a3e49 |
}
|
|
|
6a3e49 |
all_opt["logging"] = {
|
|
|
6a3e49 |
@@ -109,6 +154,7 @@ def define_new_opts():
|
|
|
6a3e49 |
"order" : 4
|
|
|
6a3e49 |
}
|
|
|
6a3e49 |
|
|
|
6a3e49 |
+
|
|
|
6a3e49 |
def main():
|
|
|
6a3e49 |
conn = None
|
|
|
6a3e49 |
global LOGGER
|
|
|
6a3e49 |
@@ -165,6 +211,19 @@ def main():
|
|
|
6a3e49 |
except Exception as err:
|
|
|
6a3e49 |
fail_usage("Failed: Create GCE compute v1 connection: {}".format(str(err)))
|
|
|
6a3e49 |
|
|
|
6a3e49 |
+ # Get project and zone
|
|
|
6a3e49 |
+ if not options.get("--project"):
|
|
|
6a3e49 |
+ try:
|
|
|
6a3e49 |
+ options["--project"] = get_metadata('project/project-id')
|
|
|
6a3e49 |
+ except Exception as err:
|
|
|
6a3e49 |
+ fail_usage("Failed retrieving GCE project. Please provide --project option: {}".format(str(err)))
|
|
|
6a3e49 |
+
|
|
|
6a3e49 |
+ if not options.get("--zone"):
|
|
|
6a3e49 |
+ try:
|
|
|
6a3e49 |
+ options["--zone"] = get_zone(conn, options['--project'], options['--plug'])
|
|
|
6a3e49 |
+ except Exception as err:
|
|
|
6a3e49 |
+ fail_usage("Failed retrieving GCE zone. Please provide --zone option: {}".format(str(err)))
|
|
|
6a3e49 |
+
|
|
|
6a3e49 |
# Operate the fencing device
|
|
|
6a3e49 |
result = fence_action(conn, options, set_power_status, get_power_status, get_nodes_list)
|
|
|
6a3e49 |
sys.exit(result)
|
|
|
6a3e49 |
diff --git a/tests/data/metadata/fence_gce.xml b/tests/data/metadata/fence_gce.xml
|
|
|
6a3e49 |
index 805ecc6b..507b8385 100644
|
|
|
6a3e49 |
--- a/tests/data/metadata/fence_gce.xml
|
|
|
6a3e49 |
+++ b/tests/data/metadata/fence_gce.xml
|
|
|
6a3e49 |
@@ -20,12 +20,12 @@ For instructions see: https://cloud.google.com/compute/docs/tutorials/python-gui
|
|
|
6a3e49 |
<content type="string" />
|
|
|
6a3e49 |
<shortdesc lang="en">Physical plug number on device, UUID or identification of machine</shortdesc>
|
|
|
6a3e49 |
</parameter>
|
|
|
6a3e49 |
- <parameter name="zone" unique="0" required="1">
|
|
|
6a3e49 |
+ <parameter name="zone" unique="0" required="0">
|
|
|
6a3e49 |
<getopt mixed="--zone=[name]" />
|
|
|
6a3e49 |
<content type="string" />
|
|
|
6a3e49 |
<shortdesc lang="en">Zone.</shortdesc>
|
|
|
6a3e49 |
</parameter>
|
|
|
6a3e49 |
- <parameter name="project" unique="0" required="1">
|
|
|
6a3e49 |
+ <parameter name="project" unique="0" required="0">
|
|
|
6a3e49 |
<getopt mixed="--project=[name]" />
|
|
|
6a3e49 |
<content type="string" />
|
|
|
6a3e49 |
<shortdesc lang="en">Project ID.</shortdesc>
|
|
|
6a3e49 |
|
|
|
6a3e49 |
From 8ae1af8068d1718a861a25bf954e14392384fa55 Mon Sep 17 00:00:00 2001
|
|
|
6a3e49 |
From: Helen Koike <helen.koike@collabora.com>
|
|
|
6a3e49 |
Date: Wed, 4 Jul 2018 09:25:46 -0300
|
|
|
6a3e49 |
Subject: [PATCH 3/7] fence_gce: add power cycle as default method
|
|
|
6a3e49 |
|
|
|
6a3e49 |
Add function to power cycle an instance and set cycle as the default
|
|
|
6a3e49 |
method to reboot.
|
|
|
6a3e49 |
---
|
|
|
6a3e49 |
agents/gce/fence_gce.py | 21 +++++++++++++++++++--
|
|
|
6a3e49 |
tests/data/metadata/fence_gce.xml | 8 ++++++++
|
|
|
6a3e49 |
2 files changed, 27 insertions(+), 2 deletions(-)
|
|
|
6a3e49 |
|
|
|
6a3e49 |
diff --git a/agents/gce/fence_gce.py b/agents/gce/fence_gce.py
|
|
|
6a3e49 |
index e53dc5a6..3f77dc24 100644
|
|
|
6a3e49 |
--- a/agents/gce/fence_gce.py
|
|
|
6a3e49 |
+++ b/agents/gce/fence_gce.py
|
|
|
6a3e49 |
@@ -83,6 +83,21 @@ def set_power_status(conn, options):
|
|
|
6a3e49 |
fail_usage("Failed: set_power_status: {}".format(str(err)))
|
|
|
6a3e49 |
|
|
|
6a3e49 |
|
|
|
6a3e49 |
+def power_cycle(conn, options):
|
|
|
6a3e49 |
+ try:
|
|
|
6a3e49 |
+ LOGGER.info('Issuing reset of %s in zone %s' % (options["--plug"], options["--zone"]))
|
|
|
6a3e49 |
+ operation = conn.instances().reset(
|
|
|
6a3e49 |
+ project=options["--project"],
|
|
|
6a3e49 |
+ zone=options["--zone"],
|
|
|
6a3e49 |
+ instance=options["--plug"]).execute()
|
|
|
6a3e49 |
+ wait_for_operation(conn, options["--project"], options["--zone"], operation)
|
|
|
6a3e49 |
+ LOGGER.info('Reset of %s in zone %s complete' % (options["--plug"], options["--zone"]))
|
|
|
6a3e49 |
+ return True
|
|
|
6a3e49 |
+ except Exception as err:
|
|
|
6a3e49 |
+ LOGGER.error("Failed: power_cycle: {}".format(str(err)))
|
|
|
6a3e49 |
+ return False
|
|
|
6a3e49 |
+
|
|
|
6a3e49 |
+
|
|
|
6a3e49 |
def get_instance(conn, project, zone, instance):
|
|
|
6a3e49 |
request = conn.instances().get(
|
|
|
6a3e49 |
project=project, zone=zone, instance=instance)
|
|
|
6a3e49 |
@@ -161,13 +176,15 @@ def main():
|
|
|
6a3e49 |
|
|
|
6a3e49 |
hostname = platform.node()
|
|
|
6a3e49 |
|
|
|
6a3e49 |
- device_opt = ["port", "no_password", "zone", "project", "logging"]
|
|
|
6a3e49 |
+ device_opt = ["port", "no_password", "zone", "project", "logging", "method"]
|
|
|
6a3e49 |
|
|
|
6a3e49 |
atexit.register(atexit_handler)
|
|
|
6a3e49 |
|
|
|
6a3e49 |
define_new_opts()
|
|
|
6a3e49 |
|
|
|
6a3e49 |
all_opt["power_timeout"]["default"] = "60"
|
|
|
6a3e49 |
+ all_opt["method"]["default"] = "cycle"
|
|
|
6a3e49 |
+ all_opt["method"]["help"] = "-m, --method=[method] Method to fence (onoff|cycle) (Default: cycle)"
|
|
|
6a3e49 |
|
|
|
6a3e49 |
options = check_input(device_opt, process_input(device_opt))
|
|
|
6a3e49 |
|
|
|
6a3e49 |
@@ -225,7 +242,7 @@ def main():
|
|
|
6a3e49 |
fail_usage("Failed retrieving GCE zone. Please provide --zone option: {}".format(str(err)))
|
|
|
6a3e49 |
|
|
|
6a3e49 |
# Operate the fencing device
|
|
|
6a3e49 |
- result = fence_action(conn, options, set_power_status, get_power_status, get_nodes_list)
|
|
|
6a3e49 |
+ result = fence_action(conn, options, set_power_status, get_power_status, get_nodes_list, power_cycle)
|
|
|
6a3e49 |
sys.exit(result)
|
|
|
6a3e49 |
|
|
|
6a3e49 |
if __name__ == "__main__":
|
|
|
6a3e49 |
diff --git a/tests/data/metadata/fence_gce.xml b/tests/data/metadata/fence_gce.xml
|
|
|
6a3e49 |
index 507b8385..f522550f 100644
|
|
|
6a3e49 |
--- a/tests/data/metadata/fence_gce.xml
|
|
|
6a3e49 |
+++ b/tests/data/metadata/fence_gce.xml
|
|
|
6a3e49 |
@@ -10,6 +10,14 @@ For instructions see: https://cloud.google.com/compute/docs/tutorials/python-gui
|
|
|
6a3e49 |
<content type="string" default="reboot" />
|
|
|
6a3e49 |
<shortdesc lang="en">Fencing action</shortdesc>
|
|
|
6a3e49 |
</parameter>
|
|
|
6a3e49 |
+ <parameter name="method" unique="0" required="0">
|
|
|
6a3e49 |
+ <getopt mixed="-m, --method=[method]" />
|
|
|
6a3e49 |
+ <content type="select" default="cycle" >
|
|
|
6a3e49 |
+ <option value="onoff" />
|
|
|
6a3e49 |
+ <option value="cycle" />
|
|
|
6a3e49 |
+ </content>
|
|
|
6a3e49 |
+ <shortdesc lang="en">Method to fence</shortdesc>
|
|
|
6a3e49 |
+ </parameter>
|
|
|
6a3e49 |
<parameter name="plug" unique="0" required="1" obsoletes="port">
|
|
|
6a3e49 |
<getopt mixed="-n, --plug=[id]" />
|
|
|
6a3e49 |
<content type="string" />
|
|
|
6a3e49 |
|
|
|
6a3e49 |
From 68644764695b79a3b75826fe009ea7da675677f7 Mon Sep 17 00:00:00 2001
|
|
|
6a3e49 |
From: Helen Koike <helen.koike@collabora.com>
|
|
|
6a3e49 |
Date: Thu, 5 Jul 2018 11:04:32 -0300
|
|
|
6a3e49 |
Subject: [PATCH 4/7] fence_gce: add missing imports to retrieve the project
|
|
|
6a3e49 |
|
|
|
6a3e49 |
---
|
|
|
6a3e49 |
agents/gce/fence_gce.py | 9 +++++++++
|
|
|
6a3e49 |
1 file changed, 9 insertions(+)
|
|
|
6a3e49 |
|
|
|
6a3e49 |
diff --git a/agents/gce/fence_gce.py b/agents/gce/fence_gce.py
|
|
|
6a3e49 |
index 3f77dc24..9b7b5e55 100644
|
|
|
6a3e49 |
--- a/agents/gce/fence_gce.py
|
|
|
6a3e49 |
+++ b/agents/gce/fence_gce.py
|
|
|
6a3e49 |
@@ -2,9 +2,18 @@
|
|
|
6a3e49 |
|
|
|
6a3e49 |
import atexit
|
|
|
6a3e49 |
import logging
|
|
|
6a3e49 |
+import os
|
|
|
6a3e49 |
import platform
|
|
|
6a3e49 |
import sys
|
|
|
6a3e49 |
import time
|
|
|
6a3e49 |
+if sys.version_info >= (3, 0):
|
|
|
6a3e49 |
+ # Python 3 imports.
|
|
|
6a3e49 |
+ import urllib.parse as urlparse
|
|
|
6a3e49 |
+ import urllib.request as urlrequest
|
|
|
6a3e49 |
+else:
|
|
|
6a3e49 |
+ # Python 2 imports.
|
|
|
6a3e49 |
+ import urllib as urlparse
|
|
|
6a3e49 |
+ import urllib2 as urlrequest
|
|
|
6a3e49 |
sys.path.append("@FENCEAGENTSLIBDIR@")
|
|
|
6a3e49 |
|
|
|
6a3e49 |
import googleapiclient.discovery
|
|
|
6a3e49 |
|
|
|
6a3e49 |
From f8f3f11187341622c26e4e439dfda6a37ad660b0 Mon Sep 17 00:00:00 2001
|
|
|
6a3e49 |
From: Helen Koike <helen.koike@collabora.com>
|
|
|
6a3e49 |
Date: Thu, 5 Jul 2018 11:05:32 -0300
|
|
|
6a3e49 |
Subject: [PATCH 5/7] fence_gce: s/--loging/--stackdriver-logging/
|
|
|
6a3e49 |
|
|
|
6a3e49 |
---
|
|
|
6a3e49 |
agents/gce/fence_gce.py | 42 ++++++++++++++++++---------------------
|
|
|
6a3e49 |
tests/data/metadata/fence_gce.xml | 11 +++++++---
|
|
|
6a3e49 |
2 files changed, 27 insertions(+), 26 deletions(-)
|
|
|
6a3e49 |
|
|
|
6a3e49 |
diff --git a/agents/gce/fence_gce.py b/agents/gce/fence_gce.py
|
|
|
6a3e49 |
index 9b7b5e55..a6befe39 100644
|
|
|
6a3e49 |
--- a/agents/gce/fence_gce.py
|
|
|
6a3e49 |
+++ b/agents/gce/fence_gce.py
|
|
|
6a3e49 |
@@ -167,14 +167,13 @@ def define_new_opts():
|
|
|
6a3e49 |
"required" : "0",
|
|
|
6a3e49 |
"order" : 3
|
|
|
6a3e49 |
}
|
|
|
6a3e49 |
- all_opt["logging"] = {
|
|
|
6a3e49 |
- "getopt" : ":",
|
|
|
6a3e49 |
- "longopt" : "logging",
|
|
|
6a3e49 |
- "help" : "--logging=[bool] Logging, true/false",
|
|
|
6a3e49 |
+ all_opt["stackdriver-logging"] = {
|
|
|
6a3e49 |
+ "getopt" : "",
|
|
|
6a3e49 |
+ "longopt" : "stackdriver-logging",
|
|
|
6a3e49 |
+ "help" : "--stackdriver-logging Enable Logging to Stackdriver",
|
|
|
6a3e49 |
"shortdesc" : "Stackdriver-logging support.",
|
|
|
6a3e49 |
- "longdesc" : "If enabled (set to true), IP failover logs will be posted to stackdriver logging.",
|
|
|
6a3e49 |
+ "longdesc" : "If enabled IP failover logs will be posted to stackdriver logging.",
|
|
|
6a3e49 |
"required" : "0",
|
|
|
6a3e49 |
- "default" : "false",
|
|
|
6a3e49 |
"order" : 4
|
|
|
6a3e49 |
}
|
|
|
6a3e49 |
|
|
|
6a3e49 |
@@ -185,7 +184,7 @@ def main():
|
|
|
6a3e49 |
|
|
|
6a3e49 |
hostname = platform.node()
|
|
|
6a3e49 |
|
|
|
6a3e49 |
- device_opt = ["port", "no_password", "zone", "project", "logging", "method"]
|
|
|
6a3e49 |
+ device_opt = ["port", "no_password", "zone", "project", "stackdriver-logging", "method"]
|
|
|
6a3e49 |
|
|
|
6a3e49 |
atexit.register(atexit_handler)
|
|
|
6a3e49 |
|
|
|
6a3e49 |
@@ -210,22 +209,19 @@ def main():
|
|
|
6a3e49 |
run_delay(options)
|
|
|
6a3e49 |
|
|
|
6a3e49 |
# Prepare logging
|
|
|
6a3e49 |
- logging_env = options.get('--logging')
|
|
|
6a3e49 |
- if logging_env:
|
|
|
6a3e49 |
- logging_env = logging_env.lower()
|
|
|
6a3e49 |
- if any(x in logging_env for x in ['yes', 'true', 'enabled']):
|
|
|
6a3e49 |
- try:
|
|
|
6a3e49 |
- import google.cloud.logging.handlers
|
|
|
6a3e49 |
- client = google.cloud.logging.Client()
|
|
|
6a3e49 |
- handler = google.cloud.logging.handlers.CloudLoggingHandler(client, name=hostname)
|
|
|
6a3e49 |
- formatter = logging.Formatter('gcp:stonish "%(message)s"')
|
|
|
6a3e49 |
- LOGGER = logging.getLogger(hostname)
|
|
|
6a3e49 |
- handler.setFormatter(formatter)
|
|
|
6a3e49 |
- LOGGER.addHandler(handler)
|
|
|
6a3e49 |
- LOGGER.setLevel(logging.INFO)
|
|
|
6a3e49 |
- except ImportError:
|
|
|
6a3e49 |
- LOGGER.error('Couldn\'t import google.cloud.logging, '
|
|
|
6a3e49 |
- 'disabling Stackdriver-logging support')
|
|
|
6a3e49 |
+ if options.get('--stackdriver-logging'):
|
|
|
6a3e49 |
+ try:
|
|
|
6a3e49 |
+ import google.cloud.logging.handlers
|
|
|
6a3e49 |
+ client = google.cloud.logging.Client()
|
|
|
6a3e49 |
+ handler = google.cloud.logging.handlers.CloudLoggingHandler(client, name=hostname)
|
|
|
6a3e49 |
+ formatter = logging.Formatter('gcp:stonish "%(message)s"')
|
|
|
6a3e49 |
+ LOGGER = logging.getLogger(hostname)
|
|
|
6a3e49 |
+ handler.setFormatter(formatter)
|
|
|
6a3e49 |
+ LOGGER.addHandler(handler)
|
|
|
6a3e49 |
+ LOGGER.setLevel(logging.INFO)
|
|
|
6a3e49 |
+ except ImportError:
|
|
|
6a3e49 |
+ LOGGER.error('Couldn\'t import google.cloud.logging, '
|
|
|
6a3e49 |
+ 'disabling Stackdriver-logging support')
|
|
|
6a3e49 |
|
|
|
6a3e49 |
# Prepare cli
|
|
|
6a3e49 |
try:
|
|
|
6a3e49 |
diff --git a/tests/data/metadata/fence_gce.xml b/tests/data/metadata/fence_gce.xml
|
|
|
6a3e49 |
index f522550f..79b82ebb 100644
|
|
|
6a3e49 |
--- a/tests/data/metadata/fence_gce.xml
|
|
|
6a3e49 |
+++ b/tests/data/metadata/fence_gce.xml
|
|
|
6a3e49 |
@@ -38,9 +38,14 @@ For instructions see: https://cloud.google.com/compute/docs/tutorials/python-gui
|
|
|
6a3e49 |
<content type="string" />
|
|
|
6a3e49 |
<shortdesc lang="en">Project ID.</shortdesc>
|
|
|
6a3e49 |
</parameter>
|
|
|
6a3e49 |
- <parameter name="logging" unique="0" required="0">
|
|
|
6a3e49 |
- <getopt mixed="--logging=[bool]" />
|
|
|
6a3e49 |
- <content type="string" default="false" />
|
|
|
6a3e49 |
+ <parameter name="stackdriver-logging" unique="0" required="0" deprecated="1">
|
|
|
6a3e49 |
+ <getopt mixed="--stackdriver-logging" />
|
|
|
6a3e49 |
+ <content type="boolean" />
|
|
|
6a3e49 |
+ <shortdesc lang="en">Stackdriver-logging support.</shortdesc>
|
|
|
6a3e49 |
+ </parameter>
|
|
|
6a3e49 |
+ <parameter name="stackdriver_logging" unique="0" required="0" obsoletes="stackdriver-logging">
|
|
|
6a3e49 |
+ <getopt mixed="--stackdriver-logging" />
|
|
|
6a3e49 |
+ <content type="boolean" />
|
|
|
6a3e49 |
<shortdesc lang="en">Stackdriver-logging support.</shortdesc>
|
|
|
6a3e49 |
</parameter>
|
|
|
6a3e49 |
<parameter name="quiet" unique="0" required="0">
|
|
|
6a3e49 |
|
|
|
6a3e49 |
From 9ae0a072424fa982e1d18a2cb661628c38601c3a Mon Sep 17 00:00:00 2001
|
|
|
6a3e49 |
From: Helen Koike <helen.koike@collabora.com>
|
|
|
6a3e49 |
Date: Sat, 7 Jul 2018 18:42:01 -0300
|
|
|
6a3e49 |
Subject: [PATCH 6/7] fence_gce: use root logger for stackdriver
|
|
|
6a3e49 |
|
|
|
6a3e49 |
---
|
|
|
6a3e49 |
agents/gce/fence_gce.py | 29 +++++++++++++++--------------
|
|
|
6a3e49 |
1 file changed, 15 insertions(+), 14 deletions(-)
|
|
|
6a3e49 |
|
|
|
6a3e49 |
diff --git a/agents/gce/fence_gce.py b/agents/gce/fence_gce.py
|
|
|
6a3e49 |
index a6befe39..1d5095ae 100644
|
|
|
6a3e49 |
--- a/agents/gce/fence_gce.py
|
|
|
6a3e49 |
+++ b/agents/gce/fence_gce.py
|
|
|
6a3e49 |
@@ -20,7 +20,6 @@
|
|
|
6a3e49 |
from fencing import fail_usage, run_delay, all_opt, atexit_handler, check_input, process_input, show_docs, fence_action
|
|
|
6a3e49 |
|
|
|
6a3e49 |
|
|
|
6a3e49 |
-LOGGER = logging
|
|
|
6a3e49 |
METADATA_SERVER = 'http://metadata.google.internal/computeMetadata/v1/'
|
|
|
6a3e49 |
METADATA_HEADERS = {'Metadata-Flavor': 'Google'}
|
|
|
6a3e49 |
|
|
|
6a3e49 |
@@ -73,37 +72,37 @@ def wait_for_operation(conn, project, zone, operation):
|
|
|
6a3e49 |
def set_power_status(conn, options):
|
|
|
6a3e49 |
try:
|
|
|
6a3e49 |
if options["--action"] == "off":
|
|
|
6a3e49 |
- LOGGER.info("Issuing poweroff of %s in zone %s" % (options["--plug"], options["--zone"]))
|
|
|
6a3e49 |
+ logging.info("Issuing poweroff of %s in zone %s" % (options["--plug"], options["--zone"]))
|
|
|
6a3e49 |
operation = conn.instances().stop(
|
|
|
6a3e49 |
project=options["--project"],
|
|
|
6a3e49 |
zone=options["--zone"],
|
|
|
6a3e49 |
instance=options["--plug"]).execute()
|
|
|
6a3e49 |
wait_for_operation(conn, options["--project"], options["--zone"], operation)
|
|
|
6a3e49 |
- LOGGER.info("Poweroff of %s in zone %s complete" % (options["--plug"], options["--zone"]))
|
|
|
6a3e49 |
+ logging.info("Poweroff of %s in zone %s complete" % (options["--plug"], options["--zone"]))
|
|
|
6a3e49 |
elif options["--action"] == "on":
|
|
|
6a3e49 |
- LOGGER.info("Issuing poweron of %s in zone %s" % (options["--plug"], options["--zone"]))
|
|
|
6a3e49 |
+ logging.info("Issuing poweron of %s in zone %s" % (options["--plug"], options["--zone"]))
|
|
|
6a3e49 |
operation = conn.instances().start(
|
|
|
6a3e49 |
project=options["--project"],
|
|
|
6a3e49 |
zone=options["--zone"],
|
|
|
6a3e49 |
instance=options["--plug"]).execute()
|
|
|
6a3e49 |
wait_for_operation(conn, options["--project"], options["--zone"], operation)
|
|
|
6a3e49 |
- LOGGER.info("Poweron of %s in zone %s complete" % (options["--plug"], options["--zone"]))
|
|
|
6a3e49 |
+ logging.info("Poweron of %s in zone %s complete" % (options["--plug"], options["--zone"]))
|
|
|
6a3e49 |
except Exception as err:
|
|
|
6a3e49 |
fail_usage("Failed: set_power_status: {}".format(str(err)))
|
|
|
6a3e49 |
|
|
|
6a3e49 |
|
|
|
6a3e49 |
def power_cycle(conn, options):
|
|
|
6a3e49 |
try:
|
|
|
6a3e49 |
- LOGGER.info('Issuing reset of %s in zone %s' % (options["--plug"], options["--zone"]))
|
|
|
6a3e49 |
+ logging.info('Issuing reset of %s in zone %s' % (options["--plug"], options["--zone"]))
|
|
|
6a3e49 |
operation = conn.instances().reset(
|
|
|
6a3e49 |
project=options["--project"],
|
|
|
6a3e49 |
zone=options["--zone"],
|
|
|
6a3e49 |
instance=options["--plug"]).execute()
|
|
|
6a3e49 |
wait_for_operation(conn, options["--project"], options["--zone"], operation)
|
|
|
6a3e49 |
- LOGGER.info('Reset of %s in zone %s complete' % (options["--plug"], options["--zone"]))
|
|
|
6a3e49 |
+ logging.info('Reset of %s in zone %s complete' % (options["--plug"], options["--zone"]))
|
|
|
6a3e49 |
return True
|
|
|
6a3e49 |
except Exception as err:
|
|
|
6a3e49 |
- LOGGER.error("Failed: power_cycle: {}".format(str(err)))
|
|
|
6a3e49 |
+ logging.error("Failed: power_cycle: {}".format(str(err)))
|
|
|
6a3e49 |
return False
|
|
|
6a3e49 |
|
|
|
6a3e49 |
|
|
|
6a3e49 |
@@ -180,7 +179,6 @@ def define_new_opts():
|
|
|
6a3e49 |
|
|
|
6a3e49 |
def main():
|
|
|
6a3e49 |
conn = None
|
|
|
6a3e49 |
- global LOGGER
|
|
|
6a3e49 |
|
|
|
6a3e49 |
hostname = platform.node()
|
|
|
6a3e49 |
|
|
|
6a3e49 |
@@ -209,18 +207,21 @@ def main():
|
|
|
6a3e49 |
run_delay(options)
|
|
|
6a3e49 |
|
|
|
6a3e49 |
# Prepare logging
|
|
|
6a3e49 |
- if options.get('--stackdriver-logging'):
|
|
|
6a3e49 |
+ if options.get('--stackdriver-logging') is not None:
|
|
|
6a3e49 |
try:
|
|
|
6a3e49 |
import google.cloud.logging.handlers
|
|
|
6a3e49 |
client = google.cloud.logging.Client()
|
|
|
6a3e49 |
handler = google.cloud.logging.handlers.CloudLoggingHandler(client, name=hostname)
|
|
|
6a3e49 |
+ handler.setLevel(logging.INFO)
|
|
|
6a3e49 |
formatter = logging.Formatter('gcp:stonish "%(message)s"')
|
|
|
6a3e49 |
- LOGGER = logging.getLogger(hostname)
|
|
|
6a3e49 |
handler.setFormatter(formatter)
|
|
|
6a3e49 |
- LOGGER.addHandler(handler)
|
|
|
6a3e49 |
- LOGGER.setLevel(logging.INFO)
|
|
|
6a3e49 |
+ root_logger = logging.getLogger()
|
|
|
6a3e49 |
+ if options.get('--verbose') is None:
|
|
|
6a3e49 |
+ root_logger.setLevel(logging.INFO)
|
|
|
6a3e49 |
+ logging.getLogger("googleapiclient").setLevel(logging.ERROR)
|
|
|
6a3e49 |
+ root_logger.addHandler(handler)
|
|
|
6a3e49 |
except ImportError:
|
|
|
6a3e49 |
- LOGGER.error('Couldn\'t import google.cloud.logging, '
|
|
|
6a3e49 |
+ logging.error('Couldn\'t import google.cloud.logging, '
|
|
|
6a3e49 |
'disabling Stackdriver-logging support')
|
|
|
6a3e49 |
|
|
|
6a3e49 |
# Prepare cli
|
|
|
6a3e49 |
|
|
|
6a3e49 |
From a52e643708908539d6e5fdb5d36a6cea935e4481 Mon Sep 17 00:00:00 2001
|
|
|
6a3e49 |
From: Helen Koike <helen.koike@collabora.com>
|
|
|
6a3e49 |
Date: Wed, 11 Jul 2018 17:16:49 -0300
|
|
|
6a3e49 |
Subject: [PATCH 7/7] fence_gce: minor changes in logging
|
|
|
6a3e49 |
|
|
|
6a3e49 |
- Remove hostname (use --plug instead).
|
|
|
6a3e49 |
- Supress messages from googleapiclient and oauth2client if not error in
|
|
|
6a3e49 |
non verbose mode.
|
|
|
6a3e49 |
- s/stonish/stonith
|
|
|
6a3e49 |
---
|
|
|
6a3e49 |
agents/gce/fence_gce.py | 13 ++++++-------
|
|
|
6a3e49 |
1 file changed, 6 insertions(+), 7 deletions(-)
|
|
|
6a3e49 |
|
|
|
6a3e49 |
diff --git a/agents/gce/fence_gce.py b/agents/gce/fence_gce.py
|
|
|
6a3e49 |
index 1d5095ae..3eca0139 100644
|
|
|
6a3e49 |
--- a/agents/gce/fence_gce.py
|
|
|
6a3e49 |
+++ b/agents/gce/fence_gce.py
|
|
|
6a3e49 |
@@ -3,7 +3,6 @@
|
|
|
6a3e49 |
import atexit
|
|
|
6a3e49 |
import logging
|
|
|
6a3e49 |
import os
|
|
|
6a3e49 |
-import platform
|
|
|
6a3e49 |
import sys
|
|
|
6a3e49 |
import time
|
|
|
6a3e49 |
if sys.version_info >= (3, 0):
|
|
|
6a3e49 |
@@ -180,8 +179,6 @@ def define_new_opts():
|
|
|
6a3e49 |
def main():
|
|
|
6a3e49 |
conn = None
|
|
|
6a3e49 |
|
|
|
6a3e49 |
- hostname = platform.node()
|
|
|
6a3e49 |
-
|
|
|
6a3e49 |
device_opt = ["port", "no_password", "zone", "project", "stackdriver-logging", "method"]
|
|
|
6a3e49 |
|
|
|
6a3e49 |
atexit.register(atexit_handler)
|
|
|
6a3e49 |
@@ -207,18 +204,20 @@ def main():
|
|
|
6a3e49 |
run_delay(options)
|
|
|
6a3e49 |
|
|
|
6a3e49 |
# Prepare logging
|
|
|
6a3e49 |
- if options.get('--stackdriver-logging') is not None:
|
|
|
6a3e49 |
+ if options.get('--verbose') is None:
|
|
|
6a3e49 |
+ logging.getLogger('googleapiclient').setLevel(logging.ERROR)
|
|
|
6a3e49 |
+ logging.getLogger('oauth2client').setLevel(logging.ERROR)
|
|
|
6a3e49 |
+ if options.get('--stackdriver-logging') is not None and options.get('--plug'):
|
|
|
6a3e49 |
try:
|
|
|
6a3e49 |
import google.cloud.logging.handlers
|
|
|
6a3e49 |
client = google.cloud.logging.Client()
|
|
|
6a3e49 |
- handler = google.cloud.logging.handlers.CloudLoggingHandler(client, name=hostname)
|
|
|
6a3e49 |
+ handler = google.cloud.logging.handlers.CloudLoggingHandler(client, name=options['--plug'])
|
|
|
6a3e49 |
handler.setLevel(logging.INFO)
|
|
|
6a3e49 |
- formatter = logging.Formatter('gcp:stonish "%(message)s"')
|
|
|
6a3e49 |
+ formatter = logging.Formatter('gcp:stonith "%(message)s"')
|
|
|
6a3e49 |
handler.setFormatter(formatter)
|
|
|
6a3e49 |
root_logger = logging.getLogger()
|
|
|
6a3e49 |
if options.get('--verbose') is None:
|
|
|
6a3e49 |
root_logger.setLevel(logging.INFO)
|
|
|
6a3e49 |
- logging.getLogger("googleapiclient").setLevel(logging.ERROR)
|
|
|
6a3e49 |
root_logger.addHandler(handler)
|
|
|
6a3e49 |
except ImportError:
|
|
|
6a3e49 |
logging.error('Couldn\'t import google.cloud.logging, '
|