Andrew Jeffery | 114010c | 2017-10-06 11:44:56 +1030 | [diff] [blame] | 1 | #!/usr/bin/env python |
Andrew Geissler | 53be16b | 2017-09-06 16:23:27 -0500 | [diff] [blame] | 2 | |
| 3 | import argparse |
| 4 | import requests |
Charles P. Hofer | 88d15b3 | 2017-09-12 11:32:55 -0500 | [diff] [blame] | 5 | import tarfile |
| 6 | import time |
Andrew Geissler | 53be16b | 2017-09-06 16:23:27 -0500 | [diff] [blame] | 7 | import json |
| 8 | |
Andrew Geissler | 53be16b | 2017-09-06 16:23:27 -0500 | [diff] [blame] | 9 | class BMC: |
| 10 | def __init__(self, server): |
| 11 | self.url = "https://{0}/".format(server) |
| 12 | self.session = requests.Session() |
| 13 | self.login() |
| 14 | |
| 15 | def login(self): |
| 16 | r = self.session.post(self.url + 'login', |
| 17 | json={'data': ['root', '0penBmc']}, |
| 18 | verify=False) |
| 19 | j = r.json() |
| 20 | if j['status'] != 'ok': |
| 21 | raise Exception("Failed to login: \n" + r.text) |
| 22 | |
| 23 | def list_sfw(self): |
| 24 | r = self.session.get(self.url + 'xyz/openbmc_project/software/', |
| 25 | verify=False) |
| 26 | j = r.json() |
| 27 | if j['status'] != 'ok': |
Andrew Geissler | 942eca2 | 2017-09-08 11:20:17 -0500 | [diff] [blame] | 28 | raise Exception("Failed to query software: \n" + r.text) |
Andrew Geissler | 53be16b | 2017-09-06 16:23:27 -0500 | [diff] [blame] | 29 | |
| 30 | events = j['data'] |
| 31 | |
| 32 | return events |
| 33 | |
| 34 | def get_image(self, image): |
| 35 | r = self.session.get(self.url + image, verify=False) |
| 36 | |
| 37 | j = r.json() |
| 38 | if j['status'] != 'ok': |
Andrew Geissler | 942eca2 | 2017-09-08 11:20:17 -0500 | [diff] [blame] | 39 | raise Exception("Failed to get image " + image + ": \n" + r.text) |
Andrew Geissler | 53be16b | 2017-09-06 16:23:27 -0500 | [diff] [blame] | 40 | |
| 41 | return j['data'] |
| 42 | |
| 43 | def upload_image(self, image): |
Charles P. Hofer | 88d15b3 | 2017-09-12 11:32:55 -0500 | [diff] [blame] | 44 | |
Andrew Geissler | 53be16b | 2017-09-06 16:23:27 -0500 | [diff] [blame] | 45 | data = open(image,'rb').read() |
| 46 | r = self.session.post(self.url + "/upload/image", |
| 47 | data=data, |
| 48 | headers={'Content-Type': 'application/octet-stream'}, |
| 49 | verify=False) |
| 50 | j = r.json() |
| 51 | if j['status'] != 'ok': |
| 52 | raise Exception("Failed to get event " + image + ": \n" + r.text) |
| 53 | |
| 54 | return j['data'] |
| 55 | |
Andrew Geissler | 942eca2 | 2017-09-08 11:20:17 -0500 | [diff] [blame] | 56 | def activate_image(self, image_id): |
Charles P. Hofer | 88d15b3 | 2017-09-12 11:32:55 -0500 | [diff] [blame] | 57 | r = self.session.put(self.url + "/xyz/openbmc_project/software/" + image_id + "/attr/RequestedActivation", |
Andrew Geissler | 64da218 | 2017-09-08 14:06:42 -0500 | [diff] [blame] | 58 | json={'data': 'xyz.openbmc_project.Software.Activation.RequestedActivations.Active'}, |
Andrew Geissler | 942eca2 | 2017-09-08 11:20:17 -0500 | [diff] [blame] | 59 | verify=False) |
Charles P. Hofer | 88d15b3 | 2017-09-12 11:32:55 -0500 | [diff] [blame] | 60 | |
Andrew Geissler | 942eca2 | 2017-09-08 11:20:17 -0500 | [diff] [blame] | 61 | j = r.json() |
| 62 | if j['status'] != 'ok': |
| 63 | raise Exception("Failed to activate image " + image_id + ": \n" + r.text) |
| 64 | |
| 65 | return j['data'] |
| 66 | |
Andrew Geissler | cb4d062 | 2017-09-21 12:54:01 -0500 | [diff] [blame] | 67 | def set_priority(self, image_id, priority): |
| 68 | r = self.session.put(self.url + "/xyz/openbmc_project/software/" + image_id + "/attr/Priority", |
| 69 | json={'data': int(priority)}, |
| 70 | verify=False) |
| 71 | |
| 72 | j = r.json() |
| 73 | if j['status'] != 'ok': |
| 74 | raise Exception("Failed to set priority of image " + image_id + ": \n" + r.text) |
| 75 | |
| 76 | return j['data'] |
| 77 | |
Andrew Geissler | e1b6a3e | 2017-09-20 21:57:37 -0500 | [diff] [blame] | 78 | def delete_image(self, image_id): |
| 79 | r = self.session.post(self.url + "/xyz/openbmc_project/software/" + image_id + "/action/delete", |
| 80 | headers={'Content-Type': 'application/json'}, |
| 81 | data='{"data":[]}', |
| 82 | verify=False) |
| 83 | |
| 84 | j = r.json() |
| 85 | if j['status'] != 'ok': |
| 86 | raise Exception("Failed to delete image " + image_id + ": \n" + r.text) |
| 87 | |
| 88 | return j['data'] |
| 89 | |
Charles P. Hofer | 88d15b3 | 2017-09-12 11:32:55 -0500 | [diff] [blame] | 90 | def update_auto(self, image, reboot): |
| 91 | image_version = self.__get_image_version(image) |
| 92 | self.upload_image(image) |
| 93 | image_id = self.__wait_for_image_id(image_version) |
| 94 | self.activate_image(image_id) |
| 95 | self.__wait_for_activation(image_id) |
| 96 | if reboot: |
| 97 | self.reboot() |
| 98 | |
| 99 | def reboot(self): |
| 100 | r = self.session.post( |
| 101 | self.url + '/org/openbmc/control/bmc0/action/warmReset', |
| 102 | headers={'Content-Type': 'application/json'}, |
| 103 | data='{"data":[]}', |
| 104 | verify=False) |
| 105 | |
| 106 | j = r.json() |
| 107 | if j['status'] != 'ok': |
| 108 | raise Exception("Failed to reboot BMC:\n" + r.text) |
| 109 | |
| 110 | def __get_image_version(self, image_file_path): |
| 111 | # Open up the manfest file. |
| 112 | image_tar = tarfile.open(name=image_file_path, mode='r') |
| 113 | manifest = image_tar.extractfile('MANIFEST') |
| 114 | |
| 115 | # Find version line. |
| 116 | for line in manifest: |
| 117 | if line.startswith('version='): |
| 118 | manifest.close() |
| 119 | image_tar.close() |
| 120 | return line.split('version=', 1)[1].strip() |
| 121 | |
| 122 | # If we didn't find the version line, print an error and return false. |
| 123 | manifest.close() |
| 124 | image_tar.close() |
| 125 | raise Exception("Could not find version line in image manifest") |
| 126 | |
| 127 | def __wait_for_image_id(self, image_version): |
| 128 | # Try 8 times, once every 15 seconds. |
| 129 | for attempt in range(8): |
| 130 | software = self.list_sfw() |
| 131 | # Look for our the image with the given version in software |
| 132 | for path in software: |
| 133 | image = self.get_image(path) |
Charles P. Hofer | c772b47 | 2017-09-20 16:02:48 -0500 | [diff] [blame] | 134 | if 'Version' in image and image_version == image['Version']: |
Charles P. Hofer | 88d15b3 | 2017-09-12 11:32:55 -0500 | [diff] [blame] | 135 | return path.split('/')[-1] |
| 136 | time.sleep(15) |
| 137 | return False |
| 138 | |
| 139 | def __wait_for_activation(self, image_id): |
| 140 | # Keep waiting until the image is active or activation fails. |
| 141 | active = False |
| 142 | while not active: |
| 143 | image = self.get_image("/xyz/openbmc_project/software/" + image_id) |
| 144 | if 'xyz.openbmc_project.Software.Activation.Activations.Active' \ |
| 145 | == image['Activation']: |
| 146 | print 'Activation Progress: 100%' |
| 147 | active = True |
| 148 | elif 'xyz.openbmc_project.Software.Activation.Activations.Activating' \ |
| 149 | == image['Activation']: |
| 150 | print 'Activation Progress: ' + str(image['Progress']) + '%' |
| 151 | else: |
| 152 | raise Exception("Image activation failed. The BMC has set " \ |
| 153 | + "the'Activation' property to " + image['Activation']) |
| 154 | time.sleep(15) |
Andrew Geissler | 53be16b | 2017-09-06 16:23:27 -0500 | [diff] [blame] | 155 | |
| 156 | def do_list_sfw(args): |
| 157 | s = BMC(server=args.server) |
| 158 | for e in s.list_sfw(): |
Andrew Geissler | cfc7e4c | 2017-09-20 14:17:12 -0500 | [diff] [blame] | 159 | if (e == '/xyz/openbmc_project/software/active'): |
| 160 | continue |
Andrew Geissler | d462384 | 2017-09-29 13:23:26 -0500 | [diff] [blame] | 161 | if (e == '/xyz/openbmc_project/software/functional'): |
| 162 | continue |
Andrew Geissler | 53be16b | 2017-09-06 16:23:27 -0500 | [diff] [blame] | 163 | info = s.get_image(e) |
Andrew Geissler | cfc7e4c | 2017-09-20 14:17:12 -0500 | [diff] [blame] | 164 | if (((info['Purpose'] == 'xyz.openbmc_project.Software.Version.VersionPurpose.BMC') and |
| 165 | (args.bmc or not args.host)) or \ |
| 166 | ((info['Purpose'] == 'xyz.openbmc_project.Software.Version.VersionPurpose.Host') and |
| 167 | (args.host or not args.bmc))): |
| 168 | print(e) |
| 169 | print json.dumps(info, indent=4) |
Charles P. Hofer | 88d15b3 | 2017-09-12 11:32:55 -0500 | [diff] [blame] | 170 | |
Andrew Geissler | 53be16b | 2017-09-06 16:23:27 -0500 | [diff] [blame] | 171 | def do_view_image(args): |
| 172 | s = BMC(server=args.server) |
| 173 | print json.dumps(s.get_image(args.image), indent=4) |
| 174 | |
| 175 | def do_upload_image(args): |
| 176 | s = BMC(server=args.server) |
| 177 | s.upload_image(args.image) |
| 178 | |
Andrew Geissler | 942eca2 | 2017-09-08 11:20:17 -0500 | [diff] [blame] | 179 | def do_activate_image(args): |
| 180 | s = BMC(server=args.server) |
| 181 | s.activate_image(args.image_id) |
| 182 | |
Charles P. Hofer | 88d15b3 | 2017-09-12 11:32:55 -0500 | [diff] [blame] | 183 | def do_update_auto(args): |
| 184 | s = BMC(server=args.server) |
| 185 | s.update_auto(args.image, args.reboot) |
| 186 | |
Andrew Geissler | e1b6a3e | 2017-09-20 21:57:37 -0500 | [diff] [blame] | 187 | def do_delete_image(args): |
| 188 | s = BMC(server=args.server) |
| 189 | s.delete_image(args.image_id) |
| 190 | |
Andrew Geissler | cb4d062 | 2017-09-21 12:54:01 -0500 | [diff] [blame] | 191 | def do_set_priority(args): |
| 192 | s = BMC(server=args.server) |
| 193 | s.set_priority(args.image_id,args.priority) |
| 194 | |
Andrew Geissler | 53be16b | 2017-09-06 16:23:27 -0500 | [diff] [blame] | 195 | parser = argparse.ArgumentParser() |
| 196 | parser.add_argument('--server', help='hostname or IP of BMC', type=str, |
| 197 | required=True) |
Andrew Jeffery | bff48b3 | 2017-10-06 12:03:00 +1030 | [diff] [blame] | 198 | parser.add_argument('--suppress-insecure-warnings', '-I', action="store_true", |
| 199 | default=False) |
Andrew Geissler | 53be16b | 2017-09-06 16:23:27 -0500 | [diff] [blame] | 200 | |
| 201 | subparsers = parser.add_subparsers() |
| 202 | list_events = subparsers.add_parser('list', help='List all software images on BMC') |
| 203 | list_events.set_defaults(func=do_list_sfw) |
Andrew Geissler | cfc7e4c | 2017-09-20 14:17:12 -0500 | [diff] [blame] | 204 | list_events.add_argument( |
| 205 | '--bmc', |
| 206 | action='store_true', |
| 207 | default=False, |
| 208 | help='Set if you want to see BMC images') |
| 209 | list_events.add_argument( |
| 210 | '--host', |
| 211 | action='store_true', |
| 212 | default=False, |
| 213 | help='Set if you want to see Host images') |
Andrew Geissler | 53be16b | 2017-09-06 16:23:27 -0500 | [diff] [blame] | 214 | |
| 215 | image_view = subparsers.add_parser('view', help='View info of input image') |
| 216 | image_view.add_argument('image', help='The image to analyze') |
| 217 | image_view.set_defaults(func=do_view_image) |
| 218 | |
| 219 | image_upload = subparsers.add_parser('upload', help='Upload input image') |
| 220 | image_upload.add_argument('image', help='The image to upload') |
| 221 | image_upload.set_defaults(func=do_upload_image) |
| 222 | |
Andrew Geissler | 942eca2 | 2017-09-08 11:20:17 -0500 | [diff] [blame] | 223 | image_activate = subparsers.add_parser('activate', help='Activate input image id') |
| 224 | image_activate.add_argument('image_id', help='The image id to activate') |
| 225 | image_activate.set_defaults(func=do_activate_image) |
| 226 | |
Charles P. Hofer | 88d15b3 | 2017-09-12 11:32:55 -0500 | [diff] [blame] | 227 | image_update_auto = subparsers.add_parser('update_auto', help='Upload and ' |
| 228 | + 'activate an image, and then reboot the BMC to apply the update ' |
| 229 | + 'if necessary') |
| 230 | image_update_auto.add_argument('image', help='The image to update to') |
| 231 | image_update_auto.add_argument( |
| 232 | '--reboot', |
Charles P. Hofer | 88d15b3 | 2017-09-12 11:32:55 -0500 | [diff] [blame] | 233 | action='store_true', |
| 234 | default=False, |
| 235 | help='Set if the BMC should reboot after the update') |
Charles P. Hofer | 88d15b3 | 2017-09-12 11:32:55 -0500 | [diff] [blame] | 236 | image_update_auto.set_defaults(func=do_update_auto) |
| 237 | |
Andrew Geissler | e1b6a3e | 2017-09-20 21:57:37 -0500 | [diff] [blame] | 238 | image_delete = subparsers.add_parser('delete', help='Delete input image id') |
| 239 | image_delete.add_argument('image_id', help='The image id to delete') |
| 240 | image_delete.set_defaults(func=do_delete_image) |
| 241 | |
Andrew Geissler | cb4d062 | 2017-09-21 12:54:01 -0500 | [diff] [blame] | 242 | image_priority = subparsers.add_parser('priority', help='Set priority of input image') |
| 243 | image_priority.add_argument('image_id', help='The image id to set priority of') |
| 244 | image_priority.add_argument('priority', help='The priority to set') |
| 245 | image_priority.set_defaults(func=do_set_priority) |
| 246 | |
Andrew Geissler | 53be16b | 2017-09-06 16:23:27 -0500 | [diff] [blame] | 247 | args = parser.parse_args() |
| 248 | |
Andrew Jeffery | bff48b3 | 2017-10-06 12:03:00 +1030 | [diff] [blame] | 249 | if args.suppress_insecure_warnings: |
Andrew Jeffery | 23da273 | 2017-10-06 12:03:51 +1030 | [diff] [blame] | 250 | from requests.packages.urllib3.exceptions import InsecureRequestWarning |
| 251 | requests.packages.urllib3.disable_warnings(InsecureRequestWarning) |
Andrew Jeffery | bff48b3 | 2017-10-06 12:03:00 +1030 | [diff] [blame] | 252 | |
Andrew Geissler | 53be16b | 2017-09-06 16:23:27 -0500 | [diff] [blame] | 253 | if 'func' in args: |
| 254 | args.func(args) |
| 255 | else: |
| 256 | parser.print_help() |