xref: /spdk/scripts/rpc.py (revision 04c48172b9879a8824de83c842087d871c433d12)
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