]>
git.proxmox.com Git - pve-storage.git/blob - PVE/Storage/NFSPlugin.pm
1 package PVE
::Storage
::NFSPlugin
;
10 use PVE
::Tools
qw(run_command);
12 use PVE
::Storage
::Plugin
;
13 use PVE
::JSONSchema
qw(get_standard_option);
15 use base
qw(PVE::Storage::Plugin);
17 # NFS helper functions
20 my ($server, $export, $mountpoint, $mountdata) = @_;
22 $server = "[$server]" if Net
::IP
::ip_is_ipv6
($server);
23 my $source = "$server:$export";
25 $mountdata = PVE
::ProcFSTools
::parse_proc_mounts
() if !$mountdata;
26 return $mountpoint if grep {
28 $_->[0] =~ m
|^\Q
$source\E/?$| &&
29 $_->[1] eq $mountpoint
35 my ($server, $export, $mountpoint, $options) = @_;
37 $server = "[$server]" if Net
::IP
::ip_is_ipv6
($server);
38 my $source = "$server:$export";
40 my $cmd = ['/bin/mount', '-t', 'nfs', $source, $mountpoint];
42 push @$cmd, '-o', $options;
45 run_command
($cmd, errmsg
=> "mount error");
56 content
=> [ { images
=> 1, rootdir
=> 1, vztmpl
=> 1, iso
=> 1, backup
=> 1, snippets
=> 1 },
58 format
=> [ { raw
=> 1, qcow2
=> 1, vmdk
=> 1 } , 'raw' ],
65 description
=> "NFS export path.",
66 type
=> 'string', format
=> 'pve-storage-path',
69 description
=> "Server IP or DNS name.",
70 type
=> 'string', format
=> 'pve-storage-server',
73 description
=> "NFS mount options (see 'man nfs')",
74 type
=> 'string', format
=> 'pve-storage-options',
81 path
=> { fixed
=> 1 },
82 server
=> { fixed
=> 1 },
83 export
=> { fixed
=> 1 },
84 nodes
=> { optional
=> 1 },
85 disable
=> { optional
=> 1 },
86 maxfiles
=> { optional
=> 1 },
87 'prune-backups' => { optional
=> 1 },
88 'max-protected-backups' => { optional
=> 1 },
89 options
=> { optional
=> 1 },
90 content
=> { optional
=> 1 },
91 format
=> { optional
=> 1 },
92 mkdir => { optional
=> 1 },
93 bwlimit
=> { optional
=> 1 },
94 preallocation
=> { optional
=> 1 },
100 my ($class, $sectionId, $config, $create, $skipSchemaCheck) = @_;
102 $config->{path
} = "/mnt/pve/$sectionId" if $create && !$config->{path
};
104 return $class->SUPER::check_config
($sectionId, $config, $create, $skipSchemaCheck);
107 # Storage implementation
110 my ($class, $storeid, $scfg, $cache) = @_;
112 $cache->{mountdata
} = PVE
::ProcFSTools
::parse_proc_mounts
()
113 if !$cache->{mountdata
};
115 my $path = $scfg->{path
};
116 my $server = $scfg->{server
};
117 my $export = $scfg->{export
};
119 return undef if !nfs_is_mounted
($server, $export, $path, $cache->{mountdata
});
121 return $class->SUPER::status
($storeid, $scfg, $cache);
124 sub activate_storage
{
125 my ($class, $storeid, $scfg, $cache) = @_;
127 $cache->{mountdata
} = PVE
::ProcFSTools
::parse_proc_mounts
()
128 if !$cache->{mountdata
};
130 my $path = $scfg->{path
};
131 my $server = $scfg->{server
};
132 my $export = $scfg->{export
};
134 if (!nfs_is_mounted
($server, $export, $path, $cache->{mountdata
})) {
135 # NOTE: only call mkpath when not mounted (avoid hang when NFS server is offline
136 mkpath
$path if !(defined($scfg->{mkdir}) && !$scfg->{mkdir});
138 die "unable to activate storage '$storeid' - " .
139 "directory '$path' does not exist\n" if ! -d
$path;
141 nfs_mount
($server, $export, $path, $scfg->{options
});
144 $class->SUPER::activate_storage
($storeid, $scfg, $cache);
147 sub deactivate_storage
{
148 my ($class, $storeid, $scfg, $cache) = @_;
150 $cache->{mountdata
} = PVE
::ProcFSTools
::parse_proc_mounts
()
151 if !$cache->{mountdata
};
153 my $path = $scfg->{path
};
154 my $server = $scfg->{server
};
155 my $export = $scfg->{export
};
157 if (nfs_is_mounted
($server, $export, $path, $cache->{mountdata
})) {
158 my $cmd = ['/bin/umount', $path];
159 run_command
($cmd, errmsg
=> 'umount error');
163 sub check_connection
{
164 my ($class, $storeid, $scfg) = @_;
166 my $server = $scfg->{server
};
167 my $opts = $scfg->{options
};
170 if (defined($opts) && $opts =~ /vers=4.*/) {
171 my $ip = PVE
::JSONSchema
::pve_verify_ip
($server, 1);
173 $ip = PVE
::Network
::get_ip_from_hostname
($server);
176 my $transport = PVE
::JSONSchema
::pve_verify_ipv4
($ip, 1) ?
'tcp' : 'tcp6';
178 # nfsv4 uses a pseudo-filesystem always beginning with /
179 # no exports are listed
180 $cmd = ['/usr/sbin/rpcinfo', '-T', $transport, $ip, 'nfs', '4'];
182 $cmd = ['/sbin/showmount', '--no-headers', '--exports', $server];
185 eval { run_command
($cmd, timeout
=> 10, outfunc
=> sub {}, errfunc
=> sub {}) };
193 # FIXME remove on the next APIAGE reset.
194 # Deprecated, use get_volume_attribute instead.
195 sub get_volume_notes
{
197 PVE
::Storage
::DirPlugin
::get_volume_notes
($class, @_);
200 # FIXME remove on the next APIAGE reset.
201 # Deprecated, use update_volume_attribute instead.
202 sub update_volume_notes
{
204 PVE
::Storage
::DirPlugin
::update_volume_notes
($class, @_);
207 sub get_volume_attribute
{
208 return PVE
::Storage
::DirPlugin
::get_volume_attribute
(@_);
211 sub update_volume_attribute
{
212 return PVE
::Storage
::DirPlugin
::update_volume_attribute
(@_);