xref: /spdk/scripts/rpc.py (revision a6014eb2adf0c95816b23ef94a911005fa047511)
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_null_bdev(args):
178    num_blocks = (args.total_size * 1024 * 1024) / args.block_size
179    params = {'name': args.name, 'num_blocks': num_blocks, 'block_size': args.block_size}
180    print_array(jsonrpc_call('construct_null_bdev', params))
181
182p = subparsers.add_parser('construct_null_bdev', help='Add a bdev with null backend')
183p.add_argument('name', help='Block device name')
184p.add_argument('total_size', help='Size of null bdev in MB (int > 0)', type=int)
185p.add_argument('block_size', help='Block size for this bdev', type=int)
186p.set_defaults(func=construct_null_bdev)
187
188
189def construct_aio_bdev(args):
190    params = {'name': args.name,
191              'fname': args.fname}
192
193    print_array(jsonrpc_call('construct_aio_bdev', params))
194
195p = subparsers.add_parser('construct_aio_bdev', help='Add a bdev with aio backend')
196p.add_argument('fname', help='Path to device or file (ex: /dev/sda)')
197p.add_argument('name', help='Block device name')
198p.set_defaults(func=construct_aio_bdev)
199
200def construct_nvme_bdev(args):
201    params = {'name': args.name,
202              'trtype': args.trtype,
203              'traddr': args.traddr}
204
205    if args.adrfam:
206        params['adrfam'] = args.adrfam
207
208    if args.trsvcid:
209        params['trsvcid'] = args.trsvcid
210
211    if args.subnqn:
212        params['subnqn'] = args.subnqn
213
214    jsonrpc_call('construct_nvme_bdev', params)
215
216p = subparsers.add_parser('construct_nvme_bdev', help='Add bdev with nvme backend')
217p.add_argument('-b', '--name', help="Name of the bdev", required=True)
218p.add_argument('-t', '--trtype', help='NVMe-oF target trtype: e.g., rdma, pcie', required=True)
219p.add_argument('-a', '--traddr', help='NVMe-oF target address: e.g., an ip address or BDF', required=True)
220p.add_argument('-f', '--adrfam', help='NVMe-oF target adrfam: e.g., ipv4, ipv6, ib, fc, intra_host')
221p.add_argument('-s', '--trsvcid', help='NVMe-oF target trsvcid: e.g., a port number')
222p.add_argument('-n', '--subnqn', help='NVMe-oF target subnqn')
223p.set_defaults(func=construct_nvme_bdev)
224
225def construct_rbd_bdev(args):
226    params = {
227        'pool_name': args.pool_name,
228        'rbd_name': args.rbd_name,
229        'block_size': args.block_size,
230    }
231    print_array(jsonrpc_call('construct_rbd_bdev', params))
232
233p = subparsers.add_parser('construct_rbd_bdev', help='Add a bdev with ceph rbd backend')
234p.add_argument('pool_name', help='rbd pool name')
235p.add_argument('rbd_name', help='rbd image name')
236p.add_argument('block_size', help='rbd block size', type=int)
237p.set_defaults(func=construct_rbd_bdev)
238
239def construct_error_bdev(args):
240    params = {'base_name': args.base_name}
241    jsonrpc_call('construct_error_bdev', params)
242p = subparsers.add_parser('construct_error_bdev', help='Add bdev with error injection backend')
243p.add_argument('base_name', help='base bdev name')
244p.set_defaults(func=construct_error_bdev)
245
246def set_trace_flag(args):
247    params = {'flag': args.flag}
248    jsonrpc_call('set_trace_flag', params)
249
250p = subparsers.add_parser('set_trace_flag', help='set trace flag')
251p.add_argument('flag', help='trace mask we want to set. (for example "debug").')
252p.set_defaults(func=set_trace_flag)
253
254
255def clear_trace_flag(args):
256    params = {'flag': args.flag}
257    jsonrpc_call('clear_trace_flag', params)
258
259p = subparsers.add_parser('clear_trace_flag', help='clear trace flag')
260p.add_argument('flag', help='trace mask we want to clear. (for example "debug").')
261p.set_defaults(func=clear_trace_flag)
262
263
264def get_trace_flags(args):
265    print_dict(jsonrpc_call('get_trace_flags'))
266
267p = subparsers.add_parser('get_trace_flags', help='get trace flags')
268p.set_defaults(func=get_trace_flags)
269
270
271def add_portal_group(args):
272    # parse out portal list host1:port1 host2:port2
273    portals = []
274    for p in args.portal_list:
275        host_port = p.split(':')
276        portals.append({'host': host_port[0], 'port': host_port[1]})
277
278    params = {'tag': args.tag, 'portals': portals}
279    jsonrpc_call('add_portal_group', params)
280
281p = subparsers.add_parser('add_portal_group', help='Add a portal group')
282p.add_argument('tag', help='Portal group tag (unique, integer > 0)', type=int)
283p.add_argument('portal_list', nargs=argparse.REMAINDER, help="""List of portals in 'host:port' format, separated by whitespace
284Example: '192.168.100.100:3260' '192.168.100.100:3261'""")
285p.set_defaults(func=add_portal_group)
286
287
288def add_initiator_group(args):
289    initiators = []
290    netmasks = []
291    for i in args.initiator_list.strip().split(' '):
292        initiators.append(i)
293    for n in args.netmask_list.strip().split(' '):
294        netmasks.append(n)
295
296    params = {'tag': args.tag, 'initiators': initiators, 'netmasks': netmasks}
297    jsonrpc_call('add_initiator_group', params)
298
299
300p = subparsers.add_parser('add_initiator_group', help='Add an initiator group')
301p.add_argument('tag', help='Initiator group tag (unique, integer > 0)', type=int)
302p.add_argument('initiator_list', help="""Whitespace-separated list of initiator hostnames or IP addresses,
303enclosed in quotes.  Example: 'ALL' or '127.0.0.1 192.168.200.100'""")
304p.add_argument('netmask_list', help="""Whitespace-separated list of initiator netmasks enclosed in quotes.
305Example: '255.255.0.0 255.248.0.0' etc""")
306p.set_defaults(func=add_initiator_group)
307
308
309def delete_target_node(args):
310    params = {'name': args.target_node_name}
311    jsonrpc_call('delete_target_node', params)
312
313p = subparsers.add_parser('delete_target_node', help='Delete a target node')
314p.add_argument('target_node_name', help='Target node name to be deleted. Example: iqn.2016-06.io.spdk:disk1.')
315p.set_defaults(func=delete_target_node)
316
317
318def delete_portal_group(args):
319    params = {'tag': args.tag}
320    jsonrpc_call('delete_portal_group', params)
321
322p = subparsers.add_parser('delete_portal_group', help='Delete a portal group')
323p.add_argument('tag', help='Portal group tag (unique, integer > 0)', type=int)
324p.set_defaults(func=delete_portal_group)
325
326
327def delete_initiator_group(args):
328    params = {'tag': args.tag}
329    jsonrpc_call('delete_initiator_group', params)
330
331p = subparsers.add_parser('delete_initiator_group', help='Delete an initiator group')
332p.add_argument('tag', help='Initiator group tag (unique, integer > 0)', type=int)
333p.set_defaults(func=delete_initiator_group)
334
335
336def get_iscsi_connections(args):
337    print_dict(jsonrpc_call('get_iscsi_connections'))
338
339p = subparsers.add_parser('get_iscsi_connections', help='Display iSCSI connections')
340p.set_defaults(func=get_iscsi_connections)
341
342
343def get_scsi_devices(args):
344    print_dict(jsonrpc_call('get_scsi_devices'))
345
346p = subparsers.add_parser('get_scsi_devices', help='Display SCSI devices')
347p.set_defaults(func=get_scsi_devices)
348
349
350def add_ip_address(args):
351    params = {'ifc_index': args.ifc_index, 'ip_address': args.ip_addr}
352    jsonrpc_call('add_ip_address', params)
353
354p = subparsers.add_parser('add_ip_address', help='Add IP address')
355p.add_argument('ifc_index', help='ifc index of the nic device.', type=int)
356p.add_argument('ip_addr', help='ip address will be added.')
357p.set_defaults(func=add_ip_address)
358
359
360def delete_ip_address(args):
361    params = {'ifc_index': args.ifc_index, 'ip_address': args.ip_addr}
362    jsonrpc_call('delete_ip_address', params)
363
364p = subparsers.add_parser('delete_ip_address', help='Delete IP address')
365p.add_argument('ifc_index', help='ifc index of the nic device.', type=int)
366p.add_argument('ip_addr', help='ip address will be deleted.')
367p.set_defaults(func=delete_ip_address)
368
369
370def get_interfaces(args):
371    print_dict(jsonrpc_call('get_interfaces'))
372
373p = subparsers.add_parser('get_interfaces', help='Display current interface list')
374p.set_defaults(func=get_interfaces)
375
376def get_bdevs(args):
377    print_dict(jsonrpc_call('get_bdevs'))
378
379p = subparsers.add_parser('get_bdevs', help='Display current blockdev list')
380p.set_defaults(func=get_bdevs)
381
382
383def delete_bdev(args):
384    params = {'name': args.bdev_name}
385    jsonrpc_call('delete_bdev', params)
386
387p = subparsers.add_parser('delete_bdev', help='Delete a blockdev')
388p.add_argument('bdev_name', help='Blockdev name to be deleted. Example: Malloc0.')
389p.set_defaults(func=delete_bdev)
390
391
392def get_nvmf_subsystems(args):
393    print_dict(jsonrpc_call('get_nvmf_subsystems'))
394
395p = subparsers.add_parser('get_nvmf_subsystems', help='Display nvmf subsystems')
396p.set_defaults(func=get_nvmf_subsystems)
397
398def construct_nvmf_subsystem(args):
399    listen_addresses = [dict(u.split(":") for u in a.split(" ")) for a in args.listen.split(",")]
400
401    params = {
402        'core': args.core,
403        'mode': args.mode,
404        'nqn': args.nqn,
405        'listen_addresses': listen_addresses,
406        'serial_number': args.serial_number,
407    }
408
409    if args.hosts:
410        hosts = []
411        for u in args.hosts.strip().split(" "):
412            hosts.append(u)
413        params['hosts'] = hosts
414
415    if args.namespaces:
416        namespaces = []
417        for u in args.namespaces.strip().split(" "):
418            namespaces.append(u)
419        params['namespaces'] = namespaces
420
421    if args.pci_address:
422        params['pci_address'] = args.pci_address
423
424    jsonrpc_call('construct_nvmf_subsystem', params)
425
426p = subparsers.add_parser('construct_nvmf_subsystem', help='Add a nvmf subsystem')
427p.add_argument("-c", "--core", help='The core Nvmf target run on', type=int, default=-1)
428p.add_argument('mode', help='Target mode: Virtual or Direct')
429p.add_argument('nqn', help='Target nqn(ASCII)')
430p.add_argument('listen', help="""comma-separated list of Listen <trtype:transport_name traddr:address trsvcid:port_id> pairs enclosed
431in quotes.  Format:  'trtype:transport0 traddr:traddr0 trsvcid:trsvcid0,trtype:transport1 traddr:traddr1 trsvcid:trsvcid1' etc
432Example: 'trtype:RDMA traddr:192.168.100.8 trsvcid:4420,trtype:RDMA traddr:192.168.100.9 trsvcid:4420'""")
433p.add_argument('hosts', help="""Whitespace-separated list of host nqn list.
434Format:  'nqn1 nqn2' etc
435Example: 'nqn.2016-06.io.spdk:init nqn.2016-07.io.spdk:init'""")
436p.add_argument("-p", "--pci_address", help="""Valid if mode == Direct.
437Format:  'domain:device:function' etc
438Example: '0000:00:01.0'""")
439p.add_argument("-s", "--serial_number", help="""Valid if mode == Virtual.
440Format:  'sn' etc
441Example: 'SPDK00000000000001'""", default='0000:00:01.0')
442p.add_argument("-n", "--namespaces", help="""Whitespace-separated list of namespaces.
443Format:  'dev1 dev2 dev3' etc
444Example: 'Malloc0 Malloc1 Malloc2'
445*** The devices must pre-exist ***""")
446p.set_defaults(func=construct_nvmf_subsystem)
447
448def delete_nvmf_subsystem(args):
449    params = {'nqn': args.subsystem_nqn}
450    jsonrpc_call('delete_nvmf_subsystem', params)
451
452p = subparsers.add_parser('delete_nvmf_subsystem', help='Delete a nvmf subsystem')
453p.add_argument('subsystem_nqn', help='subsystem nqn to be deleted. Example: nqn.2016-06.io.spdk:cnode1.')
454p.set_defaults(func=delete_nvmf_subsystem)
455
456def bdev_inject_error(args):
457    params = {
458        'name': args.name,
459        'io_type': args.io_type,
460        'error_type': args.error_type,
461        'num': args.num,
462    }
463
464    jsonrpc_call('bdev_inject_error', params)
465
466p = subparsers.add_parser('bdev_inject_error', help='bdev inject error')
467p.add_argument('name', help="""the name of the error injection bdev""")
468p.add_argument('io_type', help="""io_type: 'clear' 'read' 'write' 'unmap' 'flush' 'all'""")
469p.add_argument('error_type', help="""error_type: 'failure' 'pending'""")
470p.add_argument('-n', '--num', help='the number of commands you want to fail', type=int, default=1)
471p.set_defaults(func=bdev_inject_error)
472
473def kill_instance(args):
474    params = {'sig_name': args.sig_name}
475    jsonrpc_call('kill_instance', params)
476
477p = subparsers.add_parser('kill_instance', help='Send signal to instance')
478p.add_argument('sig_name', help='signal will be sent to server.')
479p.set_defaults(func=kill_instance)
480
481def get_vhost_scsi_controllers(args):
482    print_dict(jsonrpc_call('get_vhost_scsi_controllers'))
483
484p = subparsers.add_parser('get_vhost_scsi_controllers', help='List vhost controllers')
485p.set_defaults(func=get_vhost_scsi_controllers)
486
487def construct_vhost_scsi_controller(args):
488    params = {'ctrlr': args.ctrlr}
489
490    if args.cpumask:
491        params['cpumask'] = args.cpumask
492
493    jsonrpc_call('construct_vhost_scsi_controller', params)
494
495p = subparsers.add_parser('construct_vhost_scsi_controller', help='Add new vhost controller')
496p.add_argument('ctrlr', help='controller name')
497p.add_argument('--cpumask', help='cpu mask for this controller')
498p.set_defaults(func=construct_vhost_scsi_controller)
499
500def remove_vhost_scsi_controller(args):
501    params = {'ctrlr': args.ctrlr}
502    jsonrpc_call('remove_vhost_scsi_controller', params)
503
504p = subparsers.add_parser('remove_vhost_scsi_controller', help='Remove vhost controller')
505p.add_argument('ctrlr', help='controller name')
506p.set_defaults(func=remove_vhost_scsi_controller)
507
508def add_vhost_scsi_lun(args):
509    params = {
510        'ctrlr': args.ctrlr,
511        'scsi_dev_num': args.scsi_dev_num,
512        'lun_name': args.lun_name
513    }
514    jsonrpc_call('add_vhost_scsi_lun', params)
515
516p = subparsers.add_parser('add_vhost_scsi_lun', help='Add lun to vhost controller')
517p.add_argument('ctrlr', help='conntroller name where add lun')
518p.add_argument('scsi_dev_num', help='scsi_dev_num', type=int)
519p.add_argument('lun_name', help='lun name')
520p.set_defaults(func=add_vhost_scsi_lun)
521
522def remove_vhost_scsi_dev(args):
523    params = {
524        'ctrlr': args.ctrlr,
525        'scsi_dev_num': args.scsi_dev_num,
526    }
527    jsonrpc_call('remove_vhost_scsi_dev', params)
528
529p = subparsers.add_parser('remove_vhost_scsi_dev', help='Remove device from vhost controller')
530p.add_argument('ctrlr', help='controller name to remove device from')
531p.add_argument('scsi_dev_num', help='scsi_dev_num', type=int)
532p.set_defaults(func=remove_vhost_scsi_dev)
533
534def construct_vhost_blk_controller(args):
535    params = {
536        'ctrlr': args.ctrlr,
537        'dev_name': args.dev_name,
538    }
539    if args.cpumask:
540        params['cpumask'] = args.cpumask
541    if args.readonly:
542        params['readonly'] = args.readonly
543    jsonrpc_call('construct_vhost_blk_controller', params)
544
545p = subparsers.add_parser('construct_vhost_blk_controller', help='Add a new vhost block controller')
546p.add_argument('ctrlr', help='controller name')
547p.add_argument('dev_name', help='device name')
548p.add_argument('--cpumask', help='cpu mask for this controller')
549p.add_argument("-r", "--readonly", action='store_true', help='Set controller as read-only')
550p.set_defaults(func=construct_vhost_blk_controller)
551
552def remove_vhost_blk_controller(args):
553    params = {'ctrlr': args.ctrlr}
554    jsonrpc_call('remove_vhost_blk_controller', params)
555
556p = subparsers.add_parser('remove_vhost_blk_controller', help='Remove a vhost block controller')
557p.add_argument('ctrlr', help='controller name')
558p.set_defaults(func=remove_vhost_blk_controller)
559
560def get_vhost_blk_controllers(args):
561    print_dict(jsonrpc_call('get_vhost_blk_controllers'))
562
563p = subparsers.add_parser('get_vhost_blk_controllers', help='List vhost block controllers')
564p.set_defaults(func=get_vhost_blk_controllers)
565
566args = parser.parse_args()
567args.func(args)
568