1#!/usr/bin/env python 2 3import argparse 4import json 5import socket 6 7try: 8 from shlex import quote 9except ImportError: 10 from pipes import quote 11 12def print_dict(d): 13 print json.dumps(d, indent=2) 14 15def print_array(a): 16 print " ".join((quote(v) for v in a)) 17 18parser = argparse.ArgumentParser(description='SPDK RPC command line interface') 19parser.add_argument('-s', dest='server_addr', help='RPC server address', default='127.0.0.1') 20parser.add_argument('-p', dest='port', help='RPC port number', default=5260, type=int) 21parser.add_argument('-v', dest='verbose', help='Verbose mode', action='store_true') 22subparsers = parser.add_subparsers(help='RPC methods') 23 24 25def int_arg(arg): 26 return int(arg, 0) 27 28 29def jsonrpc_call(method, params={}): 30 if args.server_addr.startswith('/'): 31 s = socket.socket(socket.AF_UNIX, socket.SOCK_STREAM) 32 s.connect(args.server_addr) 33 else: 34 s = socket.socket(socket.AF_INET, socket.SOCK_STREAM) 35 s.connect((args.server_addr, args.port)) 36 req = {} 37 req['jsonrpc'] = '2.0' 38 req['method'] = method 39 req['id'] = 1 40 if (params): 41 req['params'] = params 42 reqstr = json.dumps(req) 43 44 if args.verbose: 45 print("request:") 46 print(json.dumps(req, indent=2)) 47 48 s.sendall(reqstr) 49 buf = '' 50 closed = False 51 response = {} 52 while not closed: 53 newdata = s.recv(4096) 54 if (newdata == b''): 55 closed = True 56 buf += newdata 57 try: 58 response = json.loads(buf) 59 except ValueError: 60 continue # incomplete response; keep buffering 61 break 62 s.close() 63 64 if not response: 65 if method == "kill_instance": 66 exit(0) 67 print "Connection closed with partial response:" 68 print buf 69 exit(1) 70 71 if 'error' in response: 72 print "Got JSON-RPC error response" 73 print "request:" 74 print_dict(json.loads(reqstr)) 75 print "response:" 76 print_dict(response['error']) 77 exit(1) 78 79 if args.verbose: 80 print("response:") 81 print(json.dumps(response, indent=2)) 82 83 return response['result'] 84 85def get_luns(args): 86 print_dict(jsonrpc_call('get_luns')) 87 88p = subparsers.add_parser('get_luns', help='Display active LUNs') 89p.set_defaults(func=get_luns) 90 91 92def get_portal_groups(args): 93 print_dict(jsonrpc_call('get_portal_groups')) 94 95p = subparsers.add_parser('get_portal_groups', help='Display current portal group configuration') 96p.set_defaults(func=get_portal_groups) 97 98 99def get_initiator_groups(args): 100 print_dict(jsonrpc_call('get_initiator_groups')) 101 102p = subparsers.add_parser('get_initiator_groups', help='Display current initiator group configuration') 103p.set_defaults(func=get_initiator_groups) 104 105 106def get_target_nodes(args): 107 print_dict(jsonrpc_call('get_target_nodes')) 108 109p = subparsers.add_parser('get_target_nodes', help='Display target nodes') 110p.set_defaults(func=get_target_nodes) 111 112 113def construct_target_node(args): 114 lun_name_id_dict = dict(u.split(":") 115 for u in args.lun_name_id_pairs.strip().split(" ")) 116 lun_names = lun_name_id_dict.keys() 117 lun_ids = list(map(int, lun_name_id_dict.values())) 118 119 pg_tags = [] 120 ig_tags = [] 121 for u in args.pg_ig_mappings.strip().split(" "): 122 pg, ig = u.split(":") 123 pg_tags.append(int(pg)) 124 ig_tags.append(int(ig)) 125 126 params = { 127 'name': args.name, 128 'alias_name': args.alias_name, 129 'pg_tags': pg_tags, 130 'ig_tags': ig_tags, 131 'lun_names': lun_names, 132 'lun_ids': lun_ids, 133 'queue_depth': args.queue_depth, 134 'chap_disabled': args.chap_disabled, 135 'chap_required': args.chap_required, 136 'chap_mutual': args.chap_mutual, 137 'chap_auth_group': args.chap_auth_group, 138 } 139 jsonrpc_call('construct_target_node', params) 140 141p = subparsers.add_parser('construct_target_node', help='Add a target node') 142p.add_argument('name', help='Target node name (ASCII)') 143p.add_argument('alias_name', help='Target node alias name (ASCII)') 144p.add_argument('lun_name_id_pairs', help="""Whitespace-separated list of LUN <name:id> pairs enclosed 145in quotes. Format: 'lun_name0:id0 lun_name1:id1' etc 146Example: 'Malloc0:0 Malloc1:1 Malloc5:2' 147*** The LUNs must pre-exist *** 148*** LUN0 (id = 0) is required *** 149*** LUN names cannot contain space or colon characters ***""") 150p.add_argument('pg_ig_mappings', help="""List of (Portal_Group_Tag:Initiator_Group_Tag) mappings 151Whitespace separated, quoted, mapping defined with colon 152separated list of "tags" (int > 0) 153Example: '1:1 2:2 2:1' 154*** The Portal/Initiator Groups must be precreated ***""") 155p.add_argument('queue_depth', help='Desired target queue depth', type=int) 156p.add_argument('chap_disabled', help="""CHAP authentication should be disabled for this target node. 157*** Mutually exclusive with chap_required ***""", type=int) 158p.add_argument('chap_required', help="""CHAP authentication should be required for this target node. 159*** Mutually exclusive with chap_disabled ***""", type=int) 160p.add_argument('chap_mutual', help='CHAP authentication should be mutual/bidirectional.', type=int) 161p.add_argument('chap_auth_group', help="""Authentication group ID for this target node. 162*** Authentication group must be precreated ***""", type=int) 163p.set_defaults(func=construct_target_node) 164 165 166def construct_malloc_bdev(args): 167 num_blocks = (args.total_size * 1024 * 1024) / args.block_size 168 params = {'num_blocks': num_blocks, 'block_size': args.block_size} 169 print_array(jsonrpc_call('construct_malloc_bdev', params)) 170 171p = subparsers.add_parser('construct_malloc_bdev', help='Add a bdev with malloc backend') 172p.add_argument('total_size', help='Size of malloc bdev in MB (int > 0)', type=int) 173p.add_argument('block_size', help='Block size for this bdev', type=int) 174p.set_defaults(func=construct_malloc_bdev) 175 176 177def construct_aio_bdev(args): 178 params = {'name': args.name, 179 'fname': args.fname} 180 181 print_array(jsonrpc_call('construct_aio_bdev', params)) 182 183p = subparsers.add_parser('construct_aio_bdev', help='Add a bdev with aio backend') 184p.add_argument('fname', help='Path to device or file (ex: /dev/sda)') 185p.add_argument('name', help='Block device name') 186p.set_defaults(func=construct_aio_bdev) 187 188def construct_nvme_bdev(args): 189 params = {'name': args.name, 190 'trtype': args.trtype, 191 'traddr': args.traddr} 192 193 if args.adrfam: 194 params['adrfam'] = args.adrfam 195 196 if args.trsvcid: 197 params['trsvcid'] = args.trsvcid 198 199 if args.subnqn: 200 params['subnqn'] = args.subnqn 201 202 jsonrpc_call('construct_nvme_bdev', params) 203 204p = subparsers.add_parser('construct_nvme_bdev', help='Add bdev with nvme backend') 205p.add_argument('-b', '--name', help="Name of the bdev", required=True) 206p.add_argument('-t', '--trtype', help='NVMe-oF target trtype: e.g., rdma, pcie', required=True) 207p.add_argument('-a', '--traddr', help='NVMe-oF target address: e.g., an ip address or BDF', required=True) 208p.add_argument('-f', '--adrfam', help='NVMe-oF target adrfam: e.g., ipv4, ipv6, ib, fc, intra_host') 209p.add_argument('-s', '--trsvcid', help='NVMe-oF target trsvcid: e.g., a port number') 210p.add_argument('-n', '--subnqn', help='NVMe-oF target subnqn') 211p.set_defaults(func=construct_nvme_bdev) 212 213def construct_rbd_bdev(args): 214 params = { 215 'pool_name': args.pool_name, 216 'rbd_name': args.rbd_name, 217 'block_size': args.block_size, 218 } 219 print_array(jsonrpc_call('construct_rbd_bdev', params)) 220 221p = subparsers.add_parser('construct_rbd_bdev', help='Add a bdev with ceph rbd backend') 222p.add_argument('pool_name', help='rbd pool name') 223p.add_argument('rbd_name', help='rbd image name') 224p.add_argument('block_size', help='rbd block size', type=int) 225p.set_defaults(func=construct_rbd_bdev) 226 227def set_trace_flag(args): 228 params = {'flag': args.flag} 229 jsonrpc_call('set_trace_flag', params) 230 231p = subparsers.add_parser('set_trace_flag', help='set trace flag') 232p.add_argument('flag', help='trace mask we want to set. (for example "debug").') 233p.set_defaults(func=set_trace_flag) 234 235 236def clear_trace_flag(args): 237 params = {'flag': args.flag} 238 jsonrpc_call('clear_trace_flag', params) 239 240p = subparsers.add_parser('clear_trace_flag', help='clear trace flag') 241p.add_argument('flag', help='trace mask we want to clear. (for example "debug").') 242p.set_defaults(func=clear_trace_flag) 243 244 245def get_trace_flags(args): 246 print_dict(jsonrpc_call('get_trace_flags')) 247 248p = subparsers.add_parser('get_trace_flags', help='get trace flags') 249p.set_defaults(func=get_trace_flags) 250 251 252def add_portal_group(args): 253 # parse out portal list host1:port1 host2:port2 254 portals = [] 255 for p in args.portal_list: 256 host_port = p.split(':') 257 portals.append({'host': host_port[0], 'port': host_port[1]}) 258 259 params = {'tag': args.tag, 'portals': portals} 260 jsonrpc_call('add_portal_group', params) 261 262p = subparsers.add_parser('add_portal_group', help='Add a portal group') 263p.add_argument('tag', help='Portal group tag (unique, integer > 0)', type=int) 264p.add_argument('portal_list', nargs=argparse.REMAINDER, help="""List of portals in 'host:port' format, separated by whitespace 265Example: '192.168.100.100:3260' '192.168.100.100:3261'""") 266p.set_defaults(func=add_portal_group) 267 268 269def add_initiator_group(args): 270 initiators = [] 271 netmasks = [] 272 for i in args.initiator_list.strip().split(' '): 273 initiators.append(i) 274 for n in args.netmask_list.strip().split(' '): 275 netmasks.append(n) 276 277 params = {'tag': args.tag, 'initiators': initiators, 'netmasks': netmasks} 278 jsonrpc_call('add_initiator_group', params) 279 280 281p = subparsers.add_parser('add_initiator_group', help='Add an initiator group') 282p.add_argument('tag', help='Initiator group tag (unique, integer > 0)', type=int) 283p.add_argument('initiator_list', help="""Whitespace-separated list of initiator hostnames or IP addresses, 284enclosed in quotes. Example: 'ALL' or '127.0.0.1 192.168.200.100'""") 285p.add_argument('netmask_list', help="""Whitespace-separated list of initiator netmasks enclosed in quotes. 286Example: '255.255.0.0 255.248.0.0' etc""") 287p.set_defaults(func=add_initiator_group) 288 289 290def delete_target_node(args): 291 params = {'name': args.target_node_name} 292 jsonrpc_call('delete_target_node', params) 293 294p = subparsers.add_parser('delete_target_node', help='Delete a target node') 295p.add_argument('target_node_name', help='Target node name to be deleted. Example: iqn.2016-06.io.spdk:disk1.') 296p.set_defaults(func=delete_target_node) 297 298 299def delete_portal_group(args): 300 params = {'tag': args.tag} 301 jsonrpc_call('delete_portal_group', params) 302 303p = subparsers.add_parser('delete_portal_group', help='Delete a portal group') 304p.add_argument('tag', help='Portal group tag (unique, integer > 0)', type=int) 305p.set_defaults(func=delete_portal_group) 306 307 308def delete_initiator_group(args): 309 params = {'tag': args.tag} 310 jsonrpc_call('delete_initiator_group', params) 311 312p = subparsers.add_parser('delete_initiator_group', help='Delete an initiator group') 313p.add_argument('tag', help='Initiator group tag (unique, integer > 0)', type=int) 314p.set_defaults(func=delete_initiator_group) 315 316 317def get_iscsi_connections(args): 318 print_dict(jsonrpc_call('get_iscsi_connections')) 319 320p = subparsers.add_parser('get_iscsi_connections', help='Display iSCSI connections') 321p.set_defaults(func=get_iscsi_connections) 322 323 324def get_scsi_devices(args): 325 print_dict(jsonrpc_call('get_scsi_devices')) 326 327p = subparsers.add_parser('get_scsi_devices', help='Display SCSI devices') 328p.set_defaults(func=get_scsi_devices) 329 330 331def add_ip_address(args): 332 params = {'ifc_index': args.ifc_index, 'ip_address': args.ip_addr} 333 jsonrpc_call('add_ip_address', params) 334 335p = subparsers.add_parser('add_ip_address', help='Add IP address') 336p.add_argument('ifc_index', help='ifc index of the nic device.', type=int) 337p.add_argument('ip_addr', help='ip address will be added.') 338p.set_defaults(func=add_ip_address) 339 340 341def delete_ip_address(args): 342 params = {'ifc_index': args.ifc_index, 'ip_address': args.ip_addr} 343 jsonrpc_call('delete_ip_address', params) 344 345p = subparsers.add_parser('delete_ip_address', help='Delete IP address') 346p.add_argument('ifc_index', help='ifc index of the nic device.', type=int) 347p.add_argument('ip_addr', help='ip address will be deleted.') 348p.set_defaults(func=delete_ip_address) 349 350 351def get_interfaces(args): 352 print_dict(jsonrpc_call('get_interfaces')) 353 354p = subparsers.add_parser('get_interfaces', help='Display current interface list') 355p.set_defaults(func=get_interfaces) 356 357def get_bdevs(args): 358 print_dict(jsonrpc_call('get_bdevs')) 359 360p = subparsers.add_parser('get_bdevs', help='Display current blockdev list') 361p.set_defaults(func=get_bdevs) 362 363 364def delete_bdev(args): 365 params = {'name': args.bdev_name} 366 jsonrpc_call('delete_bdev', params) 367 368p = subparsers.add_parser('delete_bdev', help='Delete a blockdev') 369p.add_argument('bdev_name', help='Blockdev name to be deleted. Example: Malloc0.') 370p.set_defaults(func=delete_bdev) 371 372 373def get_nvmf_subsystems(args): 374 print_dict(jsonrpc_call('get_nvmf_subsystems')) 375 376p = subparsers.add_parser('get_nvmf_subsystems', help='Display nvmf subsystems') 377p.set_defaults(func=get_nvmf_subsystems) 378 379def construct_nvmf_subsystem(args): 380 listen_addresses = [dict(u.split(":") for u in a.split(" ")) for a in args.listen.split(",")] 381 382 params = { 383 'core': args.core, 384 'mode': args.mode, 385 'nqn': args.nqn, 386 'listen_addresses': listen_addresses, 387 'serial_number': args.serial_number, 388 } 389 390 if args.hosts: 391 hosts = [] 392 for u in args.hosts.strip().split(" "): 393 hosts.append(u) 394 params['hosts'] = hosts 395 396 if args.namespaces: 397 namespaces = [] 398 for u in args.namespaces.strip().split(" "): 399 namespaces.append(u) 400 params['namespaces'] = namespaces 401 402 if args.pci_address: 403 params['pci_address'] = args.pci_address 404 405 jsonrpc_call('construct_nvmf_subsystem', params) 406 407p = subparsers.add_parser('construct_nvmf_subsystem', help='Add a nvmf subsystem') 408p.add_argument("-c", "--core", help='The core Nvmf target run on', type=int, default=-1) 409p.add_argument('mode', help='Target mode: Virtual or Direct') 410p.add_argument('nqn', help='Target nqn(ASCII)') 411p.add_argument('listen', help="""comma-separated list of Listen <transport:transport_name traddr:address trsvcid:port_id> pairs enclosed 412in quotes. Format: 'transport:transport0 traddr:traddr0 trsvcid:trsvcid0,transport:transport1 traddr:traddr1 trsvcid:trsvcid1' etc 413Example: 'transport:RDMA traddr:192.168.100.8 trsvcid:4420,transport:RDMA traddr:192.168.100.9 trsvcid:4420'""") 414p.add_argument('hosts', help="""Whitespace-separated list of host nqn list. 415Format: 'nqn1 nqn2' etc 416Example: 'nqn.2016-06.io.spdk:init nqn.2016-07.io.spdk:init'""") 417p.add_argument("-p", "--pci_address", help="""Valid if mode == Direct. 418Format: 'domain:device:function' etc 419Example: '0000:00:01.0'""") 420p.add_argument("-s", "--serial_number", help="""Valid if mode == Virtual. 421Format: 'sn' etc 422Example: 'SPDK00000000000001'""", default='0000:00:01.0') 423p.add_argument("-n", "--namespaces", help="""Whitespace-separated list of namespaces. 424Format: 'dev1 dev2 dev3' etc 425Example: 'Malloc0 Malloc1 Malloc2' 426*** The devices must pre-exist ***""") 427p.set_defaults(func=construct_nvmf_subsystem) 428 429def delete_nvmf_subsystem(args): 430 params = {'nqn': args.subsystem_nqn} 431 jsonrpc_call('delete_nvmf_subsystem', params) 432 433p = subparsers.add_parser('delete_nvmf_subsystem', help='Delete a nvmf subsystem') 434p.add_argument('subsystem_nqn', help='subsystem nqn to be deleted. Example: nqn.2016-06.io.spdk:cnode1.') 435p.set_defaults(func=delete_nvmf_subsystem) 436 437def kill_instance(args): 438 params = {'sig_name': args.sig_name} 439 jsonrpc_call('kill_instance', params) 440 441p = subparsers.add_parser('kill_instance', help='Send signal to instance') 442p.add_argument('sig_name', help='signal will be sent to server.') 443p.set_defaults(func=kill_instance) 444 445def get_vhost_scsi_controllers(args): 446 print_dict(jsonrpc_call('get_vhost_scsi_controllers')) 447 448p = subparsers.add_parser('get_vhost_scsi_controllers', help='List vhost controllers') 449p.set_defaults(func=get_vhost_scsi_controllers) 450 451def construct_vhost_scsi_controller(args): 452 params = {'ctrlr': args.ctrlr} 453 454 if args.cpumask: 455 params['cpumask'] = args.cpumask 456 457 jsonrpc_call('construct_vhost_scsi_controller', params) 458 459p = subparsers.add_parser('construct_vhost_scsi_controller', help='Add new vhost controller') 460p.add_argument('ctrlr', help='controller name') 461p.add_argument('--cpumask', help='cpu mask for this controller') 462p.set_defaults(func=construct_vhost_scsi_controller) 463 464def add_vhost_scsi_lun(args): 465 params = { 466 'ctrlr': args.ctrlr, 467 'scsi_dev_num': args.scsi_dev_num, 468 'lun_name': args.lun_name 469 } 470 jsonrpc_call('add_vhost_scsi_lun', params) 471 472p = subparsers.add_parser('add_vhost_scsi_lun', help='Add lun to vhost controller') 473p.add_argument('ctrlr', help='conntroller name where add lun') 474p.add_argument('scsi_dev_num', help='scsi_dev_num', type=int) 475p.add_argument('lun_name', help='lun name') 476p.set_defaults(func=add_vhost_scsi_lun) 477 478args = parser.parse_args() 479args.func(args) 480