Source code for freezer_dr.fencers.drivers.ipmi.ipmitool
# (c) Copyright 2014,2015 Hewlett-Packard Development Company, L.P.
#
# 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.
from distutils import spawn
from oslo_log import log
import subprocess
import sys
LOG = log.getLogger(__name__)
[docs]class IpmiInterface:
_IPMI = 'ipmitool'
_RAW_CMD = '{0} -I {1} -H {2} -U {3} -P {4} '
_SUPPORTED_INTERFACES = ['lan', 'lanplus']
def __init__(self, host, username, password, verbose=False,
interface='lanplus'):
self._IPMI = spawn.find_executable('ipmitool')
if not self._IPMI:
self._IPMI = spawn.find_executable('ipmitool',
path=':'.join(sys.path))
if interface not in self._SUPPORTED_INTERFACES:
raise Exception("Provided Interface is not supported")
self._host = host
self._username = username
self._password = password
self._verbose = verbose
self._interface = interface
self._update_cmd_credentials(
host=host,
username=username,
password=password,
interface=interface
)
LOG.debug('IPMI Interface initialized')
def _update_cmd_credentials(self, host, username, password, interface):
"""
Update credentials to work with different server
:param host: IPMI IP address of the server
:param username: IPMI username
:param password: IPMI password
:param interface: IPMI Interface lan, lanplus
"""
cmd = self._RAW_CMD.format(
self._IPMI,
interface,
host,
username,
password
)
self._cmd = cmd
[docs] def get_power_status(self):
"""
get the machine power status
:return: 1 if the power is on and 0 if the power is off. otherwise it
will return -1 for unknown state
"""
cmd = self._cmd + ' chassis power status'
output = self._process_request(cmd)
if self._verbose:
LOG.debug(output)
if 'is on'.lower() in output.lower():
return 1
elif 'is off'.lower() in output.lower():
return 0
return -1 # power status unknown
[docs] def power_down(self):
"""
Force shutdown the machine
"""
cmd = self._cmd + ' chassis power down'
output = self._process_request(cmd)
LOG.info('IPMI interface force shutdown node: %s, output: %s' %
(self._host, output))
return output
[docs] def power_soft(self):
"""
Softly shutdown the machine
"""
cmd = self._cmd + 'chassis power soft'
output = self._process_request(cmd)
LOG.info('IPMI interface soft shutdown node: %s, output: %s' %
(self._host, output))
return output
[docs] def power_reset(self):
"""
restart the machine
"""
cmd = self._cmd + ' chassis power reset'
return self._process_request(cmd)
[docs] def power_on(self):
"""
power on the machine
"""
cmd = self._cmd + ' chassis power on'
return self._process_request(cmd)
def _process_request(self, cmd):
if self._verbose:
LOG.debug('Executing IPMI command:', cmd)
process = subprocess.Popen(cmd, shell=True,
stdout=subprocess.PIPE,
stderr=subprocess.PIPE)
output, error = process.communicate()
if self._verbose:
LOG.debug('IPMI Output: ', output)
LOG.debug('IPMI Error', error)
if process.returncode:
LOG.error(cmd)
raise Exception(error)
return output
def _custom_cmd(self, cmd):
"""
execute custom ipmitool commands
:param cmd: string contains the command, for credentials and interface
you should _update_cmd_credentials to update them first
:return: output of the command you sent or raise error
"""
cmd = self._cmd + cmd
output = self._process_request(cmd)
LOG.info('Executing IPMI custom command: %s with output: %s' %
(cmd, output))
return output