Files
python3-cookbook/gui/examples/zupload.py

574 lines
21 KiB
Python
Raw Normal View History

2015-01-20 17:00:37 +08:00
#!/usr/bin/env python
# -*- encoding: utf-8 -*-
"""
Topic: 小令自动发布工具
Desc :
"""
import wx
import wx.html
2015-01-21 15:24:12 +08:00
import examples.ztransfer as ztransfer
2015-01-22 11:38:55 +08:00
import commons.util as zutil
2015-01-20 17:00:37 +08:00
import os
2015-01-21 15:24:12 +08:00
import subprocess
import logging
2015-01-21 17:56:20 +08:00
import time
2015-01-21 15:24:12 +08:00
import tempfile
import sys
2015-01-21 17:56:20 +08:00
from threading import Thread
from wx.lib.pubsub import pub
2015-01-22 11:38:55 +08:00
import ConfigParser
2015-01-21 17:56:20 +08:00
import wx.lib.agw.pyprogress as PP
2015-01-20 17:00:37 +08:00
HOSTNAME_ = '115.29.145.245' # remote hostname where SSH server is running
USERNAME_ = 'winhong'
PASSWORD_ = 'jianji2014'
2015-01-22 11:38:55 +08:00
TOMAT_REMOTE_ = r"/usr/local/apache-tomcat-8.0.15"
# DIR_REMOTE = r"/usr/local/apache-tomcat-8.0.15/webapps/ROOT/WEB-INF/classes/com"
COMMAND_01_ = '/home/winhong/ling01.sh'
COMMAND_02_ = '/home/winhong/ling02.sh'
2015-01-20 17:00:37 +08:00
2015-01-21 15:24:12 +08:00
_LOGGING = logging.getLogger('zupload')
IS_WIN32 = 'win32' in str(sys.platform).lower()
def subprocess_call(*args, **kwargs):
# also works for Popen. It creates a new *hidden* window,
# so it will work in frozen apps (.exe).
if IS_WIN32:
_LOGGING.info('subprocess_call==IS_WIN32')
startupinfo = subprocess.STARTUPINFO()
startupinfo.dwFlags = subprocess.CREATE_NEW_CONSOLE | subprocess.STARTF_USESHOWWINDOW
startupinfo.wShowWindow = subprocess.SW_HIDE
kwargs['startupinfo'] = startupinfo
retcode = subprocess.call(*args, **kwargs)
return retcode
def subprocess_popen(*args, **kwargs):
# also works for Popen. It creates a new *hidden* window,
# so it will work in frozen apps (.exe).
if IS_WIN32:
_LOGGING.info('subprocess_call==IS_WIN32')
startupinfo = subprocess.STARTUPINFO()
startupinfo.dwFlags = subprocess.CREATE_NEW_CONSOLE | subprocess.STARTF_USESHOWWINDOW
startupinfo.wShowWindow = subprocess.SW_HIDE
kwargs['startupinfo'] = startupinfo
ppopen = subprocess.Popen(*args, **kwargs)
return ppopen
2015-01-20 17:00:37 +08:00
2015-01-22 11:38:55 +08:00
class MyConfig():
def __init__(self, hostname=HOSTNAME_, username=USERNAME_, password=PASSWORD_,
projdir='', mavendir='', tomcatdir=TOMAT_REMOTE_,
command1=COMMAND_01_, command2=COMMAND_02_):
self.hostname = hostname
self.username = username
self.password = password
self.projdir = projdir
self.mavendir = mavendir
self.tomcatdir = tomcatdir
self.command1 = command1
self.command2 = command2
def saveConfig(myconfig):
config_file = zutil.userhome_file('ztool.conf')
# check if file exists
cf = ConfigParser.ConfigParser()
if os.path.isfile(config_file):
# update file
cf.read(config_file)
else:
# create file
cf.add_section('main')
for k, v in vars(myconfig).iteritems():
cf.set("main", k, v)
with open(config_file, "w") as f:
cf.write(f)
def loadConfig():
config = MyConfig()
userhome = os.path.expanduser('~')
config_file = os.path.join(userhome, 'ztool.conf')
if os.path.isfile(config_file):
cf = ConfigParser.ConfigParser()
cf.read(config_file)
for k, v in vars(config).iteritems():
setattr(config, k, cf.get('main', k))
return config
2015-01-22 16:02:57 +08:00
class Ling01(wx.Dialog):
text = u'''
<html>
<body bgcolor="#ACAA60">
<center>
<table bgcolor="#455481" width="100%" cellspacing="0"
cellpadding=0 border=1>
<tr>
<td align="center"><h1>ling01.sh示例代码</h1></td>
</tr>
</table>
</center>
<p>
#!/bin/bash<br><br>
echo '先停止tomcat...'<br>
ps aux |grep tomcat |grep -v 'grep tomcat' |awk '{print $2}'|sudo xargs kill -9<br>
wait<br><br>
echo '成功停止!开始替换class文件'<br>
cd /usr/local/apache-tomcat-8.0.15/webapps/ROOT/WEB-INF/classes/com<br>
rm -rf winhong/<br>
unzip ling.zip<br>
wait<br><br>
echo '解压成功'<br>
rm -f ling.zip<br>
</p>
</body>
</html>
'''
def __init__(self, parent):
wx.Dialog.__init__(self, parent, -1, u'ling01.sh示例', size=(600, 400))
html = wx.html.HtmlWindow(self)
html.SetPage(self.text)
button = wx.Button(self, wx.ID_OK, u'确定')
sizer = wx.BoxSizer(wx.VERTICAL)
sizer.Add(html, 1, wx.EXPAND | wx.ALL, 5)
sizer.Add(button, 0, wx.ALIGN_CENTER | wx.ALL, 5)
self.SetSizer(sizer)
self.Layout()
class Ling02(wx.Dialog):
text = u'''
<html>
<body bgcolor="#ACAA60">
<center>
<table bgcolor="#455481" width="100%" cellspacing="0"
cellpadding=0 border=1>
<tr>
<td align="center"><h1>ling02.sh示例代码</h1></td>
</tr>
</table>
</center>
<p>
#!/bin/bash<br><br>
echo '开始重启tomcat....'<br>
/usr/local/apache-tomcat-8.0.15/bin/startup.sh<br>
wait<br><br>
echo '重启成功...'<br>
</p>
</body>
</html>
'''
def __init__(self, parent):
wx.Dialog.__init__(self, parent, -1, u'ling02.sh示例', size=(550, 360))
html = wx.html.HtmlWindow(self)
html.SetPage(self.text)
button = wx.Button(self, wx.ID_OK, u'确定')
sizer = wx.BoxSizer(wx.VERTICAL)
sizer.Add(html, 1, wx.EXPAND | wx.ALL, 5)
sizer.Add(button, 0, wx.ALIGN_CENTER | wx.ALL, 5)
self.SetSizer(sizer)
self.Layout()
class SketchGuide(wx.Dialog):
text = u'''
<html>
<body bgcolor="#ACAA60">
<center>
<table bgcolor="#455481" width="100%" cellspacing="0"
cellpadding=0 border=1>
<tr>
<td align="center"><h1>使用帮助</h1></td>
</tr>
</table>
</center>
<p>
2015-01-22 16:02:57 +08:00
<b>简介</b>自动编译代码并上传到服务器然后替换class文件并重启tomcat
</p>
<p>
<b>1.更新代码</b>请先确保代码源代码已经从svn更新到最新了
</p>
<p>
2015-01-22 16:02:57 +08:00
<b>2.Tomcat目录</b> 服务器上面Tomcat的根目录
</p>
<p>
<b>3.远程脚本1</b> 使用sudo命令去kill掉tomcat进程(参考ling01.sh)
</p>
<p>
<b>4.远程脚本2</b> 非sudo方式启动tomcat需要设置sudoers(参考ling02.sh)
</p>
<p>
<b>5.Maven目录(选填)</b> 如果mvn命令还没有添加到系统PATH中就需要填这个
</p>
</body>
</html>
'''
def __init__(self, parent):
2015-01-22 16:02:57 +08:00
wx.Dialog.__init__(self, parent, -1, 'Use Guide', size=(550, 360))
html = wx.html.HtmlWindow(self)
html.SetPage(self.text)
button = wx.Button(self, wx.ID_OK, u'确定')
sizer = wx.BoxSizer(wx.VERTICAL)
sizer.Add(html, 1, wx.EXPAND | wx.ALL, 5)
sizer.Add(button, 0, wx.ALIGN_CENTER | wx.ALL, 5)
self.SetSizer(sizer)
self.Layout()
class SketchAbout(wx.Dialog):
text = u'''
<html>
<body bgcolor="#ACAA60">
<center>
<table bgcolor="#455481" width="100%" cellspacing="0"
cellpadding=0 border=1>
<tr>
<td align="center"><h1>小令发布工具</h1></td>
</tr>
<tr>
<td align="center"><h4>Profession Edition 0.9.0</h4></td>
</tr>
</table>
</center>
2015-01-22 11:38:55 +08:00
<br/><br/><br/><br/><br/>
<p style="text-align: right; margin-top: 100px;">
Powered By XiongNeng 2015/01/21
</p>
</body>
</html>
'''
def __init__(self, parent):
wx.Dialog.__init__(self, parent, -1, 'About', size=(550, 300))
html = wx.html.HtmlWindow(self)
html.SetPage(self.text)
button = wx.Button(self, wx.ID_OK, u'确定')
sizer = wx.BoxSizer(wx.VERTICAL)
sizer.Add(html, 1, wx.EXPAND | wx.ALL, 5)
sizer.Add(button, 0, wx.ALIGN_CENTER | wx.ALL, 5)
self.SetSizer(sizer)
self.Layout()
2015-01-21 17:56:20 +08:00
########################################################################
class ButtonThread(Thread):
2015-01-22 11:38:55 +08:00
"""Button Worker Thread Class."""
2015-01-21 17:56:20 +08:00
# ----------------------------------------------------------------------
def __init__(self, myframe):
"""Init Worker Thread Class."""
Thread.__init__(self)
self.myframe = myframe
# ----------------------------------------------------------------------
def run(self):
self.myframe.buttonEnd = False
self.myframe.buttonResult = False
2015-01-22 11:38:55 +08:00
# save config
hostname = self.myframe.hostname.GetValue()
username = self.myframe.username.GetValue()
password = self.myframe.password.GetValue()
tomcathome = self.myframe.tomcat.GetValue()
command1 = self.myframe.command1.GetValue()
command2 = self.myframe.command2.GetValue()
2015-01-21 17:56:20 +08:00
projdir = self.myframe.proj.GetValue()
maven_home = self.myframe.maven.GetValue()
2015-01-22 11:38:55 +08:00
myconf = MyConfig(hostname=hostname, username=username, password=password,
projdir=projdir, mavendir=maven_home, tomcatdir=tomcathome,
command1=command1, command2=command2)
# 先更新模块的配置
self.myframe.ChangeConfig(myconfig=myconf)
# 然后同步到本地配置文件中
saveConfig(myconfig=myconf)
2015-01-21 17:56:20 +08:00
errmsg = ''
try:
# 先本地编译
if not maven_home:
exe_command = 'cd /d %s && mvn clean && mvn compile' % projdir
else:
mvn_full = os.path.join(maven_home, 'bin', 'mvn')
exe_command = 'cd /d %s && %s clean && %s compile' % (projdir, mvn_full, mvn_full)
_LOGGING.info('#subprocess exe_command start: %s' % exe_command)
# 执行命令,但是捕捉输出
# if os.name == 'nt':
# _LOGGING.info('os.name==nt')
# startupinfo = subprocess.STARTUPINFO()
# startupinfo.dwFlags |= subprocess.STARTF_USESHOWWINDOW
exresult = subprocess_call(exe_command, shell=True)
# exresult = subprocess_popen(exe_command, shell=True, stdout=subprocess.PIPE)
# out = exresult.stdout.read()
# _LOGGING.info(out)
_LOGGING.info('#subprocess_call result is %s' % exresult)
_LOGGING.info('#subprocess_call exe_command end')
if exresult != 0:
result = False
errmsg = 'execute maven command failure.'
else:
result = ztransfer.main()
except Exception as e:
result = False
errmsg = e.message
self.myframe.buttonEnd = True
self.myframe.buttonResult = result
self.myframe.buttonMsg = errmsg
class CheckThread(Thread):
2015-01-22 11:38:55 +08:00
"""每隔0.5秒检查是否已经完成"""
2015-01-21 17:56:20 +08:00
# ----------------------------------------------------------------------
def __init__(self, myframe):
"""Init Worker Thread Class."""
Thread.__init__(self)
self.myframe = myframe
# ----------------------------------------------------------------------
def run(self):
"""Run Worker Thread."""
# This is the code executing in the new thread.
while not self.myframe.buttonEnd:
2015-01-22 11:38:55 +08:00
time.sleep(0.5)
2015-01-21 17:56:20 +08:00
wx.CallAfter(pub.sendMessage, "update",
2015-01-22 11:38:55 +08:00
msg=(self.myframe.buttonEnd, self.myframe.buttonResult,
self.myframe.buttonMsg))
2015-01-21 17:56:20 +08:00
########################################################################
class MyProgressDialog(wx.Dialog):
2015-01-22 11:38:55 +08:00
"""进度条弹出框"""
2015-01-21 17:56:20 +08:00
# ----------------------------------------------------------------------
def __init__(self):
"""Constructor"""
wx.Dialog.__init__(self, None, title=u"同步进度条", size=(350, 150))
self.count = 0
self.progress = wx.Gauge(self, -1, 15, (20, 50), size=(300, 30))
self.tips = wx.StaticText(self, -1, u'正在同步,请耐心等待几秒钟...')
sizer = wx.BoxSizer(wx.VERTICAL)
sizer.Add(self.progress, 0, wx.EXPAND)
2015-01-22 11:38:55 +08:00
sizer.Add((20, 20), 0, wx.EXPAND)
2015-01-21 17:56:20 +08:00
sizer.Add(self.tips, 0, wx.EXPAND)
self.SetSizer(sizer)
# create a pubsub receiver
pub.subscribe(self.updateProgress, "update")
# ----------------------------------------------------------------------
def updateProgress(self, msg):
"""更新进度条"""
self.count += 1
if msg[0]:
self.Destroy()
if msg[1]:
_LOGGING.info('MessageDialog.upload.success!!!')
wx.MessageDialog(self, u'上传成功了!',
'MessageDialog', wx.ICON_INFORMATION).ShowModal()
else:
_LOGGING.error('MessageDialog.upload.error!!!')
wx.MessageDialog(self, u'上传失败error:%s' % msg[2],
'MessageDialog', wx.ICON_INFORMATION).ShowModal()
elif self.count >= 15:
self.count = 1
self.progress.SetValue(self.count)
2015-01-20 17:00:37 +08:00
class UploadFrame(wx.Frame):
def __init__(self):
wx.Frame.__init__(self, None, -1, u'小令自动发布工具')
2015-01-22 14:51:46 +08:00
ico = wx.Icon(zutil.resource_path('resources/main.ico'), wx.BITMAP_TYPE_ICO)
self.SetIcon(ico)
2015-01-21 17:56:20 +08:00
self.buttonEnd = False
self.buttonResult = False
self.buttonMsg = ''
# 创建一个菜单栏
menuBar = wx.MenuBar()
# 创建几个菜单
menu1 = wx.Menu()
menuBar.Append(menu1, '&File')
2015-01-22 11:38:55 +08:00
menu1.Append(-1, "&Open", 'Open new file')
menuItem = menu1.Append(-1, "&Exit", 'Exit System')
# 菜单项绑定事件
self.Bind(wx.EVT_MENU, self.OnCloseMe, menuItem)
menu2 = wx.Menu()
# 创建菜单项MenuItem
menu2.Append(wx.NewId(), '&Copy', 'Copy in status bar')
menu2.Append(wx.NewId(), '&Cut', '')
menu2.Append(wx.NewId(), '&Paste', '')
menu2.AppendSeparator()
menu2.Append(wx.NewId(), '&Options', 'Display Options')
menuBar.Append(menu2, '&Edit') # 在菜单栏上附上菜单
menu3 = wx.Menu()
menuBar.Append(menu3, '&Help')
guideItems = menu3.Append(-1, "&Use Guide", '')
2015-01-22 16:02:57 +08:00
ling01Items = menu3.Append(-1, u"&ling01.sh", '')
ling02Items = menu3.Append(-1, u"&ling02.sh", '')
aboutItem3 = menu3.Append(-1, "&About", '')
# 菜单项绑定事件
self.Bind(wx.EVT_MENU, self.OnGuide, guideItems)
2015-01-22 16:02:57 +08:00
self.Bind(wx.EVT_MENU, self.OnLing01, ling01Items)
self.Bind(wx.EVT_MENU, self.OnLing02, ling02Items)
self.Bind(wx.EVT_MENU, self.OnAbout, aboutItem3)
self.SetMenuBar(menuBar) # 在Frame上面附加菜单
# ----------------------------分割线----------------------------------
2015-01-22 11:38:55 +08:00
# 加载配置文件
myconfig = loadConfig()
2015-01-20 17:00:37 +08:00
panel = wx.Panel(self)
# 首先创建controls
2015-01-21 15:24:12 +08:00
topLbl = wx.StaticText(panel, -1, u'================小令自动发布工具===============')
2015-01-20 17:00:37 +08:00
topLbl.SetFont(wx.Font(18, wx.SWISS, wx.NORMAL, wx.BOLD))
2015-01-22 11:38:55 +08:00
hostLbl = wx.StaticText(panel, -1, u'服务器地址:')
self.hostname = wx.TextCtrl(panel, -1, myconfig.hostname)
usernameLbl = wx.StaticText(panel, -1, u'登录用户名:')
self.username = wx.TextCtrl(panel, -1, myconfig.username)
passwordLbl = wx.StaticText(panel, -1, u'密码:')
self.password = wx.TextCtrl(panel, -1, myconfig.password, style=wx.TE_PASSWORD)
tomcatLbl = wx.StaticText(panel, -1, u'Tomcat目录:')
self.tomcat = wx.TextCtrl(panel, -1, myconfig.tomcatdir)
command1Lbl = wx.StaticText(panel, -1, u'远程脚本1:')
self.command1 = wx.TextCtrl(panel, -1, myconfig.command1)
command2Lbl = wx.StaticText(panel, -1, u'远程脚本2:')
self.command2 = wx.TextCtrl(panel, -1, myconfig.command2)
2015-01-20 17:00:37 +08:00
projLbl = wx.StaticText(panel, -1, u'本地工程目录:')
2015-01-22 11:38:55 +08:00
self.proj = wx.TextCtrl(panel, -1, myconfig.projdir)
mavenbl = wx.StaticText(panel, -1, u'Maven目录(选填):')
self.maven = wx.TextCtrl(panel, -1, myconfig.mavendir)
2015-01-20 17:00:37 +08:00
saveBtn = wx.Button(panel, -1, u'开始发布')
2015-01-21 15:24:12 +08:00
cancelBtn = wx.Button(panel, -1, u'关闭')
2015-01-20 17:00:37 +08:00
# 下面开始布局
# mainSizer是顶级sizer控制所有部件使用box sizer
# 垂直sizer
mainSizer = wx.BoxSizer(wx.VERTICAL)
# boxsizer.Add(btn1, proportion=0, flag=wx.ALL, border=2)
mainSizer.Add(topLbl, 0, wx.ALL, 5)
mainSizer.Add(wx.StaticLine(panel), 0, wx.EXPAND | wx.TOP | wx.BOTTOM, 5)
# addrSizer控制所有地址信息使用gridbag sizer
addrSizer = wx.GridBagSizer(hgap=5, vgap=5)
# sizer.Add(bw, pos=(3,0), span=(1,4), flag=wx.EXPAND)
2015-01-22 11:38:55 +08:00
addrSizer.Add(hostLbl, pos=(0, 0), flag=wx.ALIGN_RIGHT | wx.ALIGN_CENTER_VERTICAL)
addrSizer.Add(self.hostname, pos=(0, 1), flag=wx.EXPAND)
addrSizer.Add(usernameLbl, pos=(1, 0), flag=wx.ALIGN_RIGHT | wx.ALIGN_CENTER_VERTICAL)
addrSizer.Add(self.username, pos=(1, 1), flag=wx.EXPAND)
addrSizer.Add(passwordLbl, pos=(2, 0), flag=wx.ALIGN_RIGHT | wx.ALIGN_CENTER_VERTICAL)
addrSizer.Add(self.password, pos=(2, 1), flag=wx.EXPAND)
addrSizer.Add(tomcatLbl, pos=(3, 0), flag=wx.ALIGN_RIGHT | wx.ALIGN_CENTER_VERTICAL)
addrSizer.Add(self.tomcat, pos=(3, 1), flag=wx.EXPAND)
addrSizer.Add(command1Lbl, pos=(4, 0), flag=wx.ALIGN_RIGHT | wx.ALIGN_CENTER_VERTICAL)
addrSizer.Add(self.command1, pos=(4, 1), flag=wx.EXPAND)
addrSizer.Add(command2Lbl, pos=(5, 0), flag=wx.ALIGN_RIGHT | wx.ALIGN_CENTER_VERTICAL)
addrSizer.Add(self.command2, pos=(5, 1), flag=wx.EXPAND)
addrSizer.Add(projLbl, pos=(6, 0), flag=wx.ALIGN_RIGHT | wx.ALIGN_CENTER_VERTICAL)
addrSizer.Add(self.proj, pos=(6, 1), flag=wx.EXPAND)
addrSizer.Add(mavenbl, pos=(7, 0), flag=wx.ALIGN_RIGHT | wx.ALIGN_CENTER_VERTICAL)
addrSizer.Add(self.maven, pos=(7, 1), flag=wx.EXPAND)
2015-01-20 17:00:37 +08:00
2015-01-21 15:24:12 +08:00
# 添加几个实际的空白行
2015-01-22 11:38:55 +08:00
addrSizer.Add((30, 30), pos=(8, 0), span=(1, 2), flag=wx.EXPAND)
2015-01-21 15:24:12 +08:00
2015-01-20 17:00:37 +08:00
addrSizer.AddGrowableCol(1)
# 然后把addrSizer添加到mainSizer中
mainSizer.Add(addrSizer, 0, wx.EXPAND | wx.ALL, 10)
# 按钮放到两边和中间都能伸缩间隔的一行中
btnSizer = wx.BoxSizer(wx.HORIZONTAL)
btnSizer.Add((20, 20), 1)
btnSizer.Add(saveBtn)
btnSizer.Add((20, 20), 1)
btnSizer.Add(cancelBtn)
btnSizer.Add((20, 20), 1)
self.Bind(wx.EVT_BUTTON, self.OnUploadMe, saveBtn)
self.Bind(wx.EVT_BUTTON, self.OnCloseMe, cancelBtn)
mainSizer.Add(btnSizer, 0, wx.EXPAND | wx.BOTTOM, 10)
panel.SetSizer(mainSizer)
# 让框架自适应sizer如果panel改变大小框架会自动调整尺寸
# 同时还能防止框架比panel最小尺寸还小
mainSizer.Fit(self)
mainSizer.SetSizeHints(self)
self.Centre()
self.Show()
def OnCloseMe(self, event):
self.Close(True)
def OnGuide(self, event):
dlg = SketchGuide(self)
dlg.ShowModal()
dlg.Destroy()
2015-01-22 16:02:57 +08:00
def OnLing01(self, event):
dlg = Ling01(self)
dlg.ShowModal()
dlg.Destroy()
def OnLing02(self, event):
dlg = Ling02(self)
dlg.ShowModal()
dlg.Destroy()
def OnAbout(self, event):
dlg = SketchAbout(self)
dlg.ShowModal()
dlg.Destroy()
2015-01-20 17:00:37 +08:00
def OnUploadMe(self, event):
2015-01-21 17:56:20 +08:00
btn = event.GetEventObject()
btn.Disable()
ButtonThread(self).start()
CheckThread(self).start()
dlg = MyProgressDialog()
dlg.ShowModal()
btn.Enable()
2015-01-20 17:00:37 +08:00
2015-01-22 11:38:55 +08:00
def ChangeConfig(self, myconfig):
ztransfer.HOSTNAME = myconfig.hostname
ztransfer.USERNAME = myconfig.username
ztransfer.PASSWORD = myconfig.password
ztransfer.DIR_LOCAL = os.path.join(myconfig.projdir, 'target', 'classes', 'com')
ztransfer.ZIPDIR_SRC = os.path.join(myconfig.projdir, 'target', 'classes', 'com')
ztransfer.ZIPDIR_DEST = myconfig.projdir
if myconfig.tomcatdir.endswith('/'):
myconfig.tomcatdir = myconfig.tomcatdir[:-1]
ztransfer.DIR_REMOTE_CONFIG = myconfig.tomcatdir + '/webapps/ROOT/WEB-INF/classes'
2015-01-22 11:38:55 +08:00
ztransfer.DIR_REMOTE = myconfig.tomcatdir + '/webapps/ROOT/WEB-INF/classes/com'
ztransfer.COMMAND_01 = myconfig.command1
ztransfer.COMMAND_02 = myconfig.command2
2015-01-20 17:00:37 +08:00
def main():
2015-01-21 15:24:12 +08:00
# app = wx.App(redirect=True, filename=tempfile.TemporaryFile().name)
2015-01-20 17:00:37 +08:00
app = wx.App()
UploadFrame()
app.MainLoop()