198 lines
5.9 KiB
Python
198 lines
5.9 KiB
Python
#/*****************************************************************************
|
|
#*
|
|
#* (C) Copyright Broadcom Corporation 2015
|
|
#*
|
|
#* 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 sys
|
|
import urllib
|
|
from xml.etree import ElementTree
|
|
import json
|
|
from sidauth import SIDAuth
|
|
from broadview_lib.config.broadviewconfig import BroadViewBSTSwitches
|
|
import os
|
|
|
|
try:
|
|
from oslo_log import log as logging
|
|
except:
|
|
import logging
|
|
|
|
LOG = logging.getLogger(__name__)
|
|
|
|
class RequestObj():
|
|
pass
|
|
|
|
class RequestFailed(Exception):
|
|
'''
|
|
class to represent a failed RESTful call
|
|
'''
|
|
def __init__(self, url, http_code, open_code, open_msg):
|
|
self._url = url
|
|
self._http_code = int(http_code)
|
|
self._open_code = int(open_code)
|
|
self._open_msg = open_msg
|
|
|
|
def __str__(self):
|
|
return repr((self._url, self._http_code, self._open_code, self._open_msg))
|
|
|
|
class AgentConnection():
|
|
|
|
def __init__(self, host, port, feature, timeout):
|
|
self.host = host
|
|
self.port = port
|
|
self.feature = feature # e.g., "bst"
|
|
self._timeout = timeout
|
|
self._swconfig = BroadViewBSTSwitches()
|
|
self._auth = None
|
|
|
|
def close(self):
|
|
if self._auth and self._auth.logout:
|
|
self._auth.logout()
|
|
self._auth = None
|
|
|
|
def _is_ok(self, r):
|
|
return r.status_code == 200
|
|
|
|
def _raise_fail_if(self, url, r, timeout):
|
|
if not self._is_ok(r):
|
|
if timeout:
|
|
j = {"response_code": r.status_code,
|
|
"error_code": 0,
|
|
"msg": "Connection timeout"}
|
|
else:
|
|
try:
|
|
j = r.json()["status"]
|
|
except:
|
|
try:
|
|
t = ElementTree.fromstring(r)
|
|
j = {"response_code": r.status_code,
|
|
"error_code": 0,
|
|
"msg": "XML failure response from web server"}
|
|
|
|
except:
|
|
j = {"response_code": r.status_code,
|
|
"error_code": 0,
|
|
"msg": "Unparsable response from web server"}
|
|
|
|
raise RequestFailed(url, j["response_code"], j["error_code"], j["msg"])
|
|
|
|
def getAuth(self):
|
|
return self._auth
|
|
|
|
def getAuthConf(self):
|
|
auth = {}
|
|
auth["auth"] = None
|
|
auth["username"] = None
|
|
auth["password"] = None
|
|
|
|
try:
|
|
conf = self._swconfig.getByIP(self.host)
|
|
if conf:
|
|
if "username" in conf:
|
|
auth["username"] = conf["username"]
|
|
if "password" in conf:
|
|
auth["password"] = conf["password"]
|
|
if "auth" in conf:
|
|
auth["auth"] = conf["auth"]
|
|
except:
|
|
pass
|
|
|
|
if auth["auth"] == None:
|
|
auth["auth"] = os.getenv("BV_AUTH")
|
|
if auth["username"] == None:
|
|
auth["username"] = os.getenv("BV_USERNAME")
|
|
if auth["password"] == None:
|
|
auth["password"] = os.getenv("BV_PASSWORD", "")
|
|
|
|
return auth
|
|
|
|
def __makeRequest(self, request):
|
|
|
|
headers = {"Content-Type": "application/json"}
|
|
timeout = False
|
|
|
|
isGet = False
|
|
if request.getHttpMethod() == "GET":
|
|
isGet = True
|
|
|
|
auth = self.getAuth()
|
|
|
|
payload = request.getjson().encode("utf-8")
|
|
if self.feature:
|
|
url = "http://%s:%d/broadview/%s/%s" % (self.host, self.port, self.feature, request.getMethod())
|
|
else:
|
|
url = "http://%s:%d/broadview/%s" % (self.host, self.port, request.getMethod())
|
|
if isGet:
|
|
try:
|
|
r = requests.get(url, timeout=self._timeout, data=payload, headers=headers, auth=auth)
|
|
except requests.exceptions.Timeout:
|
|
timeout = True
|
|
else:
|
|
try:
|
|
r = requests.post(url, timeout=self._timeout, data=payload, headers=headers, auth=auth)
|
|
except requests.exceptions.Timeout:
|
|
timeout = True
|
|
|
|
json_data = {}
|
|
if timeout:
|
|
r = RequestObj()
|
|
r.status_code = 500
|
|
|
|
if r.status_code == 200:
|
|
try:
|
|
# Data can come back with leading.trailing spaces, which trips
|
|
# up requests' json parser. So get as test, string, and let
|
|
# Python json do the parsing
|
|
|
|
json_data = json.loads(r.text.strip())
|
|
except:
|
|
pass
|
|
|
|
return (r, json_data)
|
|
|
|
def makeRequest(self, request):
|
|
r, json_data = self.__makeRequest(request)
|
|
|
|
if r.status_code == 401:
|
|
conf = self.getAuthConf()
|
|
try:
|
|
auth_method = r.headers["WWW-Authenticate"]
|
|
except:
|
|
# RFC 2616 requires a WWW-Authenticate header in 401 responses. If
|
|
# we get here, it was missing. Check if there is configuration that
|
|
# declares an auth method and use that.
|
|
LOG.info("makeRequest: 401 but no WWW-Authenticate")
|
|
auth_method = conf["auth"]
|
|
if auth_method:
|
|
auth_method = auth_method.lower()
|
|
if auth_method == "basic":
|
|
self._auth = requests.HTTPBasicAuth(conf["username"], conf["password"])
|
|
elif auth_method == "digest":
|
|
self._auth = requests.HTTPDigestAuth(conf["username"], conf["password"])
|
|
elif auth_method == "sidauth":
|
|
self._auth = SIDAuth(self.host, self.port, conf["username"], conf["password"])
|
|
else:
|
|
LOG.info("unknown auth {}".format(auth_method))
|
|
# return the 401 here
|
|
return (r.status_code, json_data)
|
|
|
|
# try again
|
|
|
|
r, json_data = self.__makeRequest(request)
|
|
|
|
return (r.status_code, json_data)
|
|
|