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