#!/usr/bin/python import os import sys, json, datetime, time, types, httplib, re import mimetypes DEFAULT_HOST = "127.0.0.1" OPENECOMP_BE = "127.0.0.1" HOST = DEFAULT_HOST DEFAULT_PORT = "8080" DEFAULT_USERNAME = "cs0008" DEFAULT_PASSWORD = "cs0008" ONBOARD_BASE_PATH = "/onboarding-api/v1.0" VSP_LIST_PATH = "{0}/vendor-software-products".format(ONBOARD_BASE_PATH) VSP_ACTIONS_PATH = "{0}/vendor-software-products/{{vspId}}/actions".format(ONBOARD_BASE_PATH) VSP_UPLOAD_PATH = "{0}/vendor-software-products/{{vspId}}/upload".format(ONBOARD_BASE_PATH) VSP_DOWNLOAD_PATH = "{0}/vendor-software-products/{{vspId}}/downloadHeat".format(ONBOARD_BASE_PATH) VSP_GET_URL = "{0}/vendor-software-products/{{vspId}}".format(ONBOARD_BASE_PATH) def main(argv): username = DEFAULT_USERNAME password = DEFAULT_PASSWORD host = DEFAULT_HOST if not argv: print("Going to use default values") else: if argv[0].lower() == 'h' or argv[0].lower() == '-h': printHelp() return if argv[0] == '-ip': host = argv[1] else: if argv[0].lower() == '-a' and '/' not in argv[1]: print('\n>>> Error: Credentials required (username/password)\n') printHelp() return else: creds = argv[1].split('/') username = creds[0] password = creds[1] # not used try: cmdIp = argv[2] host = argv[3] except IndexError: host = DEFAULT_HOST print("Going to use user defined values") Service.server(host) webHandler = WebHandler(host=host, port=DEFAULT_PORT) response, headers = webHandler.rest(url=VSP_LIST_PATH, method='GET', data=None, userId=username) jResult = json.loads(response) jSrvices = jResult["results"] reportFileName = 'upgradereport.csv' # datetime.now() reportFile = open(reportFileName, 'w') reportFile.write(Service.header()) for jService in jSrvices: serviceName = jService["name"] vendorName = jService["vendorName"] vspId = jService["id"] status = jService["status"] if status != "Locked": lockingUser = "None" else: lockingUser = jService["lockingUser"] service = Service(serviceName=serviceName, vspId=vspId, vendorName=vendorName, lockingUser=lockingUser) print(service) # Will try to GET the service res = service.Get() if res == 500: serviceMigration(service, status, username) else: print("Service {0} was tested and does not need a migration".format(serviceName)) reportFile.write(service.line()) reportFile.close() def serviceMigration(service, serviceStatus, username): print("Service {0} was tested and it needs a migration".format(service.serviceName)) print("Service {0} - Migration start") if serviceStatus == "Locked": print("Service {0} is locked - forcing checkin".format(service.serviceName)) service.Checkin() print("Doing new checkout") service.Checkout(username) zipName = service.DownloadHeat() if not zipName: print("no heat found") service.uploadStatus = "no heat found" else: uploadResponse = service.UploadHeat(zipName) uploadResults = json.loads(uploadResponse) if uploadResults['status'] == 'Success' and uploadResults['errors'].__len__() == 0: service.uploadStatus = "Heat uploaded successfully" else: service.uploadStatus = "Heat uploaded with errors" print("Doing new checkin") service.Checkin() print("Service {0} - Migration end") def printHelp(): print("Upgrade script Help:") print("==================================") print("1607_to_1610 -h --> get help") print("1607_to_1610 -a / [-ip {ip}]") print("Example: 1607_to_1610 -a root/secret") class Service(object): def __init__(self, serviceName, vspId, vendorName, lockingUser): self.serviceName = serviceName self.vspId = vspId self.vendorName = vendorName self.lockingUser = lockingUser self.webHandler = WebHandler(host=Service.serveraddress, port=DEFAULT_PORT) # Schema? self.uploadStatus = "not started" def __repr__(self): return 'Name: {0}, Id: {1}, Vendor: {2}, locked by: {3}, status {4}'.format(self.serviceName, self.vspId, self.vendorName, self.lockingUser, self.uploadStatus) @classmethod def header(cls): return 'Name,Id,Vendor,locked-by,status\n' @classmethod def server(cls, address): cls.serveraddress = address def line(self): return '{0},{1},{2},{3},{4}\n'.format(self.serviceName, self.vspId, self.vendorName, self.lockingUser, self.uploadStatus) def Checkout(self, userId): # /v1.0/vendor-software-products/{vspId}/actions urlpath = VSP_ACTIONS_PATH.format(vspId=self.vspId) response, headers = self.webHandler.rest(url=urlpath, method='PUT', data={"action": "Checkout"}, userId=userId) self.lockingUser = userId # we will later use this user to checkin return response def Checkin(self): # /v1.0/vendor-software-products/{vspId}/actions urlpath = VSP_ACTIONS_PATH.format(vspId=self.vspId) response, headers = self.webHandler.rest(url=urlpath, method='PUT', data={"action": "Checkin"}, userId=self.lockingUser) return response def Get(self): # /v1.0/vendor-software-products/{vspId} urlpath = VSP_GET_URL.format(vspId=self.vspId) try: response, headers = self.webHandler.rest(url=urlpath, method='GET', data=None, userId=self.lockingUser) except HttpError as e: print(e.message) response = e.status return response def UploadHeat(self, zipName): # /v1.0/vendor-software-products/{vspId}/upload urlpath = VSP_UPLOAD_PATH.format(vspId=self.vspId) try: fields = [] with open(zipName, 'rb') as fin: buffer = fin.read() fin.close() files = [('upload', 'heatfile.zip', buffer)] response = self.webHandler.post_multipart('HTTP', urlpath, fields, files, self.lockingUser) return response finally: print("done upload") def DownloadHeat(self): urlpath = VSP_DOWNLOAD_PATH.format(vspId=self.vspId) try: response, headers = self.webHandler.rest(url=urlpath, method='Get', data=None, userId=self.lockingUser, accept='application/octet-stream') except HttpError as e: if e.status == 404: return "" for (key, value) in headers: if key.lower() == "content-disposition": file_name = value[value.index('=') + 1:] break heatsDir = os.path.join(os.path.dirname(__file__), 'heats') if not os.path.exists(heatsDir): os.makedirs(heatsDir) file_name = os.path.join(heatsDir, file_name) with open(file_name, "wb") as fout: fout.write(response) fout.close() return file_name class WebHandler(object): def __init__(self, host, port): self.host = host self.port = port def rest(self, url, method, data, userId, accept='application/json', content_type='application/json'): connection = httplib.HTTPConnection(host=self.host, port=self.port) try: headers = {'Content-Type': content_type, 'Accept': accept} headers['USER_ID'] = userId connection.request(method=method, headers=headers, body=json.dumps(data), url=url) response = connection.getresponse() if response.status not in range(200, 300): raise HttpError(status=response.status, message=response.reason) return response.read(), response.getheaders() finally: connection.close() def post_multipart(self, scheme, selector, fields, files, userId): """ Post fields and files to an http host as multipart/form-data. fields is a sequence of (name, value) elements for regular form fields. files is a sequence of (name, filename, value) elements for data to be uploaded as files Return the server's response page. """ content_type, body = self.encode_multipart_form_data(fields, files) if scheme and scheme.lower() == "http": h = httplib.HTTP(self.host, self.port) else: h = httplib.HTTPS(self.host, self.port) h.putrequest('POST', selector) h.putheader('content-type', content_type) h.putheader('content-length', str(len(body))) h.putheader('Accept', 'application/json') h.putheader('USER_ID', userId) h.endheaders() h.send(body) errcode, errmsg, headers = h.getreply() print(errcode, errmsg, headers) return h.file.read() def encode_multipart_form_data(self, fields, files): LIMIT = '----------lImIt_of_THE_fIle_eW_$' CRLF = '\r\n' L = [] for (key, value) in fields: L.append('--' + LIMIT) L.append('Content-Disposition: form-data; name="%s"' % key) L.append('') L.append(value) for (key, filename, value) in files: L.append('--' + LIMIT) L.append('Content-Disposition: form-data; name="%s"; filename="%s"' % (key, filename)) L.append('Content-Type: %s' % self.get_content_type(filename)) L.append('') L.append(value) L.append('--' + LIMIT + '--') L.append('') body = CRLF.join(L) content_type = 'multipart/form-data; boundary=%s' % LIMIT return content_type, body def get_content_type(self, filename): return mimetypes.guess_type(filename)[0] or 'application/octet-stream' class HttpError(Exception): def __init__(self, status, message): self.status = status self.message = message def __str__(self): return repr(self.value, self.message) if __name__ == "__main__": main(sys.argv[1:])