]> git.proxmox.com Git - pve-container.git/blame - src/PVE/LXC/Setup/Debian.pm
allow /32 CIDRs and remove duplicated mask array
[pve-container.git] / src / PVE / LXC / Setup / Debian.pm
CommitLineData
7af97ad5 1package PVE::LXC::Setup::Debian;
1c7f4f65
DM
2
3use strict;
4use warnings;
55fa4e09 5use Data::Dumper;
a1b1a247 6use PVE::Tools qw($IPV6RE);
55fa4e09
DM
7use PVE::LXC;
8use File::Path;
1c7f4f65 9
7af97ad5 10use PVE::LXC::Setup::Base;
1c7f4f65 11
7af97ad5 12use base qw(PVE::LXC::Setup::Base);
1c7f4f65 13
633a7bd8 14sub new {
5b4657d0 15 my ($class, $conf, $rootdir) = @_;
633a7bd8 16
5b4657d0 17 my $version = PVE::Tools::file_read_firstline("$rootdir/etc/debian_version");
633a7bd8
DM
18
19 die "unable to read version info\n" if !defined($version);
adc8f577
DM
20
21 die "unable to parse version info\n"
22 if $version !~ m/^(\d+(\.\d+)?)(\.\d+)?/;
23
24 $version = $1;
25
1c0a5ac7 26 die "unsupported debian version '$version'\n"
4596e30d 27 if !($version >= 4 && $version < 9);
633a7bd8 28
5b4657d0
DM
29 my $self = { conf => $conf, rootdir => $rootdir, version => $version };
30
27916659 31 $conf->{ostype} = "debian";
633a7bd8
DM
32
33 return bless $self, $class;
34}
35
d66768a2 36sub setup_init {
633a7bd8 37 my ($self, $conf) = @_;
d66768a2 38
2063d380 39 my $filename = "/etc/inittab";
fe8a16b8 40 return if !$self->ct_file_exists($filename);
d66768a2 41
fe8a16b8
WB
42 my $ttycount = PVE::LXC::get_tty_count($conf);
43 my $inittab = $self->ct_file_get_contents($filename);
44
45 my @lines = grep {
46 # remove getty lines
47 !/^\s*\d+:\d+:[^:]*:.*getty/ &&
48 # remove power lines
49 !/^\s*p[fno0]:/
50 } split(/\n/, $inittab);
51
52 $inittab = join("\n", @lines) . "\n";
53
54 $inittab .= "p0::powerfail:/sbin/init 0\n";
55
56 my $version = $self->{version};
39630c0d
DM
57 for (my $id = 1; $id <= $ttycount; $id++) {
58 next if $id == 7; # reserved for X11
59 my $levels = ($id == 1) ? '2345' : '23';
fe8a16b8
WB
60 if ($version < 7) {
61 $inittab .= "$id:$levels:respawn:/sbin/getty -L 38400 tty$id\n";
62 } else {
63 $inittab .= "$id:$levels:respawn:/sbin/getty --noclear 38400 tty$id\n";
d66768a2 64 }
d66768a2 65 }
fe8a16b8
WB
66
67 $self->ct_file_set_contents($filename, $inittab);
d66768a2
DM
68}
69
55fa4e09 70sub setup_network {
633a7bd8 71 my ($self, $conf) = @_;
55fa4e09 72
55fa4e09
DM
73 my $networks = {};
74 foreach my $k (keys %$conf) {
75 next if $k !~ m/^net(\d+)$/;
93285df8 76 my $ind = $1;
27916659 77 my $d = PVE::LXC::parse_lxc_network($conf->{$k});
55fa4e09
DM
78 if ($d->{name}) {
79 my $net = {};
93285df8 80 if (defined($d->{ip})) {
ecf84da0 81 if ($d->{ip} =~ /^(?:dhcp|manual)$/) {
0178cffa
WB
82 $net->{address} = $d->{ip};
83 } else {
84 my $ipinfo = PVE::LXC::parse_ipv4_cidr($d->{ip});
85 $net->{address} = $ipinfo->{address};
86 $net->{netmask} = $ipinfo->{netmask};
87 }
55fa4e09 88 }
93285df8 89 if (defined($d->{'gw'})) {
a1b1a247 90 $net->{gateway} = $d->{'gw'};
55fa4e09 91 }
93285df8 92 if (defined($d->{ip6})) {
c6b8740b 93 if ($d->{ip6} =~ /^(?:auto|dhcp|manual)$/) {
0178cffa
WB
94 $net->{address6} = $d->{ip6};
95 } elsif ($d->{ip6} !~ /^($IPV6RE)\/(\d+)$/) {
a1b1a247 96 die "unable to parse ipv6 address/prefix\n";
0178cffa
WB
97 } else {
98 $net->{address6} = $1;
99 $net->{netmask6} = $2;
a1b1a247 100 }
a1b1a247
WB
101 }
102 if (defined($d->{'gw6'})) {
103 $net->{gateway6} = $d->{'gw6'};
55fa4e09 104 }
0178cffa 105 $networks->{$d->{name}} = $net if keys %$net;
55fa4e09
DM
106 }
107 }
108
f44d23a5 109 return if !scalar(keys %$networks);
55fa4e09 110
2063d380 111 my $filename = "/etc/network/interfaces";
55fa4e09
DM
112 my $interfaces = "";
113
114 my $section;
115
56be201e 116 my $done_auto = {};
55fa4e09
DM
117 my $done_v4_hash = {};
118 my $done_v6_hash = {};
1c0a5ac7 119
55fa4e09 120 my $print_section = sub {
55fa4e09
DM
121 return if !$section;
122
6fd46881
WB
123 my $ifname = $section->{ifname};
124 my $net = $networks->{$ifname};
55fa4e09 125
6b0a0043 126 if (!$done_auto->{$ifname}) {
6fd46881
WB
127 $interfaces .= "auto $ifname\n";
128 $done_auto->{$ifname} = 1;
56be201e
WB
129 }
130
55fa4e09 131 if ($section->{type} eq 'ipv4') {
6fd46881 132 $done_v4_hash->{$ifname} = 1;
55fa4e09 133
ecf84da0 134 if ($net->{address} =~ /^(dhcp|manual)$/) {
6fd46881 135 $interfaces .= "iface $ifname inet $1\n";
fcaca113 136 } else {
6fd46881 137 $interfaces .= "iface $ifname inet static\n";
a1b1a247
WB
138 $interfaces .= "\taddress $net->{address}\n" if defined($net->{address});
139 $interfaces .= "\tnetmask $net->{netmask}\n" if defined($net->{netmask});
140 $interfaces .= "\tgateway $net->{gateway}\n" if defined($net->{gateway});
55fa4e09
DM
141 foreach my $attr (@{$section->{attr}}) {
142 $interfaces .= "\t$attr\n";
143 }
55fa4e09 144 }
1c0a5ac7 145
55fa4e09 146 $interfaces .= "\n";
1c0a5ac7 147
55fa4e09 148 } elsif ($section->{type} eq 'ipv6') {
6fd46881 149 $done_v6_hash->{$ifname} = 1;
1c0a5ac7 150
c6b8740b 151 if ($net->{address6} =~ /^(auto|dhcp|manual)$/) {
6fd46881 152 $interfaces .= "iface $ifname inet6 $1\n";
fcaca113 153 } else {
6fd46881 154 $interfaces .= "iface $ifname inet6 static\n";
a1b1a247
WB
155 $interfaces .= "\taddress $net->{address6}\n" if defined($net->{address6});
156 $interfaces .= "\tnetmask $net->{netmask6}\n" if defined($net->{netmask6});
157 $interfaces .= "\tgateway $net->{gateway6}\n" if defined($net->{gateway6});
55fa4e09
DM
158 foreach my $attr (@{$section->{attr}}) {
159 $interfaces .= "\t$attr\n";
160 }
161 }
1c0a5ac7
WB
162
163 $interfaces .= "\n";
55fa4e09
DM
164 } else {
165 die "unknown section type '$section->{type}'";
166 }
167
168 $section = undef;
169 };
1c0a5ac7 170
2063d380 171 if (my $fh = $self->ct_open_file($filename, "r")) {
55fa4e09
DM
172 while (defined (my $line = <$fh>)) {
173 chomp $line;
174 if ($line =~ m/^#/) {
175 $interfaces .= "$line\n";
176 next;
177 }
178 if ($line =~ m/^\s*$/) {
179 if ($section) {
180 &$print_section();
181 } else {
182 $interfaces .= "$line\n";
183 }
184 next;
185 }
0178cffa 186 if ($line =~ m/^\s*iface\s+(\S+)\s+inet\s+(\S+)\s*$/) {
55fa4e09 187 my $ifname = $1;
0178cffa 188 &$print_section(); # print previous section
55fa4e09
DM
189 if (!$networks->{$ifname}) {
190 $interfaces .= "$line\n";
191 next;
192 }
193 $section = { type => 'ipv4', ifname => $ifname, attr => []};
194 next;
195 }
0178cffa 196 if ($line =~ m/^\s*iface\s+(\S+)\s+inet6\s+(\S+)\s*$/) {
55fa4e09 197 my $ifname = $1;
0178cffa 198 &$print_section(); # print previous section
55fa4e09
DM
199 if (!$networks->{$ifname}) {
200 $interfaces .= "$line\n";
201 next;
202 }
203 $section = { type => 'ipv6', ifname => $ifname, attr => []};
204 next;
205 }
56be201e
WB
206 # Handle 'auto'
207 if ($line =~ m/^\s*auto\s*(.*)$/) {
208 foreach my $iface (split(/\s+/, $1)) {
209 $done_auto->{$iface} = 1;
210 }
211 &$print_section();
212 $interfaces .= "$line\n";
213 next;
214 }
0178cffa
WB
215 # Handle other section delimiters:
216 if ($line =~ m/^\s*(?:mapping\s
0178cffa
WB
217 |allow-
218 |source\s
219 |source-directory\s
220 )/x) {
221 &$print_section();
222 $interfaces .= "$line\n";
223 next;
224 }
55fa4e09
DM
225 if ($section && $line =~ m/^\s*((\S+)\s(.*))$/) {
226 my ($adata, $aname) = ($1, $2);
227 if ($aname eq 'address' || $aname eq 'netmask' ||
228 $aname eq 'gateway' || $aname eq 'broadcast') {
229 # skip
230 } else {
1c0a5ac7 231 push @{$section->{attr}}, $adata;
55fa4e09
DM
232 }
233 next;
234 }
1c0a5ac7
WB
235
236 $interfaces .= "$line\n";
55fa4e09
DM
237 }
238 &$print_section();
55fa4e09
DM
239 }
240
e90673a8 241 my $need_separator = length($interfaces) && ($interfaces !~ /\n\n$/);
55fa4e09
DM
242 foreach my $ifname (sort keys %$networks) {
243 my $net = $networks->{$ifname};
1c0a5ac7 244
fcaca113 245 if (!$done_v4_hash->{$ifname} && defined($net->{address})) {
1c0a5ac7 246 if ($need_separator) { $interfaces .= "\n"; $need_separator = 0; };
55fa4e09 247 $section = { type => 'ipv4', ifname => $ifname, attr => []};
6b0a0043 248 &$print_section();
55fa4e09 249 }
a1b1a247 250 if (!$done_v6_hash->{$ifname} && defined($net->{address6})) {
1c0a5ac7 251 if ($need_separator) { $interfaces .= "\n"; $need_separator = 0; };
55fa4e09 252 $section = { type => 'ipv6', ifname => $ifname, attr => []};
6b0a0043 253 &$print_section();
55fa4e09
DM
254 }
255 }
1c0a5ac7 256
2063d380 257 $self->ct_file_set_contents($filename, $interfaces);
55fa4e09 258}
1c7f4f65
DM
259
2601;