Mini Shell
import time
import re
from netmiko.cisco_base_connection import CiscoBaseConnection
from netmiko.ssh_exception import NetmikoAuthenticationException
from netmiko import log
class HuaweiBase(CiscoBaseConnection):
def session_preparation(self):
"""Prepare the session after the connection has been established."""
self.ansi_escape_codes = True
self._test_channel_read()
self.set_base_prompt()
self.disable_paging(command="screen-length 0 temporary")
# Clear the read buffer
time.sleep(0.3 * self.global_delay_factor)
self.clear_buffer()
def strip_ansi_escape_codes(self, string_buffer):
"""
Huawei does a strange thing where they add a space and then add ESC[1D
to move the cursor to the left one.
The extra space is problematic.
"""
code_cursor_left = chr(27) + r"\[\d+D"
output = string_buffer
pattern = rf" {code_cursor_left}"
output = re.sub(pattern, "", output)
return super().strip_ansi_escape_codes(output)
def config_mode(self, config_command="system-view"):
"""Enter configuration mode."""
return super().config_mode(config_command=config_command)
def exit_config_mode(self, exit_config="return", pattern=r">"):
"""Exit configuration mode."""
return super().exit_config_mode(exit_config=exit_config, pattern=pattern)
def check_config_mode(self, check_string="]"):
"""Checks whether in configuration mode. Returns a boolean."""
return super().check_config_mode(check_string=check_string)
def check_enable_mode(self, *args, **kwargs):
"""Huawei has no enable mode."""
pass
def enable(self, *args, **kwargs):
"""Huawei has no enable mode."""
return ""
def exit_enable_mode(self, *args, **kwargs):
"""Huawei has no enable mode."""
return ""
def set_base_prompt(
self, pri_prompt_terminator=">", alt_prompt_terminator="]", delay_factor=1
):
"""
Sets self.base_prompt
Used as delimiter for stripping of trailing prompt in output.
Should be set to something that is general and applies in multiple contexts.
For Huawei this will be the router prompt with < > or [ ] stripped off.
This will be set on logging in, but not when entering system-view
"""
# log.debug("In set_base_prompt")
delay_factor = self.select_delay_factor(delay_factor)
self.clear_buffer()
self.write_channel(self.RETURN)
time.sleep(0.5 * delay_factor)
prompt = self.read_channel()
prompt = self.normalize_linefeeds(prompt)
# If multiple lines in the output take the last line
prompt = prompt.split(self.RESPONSE_RETURN)[-1]
prompt = prompt.strip()
# Check that ends with a valid terminator character
if not prompt[-1] in (pri_prompt_terminator, alt_prompt_terminator):
raise ValueError(f"Router prompt not found: {prompt}")
# Strip off any leading HRP_. characters for USGv5 HA
prompt = re.sub(r"^HRP_.", "", prompt, flags=re.M)
# Strip off leading and trailing terminator
prompt = prompt[1:-1]
prompt = prompt.strip()
self.base_prompt = prompt
log.debug(f"prompt: {self.base_prompt}")
return self.base_prompt
def save_config(self, cmd="save", confirm=True, confirm_response="y"):
""" Save Config for HuaweiSSH"""
return super().save_config(
cmd=cmd, confirm=confirm, confirm_response=confirm_response
)
class HuaweiSSH(HuaweiBase):
"""Huawei SSH driver."""
def special_login_handler(self):
"""Handle password change request by ignoring it"""
# Huawei can prompt for password change. Search for that or for normal prompt
password_change_prompt = r"((Change now|Please choose))|([\]>]\s*$)"
output = self.read_until_pattern(password_change_prompt)
if re.search(password_change_prompt, output):
self.write_channel("N\n")
self.clear_buffer()
return output
class HuaweiTelnet(HuaweiBase):
"""Huawei Telnet driver."""
def telnet_login(
self,
pri_prompt_terminator=r"]\s*$",
alt_prompt_terminator=r">\s*$",
username_pattern=r"(?:user:|username|login|user name)",
pwd_pattern=r"assword",
delay_factor=1,
max_loops=20,
):
"""Telnet login for Huawei Devices"""
delay_factor = self.select_delay_factor(delay_factor)
password_change_prompt = r"(Change now|Please choose 'YES' or 'NO').+"
combined_pattern = r"({}|{}|{})".format(
pri_prompt_terminator, alt_prompt_terminator, password_change_prompt
)
output = ""
return_msg = ""
i = 1
while i <= max_loops:
try:
# Search for username pattern / send username
output = self.read_until_pattern(
pattern=username_pattern, re_flags=re.I
)
return_msg += output
self.write_channel(self.username + self.TELNET_RETURN)
# Search for password pattern / send password
output = self.read_until_pattern(pattern=pwd_pattern, re_flags=re.I)
return_msg += output
self.write_channel(self.password + self.TELNET_RETURN)
# Waiting for combined output
output = self.read_until_pattern(pattern=combined_pattern)
return_msg += output
# Search for password change prompt, send "N"
if re.search(password_change_prompt, output):
self.write_channel("N" + self.TELNET_RETURN)
output = self.read_until_pattern(pattern=combined_pattern)
return_msg += output
# Check if proper data received
if re.search(pri_prompt_terminator, output, flags=re.M) or re.search(
alt_prompt_terminator, output, flags=re.M
):
return return_msg
self.write_channel(self.TELNET_RETURN)
time.sleep(0.5 * delay_factor)
i += 1
except EOFError:
self.remote_conn.close()
msg = f"Login failed: {self.host}"
raise NetmikoAuthenticationException(msg)
# Last try to see if we already logged in
self.write_channel(self.TELNET_RETURN)
time.sleep(0.5 * delay_factor)
output = self.read_channel()
return_msg += output
if re.search(pri_prompt_terminator, output, flags=re.M) or re.search(
alt_prompt_terminator, output, flags=re.M
):
return return_msg
self.remote_conn.close()
msg = f"Login failed: {self.host}"
raise NetmikoAuthenticationException(msg)
class HuaweiVrpv8SSH(HuaweiSSH):
def commit(self, comment="", delay_factor=1):
"""
Commit the candidate configuration.
Commit the entered configuration. Raise an error and return the failure
if the commit fails.
default:
command_string = commit
comment:
command_string = commit comment <comment>
"""
delay_factor = self.select_delay_factor(delay_factor)
error_marker = "Failed to generate committed config"
command_string = "commit"
if comment:
command_string += f' comment "{comment}"'
output = self.config_mode()
output += self.send_command_expect(
command_string,
strip_prompt=False,
strip_command=False,
delay_factor=delay_factor,
expect_string=r"]",
)
output += self.exit_config_mode()
if error_marker in output:
raise ValueError(f"Commit failed with following errors:\n\n{output}")
return output
def save_config(self, *args, **kwargs):
"""Not Implemented"""
raise NotImplementedError
Zerion Mini Shell 1.0