392 lines
12 KiB
Python
392 lines
12 KiB
Python
# vim: tabstop=4 shiftwidth=4 softtabstop=4
|
|
|
|
# Copyright 2011 Nicira Networks, Inc.
|
|
# Copyright 2011 Citrix Systems
|
|
#
|
|
# Licensed under the Apache License, Version 2.0 (the "License"); you may
|
|
# not use this file except in compliance with the License. You may obtain
|
|
# a copy of the License at
|
|
#
|
|
# http://www.apache.org/licenses/LICENSE-2.0
|
|
#
|
|
# Unless required by applicable law or agreed to in writing, software
|
|
# distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
|
|
# WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
|
|
# License for the specific language governing permissions and limitations
|
|
# under the License.
|
|
# @author: Somik Behera, Nicira Networks, Inc.
|
|
# @author: Brad Hall, Nicira Networks, Inc.
|
|
|
|
import httplib
|
|
import logging as LOG
|
|
import json
|
|
import socket
|
|
import sys
|
|
import urllib
|
|
|
|
from manager import QuantumManager
|
|
from optparse import OptionParser
|
|
from client import Client
|
|
|
|
FORMAT = "json"
|
|
|
|
### -- Core CLI functions
|
|
|
|
|
|
def list_nets(manager, *args):
|
|
tenant_id = args[0]
|
|
networks = manager.get_all_networks(tenant_id)
|
|
print "Virtual Networks on Tenant:%s\n" % tenant_id
|
|
for net in networks:
|
|
id = net["net-id"]
|
|
name = net["net-name"]
|
|
print "\tNetwork ID:%s \n\tNetwork Name:%s \n" % (id, name)
|
|
|
|
|
|
def api_list_nets(client, *args):
|
|
tenant_id = args[0]
|
|
res = client.list_networks()
|
|
LOG.debug(res)
|
|
print "Virtual Networks on Tenant:%s\n" % tenant_id
|
|
for n in res["networks"]:
|
|
net_id = n["id"]
|
|
print "\tNetwork ID:%s\n" % (net_id)
|
|
# TODO(bgh): we should make this call pass back the name too
|
|
# name = n["net-name"]
|
|
# LOG.info("\tNetwork ID:%s \n\tNetwork Name:%s \n" % (id, name))
|
|
|
|
|
|
def create_net(manager, *args):
|
|
tid, name = args
|
|
new_net_id = manager.create_network(tid, name)
|
|
print "Created a new Virtual Network with ID:%s\n" % new_net_id
|
|
|
|
|
|
def api_create_net(client, *args):
|
|
tid, name = args
|
|
data = {'network': {'net-name': '%s' % name}}
|
|
res = client.create_network(data)
|
|
LOG.debug(res)
|
|
nid = None
|
|
try:
|
|
nid = res["networks"]["network"]["id"]
|
|
except Exception, e:
|
|
print "Failed to create network"
|
|
# TODO(bgh): grab error details from ws request result
|
|
return
|
|
print "Created a new Virtual Network with ID:%s\n" % nid
|
|
|
|
|
|
def delete_net(manager, *args):
|
|
tid, nid = args
|
|
manager.delete_network(tid, nid)
|
|
print "Deleted Virtual Network with ID:%s" % nid
|
|
|
|
|
|
def api_delete_net(client, *args):
|
|
tid, nid = args
|
|
try:
|
|
res = client.delete_network(nid)
|
|
print "Deleted Virtual Network with ID:%s" % nid
|
|
except Exception, e:
|
|
print "Failed to delete network"
|
|
LOG.error("Failed to delete network: %s" % e)
|
|
|
|
|
|
def detail_net(manager, *args):
|
|
tid, nid = args
|
|
iface_list = manager.get_network_details(tid, nid)
|
|
print "Remote Interfaces on Virtual Network:%s\n" % nid
|
|
for iface in iface_list:
|
|
print "\tRemote interface:%s" % iface
|
|
|
|
|
|
def api_detail_net(client, *args):
|
|
tid, nid = args
|
|
try:
|
|
res = client.list_network_details(nid)["networks"]["network"]
|
|
except Exception, e:
|
|
LOG.error("Failed to get network details: %s" % e)
|
|
return
|
|
|
|
try:
|
|
ports = client.list_ports(nid)
|
|
except Exception, e:
|
|
LOG.error("Failed to list ports: %s" % e)
|
|
return
|
|
|
|
print "Network %s (%s)" % (res['name'], res['id'])
|
|
print "Remote Interfaces on Virtual Network:%s\n" % nid
|
|
for port in ports["ports"]:
|
|
pid = port["id"]
|
|
res = client.list_port_attachments(nid, pid)
|
|
LOG.debug(res)
|
|
remote_iface = res["attachment"]
|
|
print "\tRemote interface:%s" % remote_iface
|
|
|
|
|
|
def rename_net(manager, *args):
|
|
tid, nid, name = args
|
|
manager.rename_network(tid, nid, name)
|
|
print "Renamed Virtual Network with ID:%s" % nid
|
|
|
|
|
|
def api_rename_net(client, *args):
|
|
tid, nid, name = args
|
|
data = {'network': {'net-name': '%s' % name}}
|
|
try:
|
|
res = client.update_network(nid, data)
|
|
except Exception, e:
|
|
LOG.error("Failed to rename network %s: %s" % (nid, e))
|
|
return
|
|
LOG.debug(res)
|
|
print "Renamed Virtual Network with ID:%s" % nid
|
|
|
|
|
|
def list_ports(manager, *args):
|
|
tid, nid = args
|
|
ports = manager.get_all_ports(tid, nid)
|
|
print "Ports on Virtual Network:%s\n" % nid
|
|
for port in ports:
|
|
print "\tVirtual Port:%s" % port["port-id"]
|
|
|
|
|
|
def api_list_ports(client, *args):
|
|
tid, nid = args
|
|
try:
|
|
ports = client.list_ports(nid)
|
|
except Exception, e:
|
|
LOG.error("Failed to list ports: %s" % e)
|
|
return
|
|
|
|
LOG.debug(ports)
|
|
print "Ports on Virtual Network:%s\n" % nid
|
|
for port in ports["ports"]:
|
|
print "\tVirtual Port:%s" % port["id"]
|
|
|
|
|
|
def create_port(manager, *args):
|
|
tid, nid = args
|
|
new_port = manager.create_port(tid, nid)
|
|
print "Created Virtual Port:%s " \
|
|
"on Virtual Network:%s" % (new_port, nid)
|
|
|
|
|
|
def api_create_port(client, *args):
|
|
tid, nid = args
|
|
try:
|
|
res = client.create_port(nid)
|
|
except Exception, e:
|
|
LOG.error("Failed to create port: %s" % e)
|
|
return
|
|
new_port = res["ports"]["port"]["id"]
|
|
print "Created Virtual Port:%s " \
|
|
"on Virtual Network:%s" % (new_port, nid)
|
|
|
|
|
|
def delete_port(manager, *args):
|
|
tid, nid, pid = args
|
|
manager.delete_port(tid, nid, pid)
|
|
LOG.info("Deleted Virtual Port:%s " \
|
|
"on Virtual Network:%s" % (pid, nid))
|
|
|
|
|
|
def api_delete_port(client, *args):
|
|
tid, nid, pid = args
|
|
try:
|
|
res = client.delete_port(nid, pid)
|
|
except Exception, e:
|
|
LOG.error("Failed to delete port: %s" % e)
|
|
return
|
|
LOG.info("Deleted Virtual Port:%s " \
|
|
"on Virtual Network:%s" % (pid, nid))
|
|
print "Deleted Virtual Port:%s " \
|
|
"on Virtual Network:%s" % (pid, nid)
|
|
|
|
|
|
def detail_port(manager, *args):
|
|
tid, nid, pid = args
|
|
port_detail = manager.get_port_details(tid, nid, pid)
|
|
print "Virtual Port:%s on Virtual Network:%s " \
|
|
"contains remote interface:%s" % (pid, nid, port_detail)
|
|
|
|
|
|
def api_detail_port(client, *args):
|
|
tid, nid, pid = args
|
|
try:
|
|
port = client.list_port_details(nid, pid)["ports"]["port"]
|
|
except Exception, e:
|
|
LOG.error("Failed to get port details: %s" % e)
|
|
return
|
|
|
|
id = port["id"]
|
|
attachment = port["attachment"]
|
|
LOG.debug(port)
|
|
print "Virtual Port:%s on Virtual Network:%s " \
|
|
"contains remote interface:%s" % (pid, nid, attachment)
|
|
|
|
|
|
def plug_iface(manager, *args):
|
|
tid, nid, pid, vid = args
|
|
manager.plug_interface(tid, nid, pid, vid)
|
|
print "Plugged remote interface:%s " \
|
|
"into Virtual Network:%s" % (vid, nid)
|
|
|
|
|
|
def api_plug_iface(client, *args):
|
|
tid, nid, pid, vid = args
|
|
try:
|
|
data = {'port': {'attachment-id': '%s' % vid}}
|
|
res = client.attach_resource(nid, pid, data)
|
|
except Exception, e:
|
|
LOG.error("Failed to plug iface \"%s\" to port \"%s\": %s" % (vid,
|
|
pid, output))
|
|
return
|
|
LOG.debug(res)
|
|
print "Plugged interface \"%s\" to port:%s on network:%s" % (vid, pid, nid)
|
|
|
|
|
|
def unplug_iface(manager, *args):
|
|
tid, nid, pid = args
|
|
manager.unplug_interface(tid, nid, pid)
|
|
print "UnPlugged remote interface " \
|
|
"from Virtual Port:%s Virtual Network:%s" % (pid, nid)
|
|
|
|
|
|
def api_unplug_iface(client, *args):
|
|
tid, nid, pid = args
|
|
try:
|
|
res = client.detach_resource(nid, pid)
|
|
except Exception, e:
|
|
LOG.error("Failed to unplug iface from port \"%s\": %s" % (pid, e))
|
|
return
|
|
LOG.debug(res)
|
|
print "Unplugged interface from port:%s on network:%s" % (pid, nid)
|
|
|
|
|
|
commands = {
|
|
"list_nets": {
|
|
"func": list_nets,
|
|
"api_func": api_list_nets,
|
|
"args": ["tenant-id"]},
|
|
"create_net": {
|
|
"func": create_net,
|
|
"api_func": api_create_net,
|
|
"args": ["tenant-id", "net-name"]},
|
|
"delete_net": {
|
|
"func": delete_net,
|
|
"api_func": api_delete_net,
|
|
"args": ["tenant-id", "net-id"]},
|
|
"detail_net": {
|
|
"func": detail_net,
|
|
"api_func": api_detail_net,
|
|
"args": ["tenant-id", "net-id"]},
|
|
"rename_net": {
|
|
"func": rename_net,
|
|
"api_func": api_rename_net,
|
|
"args": ["tenant-id", "net-id", "new-name"]},
|
|
"list_ports": {
|
|
"func": list_ports,
|
|
"api_func": api_list_ports,
|
|
"args": ["tenant-id", "net-id"]},
|
|
"create_port": {
|
|
"func": create_port,
|
|
"api_func": api_create_port,
|
|
"args": ["tenant-id", "net-id"]},
|
|
"delete_port": {
|
|
"func": delete_port,
|
|
"api_func": api_delete_port,
|
|
"args": ["tenant-id", "net-id", "port-id"]},
|
|
"detail_port": {
|
|
"func": detail_port,
|
|
"api_func": api_detail_port,
|
|
"args": ["tenant-id", "net-id", "port-id"]},
|
|
"plug_iface": {
|
|
"func": plug_iface,
|
|
"api_func": api_plug_iface,
|
|
"args": ["tenant-id", "net-id", "port-id", "iface-id"]},
|
|
"unplug_iface": {
|
|
"func": unplug_iface,
|
|
"api_func": api_unplug_iface,
|
|
"args": ["tenant-id", "net-id", "port-id"]}, }
|
|
|
|
|
|
def help():
|
|
print "\nCommands:"
|
|
for k in commands.keys():
|
|
print " %s %s" % (k,
|
|
" ".join(["<%s>" % y for y in commands[k]["args"]]))
|
|
|
|
|
|
def build_args(cmd, cmdargs, arglist):
|
|
args = []
|
|
orig_arglist = arglist[:]
|
|
try:
|
|
for x in cmdargs:
|
|
args.append(arglist[0])
|
|
del arglist[0]
|
|
except Exception, e:
|
|
LOG.error("Not enough arguments for \"%s\" (expected: %d, got: %d)" % (
|
|
cmd, len(cmdargs), len(orig_arglist)))
|
|
print "Usage:\n %s %s" % (cmd,
|
|
" ".join(["<%s>" % y for y in commands[cmd]["args"]]))
|
|
return None
|
|
if len(arglist) > 0:
|
|
LOG.error("Too many arguments for \"%s\" (expected: %d, got: %d)" % (
|
|
cmd, len(cmdargs), len(orig_arglist)))
|
|
print "Usage:\n %s %s" % (cmd,
|
|
" ".join(["<%s>" % y for y in commands[cmd]["args"]]))
|
|
return None
|
|
return args
|
|
|
|
|
|
if __name__ == "__main__":
|
|
usagestr = "Usage: %prog [OPTIONS] <command> [args]"
|
|
parser = OptionParser(usage=usagestr)
|
|
parser.add_option("-l", "--load-plugin", dest="load_plugin",
|
|
action="store_true", default=False,
|
|
help="Load plugin directly instead of using WS API")
|
|
parser.add_option("-H", "--host", dest="host",
|
|
type="string", default="127.0.0.1", help="ip address of api host")
|
|
parser.add_option("-p", "--port", dest="port",
|
|
type="int", default=9696, help="api poort")
|
|
parser.add_option("-s", "--ssl", dest="ssl",
|
|
action="store_true", default=False, help="use ssl")
|
|
parser.add_option("-v", "--verbose", dest="verbose",
|
|
action="store_true", default=False, help="turn on verbose logging")
|
|
|
|
options, args = parser.parse_args()
|
|
|
|
if options.verbose:
|
|
LOG.basicConfig(level=LOG.DEBUG)
|
|
else:
|
|
LOG.basicConfig(level=LOG.WARN)
|
|
|
|
if len(args) < 1:
|
|
parser.print_help()
|
|
help()
|
|
sys.exit(1)
|
|
|
|
cmd = args[0]
|
|
if cmd not in commands.keys():
|
|
LOG.error("Unknown command: %s" % cmd)
|
|
help()
|
|
sys.exit(1)
|
|
|
|
args = build_args(cmd, commands[cmd]["args"], args[1:])
|
|
if not args:
|
|
sys.exit(1)
|
|
LOG.debug("Executing command \"%s\" with args: %s" % (cmd, args))
|
|
if not options.load_plugin:
|
|
client = Client(options.host, options.port, options.ssl,
|
|
args[0], FORMAT)
|
|
if "api_func" not in commands[cmd]:
|
|
LOG.error("API version of \"%s\" is not yet implemented" % cmd)
|
|
sys.exit(1)
|
|
commands[cmd]["api_func"](client, *args)
|
|
else:
|
|
quantum = QuantumManager()
|
|
manager = quantum.get_manager()
|
|
commands[cmd]["func"](manager, *args)
|
|
sys.exit(0)
|