From: Stefan Hanreich <s.hanreich@proxmox.com>
To: pve-devel@lists.proxmox.com
Subject: [PATCH pve-network v6 07/24] api: refactor route map api structure
Date: Fri, 8 May 2026 18:31:16 +0200 [thread overview]
Message-ID: <20260508163134.481912-8-s.hanreich@proxmox.com> (raw)
In-Reply-To: <20260508163134.481912-1-s.hanreich@proxmox.com>
Reworks the existing rotue map API structure to look as follows:
GET /route-maps
Returns a list of all route maps that exist in the configuration, the
returned data only contains the ID of the route maps. It does not
provide the usual pending parameter, since different entries could
have different pending states. This could potentially be added in the
future by looking at the pending state of all entries, and treating
any pending state of any entry as changed state for the route map. For
now this hasn't been implemented, since the endpoint is only used by
the route map selector component, which doesn't require knowledge
about the pending state of route maps.
GET /route-maps/entries
Returns a list of all entries of all route maps.
POST /route-maps/entries
Creates a new route map entry.
GET /route-maps/entries/{route-map-id}
Returns a list of all entries of a specific route map.
GET/PUT/DELETE /route-maps/entries/{route-map-id}/entry/{order}
Returns / updates / deletes a specific route map entry.
Signed-off-by: Stefan Hanreich <s.hanreich@proxmox.com>
---
src/PVE/API2/Network/SDN/RouteMaps.pm | 93 ++----------
src/PVE/API2/Network/SDN/RouteMaps/Makefile | 2 +-
.../API2/Network/SDN/RouteMaps/RouteMap.pm | 7 +-
.../Network/SDN/RouteMaps/RouteMapEntries.pm | 142 ++++++++++++++++++
.../Network/SDN/RouteMaps/RouteMapEntry.pm | 9 +-
5 files changed, 164 insertions(+), 89 deletions(-)
create mode 100644 src/PVE/API2/Network/SDN/RouteMaps/RouteMapEntries.pm
diff --git a/src/PVE/API2/Network/SDN/RouteMaps.pm b/src/PVE/API2/Network/SDN/RouteMaps.pm
index 8a7936d9..4baffe31 100644
--- a/src/PVE/API2/Network/SDN/RouteMaps.pm
+++ b/src/PVE/API2/Network/SDN/RouteMaps.pm
@@ -3,7 +3,7 @@ package PVE::API2::Network::SDN::RouteMaps;
use strict;
use warnings;
-use PVE::API2::Network::SDN::RouteMaps::RouteMap;
+use PVE::API2::Network::SDN::RouteMaps::RouteMapEntries;
use PVE::Exception qw(raise_param_exc);
use PVE::JSONSchema qw(get_standard_option);
use PVE::Network::SDN::RouteMaps;
@@ -13,8 +13,8 @@ use PVE::RESTHandler;
use base qw(PVE::RESTHandler);
__PACKAGE__->register_method({
- subclass => "PVE::API2::Network::SDN::RouteMaps::RouteMap",
- path => '{route-map-id}',
+ subclass => "PVE::API2::Network::SDN::RouteMaps::RouteMapEntries",
+ path => 'entries',
});
__PACKAGE__->register_method({
@@ -23,7 +23,7 @@ __PACKAGE__->register_method({
method => 'GET',
permissions => {
description =>
- "Only returns route map entries where you have 'SDN.Audit' or 'SDN.Allocate' permissions.",
+ "Only returns route maps where you have 'SDN.Audit' or 'SDN.Allocate' permissions.",
user => 'all',
},
description => "List Route Maps",
@@ -34,108 +34,41 @@ __PACKAGE__->register_method({
optional => 1,
description => "Display running config.",
},
- pending => {
- type => 'boolean',
- optional => 1,
- description => "Display pending config.",
- },
},
},
returns => {
type => 'array',
items => {
type => "object",
- properties => PVE::Network::SDN::RouteMaps::route_map_properties(0),
+ properties => {
+ id => get_standard_option('pve-sdn-route-map-id'),
+ }
},
- links => [{ rel => 'child', href => "{route-map-id}" }],
+ links => [{ rel => 'child', href => "entries/{id}" }],
},
code => sub {
my ($param) = @_;
- my $pending = extract_param($param, 'pending');
my $running = extract_param($param, 'running');
-
- my $digest;
- my $route_maps;
-
- if ($pending) {
- my $current_config = PVE::Network::SDN::RouteMaps::config();
- my $running_config = PVE::Network::SDN::RouteMaps::config(1);
-
- my $pending_route_maps = PVE::Network::SDN::pending_config(
- { 'route-maps' => { ids => $running_config->list() } },
- { ids => $current_config->list() },
- 'route-maps',
- );
-
- $digest = $current_config->digest();
- $route_maps = $pending_route_maps->{ids};
- } elsif ($running) {
- $route_maps = PVE::Network::SDN::RouteMaps::config(1)->list();
- } else {
- my $current_config = PVE::Network::SDN::RouteMaps::config();
-
- $digest = $current_config->digest();
- $route_maps = $current_config->list();
- }
+ my $route_maps = PVE::Network::SDN::RouteMaps::config($running)->list_route_maps();
my $rpcenv = PVE::RPCEnvironment::get();
my $authuser = $rpcenv->get_user();
my $route_map_privs = ['SDN.Audit', 'SDN.Allocate'];
my @res;
- for my $route_map_id (sort keys $route_maps->%*) {
+ for my $route_map ($route_maps->@*) {
next
- if !$rpcenv->check_any($authuser, "/sdn/route-maps/$route_map_id",
+ if !$rpcenv->check_any($authuser, "/sdn/route-maps/$route_map->{id}",
$route_map_privs, 1);
- $route_maps->{$route_map_id}->{digest} = $digest if $digest;
- push @res, $route_maps->{$route_map_id};
+
+ push @res, $route_map;
}
return \@res;
},
});
-__PACKAGE__->register_method({
- name => 'create_route_map_entry',
- path => '',
- method => 'POST',
- protected => 1,
- permissions => {
- check => ['perm', '/sdn/route-maps', ['SDN.Allocate']],
- },
- description => "Create Route Map entry",
- parameters => {
- properties => {
- digest => get_standard_option('pve-config-digest'),
- 'lock-token' => get_standard_option('pve-sdn-lock-token'),
- PVE::Network::SDN::RouteMaps::route_map_properties(0)->%*,
- },
- },
- returns => {
- type => "null",
- },
- code => sub {
- my ($param) = @_;
- my $lock_token = extract_param($param, 'lock-token');
-
- PVE::Network::SDN::lock_sdn_config(
- sub {
- my $config = PVE::Network::SDN::RouteMaps::config();
-
- my $digest = extract_param($param, 'digest');
- PVE::Tools::assert_if_modified($config->digest(), $digest) if $digest;
-
- $config->create($param);
- PVE::Network::SDN::RouteMaps::write_config($config);
- },
- "creating route map entry failed",
- $lock_token,
- );
-
- return;
- },
-});
1;
diff --git a/src/PVE/API2/Network/SDN/RouteMaps/Makefile b/src/PVE/API2/Network/SDN/RouteMaps/Makefile
index 07b45e91..46ed800f 100644
--- a/src/PVE/API2/Network/SDN/RouteMaps/Makefile
+++ b/src/PVE/API2/Network/SDN/RouteMaps/Makefile
@@ -1,7 +1,7 @@
SOURCES=RouteMap.pm\
+ RouteMapEntries.pm\
RouteMapEntry.pm
-
PERL5DIR=${DESTDIR}/usr/share/perl5
.PHONY: install
diff --git a/src/PVE/API2/Network/SDN/RouteMaps/RouteMap.pm b/src/PVE/API2/Network/SDN/RouteMaps/RouteMap.pm
index 3b027a0b..7565c377 100644
--- a/src/PVE/API2/Network/SDN/RouteMaps/RouteMap.pm
+++ b/src/PVE/API2/Network/SDN/RouteMaps/RouteMap.pm
@@ -13,11 +13,11 @@ use base qw(PVE::RESTHandler);
__PACKAGE__->register_method({
subclass => "PVE::API2::Network::SDN::RouteMaps::RouteMapEntry",
- path => '{order}',
+ path => 'entry',
});
__PACKAGE__->register_method({
- name => 'list_route_map_entries',
+ name => 'list_route_map_entries_for_route_map',
path => '',
method => 'GET',
permissions => {
@@ -46,13 +46,14 @@ __PACKAGE__->register_method({
type => "object",
properties => PVE::Network::SDN::RouteMaps::route_map_properties(0),
},
- links => [{ rel => 'child', href => "{order}" }],
+ links => [{ rel => 'child', href => "entry/{route-map-id}" }],
},
code => sub {
my ($param) = @_;
my $pending = extract_param($param, 'pending');
my $running = extract_param($param, 'running');
+
my $route_map_id = extract_param($param, 'route-map-id');
my $digest;
diff --git a/src/PVE/API2/Network/SDN/RouteMaps/RouteMapEntries.pm b/src/PVE/API2/Network/SDN/RouteMaps/RouteMapEntries.pm
new file mode 100644
index 00000000..220e5a7b
--- /dev/null
+++ b/src/PVE/API2/Network/SDN/RouteMaps/RouteMapEntries.pm
@@ -0,0 +1,142 @@
+package PVE::API2::Network::SDN::RouteMaps::RouteMapEntries;
+
+use strict;
+use warnings;
+
+use PVE::API2::Network::SDN::RouteMaps::RouteMap;
+use PVE::Exception qw(raise_param_exc);
+use PVE::JSONSchema qw(get_standard_option);
+use PVE::Network::SDN::RouteMaps;
+use PVE::Tools qw(extract_param);
+
+use PVE::RESTHandler;
+use base qw(PVE::RESTHandler);
+
+__PACKAGE__->register_method({
+ subclass => "PVE::API2::Network::SDN::RouteMaps::RouteMap",
+ path => '{route-map-id}',
+});
+
+__PACKAGE__->register_method({
+ name => 'list_route_map_entries',
+ path => '',
+ method => 'GET',
+ permissions => {
+ description =>
+ "Only returns route map entries where you have 'SDN.Audit' or 'SDN.Allocate' permissions.",
+ user => 'all',
+ },
+ description => "Lists all route map entries.",
+ parameters => {
+ properties => {
+ 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 => PVE::Network::SDN::RouteMaps::route_map_properties(0),
+ },
+ links => [{ rel => 'child', href => "{route-map-id}" }],
+ },
+ code => sub {
+ my ($param) = @_;
+
+ my $pending = extract_param($param, 'pending');
+ my $running = extract_param($param, 'running');
+
+ my $digest;
+ my $route_maps;
+
+ if ($pending) {
+ my $current_config = PVE::Network::SDN::RouteMaps::config();
+ my $running_config = PVE::Network::SDN::RouteMaps::config(1);
+
+ my $pending_route_maps = PVE::Network::SDN::pending_config(
+ { 'route-maps' => { ids => $running_config->list() } },
+ { ids => $current_config->list() },
+ 'route-maps',
+ );
+
+ $digest = $current_config->digest();
+ $route_maps = $pending_route_maps->{ids};
+ } elsif ($running) {
+ $route_maps = PVE::Network::SDN::RouteMaps::config(1)->list();
+ } else {
+ my $current_config = PVE::Network::SDN::RouteMaps::config();
+
+ $digest = $current_config->digest();
+ $route_maps = $current_config->list();
+ }
+
+ my $rpcenv = PVE::RPCEnvironment::get();
+ my $authuser = $rpcenv->get_user();
+ my $route_map_privs = ['SDN.Audit', 'SDN.Allocate'];
+
+ my @res;
+ for my $route_map_id (sort keys $route_maps->%*) {
+ next
+ if !$rpcenv->check_any($authuser, "/sdn/route-maps/$route_map_id",
+ $route_map_privs, 1);
+
+ $route_maps->{$route_map_id}->{digest} = $digest if $digest;
+ push @res, $route_maps->{$route_map_id};
+ }
+
+ return \@res;
+ },
+});
+
+__PACKAGE__->register_method({
+ name => 'create_route_map_entry',
+ path => '',
+ method => 'POST',
+ protected => 1,
+ permissions => {
+ check => ['perm', '/sdn/route-maps', ['SDN.Allocate']],
+ },
+ description => "Create Route Map entry",
+ parameters => {
+ properties => {
+ digest => get_standard_option('pve-config-digest'),
+ 'lock-token' => get_standard_option('pve-sdn-lock-token'),
+ PVE::Network::SDN::RouteMaps::route_map_properties(0)->%*,
+ },
+ },
+ returns => {
+ type => "null",
+ },
+ code => sub {
+ my ($param) = @_;
+
+ my $lock_token = extract_param($param, 'lock-token');
+
+ PVE::Network::SDN::lock_sdn_config(
+ sub {
+ my $config = PVE::Network::SDN::RouteMaps::config();
+
+ my $digest = extract_param($param, 'digest');
+ PVE::Tools::assert_if_modified($config->digest(), $digest) if $digest;
+
+ $config->create($param);
+ PVE::Network::SDN::RouteMaps::write_config($config);
+ },
+ "creating route map entry failed",
+ $lock_token,
+ );
+
+ return;
+ },
+});
+
+
diff --git a/src/PVE/API2/Network/SDN/RouteMaps/RouteMapEntry.pm b/src/PVE/API2/Network/SDN/RouteMaps/RouteMapEntry.pm
index f53cfad3..a7561f44 100644
--- a/src/PVE/API2/Network/SDN/RouteMaps/RouteMapEntry.pm
+++ b/src/PVE/API2/Network/SDN/RouteMaps/RouteMapEntry.pm
@@ -3,9 +3,8 @@ package PVE::API2::Network::SDN::RouteMaps::RouteMapEntry;
use strict;
use warnings;
-use PVE::Exception qw(raise_param_exc);
use PVE::JSONSchema qw(get_standard_option);
-use PVE::Network::SDN::RouteMaps;
+use PVE::Exception qw(raise_param_exc);
use PVE::Tools qw(extract_param);
use PVE::RESTHandler;
@@ -13,7 +12,7 @@ use base qw(PVE::RESTHandler);
__PACKAGE__->register_method({
name => 'get_route_map_entry',
- path => '',
+ path => '{order}',
method => 'GET',
permissions => {
check =>
@@ -48,7 +47,7 @@ __PACKAGE__->register_method({
__PACKAGE__->register_method({
name => 'update_route_map_entry',
- path => '',
+ path => '{order}',
method => 'PUT',
protected => 1,
permissions => {
@@ -94,7 +93,7 @@ __PACKAGE__->register_method({
__PACKAGE__->register_method({
name => 'delete_route_map_entry',
- path => '',
+ path => '{order}',
method => 'DELETE',
protected => 1,
permissions => {
--
2.47.3
next prev parent reply other threads:[~2026-05-08 16:34 UTC|newest]
Thread overview: 25+ messages / expand[flat|nested] mbox.gz Atom feed top
2026-05-08 16:31 [PATCH manager/network/proxmox{-ve-rs,-perl-rs} v6 00/24] Add support for route maps / prefix lists to SDN Stefan Hanreich
2026-05-08 16:31 ` [PATCH proxmox-ve-rs v6 01/24] sdn: prefix lists: refactor section config and api format Stefan Hanreich
2026-05-08 16:31 ` [PATCH proxmox-ve-rs v6 02/24] prefix lists: implement validation for prefix lists Stefan Hanreich
2026-05-08 16:31 ` [PATCH proxmox-perl-rs v6 03/24] sdn: prefix lists: refactor existing API endpoint Stefan Hanreich
2026-05-08 16:31 ` [PATCH proxmox-perl-rs v6 04/24] sdn: prefix lists: add crud methods for prefix list entries Stefan Hanreich
2026-05-08 16:31 ` [PATCH proxmox-perl-rs v6 05/24] sdn: prefix lists: validate prefix lists Stefan Hanreich
2026-05-08 16:31 ` [PATCH proxmox-perl-rs v6 06/24] sdn: route maps: add route map list method Stefan Hanreich
2026-05-08 16:31 ` Stefan Hanreich [this message]
2026-05-08 16:31 ` [PATCH pve-network v6 08/24] api: refactor prefix list api structure Stefan Hanreich
2026-05-08 16:31 ` [PATCH pve-manager v6 09/24] ui: sdn: add route map selector Stefan Hanreich
2026-05-08 16:31 ` [PATCH pve-manager v6 10/24] ui: sdn: add prefix list selector Stefan Hanreich
2026-05-08 16:31 ` [PATCH pve-manager v6 11/24] ui: sdn: add panel for managing prefix lists Stefan Hanreich
2026-05-08 16:31 ` [PATCH pve-manager v6 12/24] ui: sdn: add panel for managing route map entries Stefan Hanreich
2026-05-08 16:31 ` [PATCH pve-manager v6 13/24] ui: sdn: bgp controller: allow configuring route maps Stefan Hanreich
2026-05-08 16:31 ` [PATCH pve-manager v6 14/24] ui: sdn: evpn " Stefan Hanreich
2026-05-08 16:31 ` [PATCH pve-manager v6 15/24] ui: sdn: openfabric: add route filter Stefan Hanreich
2026-05-08 16:31 ` [PATCH pve-manager v6 16/24] ui: sdn: ospf: add route filter setting Stefan Hanreich
2026-05-08 16:31 ` [PATCH pve-manager v6 17/24] ui: sdn: prefix list: add missing subjects Stefan Hanreich
2026-05-08 16:31 ` [PATCH pve-manager v6 18/24] sdn: do not fail rendering record data if pending property is missing Stefan Hanreich
2026-05-08 16:31 ` [PATCH pve-manager v6 19/24] ui: sdn: prefix list: adapt to changed api structure Stefan Hanreich
2026-05-08 16:31 ` [PATCH pve-manager v6 20/24] ui: sdn: route maps: adapt to new route map " Stefan Hanreich
2026-05-08 16:31 ` [PATCH pve-manager v6 21/24] ui: sdn: prefix lists: route maps: use integerfields for numbers Stefan Hanreich
2026-05-08 16:31 ` [PATCH pve-manager v6 22/24] ui: sdn: prefix list panel: reload data on deleting prefix list entry Stefan Hanreich
2026-05-08 16:31 ` [PATCH pve-manager v6 23/24] ui: prefix list panel: delete empty le and get properties Stefan Hanreich
2026-05-08 16:31 ` [PATCH pve-manager v6 24/24] ui: prefix list entry panel: make prefix required Stefan Hanreich
Reply instructions:
You may reply publicly to this message via plain-text email
using any one of the following methods:
* Save the following mbox file, import it into your mail client,
and reply-to-all from there: mbox
Avoid top-posting and favor interleaved quoting:
https://en.wikipedia.org/wiki/Posting_style#Interleaved_style
* Reply using the --to, --cc, and --in-reply-to
switches of git-send-email(1):
git send-email \
--in-reply-to=20260508163134.481912-8-s.hanreich@proxmox.com \
--to=s.hanreich@proxmox.com \
--cc=pve-devel@lists.proxmox.com \
/path/to/YOUR_REPLY
https://kernel.org/pub/software/scm/git/docs/git-send-email.html
* If your mail client supports setting the In-Reply-To header
via mailto: links, try the mailto: link
Be sure your reply has a Subject: header at the top and a blank line
before the message body.
This is a public inbox, see mirroring instructions
for how to clone and mirror all data and code used for this inbox