|
|
2d8bb4 |
From a4e8b77ac51a0e4a6de489823ee1be47cbc7eb18 Mon Sep 17 00:00:00 2001
|
|
|
2d8bb4 |
From: Oyvind Albrigtsen <oalbrigt@redhat.com>
|
|
|
2d8bb4 |
Date: Thu, 9 May 2019 12:09:48 +0200
|
|
|
2d8bb4 |
Subject: [PATCH] fence_rhevm: add RHEV v4 API support and auto-detection
|
|
|
2d8bb4 |
|
|
|
2d8bb4 |
---
|
|
|
2d8bb4 |
agents/rhevm/fence_rhevm.py | 44 +++++++++++++++++++++++------
|
|
|
2d8bb4 |
tests/data/metadata/fence_rhevm.xml | 7 ++++-
|
|
|
2d8bb4 |
2 files changed, 41 insertions(+), 10 deletions(-)
|
|
|
2d8bb4 |
|
|
|
2d8bb4 |
diff --git a/agents/rhevm/fence_rhevm.py b/agents/rhevm/fence_rhevm.py
|
|
|
2d8bb4 |
index a1cdaf605..6012c4239 100644
|
|
|
2d8bb4 |
--- a/agents/rhevm/fence_rhevm.py
|
|
|
2d8bb4 |
+++ b/agents/rhevm/fence_rhevm.py
|
|
|
2d8bb4 |
@@ -9,7 +9,8 @@
|
|
|
2d8bb4 |
from fencing import fail, EC_FETCH_VM_UUID, run_delay
|
|
|
2d8bb4 |
|
|
|
2d8bb4 |
RE_GET_ID = re.compile("
|
|
|
2d8bb4 |
-RE_STATUS = re.compile("<state>(.*?)</state>", re.IGNORECASE)
|
|
|
2d8bb4 |
+RE_STATUS = re.compile("<status>(.*?)</status>", re.IGNORECASE)
|
|
|
2d8bb4 |
+RE_STATE = re.compile("<state>(.*?)</state>", re.IGNORECASE)
|
|
|
2d8bb4 |
RE_GET_NAME = re.compile("<name>(.*?)</name>", re.IGNORECASE)
|
|
|
2d8bb4 |
|
|
|
2d8bb4 |
def get_power_status(conn, options):
|
|
|
2d8bb4 |
@@ -25,7 +26,10 @@ def get_power_status(conn, options):
|
|
|
2d8bb4 |
|
|
|
2d8bb4 |
options["id"] = result.group(2)
|
|
|
2d8bb4 |
|
|
|
2d8bb4 |
- result = RE_STATUS.search(res)
|
|
|
2d8bb4 |
+ if tuple(map(int, options["--api-version"].split(".")))[0] > 3:
|
|
|
2d8bb4 |
+ result = RE_STATUS.search(res)
|
|
|
2d8bb4 |
+ else:
|
|
|
2d8bb4 |
+ result = RE_STATE.search(res)
|
|
|
2d8bb4 |
if result == None:
|
|
|
2d8bb4 |
# We were able to parse ID so output is correct
|
|
|
2d8bb4 |
# in some cases it is possible that RHEV-M output does not
|
|
|
2d8bb4 |
@@ -59,7 +63,10 @@ def get_list(conn, options):
|
|
|
2d8bb4 |
lines = res.split("
|
|
|
2d8bb4 |
for i in range(1, len(lines)):
|
|
|
2d8bb4 |
name = RE_GET_NAME.search(lines[i]).group(1)
|
|
|
2d8bb4 |
- status = RE_STATUS.search(lines[i]).group(1)
|
|
|
2d8bb4 |
+ if tuple(map(int, options["--api-version"].split(".")))[0] > 3:
|
|
|
2d8bb4 |
+ status = RE_STATUS.search(lines[i]).group(1)
|
|
|
2d8bb4 |
+ else:
|
|
|
2d8bb4 |
+ status = RE_STATE.search(lines[i]).group(1)
|
|
|
2d8bb4 |
outlets[name] = ("", status)
|
|
|
2d8bb4 |
except AttributeError:
|
|
|
2d8bb4 |
return {}
|
|
|
2d8bb4 |
@@ -69,6 +76,13 @@ def get_list(conn, options):
|
|
|
2d8bb4 |
return outlets
|
|
|
2d8bb4 |
|
|
|
2d8bb4 |
def send_command(opt, command, method="GET"):
|
|
|
2d8bb4 |
+ if opt["--api-version"] == "auto":
|
|
|
2d8bb4 |
+ opt["--api-version"] = "4"
|
|
|
2d8bb4 |
+ res = send_command(opt, "")
|
|
|
2d8bb4 |
+ if re.search("<title>Error</title>", res):
|
|
|
2d8bb4 |
+ opt["--api-version"] = "3"
|
|
|
2d8bb4 |
+ logging.debug("auto-detected API version: " + opt["--api-version"])
|
|
|
2d8bb4 |
+
|
|
|
2d8bb4 |
## setup correct URL
|
|
|
2d8bb4 |
if "--ssl" in opt or "--ssl-secure" in opt or "--ssl-insecure" in opt:
|
|
|
2d8bb4 |
url = "https:"
|
|
|
2d8bb4 |
@@ -90,7 +104,7 @@ def send_command(opt, command, method="GET"):
|
|
|
2d8bb4 |
web_buffer = io.BytesIO()
|
|
|
2d8bb4 |
conn.setopt(pycurl.URL, url.encode("UTF-8"))
|
|
|
2d8bb4 |
conn.setopt(pycurl.HTTPHEADER, [
|
|
|
2d8bb4 |
- "Version: 3",
|
|
|
2d8bb4 |
+ "Version: {}".format(opt["--api-version"]),
|
|
|
2d8bb4 |
"Content-type: application/xml",
|
|
|
2d8bb4 |
"Accept: application/xml",
|
|
|
2d8bb4 |
"Prefer: persistent-auth",
|
|
|
2d8bb4 |
@@ -130,8 +144,9 @@ def send_command(opt, command, method="GET"):
|
|
|
2d8bb4 |
|
|
|
2d8bb4 |
result = web_buffer.getvalue().decode("UTF-8")
|
|
|
2d8bb4 |
|
|
|
2d8bb4 |
- logging.debug("%s\n", command)
|
|
|
2d8bb4 |
- logging.debug("%s\n", result)
|
|
|
2d8bb4 |
+ logging.debug("url: %s\n", url)
|
|
|
2d8bb4 |
+ logging.debug("command: %s\n", command)
|
|
|
2d8bb4 |
+ logging.debug("result: %s\n", result)
|
|
|
2d8bb4 |
|
|
|
2d8bb4 |
return result
|
|
|
2d8bb4 |
|
|
|
2d8bb4 |
@@ -151,6 +166,15 @@ def define_new_opts():
|
|
|
2d8bb4 |
"required" : "0",
|
|
|
2d8bb4 |
"shortdesc" : "Reuse cookies for authentication",
|
|
|
2d8bb4 |
"order" : 1}
|
|
|
2d8bb4 |
+ all_opt["api_version"] = {
|
|
|
2d8bb4 |
+ "getopt" : ":",
|
|
|
2d8bb4 |
+ "longopt" : "api-version",
|
|
|
2d8bb4 |
+ "help" : "--api-version "
|
|
|
2d8bb4 |
+ "Version of RHEV API (default: auto)",
|
|
|
2d8bb4 |
+ "required" : "0",
|
|
|
2d8bb4 |
+ "order" : 2,
|
|
|
2d8bb4 |
+ "default" : "auto",
|
|
|
2d8bb4 |
+ }
|
|
|
2d8bb4 |
all_opt["api_path"] = {
|
|
|
2d8bb4 |
"getopt" : ":",
|
|
|
2d8bb4 |
"longopt" : "api-path",
|
|
|
2d8bb4 |
@@ -158,20 +182,19 @@ def define_new_opts():
|
|
|
2d8bb4 |
"default" : "/ovirt-engine/api",
|
|
|
2d8bb4 |
"required" : "0",
|
|
|
2d8bb4 |
"shortdesc" : "The path part of the API URL",
|
|
|
2d8bb4 |
- "order" : 2}
|
|
|
2d8bb4 |
+ "order" : 3}
|
|
|
2d8bb4 |
all_opt["disable_http_filter"] = {
|
|
|
2d8bb4 |
"getopt" : "",
|
|
|
2d8bb4 |
"longopt" : "disable-http-filter",
|
|
|
2d8bb4 |
"help" : "--disable-http-filter Set HTTP Filter header to false",
|
|
|
2d8bb4 |
"required" : "0",
|
|
|
2d8bb4 |
"shortdesc" : "Set HTTP Filter header to false",
|
|
|
2d8bb4 |
- "order" : 3}
|
|
|
2d8bb4 |
+ "order" : 4}
|
|
|
2d8bb4 |
|
|
|
2d8bb4 |
|
|
|
2d8bb4 |
def main():
|
|
|
2d8bb4 |
device_opt = [
|
|
|
2d8bb4 |
"ipaddr",
|
|
|
2d8bb4 |
- "api_path",
|
|
|
2d8bb4 |
"login",
|
|
|
2d8bb4 |
"passwd",
|
|
|
2d8bb4 |
"ssl",
|
|
|
2d8bb4 |
@@ -179,6 +202,8 @@ def main():
|
|
|
2d8bb4 |
"web",
|
|
|
2d8bb4 |
"port",
|
|
|
2d8bb4 |
"use_cookies",
|
|
|
2d8bb4 |
+ "api_version",
|
|
|
2d8bb4 |
+ "api_path",
|
|
|
2d8bb4 |
"disable_http_filter",
|
|
|
2d8bb4 |
]
|
|
|
2d8bb4 |
|
|
|
2d8bb4 |
@@ -186,6 +211,7 @@ def main():
|
|
|
2d8bb4 |
define_new_opts()
|
|
|
2d8bb4 |
|
|
|
2d8bb4 |
all_opt["power_wait"]["default"] = "1"
|
|
|
2d8bb4 |
+ all_opt["shell_timeout"]["default"] = "5"
|
|
|
2d8bb4 |
|
|
|
2d8bb4 |
options = check_input(device_opt, process_input(device_opt))
|
|
|
2d8bb4 |
|
|
|
2d8bb4 |
diff --git a/tests/data/metadata/fence_rhevm.xml b/tests/data/metadata/fence_rhevm.xml
|
|
|
2d8bb4 |
index 6344db79f..c56cf64b6 100644
|
|
|
2d8bb4 |
--- a/tests/data/metadata/fence_rhevm.xml
|
|
|
2d8bb4 |
+++ b/tests/data/metadata/fence_rhevm.xml
|
|
|
2d8bb4 |
@@ -98,6 +98,11 @@
|
|
|
2d8bb4 |
<content type="string" />
|
|
|
2d8bb4 |
<shortdesc lang="en">Login name</shortdesc>
|
|
|
2d8bb4 |
</parameter>
|
|
|
2d8bb4 |
+ <parameter name="api_version" unique="0" required="0">
|
|
|
2d8bb4 |
+ <getopt mixed="--api-version" />
|
|
|
2d8bb4 |
+ <content type="string" default="auto" />
|
|
|
2d8bb4 |
+ <shortdesc lang="en">Version of RHEV API (default: auto)</shortdesc>
|
|
|
2d8bb4 |
+ </parameter>
|
|
|
2d8bb4 |
<parameter name="api_path" unique="0" required="0">
|
|
|
2d8bb4 |
<getopt mixed="--api-path=[path]" />
|
|
|
2d8bb4 |
<shortdesc lang="en">The path part of the API URL</shortdesc>
|
|
|
2d8bb4 |
@@ -164,7 +169,7 @@
|
|
|
2d8bb4 |
</parameter>
|
|
|
2d8bb4 |
<parameter name="shell_timeout" unique="0" required="0">
|
|
|
2d8bb4 |
<getopt mixed="--shell-timeout=[seconds]" />
|
|
|
2d8bb4 |
- <content type="second" default="3" />
|
|
|
2d8bb4 |
+ <content type="second" default="5" />
|
|
|
2d8bb4 |
<shortdesc lang="en">Wait X seconds for cmd prompt after issuing command</shortdesc>
|
|
|
2d8bb4 |
</parameter>
|
|
|
2d8bb4 |
<parameter name="retry_on" unique="0" required="0">
|