pve-network/PVE/API2/Network/SDN/Controllers.pm
Alexandre Derumier f23633dc24 controllers: improve bgp-evpn
- add new bgp plugin
- add ebgp support
- add loopback support
- move gateway-nodes option to zone as 'exitnodes'
- move external-peers to bgp plugin
2020-11-25 12:04:00 +01:00

291 lines
8 KiB
Perl

package PVE::API2::Network::SDN::Controllers;
use strict;
use warnings;
use PVE::SafeSyslog;
use PVE::Tools qw(extract_param);
use PVE::Cluster qw(cfs_read_file cfs_write_file);
use PVE::Network::SDN;
use PVE::Network::SDN::Zones;
use PVE::Network::SDN::Controllers;
use PVE::Network::SDN::Controllers::Plugin;
use PVE::Network::SDN::Controllers::EvpnPlugin;
use PVE::Network::SDN::Controllers::BgpPlugin;
use PVE::Network::SDN::Controllers::FaucetPlugin;
use Storable qw(dclone);
use PVE::JSONSchema qw(get_standard_option);
use PVE::RPCEnvironment;
use PVE::RESTHandler;
use base qw(PVE::RESTHandler);
my $sdn_controllers_type_enum = PVE::Network::SDN::Controllers::Plugin->lookup_types();
my $api_sdn_controllers_config = sub {
my ($cfg, $id) = @_;
my $scfg = dclone(PVE::Network::SDN::Controllers::sdn_controllers_config($cfg, $id));
$scfg->{controller} = $id;
$scfg->{digest} = $cfg->{digest};
return $scfg;
};
__PACKAGE__->register_method ({
name => 'index',
path => '',
method => 'GET',
description => "SDN controllers index.",
permissions => {
description => "Only list entries where you have 'SDN.Audit' or 'SDN.Allocate' permissions on '/sdn/controllers/<controller>'",
user => 'all',
},
parameters => {
additionalProperties => 0,
properties => {
type => {
description => "Only list sdn controllers of specific type",
type => 'string',
enum => $sdn_controllers_type_enum,
optional => 1,
},
running => {
type => 'boolean',
optional => 1,
description => "Display running config.",
},
pending => {
type => 'boolean',
optional => 1,
description => "Display pending config.",
},
},
},
returns => {
type => 'array',
items => {
type => "object",
properties => { controller => { type => 'string' },
type => { type => 'string' },
state => { type => 'string', optional => 1 },
pending => { optional => 1},
},
},
links => [ { rel => 'child', href => "{controller}" } ],
},
code => sub {
my ($param) = @_;
my $rpcenv = PVE::RPCEnvironment::get();
my $authuser = $rpcenv->get_user();
my $cfg = {};
if($param->{pending}) {
my $running_cfg = PVE::Network::SDN::config();
my $config = PVE::Network::SDN::Controllers::config();
$cfg = PVE::Network::SDN::pending_config($running_cfg, $config, 'controllers');
} elsif ($param->{running}) {
my $running_cfg = PVE::Network::SDN::config();
$cfg = $running_cfg->{controllers};
} else {
$cfg = PVE::Network::SDN::Controllers::config();
}
my @sids = PVE::Network::SDN::Controllers::sdn_controllers_ids($cfg);
my $res = [];
foreach my $id (@sids) {
my $privs = [ 'SDN.Audit', 'SDN.Allocate' ];
next if !$rpcenv->check_any($authuser, "/sdn/controllers/$id", $privs, 1);
my $scfg = &$api_sdn_controllers_config($cfg, $id);
next if $param->{type} && $param->{type} ne $scfg->{type};
my $plugin_config = $cfg->{ids}->{$id};
my $plugin = PVE::Network::SDN::Controllers::Plugin->lookup($plugin_config->{type});
push @$res, $scfg;
}
return $res;
}});
__PACKAGE__->register_method ({
name => 'read',
path => '{controller}',
method => 'GET',
description => "Read sdn controller configuration.",
permissions => {
check => ['perm', '/sdn/controllers/{controller}', ['SDN.Allocate']],
},
parameters => {
additionalProperties => 0,
properties => {
controller => get_standard_option('pve-sdn-controller-id'),
running => {
type => 'boolean',
optional => 1,
description => "Display running config.",
},
pending => {
type => 'boolean',
optional => 1,
description => "Display pending config.",
},
},
},
returns => { type => 'object' },
code => sub {
my ($param) = @_;
my $cfg = {};
if($param->{pending}) {
my $running_cfg = PVE::Network::SDN::config();
my $config = PVE::Network::SDN::Controllers::config();
$cfg = PVE::Network::SDN::pending_config($running_cfg, $config, 'controllers');
} elsif ($param->{running}) {
my $running_cfg = PVE::Network::SDN::config();
$cfg = $running_cfg->{controllers};
} else {
$cfg = PVE::Network::SDN::Controllers::config();
}
return &$api_sdn_controllers_config($cfg, $param->{controller});
}});
__PACKAGE__->register_method ({
name => 'create',
protected => 1,
path => '',
method => 'POST',
description => "Create a new sdn controller object.",
permissions => {
check => ['perm', '/sdn/controllers', ['SDN.Allocate']],
},
parameters => PVE::Network::SDN::Controllers::Plugin->createSchema(),
returns => { type => 'null' },
code => sub {
my ($param) = @_;
my $type = extract_param($param, 'type');
my $id = extract_param($param, 'controller');
my $plugin = PVE::Network::SDN::Controllers::Plugin->lookup($type);
my $opts = $plugin->check_config($id, $param, 1, 1);
# create /etc/pve/sdn directory
PVE::Cluster::check_cfs_quorum();
mkdir("/etc/pve/sdn");
PVE::Network::SDN::lock_sdn_config(
sub {
my $controller_cfg = PVE::Network::SDN::Controllers::config();
my $scfg = undef;
if ($scfg = PVE::Network::SDN::Controllers::sdn_controllers_config($controller_cfg, $id, 1)) {
die "sdn controller object ID '$id' already defined\n";
}
$controller_cfg->{ids}->{$id} = $opts;
$plugin->on_update_hook($id, $controller_cfg);
PVE::Network::SDN::Controllers::write_config($controller_cfg);
}, "create sdn controller object failed");
return undef;
}});
__PACKAGE__->register_method ({
name => 'update',
protected => 1,
path => '{controller}',
method => 'PUT',
description => "Update sdn controller object configuration.",
permissions => {
check => ['perm', '/sdn/controllers', ['SDN.Allocate']],
},
parameters => PVE::Network::SDN::Controllers::Plugin->updateSchema(),
returns => { type => 'null' },
code => sub {
my ($param) = @_;
my $id = extract_param($param, 'controller');
my $digest = extract_param($param, 'digest');
PVE::Network::SDN::lock_sdn_config(
sub {
my $controller_cfg = PVE::Network::SDN::Controllers::config();
PVE::SectionConfig::assert_if_modified($controller_cfg, $digest);
my $scfg = PVE::Network::SDN::Controllers::sdn_controllers_config($controller_cfg, $id);
my $plugin = PVE::Network::SDN::Controllers::Plugin->lookup($scfg->{type});
my $opts = $plugin->check_config($id, $param, 0, 1);
foreach my $k (%$opts) {
$scfg->{$k} = $opts->{$k};
}
$plugin->on_update_hook($id, $controller_cfg);
PVE::Network::SDN::Controllers::write_config($controller_cfg);
}, "update sdn controller object failed");
return undef;
}});
__PACKAGE__->register_method ({
name => 'delete',
protected => 1,
path => '{controller}',
method => 'DELETE',
description => "Delete sdn controller object configuration.",
permissions => {
check => ['perm', '/sdn/controllers', ['SDN.Allocate']],
},
parameters => {
additionalProperties => 0,
properties => {
controller => get_standard_option('pve-sdn-controller-id', {
completion => \&PVE::Network::SDN::Controllers::complete_sdn_controllers,
}),
},
},
returns => { type => 'null' },
code => sub {
my ($param) = @_;
my $id = extract_param($param, 'controller');
PVE::Network::SDN::lock_sdn_config(
sub {
my $cfg = PVE::Network::SDN::Controllers::config();
my $scfg = PVE::Network::SDN::Controllers::sdn_controllers_config($cfg, $id);
my $plugin = PVE::Network::SDN::Controllers::Plugin->lookup($scfg->{type});
my $zone_cfg = PVE::Network::SDN::Zones::config();
$plugin->on_delete_hook($id, $zone_cfg);
delete $cfg->{ids}->{$id};
PVE::Network::SDN::Controllers::write_config($cfg);
}, "delete sdn controller object failed");
return undef;
}});
1;