Merge "Add IBM GPFS Manila driver"
This commit is contained in:
commit
e3943acf73
102
doc/source/devref/gpfs_driver.rst
Normal file
102
doc/source/devref/gpfs_driver.rst
Normal file
@ -0,0 +1,102 @@
|
|||||||
|
..
|
||||||
|
Copyright 2015 IBM Corp.
|
||||||
|
All Rights Reserved.
|
||||||
|
|
||||||
|
Licensed under the Apache License, Version 2.0 (the "License"); you may
|
||||||
|
not use this file except in compliance with the License. You may obtain
|
||||||
|
a copy of the License at
|
||||||
|
|
||||||
|
http://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
|
||||||
|
Unless required by applicable law or agreed to in writing, software
|
||||||
|
distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
|
||||||
|
WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
|
||||||
|
License for the specific language governing permissions and limitations
|
||||||
|
under the License.
|
||||||
|
|
||||||
|
GPFS Driver
|
||||||
|
===========
|
||||||
|
|
||||||
|
GPFS driver uses IBM General Parallel File System (GPFS), a high-performance,
|
||||||
|
clustered file system, developed by IBM, as the storage backend for serving
|
||||||
|
file shares to the Manila clients.
|
||||||
|
|
||||||
|
Supported shared filesystems
|
||||||
|
----------------------------
|
||||||
|
|
||||||
|
- NFS (access by IP)
|
||||||
|
|
||||||
|
|
||||||
|
Supported Operations
|
||||||
|
--------------------
|
||||||
|
|
||||||
|
- Create NFS Share
|
||||||
|
- Delete NFS Share
|
||||||
|
- Create Share Snapshot
|
||||||
|
- Delete Share Snapshot
|
||||||
|
- Create Share from a Share Snapshot
|
||||||
|
- Allow NFS Share access
|
||||||
|
|
||||||
|
* Currently only 'rw' access level is supported
|
||||||
|
|
||||||
|
- Deny NFS Share access
|
||||||
|
|
||||||
|
Requirements
|
||||||
|
------------
|
||||||
|
|
||||||
|
- Install GPFS with server license, version >= 2.0, on the storage backend.
|
||||||
|
- Install Kernel NFS or Ganesha NFS server on the storage backend servers.
|
||||||
|
- If using Ganesha NFS, currently NFS Ganesha v1.5 and v2.0 are supported.
|
||||||
|
- Create a GPFS cluster and create a filesystem on the cluster, that will be
|
||||||
|
used to create the Manila shares.
|
||||||
|
- Enable quotas for the GPFS file system (`mmchfs -Q yes`).
|
||||||
|
- Establish network connection between the Manila host and the storage backend.
|
||||||
|
|
||||||
|
|
||||||
|
Manila driver configuration setting
|
||||||
|
-----------------------------------
|
||||||
|
|
||||||
|
The following parameters in the Manila configuration file need to be set:
|
||||||
|
|
||||||
|
- `share_driver` = manila.share.drivers.ibm.gpfs.GPFSShareDriver
|
||||||
|
- `gpfs_share_export_ip` = <IP to be added to GPFS export string>
|
||||||
|
- If the backend GPFS server is not running on the Manila host machine, the
|
||||||
|
following options are required to SSH to the remote GPFS backend server:
|
||||||
|
|
||||||
|
- `gpfs_ssh_login` = <GPFS server SSH login name>
|
||||||
|
|
||||||
|
and one of the following settings is required to execute commands over SSH:
|
||||||
|
|
||||||
|
- `gpfs_ssh_private_key` = <path to GPFS server SSH private key for login>
|
||||||
|
- `gpfs_ssh_password` = <GPFS server SSH login password>
|
||||||
|
|
||||||
|
The following configuration parameters are optional:
|
||||||
|
|
||||||
|
- `gpfs_mount_point_base` = <base folder where exported shares are located>
|
||||||
|
- `gpfs_nfs_server_type` = <KNFS|GNFS>
|
||||||
|
- `gpfs_nfs_server_list` = <list of the fully qualified NFS server names>
|
||||||
|
- `gpfs_ssh_port` = <ssh port number>
|
||||||
|
- `knfs_export_options` = <options to use when creating a share using kernel
|
||||||
|
NFS server>
|
||||||
|
|
||||||
|
Restart of :term:`manila-share` service is needed for the configuration changes to take
|
||||||
|
effect.
|
||||||
|
|
||||||
|
Known Restrictions
|
||||||
|
------------------
|
||||||
|
|
||||||
|
- The driver does not support a segmented-network multi-tenancy model but
|
||||||
|
instead works over a flat network where the tenants share a network.
|
||||||
|
- While using remote GPFS node, with Ganesha NFS, 'gpfs_ssh_private_key' for
|
||||||
|
remote login to the GPFS node must be specified and there must be a
|
||||||
|
passwordless authentication already setup between the Manila share service
|
||||||
|
and the remote GPFS node.
|
||||||
|
|
||||||
|
The :mod:`manila.share.drivers.ibm.gpfs` Module
|
||||||
|
~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
|
||||||
|
|
||||||
|
.. automodule:: manila.share.drivers.ibm.gpfs
|
||||||
|
:noindex:
|
||||||
|
:members:
|
||||||
|
:undoc-members:
|
||||||
|
:show-inheritance:
|
@ -102,6 +102,7 @@ Share backends
|
|||||||
generic_driver
|
generic_driver
|
||||||
glusterfs_driver
|
glusterfs_driver
|
||||||
glusterfs_native_driver
|
glusterfs_native_driver
|
||||||
|
gpfs_driver
|
||||||
huawei_nas_driver
|
huawei_nas_driver
|
||||||
hdfs_native_driver
|
hdfs_native_driver
|
||||||
hds_hnas_driver
|
hds_hnas_driver
|
||||||
|
@ -52,6 +52,8 @@ Mapping of share drivers and share features support
|
|||||||
| HP 3PAR | DHSS = True (L) & False (K) | \- | \- | \- | K | K |
|
| HP 3PAR | DHSS = True (L) & False (K) | \- | \- | \- | K | K |
|
||||||
+----------------------------------------+-----------------------------+-----------------------+--------------+--------------+------------------------+----------------------------+
|
+----------------------------------------+-----------------------------+-----------------------+--------------+--------------+------------------------+----------------------------+
|
||||||
| Huawei | DHSS = False(K) | L | L | L | K | \- |
|
| Huawei | DHSS = False(K) | L | L | L | K | \- |
|
||||||
|
+----------------------------------------------------------------------------------------------------------------------------------------------------------------------------------+
|
||||||
|
| IBM GPFS | DHSS = False(K) | \- | L | \- | K | K |
|
||||||
+----------------------------------------+-----------------------------+-----------------------+--------------+--------------+------------------------+----------------------------+
|
+----------------------------------------+-----------------------------+-----------------------+--------------+--------------+------------------------+----------------------------+
|
||||||
| Quobyte | DHSS = False (K) | \- | \- | \- | \- | \- |
|
| Quobyte | DHSS = False (K) | \- | \- | \- | \- | \- |
|
||||||
+----------------------------------------+-----------------------------+-----------------------+--------------+--------------+------------------------+----------------------------+
|
+----------------------------------------+-----------------------------+-----------------------+--------------+--------------+------------------------+----------------------------+
|
||||||
@ -95,6 +97,8 @@ Mapping of share drivers and share access rules support
|
|||||||
| Quobyte | NFS (K) | \- | \- | NFS (K) | \- | \- |
|
| Quobyte | NFS (K) | \- | \- | NFS (K) | \- | \- |
|
||||||
+----------------------------------------+--------------+------------+------------+--------------+------------+------------+
|
+----------------------------------------+--------------+------------+------------+--------------+------------+------------+
|
||||||
| Windows SMB | \- | CIFS (L) | \- | \- | CIFS (L) | \- |
|
| Windows SMB | \- | CIFS (L) | \- | \- | CIFS (L) | \- |
|
||||||
|
+--------------------------------------------------------------------------------------------------------------------------+
|
||||||
|
| IBM GPFS | NFS (K) | \- | \- | NFS (K) | \- | \- |
|
||||||
+----------------------------------------+--------------+------------+------------+--------------+------------+------------+
|
+----------------------------------------+--------------+------------+------------+--------------+------------+------------+
|
||||||
| ZFS | ? | ? | ? | ? | ? | ? |
|
| ZFS | ? | ? | ? | ? | ? | ? |
|
||||||
+----------------------------------------+--------------+------------+------------+--------------+------------+------------+
|
+----------------------------------------+--------------+------------+------------+--------------+------------+------------+
|
||||||
@ -128,6 +132,8 @@ Mapping of share drivers and security services support
|
|||||||
| Quobyte | \- | \- | \- |
|
| Quobyte | \- | \- | \- |
|
||||||
+----------------------------------------+------------------+-----------------+------------------+
|
+----------------------------------------+------------------+-----------------+------------------+
|
||||||
| Windows SMB | L | \- | \- |
|
| Windows SMB | L | \- | \- |
|
||||||
|
+------------------------------------------------------------------------------------------------+
|
||||||
|
| IBM GPFS | \- | \- | \- |
|
||||||
+----------------------------------------+------------------+-----------------+------------------+
|
+----------------------------------------+------------------+-----------------+------------------+
|
||||||
| ZFS | ? | ? | ? |
|
| ZFS | ? | ? | ? |
|
||||||
+----------------------------------------+------------------+-----------------+------------------+
|
+----------------------------------------+------------------+-----------------+------------------+
|
||||||
|
@ -35,9 +35,48 @@ find: CommandFilter, find, root
|
|||||||
# manila/share/drivers/glusterfs_native.py: 'umount', '%s'
|
# manila/share/drivers/glusterfs_native.py: 'umount', '%s'
|
||||||
umount: CommandFilter, umount, root
|
umount: CommandFilter, umount, root
|
||||||
|
|
||||||
|
# GPFS commands
|
||||||
|
# manila/share/drivers/ibm/gpfs.py: 'mmgetstate', '-Y'
|
||||||
|
mmgetstate: CommandFilter, /usr/lpp/mmfs/bin/mmgetstate, root
|
||||||
|
# manila/share/drivers/ibm/gpfs.py: 'mmlsattr', '%s'
|
||||||
|
mmlsattr: CommandFilter, /usr/lpp/mmfs/bin/mmlsattr, root
|
||||||
|
# manila/share/drivers/ibm/gpfs.py: 'mmcrfileset', '%s', '%s', '--inode-space', 'new'
|
||||||
|
mmcrfileset: CommandFilter, /usr/lpp/mmfs/bin/mmcrfileset, root
|
||||||
|
# manila/share/drivers/ibm/gpfs.py: 'mmlinkfileset', '%s', '%s', '-J', '%s'
|
||||||
|
mmlinkfileset: CommandFilter, /usr/lpp/mmfs/bin/mmlinkfileset, root
|
||||||
|
# manila/share/drivers/ibm/gpfs.py: 'mmsetquota', '-j', '%s', '-h', '%s', '%s'
|
||||||
|
mmsetquota: CommandFilter, /usr/lpp/mmfs/bin/mmsetquota, root
|
||||||
|
# manila/share/drivers/ibm/gpfs.py: 'mmunlinkfileset', '%s', '%s', '-f'
|
||||||
|
mmunlinkfileset: CommandFilter, /usr/lpp/mmfs/bin/mmunlinkfileset, root
|
||||||
|
# manila/share/drivers/ibm/gpfs.py: 'mmdelfileset', '%s', '%s', '-f'
|
||||||
|
mmdelfileset: CommandFilter, /usr/lpp/mmfs/bin/mmdelfileset, root
|
||||||
|
# manila/share/drivers/ibm/gpfs.py: 'mmcrsnapshot', '%s', '%s', '-j', '%s'
|
||||||
|
mmcrsnapshot: CommandFilter, /usr/lpp/mmfs/bin/mmcrsnapshot, root
|
||||||
|
# manila/share/drivers/ibm/gpfs.py: 'mmdelsnapshot', '%s', '%s', '-j', '%s'
|
||||||
|
mmdelsnapshot: CommandFilter, /usr/lpp/mmfs/bin/mmdelsnapshot, root
|
||||||
|
# manila/share/drivers/ibm/gpfs.py: 'rsync', '-rp', '%s', '%s'
|
||||||
|
rsync: CommandFilter, rsync, root
|
||||||
|
# manila/share/drivers/ibm/gpfs.py: 'exportfs'
|
||||||
|
exportfs: CommandFilter, exportfs, root
|
||||||
|
# manila/share/drivers/ibm/gpfs.py: 'stat', '--format=%F', '%s'
|
||||||
|
stat: CommandFilter, stat, root
|
||||||
|
# manila/share/drivers/ibm/gpfs.py: 'df', '-P', '-B', '1', '%s'
|
||||||
|
df: CommandFilter, df, root
|
||||||
|
|
||||||
# Ganesha commands
|
# Ganesha commands
|
||||||
|
# manila/share/drivers/ibm/ganesha_utils.py: 'mv', '%s', '%s'
|
||||||
# manila/share/drivers/ganesha/manager.py: 'mv', '%s', '%s'
|
# manila/share/drivers/ganesha/manager.py: 'mv', '%s', '%s'
|
||||||
mv: CommandFilter, mv, root
|
mv: CommandFilter, mv, root
|
||||||
|
# manila/share/drivers/ibm/ganesha_utils.py: 'cp', '%s', '%s'
|
||||||
|
cp: CommandFilter, cp, root
|
||||||
|
# manila/share/drivers/ibm/ganesha_utils.py: 'scp', '-i', '%s', '%s', '%s'
|
||||||
|
scp: CommandFilter, scp, root
|
||||||
|
# manila/share/drivers/ibm/ganesha_utils.py: 'ssh', '%s', '%s'
|
||||||
|
ssh: CommandFilter, ssh, root
|
||||||
|
# manila/share/drivers/ibm/ganesha_utils.py: 'chmod', '%s', '%s'
|
||||||
|
chmod: CommandFilter, chmod, root
|
||||||
|
# manila/share/drivers/ibm/ganesha_utils.py: 'service', '%s', 'restart'
|
||||||
|
service: CommandFilter, service, root
|
||||||
|
|
||||||
# manila/share/drivers/ganesha/manager.py: 'mktemp', '-p', '%s', '-t', '%s'
|
# manila/share/drivers/ganesha/manager.py: 'mktemp', '-p', '%s', '-t', '%s'
|
||||||
mktemp: CommandFilter, mktemp, root
|
mktemp: CommandFilter, mktemp, root
|
||||||
|
@ -607,6 +607,14 @@ class HP3ParUnexpectedError(ManilaException):
|
|||||||
message = _("%(err)s")
|
message = _("%(err)s")
|
||||||
|
|
||||||
|
|
||||||
|
class GPFSException(ManilaException):
|
||||||
|
message = _("GPFS exception occurred.")
|
||||||
|
|
||||||
|
|
||||||
|
class GPFSGaneshaException(ManilaException):
|
||||||
|
message = _("GPFS Ganesha exception occurred.")
|
||||||
|
|
||||||
|
|
||||||
class GaneshaCommandFailure(ProcessExecutionError):
|
class GaneshaCommandFailure(ProcessExecutionError):
|
||||||
_description = _("Ganesha management command failed.")
|
_description = _("Ganesha management command failed.")
|
||||||
|
|
||||||
|
@ -62,6 +62,7 @@ import manila.share.drivers.hdfs.hdfs_native
|
|||||||
import manila.share.drivers.hitachi.hds_hnas
|
import manila.share.drivers.hitachi.hds_hnas
|
||||||
import manila.share.drivers.hp.hp_3par_driver
|
import manila.share.drivers.hp.hp_3par_driver
|
||||||
import manila.share.drivers.huawei.huawei_nas
|
import manila.share.drivers.huawei.huawei_nas
|
||||||
|
import manila.share.drivers.ibm.gpfs
|
||||||
import manila.share.drivers.netapp.options
|
import manila.share.drivers.netapp.options
|
||||||
import manila.share.drivers.quobyte.quobyte
|
import manila.share.drivers.quobyte.quobyte
|
||||||
import manila.share.drivers.service_instance
|
import manila.share.drivers.service_instance
|
||||||
@ -123,6 +124,7 @@ _global_opt_lists = [
|
|||||||
manila.share.drivers.hitachi.hds_hnas.hds_hnas_opts,
|
manila.share.drivers.hitachi.hds_hnas.hds_hnas_opts,
|
||||||
manila.share.drivers.hp.hp_3par_driver.HP3PAR_OPTS,
|
manila.share.drivers.hp.hp_3par_driver.HP3PAR_OPTS,
|
||||||
manila.share.drivers.huawei.huawei_nas.huawei_opts,
|
manila.share.drivers.huawei.huawei_nas.huawei_opts,
|
||||||
|
manila.share.drivers.ibm.gpfs.gpfs_share_opts,
|
||||||
manila.share.drivers.netapp.options.netapp_proxy_opts,
|
manila.share.drivers.netapp.options.netapp_proxy_opts,
|
||||||
manila.share.drivers.netapp.options.netapp_connection_opts,
|
manila.share.drivers.netapp.options.netapp_connection_opts,
|
||||||
manila.share.drivers.netapp.options.netapp_transport_opts,
|
manila.share.drivers.netapp.options.netapp_transport_opts,
|
||||||
|
@ -140,6 +140,14 @@ ganesha_opts = [
|
|||||||
cfg.StrOpt('ganesha_config_path',
|
cfg.StrOpt('ganesha_config_path',
|
||||||
default='$ganesha_config_dir/ganesha.conf',
|
default='$ganesha_config_dir/ganesha.conf',
|
||||||
help='Path to main Ganesha config file.'),
|
help='Path to main Ganesha config file.'),
|
||||||
|
cfg.StrOpt('ganesha_nfs_export_options',
|
||||||
|
default='maxread = 65536, prefread = 65536',
|
||||||
|
help='Options to use when exporting a share using ganesha '
|
||||||
|
'NFS server. Note that these defaults can be overridden '
|
||||||
|
'when a share is created by passing metadata with key '
|
||||||
|
'name export_options. Also note the complete set of '
|
||||||
|
'default ganesha export options is specified in '
|
||||||
|
'ganesha_utils. (GPFS only.)'),
|
||||||
cfg.StrOpt('ganesha_service_name',
|
cfg.StrOpt('ganesha_service_name',
|
||||||
default='ganesha.nfsd',
|
default='ganesha.nfsd',
|
||||||
help='Name of the ganesha nfs service.'),
|
help='Name of the ganesha nfs service.'),
|
||||||
|
0
manila/share/drivers/ibm/__init__.py
Normal file
0
manila/share/drivers/ibm/__init__.py
Normal file
332
manila/share/drivers/ibm/ganesha_utils.py
Normal file
332
manila/share/drivers/ibm/ganesha_utils.py
Normal file
@ -0,0 +1,332 @@
|
|||||||
|
# Copyright 2014 IBM Corp.
|
||||||
|
#
|
||||||
|
# Licensed under the Apache License, Version 2.0 (the "License"); you may
|
||||||
|
# not use this file except in compliance with the License. You may obtain
|
||||||
|
# a copy of the License at
|
||||||
|
#
|
||||||
|
# http://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
#
|
||||||
|
# Unless required by applicable law or agreed to in writing, software
|
||||||
|
# distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
|
||||||
|
# WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
|
||||||
|
# License for the specific language governing permissions and limitations
|
||||||
|
# under the License.
|
||||||
|
"""
|
||||||
|
Ganesha Admin Utilities
|
||||||
|
|
||||||
|
Ganesha NFS does not provide many tools for automating the process of creating
|
||||||
|
and managing export defintions. This module provides utilities to help parse
|
||||||
|
a specified ganesha config file and return a map containing the export
|
||||||
|
definitions and attributes. A method republishing updated export definitions
|
||||||
|
is also provided. And there are methods for requesting the ganesha server
|
||||||
|
to reload the export definitions.
|
||||||
|
|
||||||
|
Consider moving this to common location for use by other manila drivers.
|
||||||
|
"""
|
||||||
|
|
||||||
|
import copy
|
||||||
|
import re
|
||||||
|
import socket
|
||||||
|
import time
|
||||||
|
|
||||||
|
import netaddr
|
||||||
|
from oslo_log import log
|
||||||
|
import six
|
||||||
|
|
||||||
|
from manila import exception
|
||||||
|
from manila.i18n import _, _LI
|
||||||
|
from manila import utils
|
||||||
|
|
||||||
|
LOG = log.getLogger(__name__)
|
||||||
|
# more simple pattern for matching a single avpair per line,
|
||||||
|
# skips lines starting with # comment char
|
||||||
|
AVPATTERN = re.compile('^\s*(?!#)\s*(?P<attr>\S+)\s*=\s*(?P<val>\S+)\s*;')
|
||||||
|
|
||||||
|
# NFS Ganesha v1.5, v2.0 format used here.
|
||||||
|
# TODO(nileshb): Upgrade it to NFS Ganesha 2.1 format.
|
||||||
|
DEFAULT_EXPORT_ATTRS = {
|
||||||
|
'export_id': 'undefined',
|
||||||
|
'path': 'undefined',
|
||||||
|
'fsal': 'undefined',
|
||||||
|
'root_access': '"*"',
|
||||||
|
'rw_access': '"*"',
|
||||||
|
'pseudo': 'undefined',
|
||||||
|
'anonymous_root_uid': '-2',
|
||||||
|
'nfs_protocols': '"3,4"',
|
||||||
|
'transport_protocols': '"UDP,TCP"',
|
||||||
|
'sectype': '"sys"',
|
||||||
|
'maxread': '65536',
|
||||||
|
'maxwrite': '65536',
|
||||||
|
'prefread': '65536',
|
||||||
|
'prefwrite': '65536',
|
||||||
|
'filesystem_id': '192.168',
|
||||||
|
'tag': 'undefined',
|
||||||
|
}
|
||||||
|
|
||||||
|
STARTING_EXPORT_ID = 100
|
||||||
|
|
||||||
|
|
||||||
|
def valid_flags():
|
||||||
|
return DEFAULT_EXPORT_ATTRS.keys()
|
||||||
|
|
||||||
|
|
||||||
|
def parse_ganesha_config(configpath):
|
||||||
|
"""Parse the specified ganesha configuration.
|
||||||
|
|
||||||
|
Parse a configuration file and return a list of lines that were found
|
||||||
|
before the first EXPORT block, and a dictionary of exports and their
|
||||||
|
attributes.
|
||||||
|
|
||||||
|
The input configuration file should be a valid ganesha config file and the
|
||||||
|
export blocks should be the last items in the file.
|
||||||
|
:returns: pre_lines -- List of lines, before the exports clause begins
|
||||||
|
exports -- Dict of exports, indexed with the 'export_id'
|
||||||
|
|
||||||
|
Hers is a sample output:
|
||||||
|
|
||||||
|
pre_lines =
|
||||||
|
[ '###################################################',
|
||||||
|
'# Export entries',
|
||||||
|
'###################################################',
|
||||||
|
'',
|
||||||
|
'',
|
||||||
|
'# First export entry']
|
||||||
|
|
||||||
|
exports =
|
||||||
|
{ '100': { 'anonymous_root_uid': '-2',
|
||||||
|
'export_id': '100',
|
||||||
|
'filesystem_id': '192.168',
|
||||||
|
'fsal': '"GPFS"',
|
||||||
|
'maxread': '65536',
|
||||||
|
'maxwrite': '65536',
|
||||||
|
'nfs_protocols': '"3,4"',
|
||||||
|
'path': '"/gpfs0/share-0d7df0c0-4792-4e2a-68dc7206a164"',
|
||||||
|
'prefread': '65536',
|
||||||
|
'prefwrite': '65536',
|
||||||
|
'pseudo': '"/gpfs0/share-0d7df0c0-4792-4e2a-68dc7206a164"',
|
||||||
|
'root_access': '"*"',
|
||||||
|
'rw_access': '""',
|
||||||
|
'sectype': '"sys"',
|
||||||
|
'tag': '"fs100"',
|
||||||
|
'transport_protocols': '"UDP,TCP"'},
|
||||||
|
'101': { 'anonymous_root_uid': '-2',
|
||||||
|
'export_id': '101',
|
||||||
|
'filesystem_id': '192.168',
|
||||||
|
'fsal': '"GPFS"',
|
||||||
|
'maxread': '65536',
|
||||||
|
'maxwrite': '65536',
|
||||||
|
'nfs_protocols': '"3,4"',
|
||||||
|
'path': '"/gpfs0/share-74bee4dc-e07a-44a9-4be619a13fb1"',
|
||||||
|
'prefread': '65536',
|
||||||
|
'prefwrite': '65536',
|
||||||
|
'pseudo': '"/gpfs0/share-74bee4dc-e07a-44a9-4be619a13fb1"',
|
||||||
|
'root_access': '"*"',
|
||||||
|
'rw_access': '"172.24.4.4"',
|
||||||
|
'sectype': '"sys"',
|
||||||
|
'tag': '"fs101"',
|
||||||
|
'transport_protocols': '"UDP,TCP"'}}
|
||||||
|
"""
|
||||||
|
export_count = 0
|
||||||
|
exports = dict()
|
||||||
|
pre_lines = []
|
||||||
|
with open(configpath) as f:
|
||||||
|
for l in f.readlines():
|
||||||
|
line = l.strip()
|
||||||
|
if export_count == 0 and line != 'EXPORT':
|
||||||
|
pre_lines.append(line)
|
||||||
|
else:
|
||||||
|
if line == 'EXPORT':
|
||||||
|
export_count += 1
|
||||||
|
expattrs = dict()
|
||||||
|
try:
|
||||||
|
match_obj = AVPATTERN.match(line)
|
||||||
|
attr = match_obj.group('attr').lower()
|
||||||
|
val = match_obj.group('val')
|
||||||
|
expattrs[attr] = val
|
||||||
|
if attr == 'export_id':
|
||||||
|
exports[val] = expattrs
|
||||||
|
except AttributeError:
|
||||||
|
pass
|
||||||
|
|
||||||
|
if export_count != len(exports):
|
||||||
|
msg = (_('Invalid export config file %(configpath)s: '
|
||||||
|
'%(exports)s export clauses found, but '
|
||||||
|
'%(export_ids)s export_ids.')
|
||||||
|
% {"configpath": configpath,
|
||||||
|
"exports": str(export_count),
|
||||||
|
"export_ids": str(len(exports))})
|
||||||
|
|
||||||
|
LOG.error(msg)
|
||||||
|
raise exception.GPFSGaneshaException(msg)
|
||||||
|
return pre_lines, exports
|
||||||
|
|
||||||
|
|
||||||
|
def _get_export_by_path(exports, path):
|
||||||
|
for index, export in exports.items():
|
||||||
|
if export and 'path' in export and export['path'].strip('"\'') == path:
|
||||||
|
return export
|
||||||
|
return None
|
||||||
|
|
||||||
|
|
||||||
|
def get_export_by_path(exports, path):
|
||||||
|
"""Return the export that matches the specified path."""
|
||||||
|
return _get_export_by_path(exports, path)
|
||||||
|
|
||||||
|
|
||||||
|
def export_exists(exports, path):
|
||||||
|
"""Return true if an export exists with the specified path."""
|
||||||
|
return _get_export_by_path(exports, path) is not None
|
||||||
|
|
||||||
|
|
||||||
|
def get_next_id(exports):
|
||||||
|
"""Return an export id that is one larger than largest existing id."""
|
||||||
|
try:
|
||||||
|
next_id = max(map(int, exports.keys())) + 1
|
||||||
|
except ValueError:
|
||||||
|
next_id = STARTING_EXPORT_ID
|
||||||
|
|
||||||
|
LOG.debug("Export id = %d", next_id)
|
||||||
|
return next_id
|
||||||
|
|
||||||
|
|
||||||
|
def get_export_template():
|
||||||
|
return copy.copy(DEFAULT_EXPORT_ATTRS)
|
||||||
|
|
||||||
|
|
||||||
|
def _convert_ipstring_to_ipn(ipstring):
|
||||||
|
"""Transform a single ip string into a list of IPNetwork objects."""
|
||||||
|
if netaddr.valid_glob(ipstring):
|
||||||
|
ipns = netaddr.glob_to_cidrs(ipstring)
|
||||||
|
else:
|
||||||
|
try:
|
||||||
|
ipns = [netaddr.IPNetwork(ipstring)]
|
||||||
|
except netaddr.AddrFormatError:
|
||||||
|
msg = (_('Invalid IP access string %s.') % ipstring)
|
||||||
|
LOG.error(msg)
|
||||||
|
raise exception.GPFSGaneshaException(msg)
|
||||||
|
return ipns
|
||||||
|
|
||||||
|
|
||||||
|
def _format_ips(iptokens):
|
||||||
|
ipaddrs = set()
|
||||||
|
for iptoken in iptokens:
|
||||||
|
ipn_list = _convert_ipstring_to_ipn(iptoken)
|
||||||
|
for ipn in ipn_list:
|
||||||
|
ips = [ip for ip in netaddr.iter_unique_ips(ipn)]
|
||||||
|
ipaddrs = ipaddrs.union(ips)
|
||||||
|
return ipaddrs
|
||||||
|
|
||||||
|
|
||||||
|
def format_access_list(access_string, deny_access=None):
|
||||||
|
"""Transform access string into a format ganesha understands."""
|
||||||
|
# handle the case where there is an access string with a trailing comma
|
||||||
|
access_string = access_string.strip(',')
|
||||||
|
iptokens = access_string.split(',')
|
||||||
|
|
||||||
|
ipaddrs = _format_ips(iptokens)
|
||||||
|
|
||||||
|
if deny_access:
|
||||||
|
deny_tokens = deny_access.split(',')
|
||||||
|
deny_ipaddrs = _format_ips(deny_tokens)
|
||||||
|
ipaddrs = ipaddrs - deny_ipaddrs
|
||||||
|
|
||||||
|
ipaddrlist = sorted(list(ipaddrs))
|
||||||
|
|
||||||
|
return ','.join([six.text_type(ip) for ip in ipaddrlist])
|
||||||
|
|
||||||
|
|
||||||
|
def _publish_local_config(configpath, pre_lines, exports):
|
||||||
|
tmp_path = '%s.tmp.%s' % (configpath, time.time())
|
||||||
|
LOG.debug("tmp_path = %s", tmp_path)
|
||||||
|
cpcmd = ['install', '-m', '666', configpath, tmp_path]
|
||||||
|
try:
|
||||||
|
utils.execute(*cpcmd, run_as_root=True)
|
||||||
|
except exception.ProcessExecutionError as e:
|
||||||
|
msg = (_('Failed while publishing ganesha config locally. '
|
||||||
|
'Error: %s.') % six.text_type(e))
|
||||||
|
LOG.error(msg)
|
||||||
|
raise exception.GPFSGaneshaException(msg)
|
||||||
|
|
||||||
|
with open(tmp_path, 'w+') as f:
|
||||||
|
for l in pre_lines:
|
||||||
|
f.write('%s\n' % l)
|
||||||
|
for e in exports:
|
||||||
|
f.write('EXPORT\n{\n')
|
||||||
|
for attr in exports[e]:
|
||||||
|
f.write('%s = %s ;\n' % (attr, exports[e][attr]))
|
||||||
|
|
||||||
|
f.write('}\n')
|
||||||
|
mvcmd = ['mv', tmp_path, configpath]
|
||||||
|
try:
|
||||||
|
utils.execute(*mvcmd, run_as_root=True)
|
||||||
|
except exception.ProcessExecutionError as e:
|
||||||
|
msg = (_('Failed while publishing ganesha config locally. '
|
||||||
|
'Error: %s.') % six.text_type(e))
|
||||||
|
LOG.error(msg)
|
||||||
|
raise exception.GPFSGaneshaException(msg)
|
||||||
|
LOG.info(_LI('Ganesha config %s published locally.'), configpath)
|
||||||
|
|
||||||
|
|
||||||
|
def _publish_remote_config(server, sshlogin, sshkey, configpath):
|
||||||
|
dest = '%s@%s:%s' % (sshlogin, server, configpath)
|
||||||
|
scpcmd = ['scp', '-i', sshkey, configpath, dest]
|
||||||
|
try:
|
||||||
|
utils.execute(*scpcmd, run_as_root=False)
|
||||||
|
except exception.ProcessExecutionError as e:
|
||||||
|
msg = (_('Failed while publishing ganesha config on remote server. '
|
||||||
|
'Error: %s.') % six.text_type(e))
|
||||||
|
LOG.error(msg)
|
||||||
|
raise exception.GPFSGaneshaException(msg)
|
||||||
|
LOG.info(_LI('Ganesha config %(path)s published to %(server)s.'),
|
||||||
|
{'path': configpath,
|
||||||
|
'server': server})
|
||||||
|
|
||||||
|
|
||||||
|
def publish_ganesha_config(servers, sshlogin, sshkey, configpath,
|
||||||
|
pre_lines, exports):
|
||||||
|
"""Publish the specified configuration information.
|
||||||
|
|
||||||
|
Save the existing configuration file and then publish a new
|
||||||
|
ganesha configuration to the specified path. The pre-export
|
||||||
|
lines are written first, followed by the collection of export
|
||||||
|
definitions.
|
||||||
|
"""
|
||||||
|
_publish_local_config(configpath, pre_lines, exports)
|
||||||
|
|
||||||
|
localserver_iplist = socket.gethostbyname_ex(socket.gethostname())[2]
|
||||||
|
for gsvr in servers:
|
||||||
|
if gsvr not in localserver_iplist:
|
||||||
|
_publish_remote_config(gsvr, sshlogin, sshkey, configpath)
|
||||||
|
|
||||||
|
|
||||||
|
def reload_ganesha_config(servers, sshlogin, service='ganesha.nfsd'):
|
||||||
|
"""Request ganesha server reload updated config."""
|
||||||
|
|
||||||
|
# Note: dynamic reload of ganesha config is not enabled
|
||||||
|
# in ganesha v2.0. Therefore, the code uses the ganesha service restart
|
||||||
|
# option to make sure the config changes are reloaded
|
||||||
|
for server in servers:
|
||||||
|
# Until reload is fully implemented and if the reload returns a bad
|
||||||
|
# status revert to service restart instead
|
||||||
|
LOG.info(_LI('Restart service %(service)s on %(server)s to force a '
|
||||||
|
'config file reload'),
|
||||||
|
{'service': service, 'server': server})
|
||||||
|
run_local = True
|
||||||
|
|
||||||
|
reload_cmd = ['service', service, 'restart']
|
||||||
|
localserver_iplist = socket.gethostbyname_ex(
|
||||||
|
socket.gethostname())[2]
|
||||||
|
if server not in localserver_iplist:
|
||||||
|
remote_login = sshlogin + '@' + server
|
||||||
|
reload_cmd = ['ssh', remote_login] + reload_cmd
|
||||||
|
run_local = False
|
||||||
|
try:
|
||||||
|
utils.execute(*reload_cmd, run_as_root=run_local)
|
||||||
|
except exception.ProcessExecutionError as e:
|
||||||
|
msg = (_('Could not restart service %(service)s on '
|
||||||
|
'%(server)s: %(excmsg)s')
|
||||||
|
% {'service': service,
|
||||||
|
'server': server,
|
||||||
|
'excmsg': six.text_type(e)})
|
||||||
|
LOG.error(msg)
|
||||||
|
raise exception.GPFSGaneshaException(msg)
|
859
manila/share/drivers/ibm/gpfs.py
Normal file
859
manila/share/drivers/ibm/gpfs.py
Normal file
@ -0,0 +1,859 @@
|
|||||||
|
# Copyright 2014 IBM Corp.
|
||||||
|
#
|
||||||
|
# Licensed under the Apache License, Version 2.0 (the "License"); you may
|
||||||
|
# not use this file except in compliance with the License. You may obtain
|
||||||
|
# a copy of the License at
|
||||||
|
#
|
||||||
|
# http://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
#
|
||||||
|
# Unless required by applicable law or agreed to in writing, software
|
||||||
|
# distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
|
||||||
|
# WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
|
||||||
|
# License for the specific language governing permissions and limitations
|
||||||
|
# under the License.
|
||||||
|
"""
|
||||||
|
GPFS Driver for shares.
|
||||||
|
|
||||||
|
Config Requirements:
|
||||||
|
GPFS file system must have quotas enabled (`mmchfs -Q yes`).
|
||||||
|
Notes:
|
||||||
|
GPFS independent fileset is used for each share.
|
||||||
|
|
||||||
|
TODO(nileshb): add support for share server creation/deletion/handling.
|
||||||
|
|
||||||
|
Limitation:
|
||||||
|
While using remote GPFS node, with Ganesha NFS, 'gpfs_ssh_private_key'
|
||||||
|
for remote login to the GPFS node must be specified and there must be
|
||||||
|
a passwordless authentication already setup between the Manila share
|
||||||
|
service and the remote GPFS node.
|
||||||
|
|
||||||
|
"""
|
||||||
|
import abc
|
||||||
|
import copy
|
||||||
|
import math
|
||||||
|
import os
|
||||||
|
import re
|
||||||
|
import socket
|
||||||
|
|
||||||
|
from oslo_config import cfg
|
||||||
|
from oslo_log import log
|
||||||
|
from oslo_utils import excutils
|
||||||
|
from oslo_utils import importutils
|
||||||
|
from oslo_utils import strutils
|
||||||
|
from oslo_utils import units
|
||||||
|
import six
|
||||||
|
|
||||||
|
from manila import exception
|
||||||
|
from manila.i18n import _, _LE, _LI
|
||||||
|
from manila.share import driver
|
||||||
|
from manila.share.drivers.ibm import ganesha_utils
|
||||||
|
from manila import utils
|
||||||
|
|
||||||
|
LOG = log.getLogger(__name__)
|
||||||
|
|
||||||
|
# matches multiple comma separated avpairs on a line. values with an embedded
|
||||||
|
# comma must be wrapped in quotation marks
|
||||||
|
AVPATTERN = re.compile(r'\s*(?P<attr>\w+)\s*=\s*(?P<val>'
|
||||||
|
'(["][a-zA-Z0-9_, ]+["])|(\w+))\s*[,]?')
|
||||||
|
|
||||||
|
ERR_FILE_NOT_FOUND = 2
|
||||||
|
|
||||||
|
gpfs_share_opts = [
|
||||||
|
cfg.StrOpt('gpfs_share_export_ip',
|
||||||
|
default=None,
|
||||||
|
help='IP to be added to GPFS export string.'),
|
||||||
|
cfg.StrOpt('gpfs_mount_point_base',
|
||||||
|
default='$state_path/mnt',
|
||||||
|
help='Base folder where exported shares are located.'),
|
||||||
|
cfg.StrOpt('gpfs_nfs_server_type',
|
||||||
|
default='KNFS',
|
||||||
|
help=('NFS Server type. Valid choices are "KNFS" (kernel NFS) '
|
||||||
|
'or "GNFS" (Ganesha NFS).')),
|
||||||
|
cfg.ListOpt('gpfs_nfs_server_list',
|
||||||
|
default=None,
|
||||||
|
help=('A list of the fully qualified NFS server names that '
|
||||||
|
'make up the OpenStack Manila configuration.')),
|
||||||
|
cfg.IntOpt('gpfs_ssh_port',
|
||||||
|
default=22,
|
||||||
|
help='GPFS server SSH port.'),
|
||||||
|
cfg.StrOpt('gpfs_ssh_login',
|
||||||
|
default=None,
|
||||||
|
help='GPFS server SSH login name.'),
|
||||||
|
cfg.StrOpt('gpfs_ssh_password',
|
||||||
|
default=None,
|
||||||
|
secret=True,
|
||||||
|
help='GPFS server SSH login password. '
|
||||||
|
'The password is not needed, if \'gpfs_ssh_private_key\' '
|
||||||
|
'is configured.'),
|
||||||
|
cfg.StrOpt('gpfs_ssh_private_key',
|
||||||
|
default=None,
|
||||||
|
help='Path to GPFS server SSH private key for login.'),
|
||||||
|
cfg.ListOpt('gpfs_share_helpers',
|
||||||
|
default=[
|
||||||
|
'KNFS=manila.share.drivers.ibm.gpfs.KNFSHelper',
|
||||||
|
'GNFS=manila.share.drivers.ibm.gpfs.GNFSHelper',
|
||||||
|
],
|
||||||
|
help='Specify list of share export helpers.'),
|
||||||
|
cfg.StrOpt('knfs_export_options',
|
||||||
|
default=('rw,sync,no_root_squash,insecure,no_wdelay,'
|
||||||
|
'no_subtree_check'),
|
||||||
|
help=('Options to use when exporting a share using kernel '
|
||||||
|
'NFS server. Note that these defaults can be overridden '
|
||||||
|
'when a share is created by passing metadata with key '
|
||||||
|
'name export_options.')),
|
||||||
|
]
|
||||||
|
|
||||||
|
|
||||||
|
CONF = cfg.CONF
|
||||||
|
CONF.register_opts(gpfs_share_opts)
|
||||||
|
|
||||||
|
|
||||||
|
class GPFSShareDriver(driver.ExecuteMixin, driver.GaneshaMixin,
|
||||||
|
driver.ShareDriver):
|
||||||
|
|
||||||
|
"""GPFS Share Driver.
|
||||||
|
|
||||||
|
Executes commands relating to Shares.
|
||||||
|
Supports creation of shares on a GPFS cluster.
|
||||||
|
|
||||||
|
API version history:
|
||||||
|
|
||||||
|
1.0 - Initial version.
|
||||||
|
1.1 - Added extend_share functionality
|
||||||
|
"""
|
||||||
|
|
||||||
|
def __init__(self, *args, **kwargs):
|
||||||
|
"""Do initialization."""
|
||||||
|
super(GPFSShareDriver, self).__init__(False, *args, **kwargs)
|
||||||
|
self._helpers = {}
|
||||||
|
self.configuration.append_config_values(gpfs_share_opts)
|
||||||
|
self.backend_name = self.configuration.safe_get(
|
||||||
|
'share_backend_name') or "IBM Storage System"
|
||||||
|
self.sshpool = None
|
||||||
|
self.ssh_connections = {}
|
||||||
|
self._gpfs_execute = None
|
||||||
|
|
||||||
|
def do_setup(self, context):
|
||||||
|
"""Any initialization the share driver does while starting."""
|
||||||
|
super(GPFSShareDriver, self).do_setup(context)
|
||||||
|
host = self.configuration.gpfs_share_export_ip
|
||||||
|
localserver_iplist = socket.gethostbyname_ex(socket.gethostname())[2]
|
||||||
|
if host in localserver_iplist: # run locally
|
||||||
|
self._gpfs_execute = self._gpfs_local_execute
|
||||||
|
else:
|
||||||
|
self._gpfs_execute = self._gpfs_remote_execute
|
||||||
|
self._setup_helpers()
|
||||||
|
|
||||||
|
def _gpfs_local_execute(self, *cmd, **kwargs):
|
||||||
|
if 'run_as_root' not in kwargs:
|
||||||
|
kwargs.update({'run_as_root': True})
|
||||||
|
|
||||||
|
return utils.execute(*cmd, **kwargs)
|
||||||
|
|
||||||
|
def _gpfs_remote_execute(self, *cmd, **kwargs):
|
||||||
|
host = self.configuration.gpfs_share_export_ip
|
||||||
|
check_exit_code = kwargs.pop('check_exit_code', True)
|
||||||
|
|
||||||
|
return self._run_ssh(host, cmd, check_exit_code)
|
||||||
|
|
||||||
|
def _run_ssh(self, host, cmd_list, ignore_exit_code=None,
|
||||||
|
check_exit_code=True):
|
||||||
|
command = ' '.join(six.moves.shlex_quote(cmd_arg)
|
||||||
|
for cmd_arg in cmd_list)
|
||||||
|
|
||||||
|
if not self.sshpool:
|
||||||
|
gpfs_ssh_login = self.configuration.gpfs_ssh_login
|
||||||
|
password = self.configuration.gpfs_ssh_password
|
||||||
|
privatekey = self.configuration.gpfs_ssh_private_key
|
||||||
|
gpfs_ssh_port = self.configuration.gpfs_ssh_port
|
||||||
|
ssh_conn_timeout = self.configuration.ssh_conn_timeout
|
||||||
|
min_size = self.configuration.ssh_min_pool_conn
|
||||||
|
max_size = self.configuration.ssh_max_pool_conn
|
||||||
|
|
||||||
|
self.sshpool = utils.SSHPool(host,
|
||||||
|
gpfs_ssh_port,
|
||||||
|
ssh_conn_timeout,
|
||||||
|
gpfs_ssh_login,
|
||||||
|
password=password,
|
||||||
|
privatekey=privatekey,
|
||||||
|
min_size=min_size,
|
||||||
|
max_size=max_size)
|
||||||
|
try:
|
||||||
|
with self.sshpool.item() as ssh:
|
||||||
|
return self._gpfs_ssh_execute(
|
||||||
|
ssh,
|
||||||
|
command,
|
||||||
|
check_exit_code=check_exit_code)
|
||||||
|
|
||||||
|
except Exception as e:
|
||||||
|
with excutils.save_and_reraise_exception():
|
||||||
|
msg = (_('Error running SSH command: %(cmd)s. '
|
||||||
|
'Error: %(excmsg)s.') %
|
||||||
|
{'cmd': command, 'excmsg': e})
|
||||||
|
LOG.error(msg)
|
||||||
|
raise exception.GPFSException(msg)
|
||||||
|
|
||||||
|
def _gpfs_ssh_execute(self, ssh, cmd, ignore_exit_code=None,
|
||||||
|
check_exit_code=True):
|
||||||
|
sanitized_cmd = strutils.mask_password(cmd)
|
||||||
|
LOG.debug('Running cmd (SSH): %s', sanitized_cmd)
|
||||||
|
|
||||||
|
stdin_stream, stdout_stream, stderr_stream = ssh.exec_command(cmd)
|
||||||
|
channel = stdout_stream.channel
|
||||||
|
|
||||||
|
stdout = stdout_stream.read()
|
||||||
|
sanitized_stdout = strutils.mask_password(stdout)
|
||||||
|
stderr = stderr_stream.read()
|
||||||
|
sanitized_stderr = strutils.mask_password(stderr)
|
||||||
|
|
||||||
|
stdin_stream.close()
|
||||||
|
|
||||||
|
exit_status = channel.recv_exit_status()
|
||||||
|
|
||||||
|
# exit_status == -1 if no exit code was returned
|
||||||
|
if exit_status != -1:
|
||||||
|
LOG.debug('Result was %s' % exit_status)
|
||||||
|
if ((check_exit_code and exit_status != 0)
|
||||||
|
and
|
||||||
|
(ignore_exit_code is None or
|
||||||
|
exit_status not in ignore_exit_code)):
|
||||||
|
raise exception.ProcessExecutionError(exit_code=exit_status,
|
||||||
|
stdout=sanitized_stdout,
|
||||||
|
stderr=sanitized_stderr,
|
||||||
|
cmd=sanitized_cmd)
|
||||||
|
|
||||||
|
return (sanitized_stdout, sanitized_stderr)
|
||||||
|
|
||||||
|
def _check_gpfs_state(self):
|
||||||
|
try:
|
||||||
|
out, __ = self._gpfs_execute('mmgetstate', '-Y')
|
||||||
|
except exception.ProcessExecutionError as e:
|
||||||
|
msg = (_('Failed to check GPFS state. Error: %(excmsg)s.') %
|
||||||
|
{'excmsg': e})
|
||||||
|
LOG.error(msg)
|
||||||
|
raise exception.GPFSException(msg)
|
||||||
|
lines = out.splitlines()
|
||||||
|
try:
|
||||||
|
state_token = lines[0].split(':').index('state')
|
||||||
|
gpfs_state = lines[1].split(':')[state_token]
|
||||||
|
except (IndexError, ValueError) as e:
|
||||||
|
msg = (_('Failed to check GPFS state. Error: %(excmsg)s.') %
|
||||||
|
{'excmsg': e})
|
||||||
|
LOG.error(msg)
|
||||||
|
raise exception.GPFSException(msg)
|
||||||
|
if gpfs_state != 'active':
|
||||||
|
return False
|
||||||
|
return True
|
||||||
|
|
||||||
|
def _is_dir(self, path):
|
||||||
|
try:
|
||||||
|
output, __ = self._gpfs_execute('stat', '--format=%F', path,
|
||||||
|
run_as_root=False)
|
||||||
|
except exception.ProcessExecutionError as e:
|
||||||
|
msg = (_('%(path)s is not a directory. Error: %(excmsg)s') %
|
||||||
|
{'path': path, 'excmsg': e})
|
||||||
|
LOG.error(msg)
|
||||||
|
raise exception.GPFSException(msg)
|
||||||
|
|
||||||
|
return output.strip() == 'directory'
|
||||||
|
|
||||||
|
def _is_gpfs_path(self, directory):
|
||||||
|
try:
|
||||||
|
self._gpfs_execute('mmlsattr', directory)
|
||||||
|
except exception.ProcessExecutionError as e:
|
||||||
|
msg = (_('%(dir)s is not on GPFS filesystem. Error: %(excmsg)s.') %
|
||||||
|
{'dir': directory, 'excmsg': e})
|
||||||
|
LOG.error(msg)
|
||||||
|
raise exception.GPFSException(msg)
|
||||||
|
|
||||||
|
return True
|
||||||
|
|
||||||
|
def _setup_helpers(self):
|
||||||
|
"""Initializes protocol-specific NAS drivers."""
|
||||||
|
self._helpers = {}
|
||||||
|
for helper_str in self.configuration.gpfs_share_helpers:
|
||||||
|
share_proto, _, import_str = helper_str.partition('=')
|
||||||
|
helper = importutils.import_class(import_str)
|
||||||
|
self._helpers[share_proto.upper()] = helper(self._gpfs_execute,
|
||||||
|
self.configuration)
|
||||||
|
|
||||||
|
def _local_path(self, sharename):
|
||||||
|
"""Get local path for a share or share snapshot by name."""
|
||||||
|
return os.path.join(self.configuration.gpfs_mount_point_base,
|
||||||
|
sharename)
|
||||||
|
|
||||||
|
def _get_gpfs_device(self):
|
||||||
|
fspath = self.configuration.gpfs_mount_point_base
|
||||||
|
try:
|
||||||
|
(out, __) = self._gpfs_execute('df', fspath)
|
||||||
|
except exception.ProcessExecutionError as e:
|
||||||
|
msg = (_('Failed to get GPFS device for %(fspath)s.'
|
||||||
|
'Error: %(excmsg)s') %
|
||||||
|
{'fspath': fspath, 'excmsg': e})
|
||||||
|
LOG.error(msg)
|
||||||
|
raise exception.GPFSException(msg)
|
||||||
|
|
||||||
|
lines = out.splitlines()
|
||||||
|
fs = lines[1].split()[0]
|
||||||
|
return fs
|
||||||
|
|
||||||
|
def _create_share(self, shareobj):
|
||||||
|
"""Create a linked fileset file in GPFS.
|
||||||
|
|
||||||
|
Note: GPFS file system must have quotas enabled
|
||||||
|
(mmchfs -Q yes).
|
||||||
|
"""
|
||||||
|
sharename = shareobj['name']
|
||||||
|
sizestr = '%sG' % shareobj['size']
|
||||||
|
sharepath = self._local_path(sharename)
|
||||||
|
fsdev = self._get_gpfs_device()
|
||||||
|
|
||||||
|
# create fileset for the share, link it to root path and set max size
|
||||||
|
try:
|
||||||
|
self._gpfs_execute('mmcrfileset', fsdev, sharename,
|
||||||
|
'--inode-space', 'new')
|
||||||
|
except exception.ProcessExecutionError as e:
|
||||||
|
msg = (_('Failed to create fileset on %(fsdev)s for '
|
||||||
|
'the share %(sharename)s. Error: %(excmsg)s.') %
|
||||||
|
{'fsdev': fsdev, 'sharename': sharename,
|
||||||
|
'excmsg': e})
|
||||||
|
LOG.error(msg)
|
||||||
|
raise exception.GPFSException(msg)
|
||||||
|
|
||||||
|
try:
|
||||||
|
self._gpfs_execute('mmlinkfileset', fsdev, sharename, '-J',
|
||||||
|
sharepath)
|
||||||
|
except exception.ProcessExecutionError as e:
|
||||||
|
msg = (_('Failed to link fileset for the share %(sharename)s. '
|
||||||
|
'Error: %(excmsg)s.') %
|
||||||
|
{'sharename': sharename, 'excmsg': e})
|
||||||
|
LOG.error(msg)
|
||||||
|
raise exception.GPFSException(msg)
|
||||||
|
|
||||||
|
try:
|
||||||
|
self._gpfs_execute('mmsetquota', '-j', sharename, '-h',
|
||||||
|
sizestr, fsdev)
|
||||||
|
except exception.ProcessExecutionError as e:
|
||||||
|
msg = (_('Failed to set quota for the share %(sharename)s. '
|
||||||
|
'Error: %(excmsg)s.') %
|
||||||
|
{'sharename': sharename, 'excmsg': e})
|
||||||
|
LOG.error(msg)
|
||||||
|
raise exception.GPFSException(msg)
|
||||||
|
|
||||||
|
try:
|
||||||
|
self._gpfs_execute('chmod', '777', sharepath)
|
||||||
|
except exception.ProcessExecutionError as e:
|
||||||
|
msg = (_('Failed to set permissions for share %(sharename)s. '
|
||||||
|
'Error: %(excmsg)s.') %
|
||||||
|
{'sharename': sharename, 'excmsg': e})
|
||||||
|
LOG.error(msg)
|
||||||
|
raise exception.GPFSException(msg)
|
||||||
|
|
||||||
|
def _delete_share(self, shareobj):
|
||||||
|
"""Remove container by removing GPFS fileset."""
|
||||||
|
sharename = shareobj['name']
|
||||||
|
fsdev = self._get_gpfs_device()
|
||||||
|
# ignore error, when the fileset does not exist
|
||||||
|
# it may happen, when the share creation failed, the share is in
|
||||||
|
# 'error' state, and the fileset was never created
|
||||||
|
# we want to ignore that error condition while deleting the fileset,
|
||||||
|
# i.e. 'Fileset name share-xyz not found', with error code '2'
|
||||||
|
# and mark the deletion successful
|
||||||
|
# ignore_exit_code = [ERR_FILE_NOT_FOUND]
|
||||||
|
|
||||||
|
# unlink and delete the share's fileset
|
||||||
|
try:
|
||||||
|
self._gpfs_execute('mmunlinkfileset', fsdev, sharename, '-f')
|
||||||
|
except exception.ProcessExecutionError as e:
|
||||||
|
msg = (_('Failed unlink fileset for share %(sharename)s. '
|
||||||
|
'Error: %(excmsg)s.') %
|
||||||
|
{'sharename': sharename, 'excmsg': e})
|
||||||
|
LOG.error(msg)
|
||||||
|
raise exception.GPFSException(msg)
|
||||||
|
|
||||||
|
try:
|
||||||
|
self._gpfs_execute('mmdelfileset', fsdev, sharename, '-f')
|
||||||
|
except exception.ProcessExecutionError as e:
|
||||||
|
msg = (_('Failed delete fileset for share %(sharename)s. '
|
||||||
|
'Error: %(excmsg)s.') %
|
||||||
|
{'sharename': sharename, 'excmsg': e})
|
||||||
|
LOG.error(msg)
|
||||||
|
raise exception.GPFSException(msg)
|
||||||
|
|
||||||
|
def _get_available_capacity(self, path):
|
||||||
|
"""Calculate available space on path."""
|
||||||
|
try:
|
||||||
|
out, __ = self._gpfs_execute('df', '-P', '-B', '1', path)
|
||||||
|
except exception.ProcessExecutionError as e:
|
||||||
|
msg = (_('Failed to check available capacity for %(path)s.'
|
||||||
|
'Error: %(excmsg)s.') %
|
||||||
|
{'path': path, 'excmsg': e})
|
||||||
|
LOG.error(msg)
|
||||||
|
raise exception.GPFSException(msg)
|
||||||
|
|
||||||
|
out = out.splitlines()[1]
|
||||||
|
size = int(out.split()[1])
|
||||||
|
available = int(out.split()[3])
|
||||||
|
return available, size
|
||||||
|
|
||||||
|
def _create_share_snapshot(self, snapshot):
|
||||||
|
"""Create a snapshot of the share."""
|
||||||
|
sharename = snapshot['share_name']
|
||||||
|
snapshotname = snapshot['name']
|
||||||
|
fsdev = self._get_gpfs_device()
|
||||||
|
LOG.debug("sharename = %s, snapshotname = %s, fsdev = %s",
|
||||||
|
(sharename, snapshotname, fsdev))
|
||||||
|
|
||||||
|
try:
|
||||||
|
self._gpfs_execute('mmcrsnapshot', fsdev, snapshot['name'],
|
||||||
|
'-j', sharename)
|
||||||
|
except exception.ProcessExecutionError as e:
|
||||||
|
msg = (_('Failed to create snapshot %(snapshot)s. '
|
||||||
|
'Error: %(excmsg)s.') %
|
||||||
|
{'snapshot': snapshot['name'], 'excmsg': e})
|
||||||
|
LOG.error(msg)
|
||||||
|
raise exception.GPFSException(msg)
|
||||||
|
|
||||||
|
def _delete_share_snapshot(self, snapshot):
|
||||||
|
"""Delete a snapshot of the share."""
|
||||||
|
sharename = snapshot['share_name']
|
||||||
|
fsdev = self._get_gpfs_device()
|
||||||
|
|
||||||
|
try:
|
||||||
|
self._gpfs_execute('mmdelsnapshot', fsdev, snapshot['name'],
|
||||||
|
'-j', sharename)
|
||||||
|
except exception.ProcessExecutionError as e:
|
||||||
|
msg = (_('Failed to delete snapshot %(snapshot)s. '
|
||||||
|
'Error: %(excmsg)s.') %
|
||||||
|
{'snapshot': snapshot['name'], 'excmsg': e})
|
||||||
|
LOG.error(msg)
|
||||||
|
raise exception.GPFSException(msg)
|
||||||
|
|
||||||
|
def _create_share_from_snapshot(self, share, snapshot, share_path):
|
||||||
|
"""Create share from a share snapshot."""
|
||||||
|
self._create_share(share)
|
||||||
|
snapshot_path = self._get_snapshot_path(snapshot)
|
||||||
|
snapshot_path = snapshot_path + "/"
|
||||||
|
try:
|
||||||
|
self._gpfs_execute('rsync', '-rp', snapshot_path, share_path)
|
||||||
|
except exception.ProcessExecutionError as e:
|
||||||
|
msg = (_('Failed to create share %(share)s from '
|
||||||
|
'snapshot %(snapshot)s. Error: %(excmsg)s.') %
|
||||||
|
{'share': share['name'], 'snapshot': snapshot['name'],
|
||||||
|
'excmsg': e})
|
||||||
|
LOG.error(msg)
|
||||||
|
raise exception.GPFSException(msg)
|
||||||
|
|
||||||
|
def _extend_share(self, shareobj, new_size):
|
||||||
|
sharename = shareobj['name']
|
||||||
|
sizestr = '%sG' % new_size
|
||||||
|
fsdev = self._get_gpfs_device()
|
||||||
|
try:
|
||||||
|
self._gpfs_execute('mmsetquota', '-j', sharename, '-h',
|
||||||
|
sizestr, fsdev)
|
||||||
|
except exception.ProcessExecutionError as e:
|
||||||
|
msg = (_('Failed to set quota for the share %(sharename)s. '
|
||||||
|
'Error: %(excmsg)s.') %
|
||||||
|
{'sharename': sharename, 'excmsg': e})
|
||||||
|
LOG.error(msg)
|
||||||
|
raise exception.GPFSException(msg)
|
||||||
|
|
||||||
|
def get_network_allocations_number(self):
|
||||||
|
return 0
|
||||||
|
|
||||||
|
def create_share(self, ctx, share, share_server=None):
|
||||||
|
"""Create GPFS directory that will be represented as share."""
|
||||||
|
self._create_share(share)
|
||||||
|
share_path = self._get_share_path(share)
|
||||||
|
location = self._get_helper(share).create_export(share_path)
|
||||||
|
return location
|
||||||
|
|
||||||
|
def create_share_from_snapshot(self, ctx, share, snapshot,
|
||||||
|
share_server=None):
|
||||||
|
"""Is called to create share from a snapshot."""
|
||||||
|
share_path = self._get_share_path(share)
|
||||||
|
self._create_share_from_snapshot(share, snapshot, share_path)
|
||||||
|
location = self._get_helper(share).create_export(share_path)
|
||||||
|
return location
|
||||||
|
|
||||||
|
def create_snapshot(self, context, snapshot, share_server=None):
|
||||||
|
"""Creates a snapshot."""
|
||||||
|
self._create_share_snapshot(snapshot)
|
||||||
|
|
||||||
|
def delete_share(self, ctx, share, share_server=None):
|
||||||
|
"""Remove and cleanup share storage."""
|
||||||
|
location = self._get_share_path(share)
|
||||||
|
self._get_helper(share).remove_export(location, share)
|
||||||
|
self._delete_share(share)
|
||||||
|
|
||||||
|
def delete_snapshot(self, context, snapshot, share_server=None):
|
||||||
|
"""Deletes a snapshot."""
|
||||||
|
self._delete_share_snapshot(snapshot)
|
||||||
|
|
||||||
|
def extend_share(self, share, new_size, share_server=None):
|
||||||
|
"""Extends the quota on the share fileset."""
|
||||||
|
self._extend_share(share, new_size)
|
||||||
|
|
||||||
|
def ensure_share(self, ctx, share, share_server=None):
|
||||||
|
"""Ensure that storage are mounted and exported."""
|
||||||
|
|
||||||
|
def allow_access(self, ctx, share, access, share_server=None):
|
||||||
|
"""Allow access to the share."""
|
||||||
|
location = self._get_share_path(share)
|
||||||
|
self._get_helper(share).allow_access(location, share,
|
||||||
|
access['access_type'],
|
||||||
|
access['access_to'])
|
||||||
|
|
||||||
|
def deny_access(self, ctx, share, access, share_server=None):
|
||||||
|
"""Deny access to the share."""
|
||||||
|
location = self._get_share_path(share)
|
||||||
|
self._get_helper(share).deny_access(location, share,
|
||||||
|
access['access_type'],
|
||||||
|
access['access_to'])
|
||||||
|
|
||||||
|
def check_for_setup_error(self):
|
||||||
|
"""Returns an error if prerequisites aren't met."""
|
||||||
|
if not self._check_gpfs_state():
|
||||||
|
msg = (_('GPFS is not active.'))
|
||||||
|
LOG.error(msg)
|
||||||
|
raise exception.GPFSException(msg)
|
||||||
|
|
||||||
|
if not self.configuration.gpfs_share_export_ip:
|
||||||
|
msg = (_('gpfs_share_export_ip must be specified.'))
|
||||||
|
LOG.error(msg)
|
||||||
|
raise exception.InvalidParameterValue(err=msg)
|
||||||
|
|
||||||
|
gpfs_base_dir = self.configuration.gpfs_mount_point_base
|
||||||
|
if not gpfs_base_dir.startswith('/'):
|
||||||
|
msg = (_('%s must be an absolute path.') % gpfs_base_dir)
|
||||||
|
LOG.error(msg)
|
||||||
|
raise exception.GPFSException(msg)
|
||||||
|
|
||||||
|
if not self._is_dir(gpfs_base_dir):
|
||||||
|
msg = (_('%s is not a directory.') % gpfs_base_dir)
|
||||||
|
LOG.error(msg)
|
||||||
|
raise exception.GPFSException(msg)
|
||||||
|
|
||||||
|
if not self._is_gpfs_path(gpfs_base_dir):
|
||||||
|
msg = (_('%s is not on GPFS. Perhaps GPFS not mounted.')
|
||||||
|
% gpfs_base_dir)
|
||||||
|
LOG.error(msg)
|
||||||
|
raise exception.GPFSException(msg)
|
||||||
|
|
||||||
|
if self.configuration.gpfs_nfs_server_type not in ['KNFS', 'GNFS']:
|
||||||
|
msg = (_('Invalid gpfs_nfs_server_type value: %s. '
|
||||||
|
'Valid values are: "KNFS", "GNFS".')
|
||||||
|
% self.configuration.gpfs_nfs_server_type)
|
||||||
|
LOG.error(msg)
|
||||||
|
raise exception.InvalidParameterValue(err=msg)
|
||||||
|
|
||||||
|
if self.configuration.gpfs_nfs_server_list is None:
|
||||||
|
msg = (_('Missing value for gpfs_nfs_server_list.'))
|
||||||
|
LOG.error(msg)
|
||||||
|
raise exception.InvalidParameterValue(err=msg)
|
||||||
|
|
||||||
|
def _update_share_stats(self):
|
||||||
|
"""Retrieve stats info from share volume group."""
|
||||||
|
|
||||||
|
data = dict(
|
||||||
|
share_backend_name=self.backend_name,
|
||||||
|
vendor_name='IBM',
|
||||||
|
storage_protocol='NFS',
|
||||||
|
reserved_percentage=self.configuration.reserved_share_percentage)
|
||||||
|
|
||||||
|
free, capacity = self._get_available_capacity(
|
||||||
|
self.configuration.gpfs_mount_point_base)
|
||||||
|
|
||||||
|
data['total_capacity_gb'] = math.ceil(capacity / units.Gi)
|
||||||
|
data['free_capacity_gb'] = math.ceil(free / units.Gi)
|
||||||
|
|
||||||
|
super(GPFSShareDriver, self)._update_share_stats(data)
|
||||||
|
|
||||||
|
def _get_helper(self, share):
|
||||||
|
if share['share_proto'] == 'NFS':
|
||||||
|
return self._helpers[self.configuration.gpfs_nfs_server_type]
|
||||||
|
else:
|
||||||
|
msg = (_('Share protocol %s not supported by GPFS driver.')
|
||||||
|
% share['share_proto'])
|
||||||
|
LOG.error(msg)
|
||||||
|
raise exception.InvalidShare(reason=msg)
|
||||||
|
|
||||||
|
def _get_share_path(self, share):
|
||||||
|
"""Returns share path on storage provider."""
|
||||||
|
return os.path.join(self.configuration.gpfs_mount_point_base,
|
||||||
|
share['name'])
|
||||||
|
|
||||||
|
def _get_snapshot_path(self, snapshot):
|
||||||
|
"""Returns share path on storage provider."""
|
||||||
|
snapshot_dir = ".snapshots"
|
||||||
|
return os.path.join(self.configuration.gpfs_mount_point_base,
|
||||||
|
snapshot["share_name"], snapshot_dir,
|
||||||
|
snapshot["name"])
|
||||||
|
|
||||||
|
|
||||||
|
@six.add_metaclass(abc.ABCMeta)
|
||||||
|
class NASHelperBase(object):
|
||||||
|
"""Interface to work with share."""
|
||||||
|
|
||||||
|
def __init__(self, execute, config_object):
|
||||||
|
self.configuration = config_object
|
||||||
|
self._execute = execute
|
||||||
|
|
||||||
|
def create_export(self, local_path):
|
||||||
|
"""Construct location of new export."""
|
||||||
|
return ':'.join([self.configuration.gpfs_share_export_ip, local_path])
|
||||||
|
|
||||||
|
@abc.abstractmethod
|
||||||
|
def remove_export(self, local_path, share):
|
||||||
|
"""Remove export."""
|
||||||
|
|
||||||
|
@abc.abstractmethod
|
||||||
|
def allow_access(self, local_path, share, access_type, access):
|
||||||
|
"""Allow access to the host."""
|
||||||
|
|
||||||
|
@abc.abstractmethod
|
||||||
|
def deny_access(self, local_path, share, access_type, access,
|
||||||
|
force=False):
|
||||||
|
"""Deny access to the host."""
|
||||||
|
|
||||||
|
|
||||||
|
class KNFSHelper(NASHelperBase):
|
||||||
|
"""Wrapper for Kernel NFS Commands."""
|
||||||
|
|
||||||
|
def __init__(self, execute, config_object):
|
||||||
|
super(KNFSHelper, self).__init__(execute, config_object)
|
||||||
|
self._execute = execute
|
||||||
|
try:
|
||||||
|
self._execute('exportfs', check_exit_code=True, run_as_root=True)
|
||||||
|
except exception.ProcessExecutionError as e:
|
||||||
|
msg = (_('NFS server not found. Error: %s.') % e)
|
||||||
|
LOG.error(msg)
|
||||||
|
raise exception.GPFSException(msg)
|
||||||
|
|
||||||
|
def _publish_access(self, *cmd):
|
||||||
|
for server in self.configuration.gpfs_nfs_server_list:
|
||||||
|
localserver_iplist = socket.gethostbyname_ex(
|
||||||
|
socket.gethostname())[2]
|
||||||
|
run_local = True
|
||||||
|
if server not in localserver_iplist:
|
||||||
|
sshlogin = self.configuration.gpfs_ssh_login
|
||||||
|
remote_login = sshlogin + '@' + server
|
||||||
|
cmd = ['ssh', remote_login] + list(cmd)
|
||||||
|
run_local = False
|
||||||
|
try:
|
||||||
|
utils.execute(*cmd,
|
||||||
|
run_as_root=run_local,
|
||||||
|
check_exit_code=True)
|
||||||
|
except exception.ProcessExecutionError:
|
||||||
|
raise
|
||||||
|
|
||||||
|
def _get_export_options(self, share):
|
||||||
|
"""Set various export attributes for share."""
|
||||||
|
|
||||||
|
metadata = share.get('share_metadata')
|
||||||
|
options = None
|
||||||
|
if metadata:
|
||||||
|
for item in metadata:
|
||||||
|
if item['key'] == 'export_options':
|
||||||
|
options = item['value']
|
||||||
|
else:
|
||||||
|
msg = (_('Unknown metadata key %s.') % item['key'])
|
||||||
|
LOG.error(msg)
|
||||||
|
raise exception.InvalidInput(reason=msg)
|
||||||
|
if not options:
|
||||||
|
options = self.configuration.knfs_export_options
|
||||||
|
|
||||||
|
return options
|
||||||
|
|
||||||
|
def remove_export(self, local_path, share):
|
||||||
|
"""Remove export."""
|
||||||
|
|
||||||
|
def allow_access(self, local_path, share, access_type, access):
|
||||||
|
"""Allow access to one or more vm instances."""
|
||||||
|
|
||||||
|
if access_type != 'ip':
|
||||||
|
raise exception.InvalidShareAccess('Only ip access type '
|
||||||
|
'supported.')
|
||||||
|
|
||||||
|
# check if present in export
|
||||||
|
try:
|
||||||
|
out, __ = self._execute('exportfs', run_as_root=True)
|
||||||
|
except exception.ProcessExecutionError as e:
|
||||||
|
msg = (_('Failed to check exports on the systems. '
|
||||||
|
' Error: %s.') % e)
|
||||||
|
LOG.error(msg)
|
||||||
|
raise exception.GPFSException(msg)
|
||||||
|
|
||||||
|
out = re.search(re.escape(local_path) + '[\s\n]*' + re.escape(access),
|
||||||
|
out)
|
||||||
|
if out is not None:
|
||||||
|
raise exception.ShareAccessExists(access_type=access_type,
|
||||||
|
access=access)
|
||||||
|
|
||||||
|
export_opts = self._get_export_options(share)
|
||||||
|
|
||||||
|
cmd = ['exportfs', '-o', export_opts,
|
||||||
|
':'.join([access, local_path])]
|
||||||
|
try:
|
||||||
|
self._publish_access(*cmd)
|
||||||
|
except exception.ProcessExecutionError as e:
|
||||||
|
msg = (_('Failed to allow access for share %(sharename)s. '
|
||||||
|
'Error: %(excmsg)s.') %
|
||||||
|
{'sharename': share['name'],
|
||||||
|
'excmsg': e})
|
||||||
|
LOG.error(msg)
|
||||||
|
raise exception.GPFSException(msg)
|
||||||
|
|
||||||
|
def deny_access(self, local_path, share, access_type, access,
|
||||||
|
force=False):
|
||||||
|
"""Remove access for one or more vm instances."""
|
||||||
|
cmd = ['exportfs', '-u', ':'.join([access, local_path])]
|
||||||
|
try:
|
||||||
|
self._publish_access(*cmd)
|
||||||
|
except exception.ProcessExecutionError as e:
|
||||||
|
msg = (_('Failed to deny access for share %(sharename)s. '
|
||||||
|
'Error: %(excmsg)s.') %
|
||||||
|
{'sharename': share['name'],
|
||||||
|
'excmsg': e})
|
||||||
|
LOG.error(msg)
|
||||||
|
raise exception.GPFSException(msg)
|
||||||
|
|
||||||
|
|
||||||
|
class GNFSHelper(NASHelperBase):
|
||||||
|
"""Wrapper for Ganesha NFS Commands."""
|
||||||
|
|
||||||
|
def __init__(self, execute, config_object):
|
||||||
|
super(GNFSHelper, self).__init__(execute, config_object)
|
||||||
|
self.default_export_options = dict()
|
||||||
|
for m in AVPATTERN.finditer(
|
||||||
|
self.configuration.ganesha_nfs_export_options
|
||||||
|
):
|
||||||
|
self.default_export_options[m.group('attr')] = m.group('val')
|
||||||
|
|
||||||
|
def _get_export_options(self, share):
|
||||||
|
"""Set various export attributes for share."""
|
||||||
|
|
||||||
|
# load default options first - any options passed as share metadata
|
||||||
|
# will take precedence
|
||||||
|
options = copy.copy(self.default_export_options)
|
||||||
|
|
||||||
|
metadata = share.get('share_metadata')
|
||||||
|
for item in metadata:
|
||||||
|
attr = item['key']
|
||||||
|
if attr in ganesha_utils.valid_flags():
|
||||||
|
options[attr] = item['value']
|
||||||
|
else:
|
||||||
|
LOG.error(_LE('Invalid metadata %(attr)s for share '
|
||||||
|
'%(share)s.'),
|
||||||
|
{'attr': attr, 'share': share['name']})
|
||||||
|
|
||||||
|
return options
|
||||||
|
|
||||||
|
@utils.synchronized("ganesha-process-req", external=True)
|
||||||
|
def _ganesha_process_request(self, req_type, local_path,
|
||||||
|
share, access_type=None,
|
||||||
|
access=None, force=False):
|
||||||
|
cfgpath = self.configuration.ganesha_config_path
|
||||||
|
gservice = self.configuration.ganesha_service_name
|
||||||
|
gservers = self.configuration.gpfs_nfs_server_list
|
||||||
|
sshlogin = self.configuration.gpfs_ssh_login
|
||||||
|
sshkey = self.configuration.gpfs_ssh_private_key
|
||||||
|
pre_lines, exports = ganesha_utils.parse_ganesha_config(cfgpath)
|
||||||
|
reload_needed = True
|
||||||
|
|
||||||
|
if (req_type == "allow_access"):
|
||||||
|
export_opts = self._get_export_options(share)
|
||||||
|
# add the new share if it's not already defined
|
||||||
|
if not ganesha_utils.export_exists(exports, local_path):
|
||||||
|
# Add a brand new export definition
|
||||||
|
new_id = ganesha_utils.get_next_id(exports)
|
||||||
|
export = ganesha_utils.get_export_template()
|
||||||
|
export['fsal'] = '"GPFS"'
|
||||||
|
export['export_id'] = new_id
|
||||||
|
export['tag'] = '"fs%s"' % new_id
|
||||||
|
export['path'] = '"%s"' % local_path
|
||||||
|
export['pseudo'] = '"%s"' % local_path
|
||||||
|
export['rw_access'] = (
|
||||||
|
'"%s"' % ganesha_utils.format_access_list(access)
|
||||||
|
)
|
||||||
|
for key in export_opts:
|
||||||
|
export[key] = export_opts[key]
|
||||||
|
|
||||||
|
exports[new_id] = export
|
||||||
|
LOG.info(_LI('Add %(share)s with access from %(access)s'),
|
||||||
|
{'share': share['name'], 'access': access})
|
||||||
|
else:
|
||||||
|
# Update existing access with new/extended access information
|
||||||
|
export = ganesha_utils.get_export_by_path(exports, local_path)
|
||||||
|
initial_access = export['rw_access'].strip('"')
|
||||||
|
merged_access = ','.join([access, initial_access])
|
||||||
|
updated_access = ganesha_utils.format_access_list(
|
||||||
|
merged_access
|
||||||
|
)
|
||||||
|
if initial_access != updated_access:
|
||||||
|
LOG.info(_LI('Update %(share)s with access from '
|
||||||
|
'%(access)s'),
|
||||||
|
{'share': share['name'], 'access': access})
|
||||||
|
export['rw_access'] = '"%s"' % updated_access
|
||||||
|
else:
|
||||||
|
LOG.info(_LI('Do not update %(share)s, access from '
|
||||||
|
'%(access)s already defined'),
|
||||||
|
{'share': share['name'], 'access': access})
|
||||||
|
reload_needed = False
|
||||||
|
|
||||||
|
elif (req_type == "deny_access"):
|
||||||
|
export = ganesha_utils.get_export_by_path(exports, local_path)
|
||||||
|
initial_access = export['rw_access'].strip('"')
|
||||||
|
updated_access = ganesha_utils.format_access_list(
|
||||||
|
initial_access,
|
||||||
|
deny_access=access
|
||||||
|
)
|
||||||
|
|
||||||
|
if initial_access != updated_access:
|
||||||
|
LOG.info(_LI('Update %(share)s removing access from '
|
||||||
|
'%(access)s'),
|
||||||
|
{'share': share['name'], 'access': access})
|
||||||
|
export['rw_access'] = '"%s"' % updated_access
|
||||||
|
else:
|
||||||
|
LOG.info(_LI('Do not update %(share)s, access from %(access)s '
|
||||||
|
'already removed'), {'share': share['name'],
|
||||||
|
'access': access})
|
||||||
|
reload_needed = False
|
||||||
|
|
||||||
|
elif (req_type == "remove_export"):
|
||||||
|
export = ganesha_utils.get_export_by_path(exports, local_path)
|
||||||
|
if export:
|
||||||
|
exports.pop(export['export_id'])
|
||||||
|
LOG.info(_LI('Remove export for %s'), share['name'])
|
||||||
|
else:
|
||||||
|
LOG.info(_LI('Export for %s is not defined in Ganesha '
|
||||||
|
'config.'),
|
||||||
|
share['name'])
|
||||||
|
reload_needed = False
|
||||||
|
|
||||||
|
if reload_needed:
|
||||||
|
# publish config to all servers and reload or restart
|
||||||
|
ganesha_utils.publish_ganesha_config(gservers, sshlogin, sshkey,
|
||||||
|
cfgpath, pre_lines, exports)
|
||||||
|
ganesha_utils.reload_ganesha_config(gservers, sshlogin, gservice)
|
||||||
|
|
||||||
|
def remove_export(self, local_path, share):
|
||||||
|
"""Remove export."""
|
||||||
|
self._ganesha_process_request("remove_export", local_path, share)
|
||||||
|
|
||||||
|
def allow_access(self, local_path, share, access_type, access):
|
||||||
|
"""Allow access to the host."""
|
||||||
|
# TODO(nileshb): add support for read only, metadata, and other
|
||||||
|
# access types
|
||||||
|
if access_type != 'ip':
|
||||||
|
raise exception.InvalidShareAccess('Only ip access type '
|
||||||
|
'supported.')
|
||||||
|
|
||||||
|
self._ganesha_process_request("allow_access", local_path,
|
||||||
|
share, access_type, access)
|
||||||
|
|
||||||
|
def deny_access(self, local_path, share, access_type, access,
|
||||||
|
force=False):
|
||||||
|
"""Deny access to the host."""
|
||||||
|
self._ganesha_process_request("deny_access", local_path,
|
||||||
|
share, access_type, access, force)
|
0
manila/tests/share/drivers/ibm/__init__.py
Normal file
0
manila/tests/share/drivers/ibm/__init__.py
Normal file
281
manila/tests/share/drivers/ibm/test_ganesha_utils.py
Normal file
281
manila/tests/share/drivers/ibm/test_ganesha_utils.py
Normal file
@ -0,0 +1,281 @@
|
|||||||
|
# Copyright (c) 2014 IBM Corp.
|
||||||
|
#
|
||||||
|
# Licensed under the Apache License, Version 2.0 (the "License"); you may
|
||||||
|
# not use this file except in compliance with the License. You may obtain
|
||||||
|
# a copy of the License at
|
||||||
|
#
|
||||||
|
# http://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
#
|
||||||
|
# Unless required by applicable law or agreed to in writing, software
|
||||||
|
# distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
|
||||||
|
# WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
|
||||||
|
# License for the specific language governing permissions and limitations
|
||||||
|
# under the License.
|
||||||
|
|
||||||
|
"""Unit tests for the Ganesha Utils module."""
|
||||||
|
|
||||||
|
import socket
|
||||||
|
import time
|
||||||
|
|
||||||
|
import mock
|
||||||
|
from oslo_config import cfg
|
||||||
|
|
||||||
|
from manila import exception
|
||||||
|
import manila.share.drivers.ibm.ganesha_utils as ganesha_utils
|
||||||
|
from manila import test
|
||||||
|
from manila import utils
|
||||||
|
|
||||||
|
|
||||||
|
CONF = cfg.CONF
|
||||||
|
|
||||||
|
|
||||||
|
def fake_pre_lines(**kwargs):
|
||||||
|
pre_lines = [
|
||||||
|
'###################################################',
|
||||||
|
'# Export entries',
|
||||||
|
'###################################################',
|
||||||
|
'',
|
||||||
|
'',
|
||||||
|
'# First export entry',
|
||||||
|
]
|
||||||
|
return pre_lines
|
||||||
|
|
||||||
|
|
||||||
|
def fake_exports(**kwargs):
|
||||||
|
exports = {
|
||||||
|
'100': {
|
||||||
|
'anonymous_root_uid': '-2',
|
||||||
|
'export_id': '100',
|
||||||
|
'filesystem_id': '192.168',
|
||||||
|
'fsal': '"GPFS"',
|
||||||
|
'maxread': '65536',
|
||||||
|
'maxwrite': '65536',
|
||||||
|
'nfs_protocols': '"3,4"',
|
||||||
|
'path': '"/fs0/share-1234"',
|
||||||
|
'prefread': '65536',
|
||||||
|
'prefwrite': '65536',
|
||||||
|
'pseudo': '"/fs0/share-1234"',
|
||||||
|
'root_access': '"*"',
|
||||||
|
'rw_access': '""',
|
||||||
|
'sectype': '"sys"',
|
||||||
|
'tag': '"fs100"',
|
||||||
|
'transport_protocols': '"UDP,TCP"',
|
||||||
|
},
|
||||||
|
'101': {
|
||||||
|
'anonymous_root_uid': '-2',
|
||||||
|
'export_id': '101',
|
||||||
|
'filesystem_id': '192.168',
|
||||||
|
'fsal': '"GPFS"',
|
||||||
|
'maxread': '65536',
|
||||||
|
'maxwrite': '65536',
|
||||||
|
'nfs_protocols': '"3,4"',
|
||||||
|
'path': '"/fs0/share-5678"',
|
||||||
|
'prefread': '65536',
|
||||||
|
'prefwrite': '65536',
|
||||||
|
'pseudo': '"/fs0/share-5678"',
|
||||||
|
'root_access': '"*"',
|
||||||
|
'rw_access': '"172.24.4.4"',
|
||||||
|
'sectype': '"sys"',
|
||||||
|
'tag': '"fs101"',
|
||||||
|
'transport_protocols': '"UDP,TCP"',
|
||||||
|
},
|
||||||
|
}
|
||||||
|
return exports
|
||||||
|
|
||||||
|
|
||||||
|
class GaneshaUtilsTestCase(test.TestCase):
|
||||||
|
"""Tests Ganesha Utils."""
|
||||||
|
|
||||||
|
def setUp(self):
|
||||||
|
super(GaneshaUtilsTestCase, self).setUp()
|
||||||
|
self.fake_path = "/fs0/share-1234"
|
||||||
|
self.fake_pre_lines = fake_pre_lines()
|
||||||
|
self.fake_exports = fake_exports()
|
||||||
|
self.fake_configpath = "/etc/ganesha/ganesha.exports.conf"
|
||||||
|
self.local_ip = ["192.11.22.1"]
|
||||||
|
self.remote_ips = ["192.11.22.2", "192.11.22.3"]
|
||||||
|
self.servers = self.local_ip + self.remote_ips
|
||||||
|
self.sshlogin = "fake_login"
|
||||||
|
self.sshkey = "fake_sshkey"
|
||||||
|
self.STARTING_EXPORT_ID = 100
|
||||||
|
self.mock_object(socket, 'gethostname',
|
||||||
|
mock.Mock(return_value="testserver"))
|
||||||
|
self.mock_object(socket, 'gethostbyname_ex', mock.Mock(
|
||||||
|
return_value=('localhost',
|
||||||
|
['localhost.localdomain', 'testserver'],
|
||||||
|
['127.0.0.1'] + self.local_ip)
|
||||||
|
))
|
||||||
|
|
||||||
|
def test_get_export_by_path(self):
|
||||||
|
fake_export = {'export_id': '100'}
|
||||||
|
self.mock_object(ganesha_utils, '_get_export_by_path',
|
||||||
|
mock.Mock(return_value=fake_export))
|
||||||
|
export = ganesha_utils.get_export_by_path(self.fake_exports,
|
||||||
|
self.fake_path)
|
||||||
|
self.assertEqual(export, fake_export)
|
||||||
|
ganesha_utils._get_export_by_path.assert_called_once_with(
|
||||||
|
self.fake_exports, self.fake_path
|
||||||
|
)
|
||||||
|
|
||||||
|
def test_export_exists(self):
|
||||||
|
fake_export = {'export_id': '100'}
|
||||||
|
self.mock_object(ganesha_utils, '_get_export_by_path',
|
||||||
|
mock.Mock(return_value=fake_export))
|
||||||
|
result = ganesha_utils.export_exists(self.fake_exports, self.fake_path)
|
||||||
|
self.assertTrue(result)
|
||||||
|
ganesha_utils._get_export_by_path.assert_called_once_with(
|
||||||
|
self.fake_exports, self.fake_path
|
||||||
|
)
|
||||||
|
|
||||||
|
def test__get_export_by_path_export_exists(self):
|
||||||
|
expected_export = {
|
||||||
|
'anonymous_root_uid': '-2',
|
||||||
|
'export_id': '100',
|
||||||
|
'filesystem_id': '192.168',
|
||||||
|
'fsal': '"GPFS"',
|
||||||
|
'maxread': '65536',
|
||||||
|
'maxwrite': '65536',
|
||||||
|
'nfs_protocols': '"3,4"',
|
||||||
|
'path': '"/fs0/share-1234"',
|
||||||
|
'prefread': '65536',
|
||||||
|
'prefwrite': '65536',
|
||||||
|
'pseudo': '"/fs0/share-1234"',
|
||||||
|
'root_access': '"*"',
|
||||||
|
'rw_access': '""',
|
||||||
|
'sectype': '"sys"',
|
||||||
|
'tag': '"fs100"',
|
||||||
|
'transport_protocols': '"UDP,TCP"',
|
||||||
|
}
|
||||||
|
export = ganesha_utils._get_export_by_path(self.fake_exports,
|
||||||
|
self.fake_path)
|
||||||
|
self.assertEqual(export, expected_export)
|
||||||
|
|
||||||
|
def test__get_export_by_path_export_does_not_exists(self):
|
||||||
|
share_path = '/fs0/share-1111'
|
||||||
|
export = ganesha_utils._get_export_by_path(self.fake_exports,
|
||||||
|
share_path)
|
||||||
|
self.assertIsNone(export)
|
||||||
|
|
||||||
|
def test_get_next_id(self):
|
||||||
|
expected_id = 102
|
||||||
|
result = ganesha_utils.get_next_id(self.fake_exports)
|
||||||
|
self.assertEqual(result, expected_id)
|
||||||
|
|
||||||
|
def test_convert_ipstring_to_ipn_exception(self):
|
||||||
|
ipstring = 'fake ip string'
|
||||||
|
self.assertRaises(exception.GPFSGaneshaException,
|
||||||
|
ganesha_utils._convert_ipstring_to_ipn,
|
||||||
|
ipstring)
|
||||||
|
|
||||||
|
@mock.patch('six.moves.builtins.map')
|
||||||
|
def test_get_next_id_first_export(self, mock_map):
|
||||||
|
expected_id = self.STARTING_EXPORT_ID
|
||||||
|
mock_map.side_effect = ValueError
|
||||||
|
result = ganesha_utils.get_next_id(self.fake_exports)
|
||||||
|
self.assertEqual(result, expected_id)
|
||||||
|
|
||||||
|
def test_format_access_list(self):
|
||||||
|
access_string = "9.123.12.1,9.123.12.2,9.122"
|
||||||
|
result = ganesha_utils.format_access_list(access_string, None)
|
||||||
|
self.assertEqual(result, "9.122.0.0,9.123.12.1,9.123.12.2")
|
||||||
|
|
||||||
|
def test_format_access_list_deny_access(self):
|
||||||
|
access_string = "9.123.12.1,9.123,12.2"
|
||||||
|
deny_access = "9.123,12.2"
|
||||||
|
result = ganesha_utils.format_access_list(access_string,
|
||||||
|
deny_access=deny_access)
|
||||||
|
self.assertEqual(result, "9.123.12.1")
|
||||||
|
|
||||||
|
def test_publish_ganesha_config(self):
|
||||||
|
configpath = self.fake_configpath
|
||||||
|
methods = ('_publish_local_config', '_publish_remote_config')
|
||||||
|
for method in methods:
|
||||||
|
self.mock_object(ganesha_utils, method)
|
||||||
|
ganesha_utils.publish_ganesha_config(self.servers, self.sshlogin,
|
||||||
|
self.sshkey, configpath,
|
||||||
|
self.fake_pre_lines,
|
||||||
|
self.fake_exports)
|
||||||
|
ganesha_utils._publish_local_config.assert_called_once_with(
|
||||||
|
configpath, self.fake_pre_lines, self.fake_exports
|
||||||
|
)
|
||||||
|
for remote_ip in self.remote_ips:
|
||||||
|
ganesha_utils._publish_remote_config.assert_any_call(
|
||||||
|
remote_ip, self.sshlogin, self.sshkey, configpath
|
||||||
|
)
|
||||||
|
|
||||||
|
def test_reload_ganesha_config(self):
|
||||||
|
self.mock_object(utils, 'execute', mock.Mock(return_value=True))
|
||||||
|
service = 'ganesha.nfsd'
|
||||||
|
ganesha_utils.reload_ganesha_config(self.servers, self.sshlogin)
|
||||||
|
reload_cmd = ['service', service, 'restart']
|
||||||
|
utils.execute.assert_any_call(*reload_cmd, run_as_root=True)
|
||||||
|
for remote_ip in self.remote_ips:
|
||||||
|
reload_cmd = ['service', service, 'restart']
|
||||||
|
remote_login = self.sshlogin + '@' + remote_ip
|
||||||
|
reload_cmd = ['ssh', remote_login] + reload_cmd
|
||||||
|
utils.execute.assert_any_call(
|
||||||
|
*reload_cmd, run_as_root=False
|
||||||
|
)
|
||||||
|
|
||||||
|
def test_reload_ganesha_config_exception(self):
|
||||||
|
self.mock_object(
|
||||||
|
utils, 'execute',
|
||||||
|
mock.Mock(side_effect=exception.ProcessExecutionError))
|
||||||
|
self.assertRaises(exception.GPFSGaneshaException,
|
||||||
|
ganesha_utils.reload_ganesha_config,
|
||||||
|
self.servers, self.sshlogin)
|
||||||
|
|
||||||
|
@mock.patch('six.moves.builtins.open')
|
||||||
|
def test__publish_local_config(self, mock_open):
|
||||||
|
self.mock_object(utils, 'execute', mock.Mock(return_value=True))
|
||||||
|
fake_timestamp = 1415506949.75
|
||||||
|
self.mock_object(time, 'time', mock.Mock(return_value=fake_timestamp))
|
||||||
|
configpath = self.fake_configpath
|
||||||
|
tmp_path = '%s.tmp.%s' % (configpath, fake_timestamp)
|
||||||
|
ganesha_utils._publish_local_config(configpath,
|
||||||
|
self.fake_pre_lines,
|
||||||
|
self.fake_exports)
|
||||||
|
cpcmd = ['install', '-m', '666', configpath, tmp_path]
|
||||||
|
utils.execute.assert_any_call(*cpcmd, run_as_root=True)
|
||||||
|
mvcmd = ['mv', tmp_path, configpath]
|
||||||
|
utils.execute.assert_any_call(*mvcmd, run_as_root=True)
|
||||||
|
self.assertTrue(time.time.called)
|
||||||
|
|
||||||
|
@mock.patch('six.moves.builtins.open')
|
||||||
|
def test__publish_local_config_exception(self, mock_open):
|
||||||
|
self.mock_object(
|
||||||
|
utils, 'execute',
|
||||||
|
mock.Mock(side_effect=exception.ProcessExecutionError))
|
||||||
|
fake_timestamp = 1415506949.75
|
||||||
|
self.mock_object(time, 'time', mock.Mock(return_value=fake_timestamp))
|
||||||
|
configpath = self.fake_configpath
|
||||||
|
tmp_path = '%s.tmp.%s' % (configpath, fake_timestamp)
|
||||||
|
self.assertRaises(exception.GPFSGaneshaException,
|
||||||
|
ganesha_utils._publish_local_config, configpath,
|
||||||
|
self.fake_pre_lines, self.fake_exports)
|
||||||
|
cpcmd = ['install', '-m', '666', configpath, tmp_path]
|
||||||
|
utils.execute.assert_called_once_with(*cpcmd, run_as_root=True)
|
||||||
|
self.assertTrue(time.time.called)
|
||||||
|
|
||||||
|
def test__publish_remote_config(self):
|
||||||
|
utils.execute = mock.Mock(return_value=True)
|
||||||
|
server = self.remote_ips[1]
|
||||||
|
dest = '%s@%s:%s' % (self.sshlogin, server, self.fake_configpath)
|
||||||
|
scpcmd = ['scp', '-i', self.sshkey, self.fake_configpath, dest]
|
||||||
|
|
||||||
|
ganesha_utils._publish_remote_config(server, self.sshlogin,
|
||||||
|
self.sshkey, self.fake_configpath)
|
||||||
|
utils.execute.assert_called_once_with(*scpcmd, run_as_root=False)
|
||||||
|
|
||||||
|
def test__publish_remote_config_exception(self):
|
||||||
|
self.mock_object(
|
||||||
|
utils, 'execute',
|
||||||
|
mock.Mock(side_effect=exception.ProcessExecutionError))
|
||||||
|
server = self.remote_ips[1]
|
||||||
|
dest = '%s@%s:%s' % (self.sshlogin, server, self.fake_configpath)
|
||||||
|
scpcmd = ['scp', '-i', self.sshkey, self.fake_configpath, dest]
|
||||||
|
|
||||||
|
self.assertRaises(exception.GPFSGaneshaException,
|
||||||
|
ganesha_utils._publish_remote_config, server,
|
||||||
|
self.sshlogin, self.sshkey, self.fake_configpath)
|
||||||
|
utils.execute.assert_called_once_with(*scpcmd, run_as_root=False)
|
916
manila/tests/share/drivers/ibm/test_gpfs.py
Normal file
916
manila/tests/share/drivers/ibm/test_gpfs.py
Normal file
@ -0,0 +1,916 @@
|
|||||||
|
# Copyright (c) 2014 IBM Corp.
|
||||||
|
#
|
||||||
|
# Licensed under the Apache License, Version 2.0 (the "License"); you may
|
||||||
|
# not use this file except in compliance with the License. You may obtain
|
||||||
|
# a copy of the License at
|
||||||
|
#
|
||||||
|
# http://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
#
|
||||||
|
# Unless required by applicable law or agreed to in writing, software
|
||||||
|
# distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
|
||||||
|
# WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
|
||||||
|
# License for the specific language governing permissions and limitations
|
||||||
|
# under the License.
|
||||||
|
|
||||||
|
"""Unit tests for the IBM GPFS driver module."""
|
||||||
|
|
||||||
|
import re
|
||||||
|
import socket
|
||||||
|
|
||||||
|
import ddt
|
||||||
|
import mock
|
||||||
|
from oslo_config import cfg
|
||||||
|
|
||||||
|
from manila import context
|
||||||
|
from manila import exception
|
||||||
|
import manila.share.configuration as config
|
||||||
|
import manila.share.drivers.ibm.ganesha_utils as ganesha_utils
|
||||||
|
import manila.share.drivers.ibm.gpfs as gpfs
|
||||||
|
from manila import test
|
||||||
|
from manila.tests import fake_share
|
||||||
|
from manila import utils
|
||||||
|
|
||||||
|
|
||||||
|
CONF = cfg.CONF
|
||||||
|
|
||||||
|
|
||||||
|
@ddt.ddt
|
||||||
|
class GPFSShareDriverTestCase(test.TestCase):
|
||||||
|
"""Tests GPFSShareDriver."""
|
||||||
|
|
||||||
|
def setUp(self):
|
||||||
|
super(GPFSShareDriverTestCase, self).setUp()
|
||||||
|
self._context = context.get_admin_context()
|
||||||
|
self._gpfs_execute = mock.Mock(return_value=('', ''))
|
||||||
|
|
||||||
|
self._helper_fake = mock.Mock()
|
||||||
|
CONF.set_default('driver_handles_share_servers', False)
|
||||||
|
self.fake_conf = config.Configuration(None)
|
||||||
|
self._driver = gpfs.GPFSShareDriver(execute=self._gpfs_execute,
|
||||||
|
configuration=self.fake_conf)
|
||||||
|
self._knfs_helper = gpfs.KNFSHelper(self._gpfs_execute,
|
||||||
|
self.fake_conf)
|
||||||
|
self._gnfs_helper = gpfs.GNFSHelper(self._gpfs_execute,
|
||||||
|
self.fake_conf)
|
||||||
|
self.fakedev = "/dev/gpfs0"
|
||||||
|
self.fakefspath = "/gpfs0"
|
||||||
|
self.fakesharepath = "/gpfs0/share-fakeid"
|
||||||
|
self.fakesnapshotpath = "/gpfs0/.snapshots/snapshot-fakesnapshotid"
|
||||||
|
self.mock_object(gpfs.os.path, 'exists', mock.Mock(return_value=True))
|
||||||
|
self._driver._helpers = {
|
||||||
|
'KNFS': self._helper_fake
|
||||||
|
}
|
||||||
|
self.share = fake_share.fake_share(share_proto='NFS')
|
||||||
|
self.server = {
|
||||||
|
'backend_details': {
|
||||||
|
'ip': '1.2.3.4',
|
||||||
|
'instance_id': 'fake'
|
||||||
|
}
|
||||||
|
}
|
||||||
|
self.access = fake_share.fake_access()
|
||||||
|
self.snapshot = fake_share.fake_snapshot()
|
||||||
|
self.local_ip = "192.11.22.1"
|
||||||
|
self.remote_ip = "192.11.22.2"
|
||||||
|
gpfs_nfs_server_list = [self.local_ip, self.remote_ip]
|
||||||
|
self._knfs_helper.configuration.gpfs_nfs_server_list = \
|
||||||
|
gpfs_nfs_server_list
|
||||||
|
self._gnfs_helper.configuration.gpfs_nfs_server_list = \
|
||||||
|
gpfs_nfs_server_list
|
||||||
|
self._gnfs_helper.configuration.ganesha_config_path = \
|
||||||
|
"fake_ganesha_config_path"
|
||||||
|
self.sshlogin = "fake_login"
|
||||||
|
self.sshkey = "fake_sshkey"
|
||||||
|
self.gservice = "fake_ganesha_service"
|
||||||
|
self._gnfs_helper.configuration.gpfs_ssh_login = self.sshlogin
|
||||||
|
self._gnfs_helper.configuration.gpfs_ssh_private_key = self.sshkey
|
||||||
|
self._gnfs_helper.configuration.ganesha_service_name = self.gservice
|
||||||
|
self.mock_object(socket, 'gethostname',
|
||||||
|
mock.Mock(return_value="testserver"))
|
||||||
|
self.mock_object(socket, 'gethostbyname_ex', mock.Mock(
|
||||||
|
return_value=('localhost',
|
||||||
|
['localhost.localdomain', 'testserver'],
|
||||||
|
['127.0.0.1', self.local_ip])
|
||||||
|
))
|
||||||
|
|
||||||
|
def test__run_ssh(self):
|
||||||
|
cmd_list = ['fake', 'cmd']
|
||||||
|
expected_cmd = 'fake cmd'
|
||||||
|
ssh_pool = mock.Mock()
|
||||||
|
ssh = mock.Mock()
|
||||||
|
self.mock_object(utils, 'SSHPool', mock.Mock(return_value=ssh_pool))
|
||||||
|
ssh_pool.item = mock.Mock(return_value=ssh)
|
||||||
|
setattr(ssh, '__enter__', mock.Mock())
|
||||||
|
setattr(ssh, '__exit__', mock.Mock())
|
||||||
|
self.mock_object(self._driver, '_gpfs_ssh_execute')
|
||||||
|
self._driver._run_ssh(self.local_ip, cmd_list)
|
||||||
|
|
||||||
|
self._driver._gpfs_ssh_execute.assert_called_once_with(
|
||||||
|
mock.ANY, expected_cmd, check_exit_code=True)
|
||||||
|
|
||||||
|
def test__run_ssh_exception(self):
|
||||||
|
cmd_list = ['fake', 'cmd']
|
||||||
|
ssh_pool = mock.Mock()
|
||||||
|
ssh = mock.Mock()
|
||||||
|
self.mock_object(utils, 'SSHPool', mock.Mock(return_value=ssh_pool))
|
||||||
|
ssh_pool.item = mock.Mock(return_value=ssh)
|
||||||
|
self.mock_object(self._driver, '_gpfs_ssh_execute')
|
||||||
|
self.assertRaises(exception.GPFSException,
|
||||||
|
self._driver._run_ssh,
|
||||||
|
self.local_ip, cmd_list)
|
||||||
|
|
||||||
|
def test__gpfs_ssh_execute(self):
|
||||||
|
cmd = 'fake cmd'
|
||||||
|
expected_out = 'cmd successful'
|
||||||
|
expected_err = 'cmd error'
|
||||||
|
ssh = mock.Mock()
|
||||||
|
stdin_stream = mock.Mock()
|
||||||
|
stdout_stream = mock.Mock()
|
||||||
|
stderr_stream = mock.Mock()
|
||||||
|
ssh.exec_command = mock.Mock(return_value=(stdin_stream,
|
||||||
|
stdout_stream,
|
||||||
|
stderr_stream))
|
||||||
|
stdout_stream.channel.recv_exit_status = mock.Mock(return_value=-1)
|
||||||
|
stdout_stream.read = mock.Mock(return_value=expected_out)
|
||||||
|
stderr_stream.read = mock.Mock(return_value=expected_err)
|
||||||
|
stdin_stream.close = mock.Mock()
|
||||||
|
actual_out, actual_err = self._driver._gpfs_ssh_execute(ssh, cmd)
|
||||||
|
|
||||||
|
self.assertEqual(actual_out, expected_out)
|
||||||
|
self.assertEqual(actual_err, expected_err)
|
||||||
|
|
||||||
|
def test__gpfs_ssh_execute_exception(self):
|
||||||
|
cmd = 'fake cmd'
|
||||||
|
ssh = mock.Mock()
|
||||||
|
stdin_stream = mock.Mock()
|
||||||
|
stdout_stream = mock.Mock()
|
||||||
|
stderr_stream = mock.Mock()
|
||||||
|
ssh.exec_command = mock.Mock(return_value=(stdin_stream,
|
||||||
|
stdout_stream,
|
||||||
|
stderr_stream))
|
||||||
|
stdout_stream.channel.recv_exit_status = mock.Mock(return_value=1)
|
||||||
|
stdout_stream.read = mock.Mock()
|
||||||
|
stderr_stream.read = mock.Mock()
|
||||||
|
stdin_stream.close = mock.Mock()
|
||||||
|
self.assertRaises(exception.ProcessExecutionError,
|
||||||
|
self._driver._gpfs_ssh_execute,
|
||||||
|
ssh, cmd)
|
||||||
|
|
||||||
|
def test_get_share_stats_refresh_false(self):
|
||||||
|
self._driver._stats = {'fake_key': 'fake_value'}
|
||||||
|
result = self._driver.get_share_stats(False)
|
||||||
|
self.assertEqual(self._driver._stats, result)
|
||||||
|
|
||||||
|
def test_get_share_stats_refresh_true(self):
|
||||||
|
self.mock_object(
|
||||||
|
self._driver, '_get_available_capacity',
|
||||||
|
mock.Mock(return_value=(11111.0, 12345.0)))
|
||||||
|
result = self._driver.get_share_stats(True)
|
||||||
|
expected_keys = [
|
||||||
|
'QoS_support', 'driver_version', 'share_backend_name',
|
||||||
|
'free_capacity_gb', 'total_capacity_gb',
|
||||||
|
'driver_handles_share_servers',
|
||||||
|
'reserved_percentage', 'vendor_name', 'storage_protocol',
|
||||||
|
]
|
||||||
|
for key in expected_keys:
|
||||||
|
self.assertIn(key, result)
|
||||||
|
self.assertEqual(False, result['driver_handles_share_servers'])
|
||||||
|
self.assertEqual('IBM', result['vendor_name'])
|
||||||
|
self._driver._get_available_capacity.assert_called_once_with(
|
||||||
|
self._driver.configuration.gpfs_mount_point_base)
|
||||||
|
|
||||||
|
def test_do_setup(self):
|
||||||
|
self.mock_object(self._driver, '_setup_helpers')
|
||||||
|
self._driver.do_setup(self._context)
|
||||||
|
self._driver._setup_helpers.assert_called_once_with()
|
||||||
|
|
||||||
|
def test_setup_helpers(self):
|
||||||
|
self._driver._helpers = {}
|
||||||
|
CONF.set_default('gpfs_share_helpers', ['KNFS=fakenfs'])
|
||||||
|
self.mock_object(gpfs.importutils, 'import_class',
|
||||||
|
mock.Mock(return_value=self._helper_fake))
|
||||||
|
self._driver._setup_helpers()
|
||||||
|
gpfs.importutils.import_class.assert_has_calls(
|
||||||
|
[mock.call('fakenfs')]
|
||||||
|
)
|
||||||
|
self.assertEqual(len(self._driver._helpers), 1)
|
||||||
|
|
||||||
|
@ddt.data(fake_share.fake_share(),
|
||||||
|
fake_share.fake_share(share_proto='NFSBOGUS'))
|
||||||
|
def test__get_helper_with_wrong_proto(self, share):
|
||||||
|
self.assertRaises(exception.InvalidShare,
|
||||||
|
self._driver._get_helper, share)
|
||||||
|
|
||||||
|
def test__local_path(self):
|
||||||
|
sharename = 'fakesharename'
|
||||||
|
self._driver.configuration.gpfs_mount_point_base =\
|
||||||
|
self.fakefspath
|
||||||
|
local_path = self._driver._local_path(sharename)
|
||||||
|
self.assertEqual(self.fakefspath + '/' + sharename,
|
||||||
|
local_path)
|
||||||
|
|
||||||
|
def test__get_share_path(self):
|
||||||
|
self._driver.configuration.gpfs_mount_point_base =\
|
||||||
|
self.fakefspath
|
||||||
|
share_path = self._driver._get_share_path(self.share)
|
||||||
|
self.assertEqual(self.fakefspath + '/' + self.share['name'],
|
||||||
|
share_path)
|
||||||
|
|
||||||
|
def test__get_snapshot_path(self):
|
||||||
|
self._driver.configuration.gpfs_mount_point_base =\
|
||||||
|
self.fakefspath
|
||||||
|
snapshot_path = self._driver._get_snapshot_path(self.snapshot)
|
||||||
|
self.assertEqual(self.fakefspath + '/' + self.snapshot['share_name'] +
|
||||||
|
'/.snapshots/' + self.snapshot['name'],
|
||||||
|
snapshot_path)
|
||||||
|
|
||||||
|
def test_check_for_setup_error_for_gpfs_state(self):
|
||||||
|
self.mock_object(self._driver, '_check_gpfs_state',
|
||||||
|
mock.Mock(return_value=False))
|
||||||
|
self.assertRaises(exception.GPFSException,
|
||||||
|
self._driver.check_for_setup_error)
|
||||||
|
|
||||||
|
def test_check_for_setup_error_for_export_ip(self):
|
||||||
|
self.mock_object(self._driver, '_check_gpfs_state',
|
||||||
|
mock.Mock(return_value=True))
|
||||||
|
|
||||||
|
self._driver.configuration.gpfs_share_export_ip = None
|
||||||
|
self.assertRaises(exception.InvalidParameterValue,
|
||||||
|
self._driver.check_for_setup_error)
|
||||||
|
|
||||||
|
def test_check_for_setup_error_for_gpfs_mount_point_base(self):
|
||||||
|
self.mock_object(self._driver, '_check_gpfs_state',
|
||||||
|
mock.Mock(return_value=True))
|
||||||
|
self._driver.configuration.gpfs_share_export_ip = self.local_ip
|
||||||
|
self._driver.configuration.gpfs_mount_point_base = 'test'
|
||||||
|
self.assertRaises(exception.GPFSException,
|
||||||
|
self._driver.check_for_setup_error)
|
||||||
|
|
||||||
|
def test_check_for_setup_error_for_directory_check(self):
|
||||||
|
self.mock_object(self._driver, '_check_gpfs_state',
|
||||||
|
mock.Mock(return_value=True))
|
||||||
|
self._driver.configuration.gpfs_share_export_ip = self.local_ip
|
||||||
|
self._driver.configuration.gpfs_mount_point_base = self.fakefspath
|
||||||
|
self.mock_object(self._driver, '_is_dir',
|
||||||
|
mock.Mock(return_value=False))
|
||||||
|
self.assertRaises(exception.GPFSException,
|
||||||
|
self._driver.check_for_setup_error)
|
||||||
|
|
||||||
|
def test_check_for_setup_error_for_gpfs_path_check(self):
|
||||||
|
self.mock_object(self._driver, '_check_gpfs_state',
|
||||||
|
mock.Mock(return_value=True))
|
||||||
|
self._driver.configuration.gpfs_share_export_ip = self.local_ip
|
||||||
|
self._driver.configuration.gpfs_mount_point_base = self.fakefspath
|
||||||
|
self.mock_object(self._driver, '_is_dir',
|
||||||
|
mock.Mock(return_value=True))
|
||||||
|
self.mock_object(self._driver, '_is_gpfs_path',
|
||||||
|
mock.Mock(return_value=False))
|
||||||
|
self.assertRaises(exception.GPFSException,
|
||||||
|
self._driver.check_for_setup_error)
|
||||||
|
|
||||||
|
def test_check_for_setup_error_for_nfs_server_type(self):
|
||||||
|
self.mock_object(self._driver, '_check_gpfs_state',
|
||||||
|
mock.Mock(return_value=True))
|
||||||
|
self._driver.configuration.gpfs_share_export_ip = self.local_ip
|
||||||
|
self._driver.configuration.gpfs_mount_point_base = self.fakefspath
|
||||||
|
self.mock_object(self._driver, '_is_dir',
|
||||||
|
mock.Mock(return_value=True))
|
||||||
|
self.mock_object(self._driver, '_is_gpfs_path',
|
||||||
|
mock.Mock(return_value=True))
|
||||||
|
self._driver.configuration.gpfs_nfs_server_type = 'test'
|
||||||
|
self.assertRaises(exception.InvalidParameterValue,
|
||||||
|
self._driver.check_for_setup_error)
|
||||||
|
|
||||||
|
def test_check_for_setup_error_for_nfs_server_list(self):
|
||||||
|
self.mock_object(self._driver, '_check_gpfs_state',
|
||||||
|
mock.Mock(return_value=True))
|
||||||
|
self._driver.configuration.gpfs_share_export_ip = self.local_ip
|
||||||
|
self._driver.configuration.gpfs_mount_point_base = self.fakefspath
|
||||||
|
self.mock_object(self._driver, '_is_dir',
|
||||||
|
mock.Mock(return_value=True))
|
||||||
|
self.mock_object(self._driver, '_is_gpfs_path',
|
||||||
|
mock.Mock(return_value=True))
|
||||||
|
self._driver.configuration.gpfs_nfs_server_type = 'KNFS'
|
||||||
|
self._driver.configuration.gpfs_nfs_server_list = None
|
||||||
|
self.assertRaises(exception.InvalidParameterValue,
|
||||||
|
self._driver.check_for_setup_error)
|
||||||
|
|
||||||
|
def test__get_available_capacity(self):
|
||||||
|
path = self.fakefspath
|
||||||
|
mock_out = "Filesystem 1-blocks Used Available Capacity Mounted on\n\
|
||||||
|
/dev/gpfs0 100 30 70 30% /gpfs0"
|
||||||
|
self.mock_object(self._driver, '_gpfs_execute',
|
||||||
|
mock.Mock(return_value=(mock_out, '')))
|
||||||
|
available, size = self._driver._get_available_capacity(path)
|
||||||
|
self.assertEqual(70, available)
|
||||||
|
self.assertEqual(100, size)
|
||||||
|
|
||||||
|
def test_create_share(self):
|
||||||
|
self._helper_fake.create_export.return_value = 'fakelocation'
|
||||||
|
methods = ('_create_share', '_get_share_path')
|
||||||
|
for method in methods:
|
||||||
|
self.mock_object(self._driver, method)
|
||||||
|
result = self._driver.create_share(self._context, self.share,
|
||||||
|
share_server=self.server)
|
||||||
|
self._driver._create_share.assert_called_once_with(self.share)
|
||||||
|
self._driver._get_share_path.assert_called_once_with(self.share)
|
||||||
|
|
||||||
|
self.assertEqual(result, 'fakelocation')
|
||||||
|
|
||||||
|
def test_create_share_from_snapshot(self):
|
||||||
|
self._helper_fake.create_export.return_value = 'fakelocation'
|
||||||
|
self._driver._get_share_path = mock.Mock(return_value=self.
|
||||||
|
fakesharepath)
|
||||||
|
self._driver._create_share_from_snapshot = mock.Mock()
|
||||||
|
result = self._driver.create_share_from_snapshot(self._context,
|
||||||
|
self.share,
|
||||||
|
self.snapshot,
|
||||||
|
share_server=None)
|
||||||
|
self._driver._get_share_path.assert_called_once_with(self.share)
|
||||||
|
self._driver._create_share_from_snapshot.assert_called_once_with(
|
||||||
|
self.share, self.snapshot,
|
||||||
|
self.fakesharepath
|
||||||
|
)
|
||||||
|
self.assertEqual(result, 'fakelocation')
|
||||||
|
|
||||||
|
def test_create_snapshot(self):
|
||||||
|
self._driver._create_share_snapshot = mock.Mock()
|
||||||
|
self._driver.create_snapshot(self._context, self.snapshot,
|
||||||
|
share_server=None)
|
||||||
|
self._driver._create_share_snapshot.assert_called_once_with(
|
||||||
|
self.snapshot
|
||||||
|
)
|
||||||
|
|
||||||
|
def test_delete_share(self):
|
||||||
|
self._driver._get_share_path = mock.Mock(
|
||||||
|
return_value=self.fakesharepath
|
||||||
|
)
|
||||||
|
self._driver._delete_share = mock.Mock()
|
||||||
|
|
||||||
|
self._driver.delete_share(self._context, self.share,
|
||||||
|
share_server=None)
|
||||||
|
|
||||||
|
self._driver._get_share_path.assert_called_once_with(self.share)
|
||||||
|
self._driver._delete_share.assert_called_once_with(self.share)
|
||||||
|
self._helper_fake.remove_export.assert_called_once_with(
|
||||||
|
self.fakesharepath, self.share
|
||||||
|
)
|
||||||
|
|
||||||
|
def test_delete_snapshot(self):
|
||||||
|
self._driver._delete_share_snapshot = mock.Mock()
|
||||||
|
self._driver.delete_snapshot(self._context, self.snapshot,
|
||||||
|
share_server=None)
|
||||||
|
self._driver._delete_share_snapshot.assert_called_once_with(
|
||||||
|
self.snapshot
|
||||||
|
)
|
||||||
|
|
||||||
|
def test__delete_share_snapshot(self):
|
||||||
|
self._driver._get_gpfs_device = mock.Mock(return_value=self.fakedev)
|
||||||
|
self._driver._gpfs_execute = mock.Mock(return_value=0)
|
||||||
|
self._driver._delete_share_snapshot(self.snapshot)
|
||||||
|
self._driver._gpfs_execute.assert_called_once_with(
|
||||||
|
'mmdelsnapshot', self.fakedev, self.snapshot['name'],
|
||||||
|
'-j', self.snapshot['share_name']
|
||||||
|
)
|
||||||
|
self._driver._get_gpfs_device.assert_called_once_with()
|
||||||
|
|
||||||
|
def test__delete_share_snapshot_exception(self):
|
||||||
|
self._driver._get_gpfs_device = mock.Mock(return_value=self.fakedev)
|
||||||
|
self._driver._gpfs_execute = mock.Mock(
|
||||||
|
side_effect=exception.ProcessExecutionError
|
||||||
|
)
|
||||||
|
self.assertRaises(exception.GPFSException,
|
||||||
|
self._driver._delete_share_snapshot, self.snapshot)
|
||||||
|
self._driver._get_gpfs_device.assert_called_once_with()
|
||||||
|
self._driver._gpfs_execute.assert_called_once_with(
|
||||||
|
'mmdelsnapshot', self.fakedev, self.snapshot['name'],
|
||||||
|
'-j', self.snapshot['share_name']
|
||||||
|
)
|
||||||
|
|
||||||
|
def test_extend_share(self):
|
||||||
|
self._driver._extend_share = mock.Mock()
|
||||||
|
self._driver.extend_share(self.share, 10)
|
||||||
|
self._driver._extend_share.assert_called_once_with(self.share, 10)
|
||||||
|
|
||||||
|
def test__extend_share(self):
|
||||||
|
self._driver._get_gpfs_device = mock.Mock(return_value=self.fakedev)
|
||||||
|
self._driver._gpfs_execute = mock.Mock(return_value=True)
|
||||||
|
self._driver._extend_share(self.share, 10)
|
||||||
|
self._driver._gpfs_execute.assert_called_once_with('mmsetquota', '-j',
|
||||||
|
self.share['name'],
|
||||||
|
'-h', '10G',
|
||||||
|
self.fakedev)
|
||||||
|
self._driver._get_gpfs_device.assert_called_once_with()
|
||||||
|
|
||||||
|
def test__extend_share_exception(self):
|
||||||
|
self._driver._get_gpfs_device = mock.Mock(return_value=self.fakedev)
|
||||||
|
self._driver._gpfs_execute = mock.Mock(
|
||||||
|
side_effect=exception.ProcessExecutionError
|
||||||
|
)
|
||||||
|
self.assertRaises(exception.GPFSException,
|
||||||
|
self._driver._extend_share, self.share, 10)
|
||||||
|
self._driver._gpfs_execute.assert_called_once_with('mmsetquota', '-j',
|
||||||
|
self.share['name'],
|
||||||
|
'-h', '10G',
|
||||||
|
self.fakedev)
|
||||||
|
self._driver._get_gpfs_device.assert_called_once_with()
|
||||||
|
|
||||||
|
def test_allow_access(self):
|
||||||
|
self._driver._get_share_path = mock.Mock(
|
||||||
|
return_value=self.fakesharepath
|
||||||
|
)
|
||||||
|
self._helper_fake.allow_access = mock.Mock()
|
||||||
|
self._driver.allow_access(self._context, self.share,
|
||||||
|
self.access, share_server=None)
|
||||||
|
self._helper_fake.allow_access.assert_called_once_with(
|
||||||
|
self.fakesharepath, self.share,
|
||||||
|
self.access['access_type'],
|
||||||
|
self.access['access_to']
|
||||||
|
)
|
||||||
|
self._driver._get_share_path.assert_called_once_with(self.share)
|
||||||
|
|
||||||
|
def test_deny_access(self):
|
||||||
|
self._driver._get_share_path = mock.Mock(return_value=self.
|
||||||
|
fakesharepath)
|
||||||
|
self._helper_fake.deny_access = mock.Mock()
|
||||||
|
self._driver.deny_access(self._context, self.share,
|
||||||
|
self.access, share_server=None)
|
||||||
|
self._helper_fake.deny_access.assert_called_once_with(
|
||||||
|
self.fakesharepath, self.share,
|
||||||
|
self.access['access_type'],
|
||||||
|
self.access['access_to']
|
||||||
|
)
|
||||||
|
self._driver._get_share_path.assert_called_once_with(self.share)
|
||||||
|
|
||||||
|
def test__check_gpfs_state_active(self):
|
||||||
|
fakeout = "mmgetstate::state:\nmmgetstate::active:"
|
||||||
|
self._driver._gpfs_execute = mock.Mock(return_value=(fakeout, ''))
|
||||||
|
result = self._driver._check_gpfs_state()
|
||||||
|
self._driver._gpfs_execute.assert_called_once_with('mmgetstate', '-Y')
|
||||||
|
self.assertEqual(result, True)
|
||||||
|
|
||||||
|
def test__check_gpfs_state_down(self):
|
||||||
|
fakeout = "mmgetstate::state:\nmmgetstate::down:"
|
||||||
|
self._driver._gpfs_execute = mock.Mock(return_value=(fakeout, ''))
|
||||||
|
result = self._driver._check_gpfs_state()
|
||||||
|
self._driver._gpfs_execute.assert_called_once_with('mmgetstate', '-Y')
|
||||||
|
self.assertEqual(result, False)
|
||||||
|
|
||||||
|
def test__check_gpfs_state_wrong_output_exception(self):
|
||||||
|
fakeout = "mmgetstate fake out"
|
||||||
|
self._driver._gpfs_execute = mock.Mock(return_value=(fakeout, ''))
|
||||||
|
self.assertRaises(exception.GPFSException,
|
||||||
|
self._driver._check_gpfs_state)
|
||||||
|
self._driver._gpfs_execute.assert_called_once_with('mmgetstate', '-Y')
|
||||||
|
|
||||||
|
def test__check_gpfs_state_exception(self):
|
||||||
|
self._driver._gpfs_execute = mock.Mock(
|
||||||
|
side_effect=exception.ProcessExecutionError
|
||||||
|
)
|
||||||
|
self.assertRaises(exception.GPFSException,
|
||||||
|
self._driver._check_gpfs_state)
|
||||||
|
self._driver._gpfs_execute.assert_called_once_with('mmgetstate', '-Y')
|
||||||
|
|
||||||
|
def test__is_dir_success(self):
|
||||||
|
fakeoutput = "directory"
|
||||||
|
self._driver._gpfs_execute = mock.Mock(return_value=(fakeoutput, ''))
|
||||||
|
result = self._driver._is_dir(self.fakefspath)
|
||||||
|
self._driver._gpfs_execute.assert_called_once_with(
|
||||||
|
'stat', '--format=%F', self.fakefspath, run_as_root=False
|
||||||
|
)
|
||||||
|
self.assertEqual(result, True)
|
||||||
|
|
||||||
|
def test__is_dir_failure(self):
|
||||||
|
fakeoutput = "regulalr file"
|
||||||
|
self._driver._gpfs_execute = mock.Mock(return_value=(fakeoutput, ''))
|
||||||
|
result = self._driver._is_dir(self.fakefspath)
|
||||||
|
self._driver._gpfs_execute.assert_called_once_with(
|
||||||
|
'stat', '--format=%F', self.fakefspath, run_as_root=False
|
||||||
|
)
|
||||||
|
self.assertEqual(result, False)
|
||||||
|
|
||||||
|
def test__is_dir_exception(self):
|
||||||
|
self._driver._gpfs_execute = mock.Mock(
|
||||||
|
side_effect=exception.ProcessExecutionError
|
||||||
|
)
|
||||||
|
self.assertRaises(exception.GPFSException,
|
||||||
|
self._driver._is_dir, self.fakefspath)
|
||||||
|
self._driver._gpfs_execute.assert_called_once_with(
|
||||||
|
'stat', '--format=%F', self.fakefspath, run_as_root=False
|
||||||
|
)
|
||||||
|
|
||||||
|
def test__is_gpfs_path_ok(self):
|
||||||
|
self._driver._gpfs_execute = mock.Mock(return_value=0)
|
||||||
|
result = self._driver._is_gpfs_path(self.fakefspath)
|
||||||
|
self._driver._gpfs_execute.assert_called_once_with('mmlsattr',
|
||||||
|
self.fakefspath)
|
||||||
|
self.assertEqual(result, True)
|
||||||
|
|
||||||
|
def test__is_gpfs_path_exception(self):
|
||||||
|
self._driver._gpfs_execute = mock.Mock(
|
||||||
|
side_effect=exception.ProcessExecutionError
|
||||||
|
)
|
||||||
|
self.assertRaises(exception.GPFSException,
|
||||||
|
self._driver._is_gpfs_path,
|
||||||
|
self.fakefspath)
|
||||||
|
self._driver._gpfs_execute.assert_called_once_with('mmlsattr',
|
||||||
|
self.fakefspath)
|
||||||
|
|
||||||
|
def test__get_gpfs_device(self):
|
||||||
|
fakeout = "Filesystem\n" + self.fakedev
|
||||||
|
orig_val = self._driver.configuration.gpfs_mount_point_base
|
||||||
|
self._driver.configuration.gpfs_mount_point_base = self.fakefspath
|
||||||
|
self._driver._gpfs_execute = mock.Mock(return_value=(fakeout, ''))
|
||||||
|
result = self._driver._get_gpfs_device()
|
||||||
|
self._driver._gpfs_execute.assert_called_once_with('df',
|
||||||
|
self.fakefspath)
|
||||||
|
self.assertEqual(result, self.fakedev)
|
||||||
|
self._driver.configuration.gpfs_mount_point_base = orig_val
|
||||||
|
|
||||||
|
def test__get_gpfs_device_exception(self):
|
||||||
|
self._driver._gpfs_execute = mock.Mock(
|
||||||
|
side_effect=exception.ProcessExecutionError)
|
||||||
|
self.assertRaises(exception.GPFSException,
|
||||||
|
self._driver._get_gpfs_device)
|
||||||
|
|
||||||
|
def test__create_share(self):
|
||||||
|
sizestr = '%sG' % self.share['size']
|
||||||
|
self._driver._gpfs_execute = mock.Mock(return_value=True)
|
||||||
|
self._driver._local_path = mock.Mock(return_value=self.fakesharepath)
|
||||||
|
self._driver._get_gpfs_device = mock.Mock(return_value=self.fakedev)
|
||||||
|
self._driver._create_share(self.share)
|
||||||
|
self._driver._gpfs_execute.assert_any_call('mmcrfileset',
|
||||||
|
self.fakedev,
|
||||||
|
self.share['name'],
|
||||||
|
'--inode-space', 'new')
|
||||||
|
self._driver._gpfs_execute.assert_any_call('mmlinkfileset',
|
||||||
|
self.fakedev,
|
||||||
|
self.share['name'],
|
||||||
|
'-J', self.fakesharepath)
|
||||||
|
self._driver._gpfs_execute.assert_any_call('mmsetquota', '-j',
|
||||||
|
self.share['name'], '-h',
|
||||||
|
sizestr,
|
||||||
|
self.fakedev)
|
||||||
|
self._driver._gpfs_execute.assert_any_call('chmod',
|
||||||
|
'777',
|
||||||
|
self.fakesharepath)
|
||||||
|
|
||||||
|
self._driver._local_path.assert_called_once_with(self.share['name'])
|
||||||
|
self._driver._get_gpfs_device.assert_called_once_with()
|
||||||
|
|
||||||
|
def test__create_share_exception(self):
|
||||||
|
self._driver._local_path = mock.Mock(return_value=self.fakesharepath)
|
||||||
|
self._driver._get_gpfs_device = mock.Mock(return_value=self.fakedev)
|
||||||
|
self._driver._gpfs_execute = mock.Mock(
|
||||||
|
side_effect=exception.ProcessExecutionError
|
||||||
|
)
|
||||||
|
self.assertRaises(exception.GPFSException,
|
||||||
|
self._driver._create_share, self.share)
|
||||||
|
self._driver._get_gpfs_device.assert_called_once_with()
|
||||||
|
self._driver._local_path.assert_called_once_with(self.share['name'])
|
||||||
|
self._driver._gpfs_execute.assert_called_once_with('mmcrfileset',
|
||||||
|
self.fakedev,
|
||||||
|
self.share['name'],
|
||||||
|
'--inode-space',
|
||||||
|
'new')
|
||||||
|
|
||||||
|
def test__delete_share(self):
|
||||||
|
self._driver._gpfs_execute = mock.Mock(return_value=True)
|
||||||
|
self._driver._get_gpfs_device = mock.Mock(return_value=self.fakedev)
|
||||||
|
self._driver._delete_share(self.share)
|
||||||
|
self._driver._gpfs_execute.assert_any_call(
|
||||||
|
'mmunlinkfileset', self.fakedev, self.share['name'],
|
||||||
|
'-f')
|
||||||
|
self._driver._gpfs_execute.assert_any_call(
|
||||||
|
'mmdelfileset', self.fakedev, self.share['name'],
|
||||||
|
'-f')
|
||||||
|
self._driver._get_gpfs_device.assert_called_once_with()
|
||||||
|
|
||||||
|
def test__delete_share_exception(self):
|
||||||
|
self._driver._get_gpfs_device = mock.Mock(return_value=self.fakedev)
|
||||||
|
self._driver._gpfs_execute = mock.Mock(
|
||||||
|
side_effect=exception.ProcessExecutionError
|
||||||
|
)
|
||||||
|
self.assertRaises(exception.GPFSException,
|
||||||
|
self._driver._delete_share, self.share)
|
||||||
|
self._driver._get_gpfs_device.assert_called_once_with()
|
||||||
|
self._driver._gpfs_execute.assert_called_once_with(
|
||||||
|
'mmunlinkfileset', self.fakedev, self.share['name'],
|
||||||
|
'-f')
|
||||||
|
|
||||||
|
def test__create_share_snapshot(self):
|
||||||
|
self._driver._gpfs_execute = mock.Mock(return_value=True)
|
||||||
|
self._driver._get_gpfs_device = mock.Mock(return_value=self.fakedev)
|
||||||
|
self._driver._create_share_snapshot(self.snapshot)
|
||||||
|
self._driver._gpfs_execute.assert_called_once_with(
|
||||||
|
'mmcrsnapshot', self.fakedev, self.snapshot['name'],
|
||||||
|
'-j', self.snapshot['share_name']
|
||||||
|
)
|
||||||
|
self._driver._get_gpfs_device.assert_called_once_with()
|
||||||
|
|
||||||
|
def test__create_share_snapshot_exception(self):
|
||||||
|
self._driver._get_gpfs_device = mock.Mock(return_value=self.fakedev)
|
||||||
|
self._driver._gpfs_execute = mock.Mock(
|
||||||
|
side_effect=exception.ProcessExecutionError
|
||||||
|
)
|
||||||
|
self.assertRaises(exception.GPFSException,
|
||||||
|
self._driver._create_share_snapshot, self.snapshot)
|
||||||
|
self._driver._get_gpfs_device.assert_called_once_with()
|
||||||
|
self._driver._gpfs_execute.assert_called_once_with(
|
||||||
|
'mmcrsnapshot', self.fakedev, self.snapshot['name'],
|
||||||
|
'-j', self.snapshot['share_name']
|
||||||
|
)
|
||||||
|
|
||||||
|
def test__create_share_from_snapshot(self):
|
||||||
|
self._driver._gpfs_execute = mock.Mock(return_value=True)
|
||||||
|
self._driver._create_share = mock.Mock(return_value=True)
|
||||||
|
self._driver._get_snapshot_path = mock.Mock(return_value=self.
|
||||||
|
fakesnapshotpath)
|
||||||
|
self._driver._create_share_from_snapshot(self.share, self.snapshot,
|
||||||
|
self.fakesharepath)
|
||||||
|
self._driver._gpfs_execute.assert_called_once_with(
|
||||||
|
'rsync', '-rp', self.fakesnapshotpath + '/', self.fakesharepath
|
||||||
|
)
|
||||||
|
self._driver._create_share.assert_called_once_with(self.share)
|
||||||
|
self._driver._get_snapshot_path.assert_called_once_with(self.snapshot)
|
||||||
|
|
||||||
|
def test__create_share_from_snapshot_exception(self):
|
||||||
|
self._driver._create_share = mock.Mock(return_value=True)
|
||||||
|
self._driver._get_snapshot_path = mock.Mock(return_value=self.
|
||||||
|
fakesnapshotpath)
|
||||||
|
self._driver._gpfs_execute = mock.Mock(
|
||||||
|
side_effect=exception.ProcessExecutionError
|
||||||
|
)
|
||||||
|
self.assertRaises(exception.GPFSException,
|
||||||
|
self._driver._create_share_from_snapshot,
|
||||||
|
self.share, self.snapshot, self.fakesharepath)
|
||||||
|
self._driver._create_share.assert_called_once_with(self.share)
|
||||||
|
self._driver._get_snapshot_path.assert_called_once_with(self.snapshot)
|
||||||
|
self._driver._gpfs_execute.assert_called_once_with(
|
||||||
|
'rsync', '-rp', self.fakesnapshotpath + '/', self.fakesharepath
|
||||||
|
)
|
||||||
|
|
||||||
|
def test__gpfs_local_execute(self):
|
||||||
|
self.mock_object(utils, 'execute', mock.Mock(return_value=True))
|
||||||
|
cmd = "testcmd"
|
||||||
|
self._driver._gpfs_local_execute(cmd)
|
||||||
|
utils.execute.assert_called_once_with(cmd, run_as_root=True)
|
||||||
|
|
||||||
|
def test__gpfs_remote_execute(self):
|
||||||
|
self._driver._run_ssh = mock.Mock(return_value=True)
|
||||||
|
cmd = "testcmd"
|
||||||
|
orig_value = self._driver.configuration.gpfs_share_export_ip
|
||||||
|
self._driver.configuration.gpfs_share_export_ip = self.local_ip
|
||||||
|
self._driver._gpfs_remote_execute(cmd, check_exit_code=True)
|
||||||
|
self._driver._run_ssh.assert_called_once_with(
|
||||||
|
self.local_ip, tuple([cmd]), True
|
||||||
|
)
|
||||||
|
self._driver.configuration.gpfs_share_export_ip = orig_value
|
||||||
|
|
||||||
|
def test_knfs_allow_access(self):
|
||||||
|
self._knfs_helper._execute = mock.Mock(
|
||||||
|
return_value=['/fs0 <world>', 0]
|
||||||
|
)
|
||||||
|
self.mock_object(re, 'search', mock.Mock(return_value=None))
|
||||||
|
export_opts = None
|
||||||
|
self._knfs_helper._get_export_options = mock.Mock(
|
||||||
|
return_value=export_opts
|
||||||
|
)
|
||||||
|
self._knfs_helper._publish_access = mock.Mock()
|
||||||
|
access_type = self.access['access_type']
|
||||||
|
access = self.access['access_to']
|
||||||
|
local_path = self.fakesharepath
|
||||||
|
self._knfs_helper.allow_access(local_path, self.share,
|
||||||
|
access_type, access)
|
||||||
|
self._knfs_helper._execute.assert_called_once_with('exportfs',
|
||||||
|
run_as_root=True)
|
||||||
|
self.assertTrue(re.search.called)
|
||||||
|
self._knfs_helper._get_export_options.assert_any_call(self.share)
|
||||||
|
cmd = ['exportfs', '-o', export_opts, ':'.join([access, local_path])]
|
||||||
|
self._knfs_helper._publish_access.assert_called_once_with(*cmd)
|
||||||
|
|
||||||
|
def test_knfs_allow_access_access_exists(self):
|
||||||
|
out = ['/fs0 <world>', 0]
|
||||||
|
self._knfs_helper._execute = mock.Mock(return_value=out)
|
||||||
|
self.mock_object(re, 'search', mock.Mock(return_value="fake"))
|
||||||
|
self._knfs_helper._get_export_options = mock.Mock()
|
||||||
|
access_type = self.access['access_type']
|
||||||
|
access = self.access['access_to']
|
||||||
|
local_path = self.fakesharepath
|
||||||
|
self.assertRaises(exception.ShareAccessExists,
|
||||||
|
self._knfs_helper.allow_access,
|
||||||
|
local_path, self.share,
|
||||||
|
access_type, access)
|
||||||
|
self._knfs_helper._execute.assert_any_call('exportfs',
|
||||||
|
run_as_root=True)
|
||||||
|
self.assertTrue(re.search.called)
|
||||||
|
self.assertFalse(self._knfs_helper._get_export_options.called)
|
||||||
|
|
||||||
|
def test_knfs_allow_access_invalid_access(self):
|
||||||
|
access_type = 'invalid_access_type'
|
||||||
|
self.assertRaises(exception.InvalidShareAccess,
|
||||||
|
self._knfs_helper.allow_access,
|
||||||
|
self.fakesharepath, self.share,
|
||||||
|
access_type,
|
||||||
|
self.access['access_to'])
|
||||||
|
|
||||||
|
def test_knfs_allow_access_exception(self):
|
||||||
|
self._knfs_helper._execute = mock.Mock(
|
||||||
|
side_effect=exception.ProcessExecutionError
|
||||||
|
)
|
||||||
|
access_type = self.access['access_type']
|
||||||
|
access = self.access['access_to']
|
||||||
|
local_path = self.fakesharepath
|
||||||
|
self.assertRaises(exception.GPFSException,
|
||||||
|
self._knfs_helper.allow_access,
|
||||||
|
local_path, self.share,
|
||||||
|
access_type, access)
|
||||||
|
self._knfs_helper._execute.assert_called_once_with('exportfs',
|
||||||
|
run_as_root=True)
|
||||||
|
|
||||||
|
def test_knfs_deny_access(self):
|
||||||
|
self._knfs_helper._publish_access = mock.Mock()
|
||||||
|
access = self.access['access_to']
|
||||||
|
access_type = self.access['access_type']
|
||||||
|
local_path = self.fakesharepath
|
||||||
|
self._knfs_helper.deny_access(local_path, self.share,
|
||||||
|
access_type, access)
|
||||||
|
cmd = ['exportfs', '-u', ':'.join([access, local_path])]
|
||||||
|
self._knfs_helper._publish_access.assert_called_once_with(*cmd)
|
||||||
|
|
||||||
|
def test_knfs_deny_access_exception(self):
|
||||||
|
self._knfs_helper._publish_access = mock.Mock(
|
||||||
|
side_effect=exception.ProcessExecutionError
|
||||||
|
)
|
||||||
|
access = self.access['access_to']
|
||||||
|
access_type = self.access['access_type']
|
||||||
|
local_path = self.fakesharepath
|
||||||
|
cmd = ['exportfs', '-u', ':'.join([access, local_path])]
|
||||||
|
self.assertRaises(exception.GPFSException,
|
||||||
|
self._knfs_helper.deny_access, local_path,
|
||||||
|
self.share, access_type, access)
|
||||||
|
self._knfs_helper._publish_access.assert_called_once_with(*cmd)
|
||||||
|
|
||||||
|
def test_knfs__publish_access(self):
|
||||||
|
self.mock_object(utils, 'execute')
|
||||||
|
cmd = ['fakecmd']
|
||||||
|
self._knfs_helper._publish_access(*cmd)
|
||||||
|
utils.execute.assert_any_call(*cmd, run_as_root=True,
|
||||||
|
check_exit_code=True)
|
||||||
|
remote_login = self.sshlogin + '@' + self.remote_ip
|
||||||
|
cmd = ['ssh', remote_login] + list(cmd)
|
||||||
|
utils.execute.assert_any_call(*cmd, run_as_root=False,
|
||||||
|
check_exit_code=True)
|
||||||
|
self.assertTrue(socket.gethostbyname_ex.called)
|
||||||
|
self.assertTrue(socket.gethostname.called)
|
||||||
|
|
||||||
|
def test_knfs__publish_access_exception(self):
|
||||||
|
self.mock_object(
|
||||||
|
utils, 'execute',
|
||||||
|
mock.Mock(side_effect=exception.ProcessExecutionError))
|
||||||
|
cmd = ['fakecmd']
|
||||||
|
self.assertRaises(exception.ProcessExecutionError,
|
||||||
|
self._knfs_helper._publish_access, *cmd)
|
||||||
|
self.assertTrue(socket.gethostbyname_ex.called)
|
||||||
|
self.assertTrue(socket.gethostname.called)
|
||||||
|
utils.execute.assert_called_once_with(*cmd, run_as_root=True,
|
||||||
|
check_exit_code=True)
|
||||||
|
|
||||||
|
def test_gnfs_allow_access(self):
|
||||||
|
self._gnfs_helper._ganesha_process_request = mock.Mock()
|
||||||
|
access = self.access['access_to']
|
||||||
|
access_type = self.access['access_type']
|
||||||
|
local_path = self.fakesharepath
|
||||||
|
self._gnfs_helper.allow_access(local_path, self.share,
|
||||||
|
access_type, access)
|
||||||
|
self._gnfs_helper._ganesha_process_request.assert_called_once_with(
|
||||||
|
"allow_access", local_path, self.share, access_type, access
|
||||||
|
)
|
||||||
|
|
||||||
|
def test_gnfs_allow_access_invalid_access(self):
|
||||||
|
access_type = 'invalid_access_type'
|
||||||
|
self.assertRaises(exception.InvalidShareAccess,
|
||||||
|
self._gnfs_helper.allow_access,
|
||||||
|
self.fakesharepath, self.share,
|
||||||
|
access_type,
|
||||||
|
self.access['access_to'])
|
||||||
|
|
||||||
|
def test_gnfs_deny_access(self):
|
||||||
|
self._gnfs_helper._ganesha_process_request = mock.Mock()
|
||||||
|
access = self.access['access_to']
|
||||||
|
access_type = self.access['access_type']
|
||||||
|
local_path = self.fakesharepath
|
||||||
|
self._gnfs_helper.deny_access(local_path, self.share,
|
||||||
|
access_type, access)
|
||||||
|
self._gnfs_helper._ganesha_process_request.assert_called_once_with(
|
||||||
|
"deny_access", local_path, self.share, access_type, access, False
|
||||||
|
)
|
||||||
|
|
||||||
|
def test_gnfs_remove_export(self):
|
||||||
|
self._gnfs_helper._ganesha_process_request = mock.Mock()
|
||||||
|
local_path = self.fakesharepath
|
||||||
|
self._gnfs_helper.remove_export(local_path, self.share)
|
||||||
|
self._gnfs_helper._ganesha_process_request.assert_called_once_with(
|
||||||
|
"remove_export", local_path, self.share
|
||||||
|
)
|
||||||
|
|
||||||
|
def test_gnfs__ganesha_process_request_allow_access(self):
|
||||||
|
access = self.access['access_to']
|
||||||
|
access_type = self.access['access_type']
|
||||||
|
local_path = self.fakesharepath
|
||||||
|
cfgpath = self._gnfs_helper.configuration.ganesha_config_path
|
||||||
|
gservers = self._gnfs_helper.configuration.gpfs_nfs_server_list
|
||||||
|
export_opts = []
|
||||||
|
pre_lines = []
|
||||||
|
exports = {}
|
||||||
|
self._gnfs_helper._get_export_options = mock.Mock(
|
||||||
|
return_value=export_opts
|
||||||
|
)
|
||||||
|
self.mock_object(ganesha_utils, 'parse_ganesha_config', mock.Mock(
|
||||||
|
return_value=(pre_lines, exports)
|
||||||
|
))
|
||||||
|
self.mock_object(ganesha_utils, 'export_exists', mock.Mock(
|
||||||
|
return_value=False
|
||||||
|
))
|
||||||
|
self.mock_object(ganesha_utils, 'get_next_id', mock.Mock(
|
||||||
|
return_value=101
|
||||||
|
))
|
||||||
|
self.mock_object(ganesha_utils, 'get_export_template', mock.Mock(
|
||||||
|
return_value={}
|
||||||
|
))
|
||||||
|
self.mock_object(ganesha_utils, 'publish_ganesha_config')
|
||||||
|
self.mock_object(ganesha_utils, 'reload_ganesha_config')
|
||||||
|
self._gnfs_helper._ganesha_process_request(
|
||||||
|
"allow_access", local_path, self.share, access_type, access
|
||||||
|
)
|
||||||
|
self._gnfs_helper._get_export_options.assert_called_once_with(
|
||||||
|
self.share
|
||||||
|
)
|
||||||
|
ganesha_utils.export_exists.assert_called_once_with(exports,
|
||||||
|
local_path)
|
||||||
|
ganesha_utils.parse_ganesha_config.assert_called_once_with(cfgpath)
|
||||||
|
ganesha_utils.publish_ganesha_config.assert_called_once_with(
|
||||||
|
gservers, self.sshlogin, self.sshkey, cfgpath, pre_lines, exports
|
||||||
|
)
|
||||||
|
ganesha_utils.reload_ganesha_config.assert_called_once_with(
|
||||||
|
gservers, self.sshlogin, self.gservice
|
||||||
|
)
|
||||||
|
|
||||||
|
def test_gnfs__ganesha_process_request_deny_access(self):
|
||||||
|
access = self.access['access_to']
|
||||||
|
access_type = self.access['access_type']
|
||||||
|
local_path = self.fakesharepath
|
||||||
|
cfgpath = self._gnfs_helper.configuration.ganesha_config_path
|
||||||
|
gservers = self._gnfs_helper.configuration.gpfs_nfs_server_list
|
||||||
|
pre_lines = []
|
||||||
|
initial_access = "10.0.0.1,10.0.0.2"
|
||||||
|
export = {"rw_access": initial_access}
|
||||||
|
exports = {}
|
||||||
|
self.mock_object(ganesha_utils, 'parse_ganesha_config', mock.Mock(
|
||||||
|
return_value=(pre_lines, exports)
|
||||||
|
))
|
||||||
|
self.mock_object(ganesha_utils, 'get_export_by_path', mock.Mock(
|
||||||
|
return_value=export
|
||||||
|
))
|
||||||
|
self.mock_object(ganesha_utils, 'format_access_list', mock.Mock(
|
||||||
|
return_value="10.0.0.1"
|
||||||
|
))
|
||||||
|
self.mock_object(ganesha_utils, 'publish_ganesha_config')
|
||||||
|
self.mock_object(ganesha_utils, 'reload_ganesha_config')
|
||||||
|
self._gnfs_helper._ganesha_process_request(
|
||||||
|
"deny_access", local_path, self.share, access_type, access
|
||||||
|
)
|
||||||
|
ganesha_utils.parse_ganesha_config.assert_called_once_with(cfgpath)
|
||||||
|
ganesha_utils.get_export_by_path.assert_called_once_with(exports,
|
||||||
|
local_path)
|
||||||
|
ganesha_utils.format_access_list.assert_called_once_with(
|
||||||
|
initial_access, deny_access=access
|
||||||
|
)
|
||||||
|
ganesha_utils.publish_ganesha_config.assert_called_once_with(
|
||||||
|
gservers, self.sshlogin, self.sshkey, cfgpath, pre_lines, exports
|
||||||
|
)
|
||||||
|
ganesha_utils.reload_ganesha_config.assert_called_once_with(
|
||||||
|
gservers, self.sshlogin, self.gservice
|
||||||
|
)
|
||||||
|
|
||||||
|
def test_gnfs__ganesha_process_request_remove_export(self):
|
||||||
|
local_path = self.fakesharepath
|
||||||
|
cfgpath = self._gnfs_helper.configuration.ganesha_config_path
|
||||||
|
pre_lines = []
|
||||||
|
exports = {}
|
||||||
|
export = {}
|
||||||
|
self.mock_object(ganesha_utils, 'parse_ganesha_config', mock.Mock(
|
||||||
|
return_value=(pre_lines, exports)
|
||||||
|
))
|
||||||
|
self.mock_object(ganesha_utils, 'get_export_by_path', mock.Mock(
|
||||||
|
return_value=export
|
||||||
|
))
|
||||||
|
self.mock_object(ganesha_utils, 'publish_ganesha_config')
|
||||||
|
self.mock_object(ganesha_utils, 'reload_ganesha_config')
|
||||||
|
self._gnfs_helper._ganesha_process_request(
|
||||||
|
"remove_export", local_path, self.share
|
||||||
|
)
|
||||||
|
ganesha_utils.parse_ganesha_config.assert_called_once_with(cfgpath)
|
||||||
|
ganesha_utils.get_export_by_path.assert_called_once_with(exports,
|
||||||
|
local_path)
|
||||||
|
self.assertFalse(ganesha_utils.publish_ganesha_config.called)
|
||||||
|
self.assertFalse(ganesha_utils.reload_ganesha_config.called)
|
Loading…
Reference in New Issue
Block a user