Skip to content

Commit

Permalink
Merge pull request #432 from oalbrigt/ibm-agents
Browse files Browse the repository at this point in the history
fence_ibm_vpc/fence_ibm_powervs: new fence agents
  • Loading branch information
oalbrigt authored Sep 7, 2021
2 parents 006bd6a + 3078e4d commit 542fb6d
Show file tree
Hide file tree
Showing 5 changed files with 724 additions and 0 deletions.
202 changes: 202 additions & 0 deletions agents/ibm_powervs/fence_ibm_powervs.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,202 @@
#!@PYTHON@ -tt

import sys
import pycurl, io, json
import logging
import atexit
sys.path.append("@FENCEAGENTSLIBDIR@")
from fencing import *
from fencing import fail, run_delay, EC_LOGIN_DENIED, EC_STATUS

state = {
"ACTIVE": "on",
"SHUTOFF": "off",
"ERROR": "unknown"
}

def get_list(conn, options):
outlets = {}

try:
command = "cloud-instances/{}/pvm-instances".format(options["--instance"])
res = send_command(conn, command)
except Exception as e:
logging.debug("Failed: {}".format(e))
return outlets

for r in res["pvmInstances"]:
if "--verbose" in options:
logging.debug(json.dumps(r, indent=2))
outlets[r["pvmInstanceID"]] = (r["serverName"], state[r["status"]])

return outlets

def get_power_status(conn, options):
try:
command = "cloud-instances/{}/pvm-instances/{}".format(
options["--instance"], options["--plug"])
res = send_command(conn, command)
result = get_list(conn, options)[options["--plug"]][1]
except KeyError as e:
logging.debug("Failed: Unable to get status for {}".format(e))
fail(EC_STATUS)

return result

def set_power_status(conn, options):
action = {
"on" : '{"action" : "start"}',
"off" : '{"action" : "immediate-shutdown"}',
}[options["--action"]]

try:
send_command(conn, "cloud-instances/{}/pvm-instances/{}/action".format(
options["--instance"], options["--plug"]), "POST", action)
except Exception as e:
logging.debug("Failed: Unable to set power to {} for {}".format(options["--action"], e))
fail(EC_STATUS)

def connect(opt):
conn = pycurl.Curl()

## setup correct URL
conn.base_url = "https://" + opt["--region"] + ".power-iaas.cloud.ibm.com/pcloud/v1/"

if opt["--verbose-level"] > 1:
conn.setopt(pycurl.VERBOSE, 1)

conn.setopt(pycurl.TIMEOUT, int(opt["--shell-timeout"]))
conn.setopt(pycurl.SSL_VERIFYPEER, 1)
conn.setopt(pycurl.SSL_VERIFYHOST, 2)

# set auth token for later requests
conn.setopt(pycurl.HTTPHEADER, [
"Content-Type: application/json",
"Authorization: Bearer {}".format(opt["--token"]),
"CRN: {}".format(opt["--crn"]),
"User-Agent: curl",
])

return conn

def disconnect(conn):
conn.close()

def send_command(conn, command, method="GET", action=None):
url = conn.base_url + command

conn.setopt(pycurl.URL, url.encode("ascii"))

web_buffer = io.BytesIO()

if method == "GET":
conn.setopt(pycurl.POST, 0)
if method == "POST":
conn.setopt(pycurl.POSTFIELDS, action)
if method == "DELETE":
conn.setopt(pycurl.CUSTOMREQUEST, "DELETE")

conn.setopt(pycurl.WRITEFUNCTION, web_buffer.write)

try:
conn.perform()
except Exception as e:
raise(e)

rc = conn.getinfo(pycurl.HTTP_CODE)
result = web_buffer.getvalue().decode("UTF-8")

web_buffer.close()

if rc != 200:
if len(result) > 0:
raise Exception("{}: {}".format(rc,
result["value"]["messages"][0]["default_message"]))
else:
raise Exception("Remote returned {} for request to {}".format(rc, url))

if len(result) > 0:
result = json.loads(result)

logging.debug("url: {}".format(url))
logging.debug("method: {}".format(method))
logging.debug("response code: {}".format(rc))
logging.debug("result: {}\n".format(result))

return result

def define_new_opts():
all_opt["token"] = {
"getopt" : ":",
"longopt" : "token",
"help" : "--token=[token] Bearer Token",
"required" : "1",
"shortdesc" : "Bearer Token",
"order" : 0
}
all_opt["crn"] = {
"getopt" : ":",
"longopt" : "crn",
"help" : "--crn=[crn] CRN",
"required" : "1",
"shortdesc" : "CRN",
"order" : 0
}
all_opt["instance"] = {
"getopt" : ":",
"longopt" : "instance",
"help" : "--instance=[instance] PowerVS Instance",
"required" : "1",
"shortdesc" : "PowerVS Instance",
"order" : 0
}
all_opt["region"] = {
"getopt" : ":",
"longopt" : "region",
"help" : "--region=[region] Region",
"required" : "1",
"shortdesc" : "Region",
"order" : 0
}


def main():
device_opt = [
"token",
"crn",
"instance",
"region",
"port",
"no_password",
]

atexit.register(atexit_handler)
define_new_opts()

all_opt["shell_timeout"]["default"] = "15"
all_opt["power_timeout"]["default"] = "30"
all_opt["power_wait"]["default"] = "1"

options = check_input(device_opt, process_input(device_opt))

docs = {}
docs["shortdesc"] = "Fence agent for IBM PowerVS"
docs["longdesc"] = """fence_ibm_powervs is an I/O Fencing agent which can be \
used with IBM PowerVS to fence virtual machines."""
docs["vendorurl"] = "https://www.ibm.com"
show_docs(options, docs)

####
## Fence operations
####
run_delay(options)

conn = connect(options)
atexit.register(disconnect, conn)

result = fence_action(conn, options, set_power_status, get_power_status, get_list)

sys.exit(result)

if __name__ == "__main__":
main()
Loading

0 comments on commit 542fb6d

Please sign in to comment.