侯体宗的博客
  • 首页
  • Hyperf版
  • beego仿版
  • 人生(杂谈)
  • 技术
  • 关于我
  • 更多分类
    • 文件下载
    • 文字修仙
    • 中国象棋ai
    • 群聊
    • 九宫格抽奖
    • 拼图
    • 消消乐
    • 相册

PyQt+socket实现远程操作服务器的方法示例

Python  /  管理员 发布于 5年前   320

来需求了。。干活啦。。

需求内容

部分时候由于缓存刷新、验证码显示不出来或者浏览器打不开或者打开速度很慢等原因,导致部分测试同事不想使用浏览器登录服务器执行命令。 期望有小工具可以替代登录浏览器的操作,直接发送指令到服务器执行并将执行结果返回。

需求设计

1、开发界面,方便用户输入IP、用户名、密码以及执行的命令。

2、IP、用户名、密码和命令输入提供默认值。特别是用户名和密码,对于测试服务器来说,通常都是固定的。

3、IP、命令行输入框可以自动补全用户输入。自动补全常用IP、命令行可以提高操作效率。

4、可以自动保存用户执行成功的IP、命令行。用于完善自动补全命令(本文代码未实现)。

需求设计

1、使用Qt Designer实现界面开发。开发后界面参考如下:

2、使用socket程序登录服务器并执行命令,并将结果显示在界面文本框中。

代码实现(程序可以直接复制运行)

1、使用Qt Designer实现界面开发。拖动4个label+4个输入框+1个按钮+1个textBrowser到主界面。开发后界面同需求设计中的截图。

2、使用pyuic5 -o commandtools.py commandtools.ui指令将.ui文件转换成.py文件。

# -*- coding: utf-8 -*-# Form implementation generated from reading ui file 'commandTools.ui'## Created by: PyQt5 UI code generator 5.11.3## WARNING! All changes made in this file will be lost!from PyQt5 import QtCore, QtGui, QtWidgetsclass Ui_Form(object):  def setupUi(self, Form):    Form.setObjectName("Form")    Form.resize(483, 347)    self.ip_label = QtWidgets.QLabel(Form)    self.ip_label.setGeometry(QtCore.QRect(30, 20, 16, 16))    font = QtGui.QFont()    font.setBold(True)    font.setWeight(75)    self.ip_label.setFont(font)    self.ip_label.setObjectName("ip_label")    self.ip_lineEdit = QtWidgets.QLineEdit(Form)    self.ip_lineEdit.setGeometry(QtCore.QRect(50, 20, 101, 20))    self.ip_lineEdit.setObjectName("ip_lineEdit")    self.username_label = QtWidgets.QLabel(Form)    self.username_label.setGeometry(QtCore.QRect(160, 20, 61, 16))    font = QtGui.QFont()    font.setBold(True)    font.setWeight(75)    self.username_label.setFont(font)    self.username_label.setObjectName("username_label")    self.username_lineEdit = QtWidgets.QLineEdit(Form)    self.username_lineEdit.setGeometry(QtCore.QRect(220, 20, 71, 20))    self.username_lineEdit.setObjectName("username_lineEdit")    self.password_label = QtWidgets.QLabel(Form)    self.password_label.setGeometry(QtCore.QRect(300, 20, 61, 16))    font = QtGui.QFont()    font.setBold(True)    font.setWeight(75)    self.password_label.setFont(font)    self.password_label.setObjectName("password_label")    self.password_lineEdit = QtWidgets.QLineEdit(Form)    self.password_lineEdit.setGeometry(QtCore.QRect(360, 20, 80, 20))    self.password_lineEdit.setObjectName("password_lineEdit")    self.command_label = QtWidgets.QLabel(Form)    self.command_label.setGeometry(QtCore.QRect(30, 70, 51, 16))    font = QtGui.QFont()    font.setBold(True)    font.setWeight(75)    self.command_label.setFont(font)    self.command_label.setObjectName("command_label")    self.command_lineEdit = QtWidgets.QLineEdit(Form)    self.command_lineEdit.setGeometry(QtCore.QRect(90, 70, 251, 20))    self.command_lineEdit.setObjectName("command_lineEdit")    self.result_textBrowser = QtWidgets.QTextBrowser(Form)    self.result_textBrowser.setGeometry(QtCore.QRect(30, 120, 410, 201))    self.result_textBrowser.setObjectName("result_textBrowser")    self.run_Button = QtWidgets.QPushButton(Form)    self.run_Button.setGeometry(QtCore.QRect(360, 70, 80, 23))    self.run_Button.setObjectName("run_Button")    self.retranslateUi(Form)    QtCore.QMetaObject.connectSlotsByName(Form)  def retranslateUi(self, Form):    _translate = QtCore.QCoreApplication.translate    Form.setWindowTitle(_translate("Form", "cmdTool"))    self.ip_label.setText(_translate("Form", "IP"))    self.ip_lineEdit.setText(_translate("Form", "127.0.0.1"))    self.username_label.setText(_translate("Form", "username"))    self.username_lineEdit.setText(_translate("Form", "admin"))    self.password_label.setText(_translate("Form", "password"))    self.password_lineEdit.setText(_translate("Form", "Winovs12!"))    self.command_label.setText(_translate("Form", "Command"))    self.command_lineEdit.setText(_translate("Form", "LST LOG"))    self.run_Button.setText(_translate("Form", "Run"))

3、实现主程序callcommand.py调用(业务与逻辑分离)。代码如下:

# -*- coding: utf-8 -*-import sysimport timeimport socketfrom PyQt5.QtWidgets import QApplication, QMainWindow,QCompleterfrom PyQt5.QtCore import Qt,QThread,pyqtSignalfrom commandTools import Ui_Formclass MyMainForm(QMainWindow, Ui_Form):  def __init__(self, parent=None):    """    构造函数    """    super(MyMainForm, self).__init__(parent)    self.setupUi(self)    self.run_Button.clicked.connect(self.execte_command)    self.ip_init_lst = ['121.1.1.1', '192.168.1.1', '172.16.1.1']    self.init_lineedit(self.ip_lineEdit,self.ip_init_lst)    self.cmd_init_lst = ['LST LOG', 'LST PARA','MOD PARA']    self.init_lineedit(self.command_lineEdit,self.cmd_init_lst)  def init_lineedit(self, lineedit, item_list):    """    用户初始化控件自动补全功能    """    # 增加自动补全    self.completer = QCompleter(item_list)    # 设置匹配模式 有三种: Qt.MatchStartsWith 开头匹配(默认) Qt.MatchContains 内容匹配 Qt.MatchEndsWith 结尾匹配    self.completer.setFilterMode(Qt.MatchContains)    # 设置补全模式 有三种: QCompleter.PopupCompletion(默认) QCompleter.InlineCompletion  QCompleter.UnfilteredPopupCompletion    self.completer.setCompletionMode(QCompleter.PopupCompletion)    # 给lineedit设置补全器    lineedit.setCompleter(self.completer)  def execte_command(self):    """    登录服务器,并执行命令    """    ip, username, password, command = self.get_input_para()    print(type(ip))    sockethandle = socket.socket(socket.AF_INET,socket.SOCK_STREAM)    sockethandle.connect((str(ip), 6000))    send_cmd = "username: %s, admin: %s, command: %s" % (username, password, command)    print(send_cmd)    sockethandle.sendall(send_cmd.encode('utf-8'))    time.sleep(0.5)    recdata = sockethandle.recv(65535)    tran_recdata = recdata.decode('utf-8')    self.result_textBrowser.setText(tran_recdata)  def get_input_para(self):    """    获取用户界面输入    """    ip = self.ip_lineEdit.text()    username = self.username_lineEdit.text()    password = self.password_lineEdit.text()    command = self.command_lineEdit.text()    return ip, username, password, commandif __name__ == "__main__":  app = QApplication(sys.argv)  myWin = MyMainForm()  myWin.show()  sys.exit(app.exec_())

4、使用pyinstaller转换成可执行的.exe文件。命令: pyinstaller -F callcommand.py -w

执行成功,生成的文件在d:\temp\dist\dist\callcommand.exe

5、运行callcommand.exe,点击run运行

关键代码

1、输入框自动补全功能函数。同样适用于下拉框控件。

  def init_lineedit(self, lineedit, item_list):    """    用户初始化控件自动补全功能    """    # 增加自动补全    self.completer = QCompleter(item_list)    # 设置匹配模式 有三种: Qt.MatchStartsWith 开头匹配(默认) Qt.MatchContains 内容匹配 Qt.MatchEndsWith 结尾匹配    self.completer.setFilterMode(Qt.MatchContains)    # 设置补全模式 有三种: QCompleter.PopupCompletion(默认) QCompleter.InlineCompletion  QCompleter.UnfilteredPopupCompletion    self.completer.setCompletionMode(QCompleter.PopupCompletion)    # 给lineedit设置补全器    lineedit.setCompleter(self.completer)

2、socket中sendall函数要将命令使用utf-8编码,否则会导致界面卡住:

sockethandle.sendall(send_cmd.encode('utf-8'))

3、需要将命令返回的内容解码再写入文本框,否则会导致界面卡住。

 recdata = sockethandle.recv(65535) tran_recdata = recdata.decode('utf-8') self.result_textBrowser.setText(tran_recdata)

附录

由于本地没有服务器用于调试程序。所以使用socket搭建1个建议服务器。服务器功能实现将接收的命令原样返回。就是接收什么命令就给客户端返回什么内容。服务器IP为本地IP127.0.0.1,绑定端口为6000。代码如下:

#!/usr/bin/env python3 # -*- coding: utf-8 -*- import socketimport syss = socket.socket(socket.AF_INET,socket.SOCK_STREAM)print("socket create success!")try:  s.bind(('127.0.0.1',6000))except socket.error as msg:  print(msg)  sys.exit(1)s.listen(10)while True:  conn, addr = s.accept()  print("success")  data = conn.recv(65535)  conn.sendall(data.decode('utf-8'))conn.close()s.close()

启动服务器:

简陋的有点过分,但是满足调试需求了。。。

小结

这个python+scoket需求实现的远程登录服务器执行命令只是把基本功能实现了。中间遇到的界面无响应甚至退出的问题(就是socket发送和接收内容编解码导致的)。。但是还有很多地方需要优化,比如对入参的判断并输出到文本框提示、对连接服务器结果的判断,还有界面的美化等内容。。正是这些小需求及实践过程中遇到问题、解决问题的过程逐步提升编码能力。。fighting


  • 上一条:
    pytorch实现用Resnet提取特征并保存为txt文件的方法
    下一条:
    解决在pycharm运行代码,调用CMD窗口的命令运行显示乱码问题
  • 昵称:

    邮箱:

    0条评论 (评论内容有缓存机制,请悉知!)
    最新最热
    • 分类目录
    • 人生(杂谈)
    • 技术
    • linux
    • Java
    • php
    • 框架(架构)
    • 前端
    • ThinkPHP
    • 数据库
    • 微信(小程序)
    • Laravel
    • Redis
    • Docker
    • Go
    • swoole
    • Windows
    • Python
    • 苹果(mac/ios)
    • 相关文章
    • 在python语言中Flask框架的学习及简单功能示例(0个评论)
    • 在Python语言中实现GUI全屏倒计时代码示例(0个评论)
    • Python + zipfile库实现zip文件解压自动化脚本示例(0个评论)
    • python爬虫BeautifulSoup快速抓取网站图片(1个评论)
    • vscode 配置 python3开发环境的方法(0个评论)
    • 近期文章
    • 在go中实现一个常用的先进先出的缓存淘汰算法示例代码(0个评论)
    • 在go+gin中使用"github.com/skip2/go-qrcode"实现url转二维码功能(0个评论)
    • 在go语言中使用api.geonames.org接口实现根据国际邮政编码获取地址信息功能(1个评论)
    • 在go语言中使用github.com/signintech/gopdf实现生成pdf分页文件功能(0个评论)
    • gmail发邮件报错:534 5.7.9 Application-specific password required...解决方案(0个评论)
    • 欧盟关于强迫劳动的规定的官方举报渠道及官方举报网站(0个评论)
    • 在go语言中使用github.com/signintech/gopdf实现生成pdf文件功能(0个评论)
    • Laravel从Accel获得5700万美元A轮融资(0个评论)
    • 在go + gin中gorm实现指定搜索/区间搜索分页列表功能接口实例(0个评论)
    • 在go语言中实现IP/CIDR的ip和netmask互转及IP段形式互转及ip是否存在IP/CIDR(0个评论)
    • 近期评论
    • 122 在

      学历:一种延缓就业设计,生活需求下的权衡之选中评论 工作几年后,报名考研了,到现在还没认真学习备考,迷茫中。作为一名北漂互联网打工人..
    • 123 在

      Clash for Windows作者删库跑路了,github已404中评论 按理说只要你在国内,所有的流量进出都在监控范围内,不管你怎么隐藏也没用,想搞你分..
    • 原梓番博客 在

      在Laravel框架中使用模型Model分表最简单的方法中评论 好久好久都没看友情链接申请了,今天刚看,已经添加。..
    • 博主 在

      佛跳墙vpn软件不会用?上不了网?佛跳墙vpn常见问题以及解决办法中评论 @1111老铁这个不行了,可以看看近期评论的其他文章..
    • 1111 在

      佛跳墙vpn软件不会用?上不了网?佛跳墙vpn常见问题以及解决办法中评论 网站不能打开,博主百忙中能否发个APP下载链接,佛跳墙或极光..
    • 2016-10
    • 2016-11
    • 2018-04
    • 2020-03
    • 2020-04
    • 2020-05
    • 2020-06
    • 2022-01
    • 2023-07
    • 2023-10
    Top

    Copyright·© 2019 侯体宗版权所有· 粤ICP备20027696号 PHP交流群

    侯体宗的博客