]> git.proxmox.com Git - pve-network.git/blob - PVE/Network/SDN/Ipams/PVEPlugin.pm
ipams: add mac address
[pve-network.git] / PVE / Network / SDN / Ipams / PVEPlugin.pm
1 package PVE::Network::SDN::Ipams::PVEPlugin;
2
3 use strict;
4 use warnings;
5 use PVE::INotify;
6 use PVE::Cluster qw(cfs_read_file cfs_write_file cfs_register_file cfs_lock_file);
7 use PVE::Tools;
8 use JSON;
9 use NetAddr::IP qw(:lower);
10
11 use Net::IP;
12 use Digest::SHA;
13
14 use base('PVE::Network::SDN::Ipams::Plugin');
15
16
17 my $ipamdb_file = "priv/ipam.db";
18
19 PVE::Cluster::cfs_register_file($ipamdb_file,
20 sub { PVE::Network::SDN::Ipams::PVEPlugin->parse_config(@_); },
21 sub { PVE::Network::SDN::Ipams::PVEPlugin->write_config(@_); });
22
23 sub type {
24 return 'pve';
25 }
26
27 sub properties {
28 }
29
30 sub options {
31 }
32
33 # Plugin implementation
34
35 sub add_subnet {
36 my ($class, $plugin_config, $subnetid, $subnet) = @_;
37
38 my $cidr = $subnet->{cidr};
39 my $zone = $subnet->{zone};
40 my $gateway = $subnet->{gateway};
41
42
43 cfs_lock_file($ipamdb_file, undef, sub {
44 my $db = {};
45 $db = read_db();
46
47 $db->{zones}->{$zone} = {} if !$db->{zones}->{$zone};
48 my $zonedb = $db->{zones}->{$zone};
49
50 if(!$zonedb->{subnets}->{$cidr}) {
51 #create subnet
52 $zonedb->{subnets}->{$cidr}->{ips} = {};
53 write_db($db);
54 }
55 });
56 die "$@" if $@;
57 }
58
59 sub del_subnet {
60 my ($class, $plugin_config, $subnetid, $subnet) = @_;
61
62 my $cidr = $subnet->{cidr};
63 my $zone = $subnet->{zone};
64
65 cfs_lock_file($ipamdb_file, undef, sub {
66
67 my $db = read_db();
68
69 my $dbzone = $db->{zones}->{$zone};
70 die "zone '$zone' doesn't exist in IPAM DB\n" if !$dbzone;
71 my $dbsubnet = $dbzone->{subnets}->{$cidr};
72 die "subnet '$cidr' doesn't exist in IPAM DB\n" if !$dbsubnet;
73
74 die "cannot delete subnet '$cidr', not empty\n" if keys %{$dbsubnet->{ips}} > 0;
75
76 delete $dbzone->{subnets}->{$cidr};
77
78 write_db($db);
79 });
80 die "$@" if $@;
81
82 }
83
84 sub add_ip {
85 my ($class, $plugin_config, $subnetid, $subnet, $ip, $hostname, $mac, $description, $is_gateway) = @_;
86
87 my $cidr = $subnet->{cidr};
88 my $zone = $subnet->{zone};
89
90 cfs_lock_file($ipamdb_file, undef, sub {
91
92 my $db = read_db();
93
94 my $dbzone = $db->{zones}->{$zone};
95 die "zone '$zone' doesn't exist in IPAM DB\n" if !$dbzone;
96 my $dbsubnet = $dbzone->{subnets}->{$cidr};
97 die "subnet '$cidr' doesn't exist in IPAM DB\n" if !$dbsubnet;
98
99 die "IP '$ip' already exist\n" if defined($dbsubnet->{ips}->{$ip});
100
101 $dbsubnet->{ips}->{$ip} = {};
102
103 write_db($db);
104 });
105 die "$@" if $@;
106 }
107
108 sub add_next_freeip {
109 my ($class, $plugin_config, $subnetid, $subnet, $hostname, $mac, $description) = @_;
110
111 my $cidr = $subnet->{cidr};
112 my $network = $subnet->{network};
113 my $zone = $subnet->{zone};
114 my $mask = $subnet->{mask};
115 my $freeip = undef;
116
117 cfs_lock_file($ipamdb_file, undef, sub {
118
119 my $db = read_db();
120 my $dbzone = $db->{zones}->{$zone};
121 die "zone '$zone' doesn't exist in IPAM DB\n" if !$dbzone;
122 my $dbsubnet = $dbzone->{subnets}->{$cidr};
123 die "subnet '$cidr' doesn't exist in IPAM DB" if !$dbsubnet;
124
125 if (Net::IP::ip_is_ipv4($network) && $mask == 32) {
126 die "cannot find free IP in subnet '$cidr'\n" if defined($dbsubnet->{ips}->{$network});
127 $freeip = $network;
128 } else {
129 my $iplist = new NetAddr::IP($cidr);
130 my $broadcast = $iplist->broadcast();
131
132 while(1) {
133 $iplist++;
134 last if $iplist eq $broadcast;
135 my $ip = $iplist->canon();
136 next if defined($dbsubnet->{ips}->{$ip});
137 $freeip = $ip;
138 last;
139 }
140 }
141
142 die "can't find free ip in subnet '$cidr'\n" if !$freeip;
143
144 $dbsubnet->{ips}->{$freeip} = {};
145
146 write_db($db);
147 });
148 die "$@" if $@;
149
150 return "$freeip/$mask";
151 }
152
153 sub del_ip {
154 my ($class, $plugin_config, $subnetid, $subnet, $ip) = @_;
155
156 my $cidr = $subnet->{cidr};
157 my $zone = $subnet->{zone};
158
159 cfs_lock_file($ipamdb_file, undef, sub {
160
161 my $db = read_db();
162 die "zone $zone don't exist in ipam db" if !$db->{zones}->{$zone};
163 my $dbzone = $db->{zones}->{$zone};
164 die "subnet $cidr don't exist in ipam db" if !$dbzone->{subnets}->{$cidr};
165 my $dbsubnet = $dbzone->{subnets}->{$cidr};
166
167 die "IP '$ip' does not exist in IPAM DB\n" if !defined($dbsubnet->{ips}->{$ip});
168 delete $dbsubnet->{ips}->{$ip};
169
170 write_db($db);
171 });
172 die "$@" if $@;
173 }
174
175 #helpers
176
177 sub read_db {
178 my $db = cfs_read_file($ipamdb_file);
179 return $db;
180 }
181
182 sub write_db {
183 my ($cfg) = @_;
184
185 my $json = to_json($cfg);
186 cfs_write_file($ipamdb_file, $json);
187 }
188
189 sub write_config {
190 my ($class, $filename, $cfg) = @_;
191
192 return $cfg;
193 }
194
195 sub parse_config {
196 my ($class, $filename, $raw) = @_;
197
198 $raw = '{}' if !defined($raw) ||$raw eq '';
199 my $cfg = from_json($raw);
200
201 return $cfg;
202 }
203
204 1;