python 借助 paramiko 库执行 SSH命令报错:input is not a terminal 解决方式

问题背景

在一个项目中,需要进入环境后台执行一个SSH命令:container_exec -n asv-con -c "sfd_cluster_cmd.sh e 'touch /sf/etc/update_disable_adeploy_upg_precheck'" (类似进入容器中执行配置修改操作),如果人为进入环境后台执行,是有交互输出的,如下图,但是如果使用 paramiko 封装的SSH client 执行这个命令,就报错提示:execing command in container: input is not a terminal

错误原因

错误解析
  1. 错误内容:
    • execing command in container: 表示试图在一个容器中执行命令。
    • input is not a terminal: 表示输入不是一个终端。
  2. 原因:
    • 这个错误通常出现在试图通过一个非交互式的方式执行一个需要终端输入的命令时,比如某些命令可能需要用户在终端交互中提供输入(如密码、确认等)。
    • 当你使用诸如 docker exec 或其他容器管理工具(如 Kubernetes)来执行命令时,如果你没有在命令中启用终端(TTY),就会出现这个问题。

解决办法:强制分配一个伪终端

Python 的 paramiko 库不支持伪终端,但可以通过 get_pty 来实现这一点。可以在执行命令时请求一个伪终端。这样就能执行一些需要终端的命令。完整代码如下

python 复制代码
# -*- coding: utf-8 -*-
# @Desc    :
import threading
import paramiko


class SSH:
    def __init__(self, host, port, username, password):
        """
        初始化ConnectManager对象, 保存连接参数
        :param host: 主机名或IP
        :param port: 端口号
        :param username: 用户名
        :param password: 密码
        """
        self.host = host
        self.port = port
        self.username = username
        self.password = password
        self.ssh_client = None
        self.connect()

    def connect(self):
        """
        建立SSH连接
        """
        self.ssh_client = paramiko.SSHClient()
        self.ssh_client.set_missing_host_key_policy(paramiko.AutoAddPolicy())
        try:
            self.ssh_client.connect(self.host, port=self.port, username=self.username, password=self.password)
            print("连接成功{}:{}".format(self.host, self.port))
        except Exception as e:
            print("连接失败:{}".format(e))
            exit(1)

    def close_connection(self):
        """
        关闭SSH连接
        """
        if self.ssh_client:
            self.ssh_client.close()
            print("已关闭连接{}:{}".format(self.host, self.port))

    def execute_command(self, command):
        """
        用于执行命令
        :param command: 执行的命令
        :return: 命令的输出结果
        """
        if not self.ssh_client:
            # SSH连接未建立, 先调用connect建立连接
            self.connect()
    
        # 接收返回值: get_pty: 请求一个伪终端
        stdin, stdout, stderr = self.ssh_client.exec_command(command, get_pty=True)
        
        # 如果需要输入(例如,密码或确认),可以写入 stdin
        # stdin.write('your_input_here\n')
        # stdin.flush()  # 确保输入被发送
    
        if stdout.channel.recv_exit_status() == 0:
            # 返回命令执行结果
            return stdout.read().decode('utf-8').strip(), True
        else:
            # 返回错误信息
            return stderr.read().decode('utf-8').strip(), False
  

相关推荐
Once_day21 小时前
C++之《程序员自我修养》读书总结(1)
c语言·开发语言·c++·程序员自我修养
喜欢喝果茶.21 小时前
QOverload<参数列表>::of(&函数名)信号槽
开发语言·qt
亓才孓21 小时前
[Class类的应用]反射的理解
开发语言·python
努力学编程呀(๑•ี_เ•ี๑)21 小时前
【在 IntelliJ IDEA 中切换项目 JDK 版本】
java·开发语言·intellij-idea
小镇敲码人21 小时前
深入剖析华为CANN框架下的Ops-CV仓库:从入门到实战指南
c++·python·华为·cann
island13141 天前
CANN GE(图引擎)深度解析:计算图优化管线、内存静态规划与异构任务的 Stream 调度机制
开发语言·人工智能·深度学习·神经网络
坚持就完事了1 天前
Java中的集合
java·开发语言
摘星编程1 天前
深入理解CANN ops-nn BatchNormalization算子:训练加速的关键技术
python
魔芋红茶1 天前
Python 项目版本控制
开发语言·python
lili-felicity1 天前
CANN批处理优化技巧:从动态批处理到流水线并行
人工智能·python