开发背景:
之前在一家IDC公司实习,负责服务器售后方面的,经常要使用ping,tracert,tcping等命令做些初步的诊断,判断服务器问题出在哪方面。于是就想集成这些常用的命令或工具到一个GUI界面中,实现一键调用。(PS:当然公司有更好的集成工具,我只是模仿公司的)
开发环境:
win 7 64位,Python 3.6.2,wxpython
具体思路:
通过wxpython实现GUI界面,通过subprocess.Popen创建新进程执行dos命令实现工具的调用。
功能实现:
GUI 界面:
实现ping,tracert,tcping命令:
关键步骤:
按钮事件代码:
self.Bind(wx.EVT_BUTTON,lambda _:subprocess.Popen(‘cmd /k ping -t ‘+tc.GetValue().split(‘:‘)[0]),b1)
wx.EVTBUTTON是按钮事件
lambda _:subprocess.Popen(‘cmd /k ping -t ‘+tc.GetValue().split(‘:‘)[0]) 是绑定的函数
b1 是ping按钮的变量名
这次的关键在按钮事件绑定的函数。
1、因为函数要传入dos命令作为参数,所以需要使用lambda 匿名函数,否则函数会在程序启动时执行,而不是在发生按钮事件时执行。
2、使用什么方法调用dos命令,os.system(),os.popen()或者subprocess.Popen()。
os.system()和os.popen()会阻塞父进程,也就是在命令执行完毕前,GUI界面是无法操作的。当然可以使用os.system(‘start dos命令‘)来解决这问题,dos的start命令是启动一个单独的窗口运行指定的程序或命令。因为启动新窗口很快,所以是感觉不到阻塞的,但是出现了个新的问题,这样会出现两个dos窗口,执行start的窗口和start启动的窗口,执行start命令的窗口会闪退(一瞬间执行完毕就退出)
所以最终我选择了subprocess.Popen(),因为subprocess.Popen()创建新的进程是不阻塞父进程的。
3、dos命令的选择
cmd /k ping -t ‘+tc.GetValue().split(‘:‘)[0]
cmd /k 是执行完命令仍保留cmd窗口
tc.GetValue()是获取输入框的IP地址,因为有时输入框的格式是IP:PORT这样子的,所以使用tc.GetValue().split(‘:‘)[0]这样去获取IP
具体源代码:
# coding:utf-8
import wx
#import os
import subprocess
class Myframe(wx.Frame):
def __init__(self, parent, title):
#设置窗口标题,窗口大小,窗口不可改变尺寸
wx.Frame.__init__(self, parent, title=title, size=(-1, 530),pos=(100,100),style=wx.DEFAULT_FRAME_STYLE ^ wx.RESIZE_BORDER ^ wx.MAXIMIZE_BOX)
self.InitUI()
#self.Centre()#窗口居中
self.Show()
def InitUI(self):#显示窗口组件内容
panel=wx.Panel(self)
font = wx.Font(13, wx.SWISS, wx.NORMAL, wx.NORMAL)
#创建输入框
text=wx.StaticText(panel, label=‘IP:‘)
text.SetFont(font)
tc=wx.TextCtrl(panel,size=(200,-1),style=wx.TE_CENTER)
tc.SetFont(font)
#创建按钮,并绑定事件
b1=wx.Button(panel, label=‘ping‘)
b1.SetFont(font)
self.Bind(wx.EVT_BUTTON,lambda _:subprocess.Popen(‘cmd /k ping -t ‘+tc.GetValue().split(‘:‘)[0]),b1)
b2=wx.Button(panel, label=‘tracert‘)
b2.SetFont(font)
self.Bind(wx.EVT_BUTTON,lambda _:subprocess.Popen(‘cmd /k tracert -d ‘+tc.GetValue().split(‘:‘)[0]),b2)
b3=wx.Button(panel, label=‘tcping‘)
b3.SetFont(font)
self.Bind(wx.EVT_BUTTON,lambda _:subprocess.Popen(‘cmd /k tools\\tcping -t ‘+tc.GetValue().replace(‘:‘,‘ ‘)),b3)
b4=wx.Button(panel, label=‘mstsc‘)
b4.SetFont(font)
self.Bind(wx.EVT_BUTTON,lambda _:subprocess.Popen(‘mstsc /v ‘+tc.GetValue()),b4)
b5=wx.Button(panel, label=‘putty‘)
b5.SetFont(font)
self.Bind(wx.EVT_BUTTON,lambda _:subprocess.Popen(‘tools\\putty ‘+tc.GetValue()+‘ -P ‘+(tc.GetValue()+‘:22‘).split(‘:‘)[1]),b5)
b6=wx.Button(panel, label=‘多ping-PingInfoView‘)
b6.SetFont(font)
self.Bind(wx.EVT_BUTTON,lambda _:subprocess.Popen(‘tools\\PingInfoView‘),b6)
font1 = wx.Font(11, wx.SWISS, wx.NORMAL, wx.NORMAL)
#操作说明
t1=wx.StaticText(panel, label=‘ 说明:‘)
t1.SetFont( wx.Font(15, wx.SWISS, wx.NORMAL, wx.NORMAL))
t2=wx.StaticText(panel, label=‘输入格式为:IP:PORT\n若只输入IP,端口则为3389\n若输入为空,则直接打开mstsc‘)
t2.SetFont(font1)
t3=wx.StaticText(panel, label=‘ mstsc说明:‘)
t3.SetFont(font1)
t4=wx.StaticText(panel, label=‘ putty说明:‘)
t4.SetFont(font1)
t5=wx.StaticText(panel, label=‘输入格式为:IP:PORT\n若只输入IP,端口则为22\n若输入为空,则直接打开putty‘)
t5.SetFont(font1)
t7=wx.StaticText(panel, label=‘tcping说明:‘)
t7.SetFont(font1)
t8=wx.StaticText(panel, label=‘输入格式为:IP:PORT\n若只输入IP,端口则为80‘)
t8.SetFont(font1)
#使用FlexGridSizer布局
fgs=wx.FlexGridSizer(cols=2,hgap=10,vgap=10)
fgs.AddMany([(1,1),(1,1),(text,0,wx.ALIGN_RIGHT | wx.ALIGN_CENTER_VERTICAL),(tc,0,wx.EXPAND),(1,1),(b1,0,wx.EXPAND),(1,1),(b2,0,wx.EXPAND),(1,1),(b3,0,wx.EXPAND),(1,1),(b4,0,wx.EXPAND),(1,1),(b5,0,wx.EXPAND),(1,1),(b6,0,wx.EXPAND),(t1),(1,1),(t7),(t8),(t3),(t2),(t4),(t5)])
panel.SetSizerAndFit(fgs)
if __name__==‘__main__‘:
app=wx.App()
Myframe(None,title=‘tools(运维工具)‘)
app.MainLoop()
源码和工具已上传在 http://down.51cto.com/data/2447030
思考总结
这次代码就六十多行,就实现了这集成工具,感觉使用Python开发还是很方便的。
原文地址:http://blog.51cto.com/eddy72/2115709