-
Notifications
You must be signed in to change notification settings - Fork 5
/
Copy pathWizard2.py
289 lines (221 loc) · 8.9 KB
/
Wizard2.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
# -*- coding: utf-8 -*-
#
# Cherokee-admin
#
# Authors:
# Alvaro Lopez Ortega <[email protected]>
#
# Copyright (C) 2001-2011 Alvaro Lopez Ortega
#
# This program is free software; you can redistribute it and/or
# modify it under the terms of version 2 of the GNU General Public
# License as published by the Free Software Foundation.
#
# This program is distributed in the hope that it will be useful,
# but WITHOUT ANY WARRANTY; without even the implied warranty of
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
# GNU General Public License for more details.
#
# You should have received a copy of the GNU General Public License
# along with this program; if not, write to the Free Software
# Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA
# 02110-1301, USA.
#
import os
import re
import CTK
import popen
import Install_Log
class Wizard (object):
"""
self.params
-----------
type : 'directory' or 'vserver'
vserver_num: 10, 20, ..
web_directory: /webdir
vserver_nick: example.com
app_fetch: /tmp/bat.tgz, http://_/foo.tgz
app_dir: /var/www_apps/worpress
targz_path: /tmp/foobar.tgz
"""
type = property (lambda s: s.params.get('type'), lambda s,v: s.params.update({'type': v}))
app_dir = property (lambda s: s.params.get('app_dir'), lambda s,v: s.params.update({'app_dir': v}))
app_fetch = property (lambda s: s.params.get('app_fetch'), lambda s,v: s.params.update({'app_fetch': v}))
targz_path = property (lambda s: s.params.get('targz_path'), lambda s,v: s.params.update({'targz_path': v}))
def __init__ (self, app_info, params=None):
self.app_info = app_info
# Do not copy the params. We do want a child wizard to
# reference its parent params dict.
#
self.params = params or {}
#
# Public
#
def Check_Parameters (self):
return []
def Check_Prerequisites (self):
return []
def Download (self):
return []
def Unpack (self):
return []
def Check_PostUnpack (self):
return []
def Configure_Cherokee (self):
return []
def Configure_Cherokee_PostApply (self):
return []
#
# Protected
#
def _Register_Child_Wizard (self, wizard_obj):
wizard_obj.params = self.params
return wizard_obj
def _Check_Params_Install_Type (self, allows_dir, allows_vserver):
errors = []
# Type
if not self.type:
errors += ["Missing 'type' property. Suitable values are: 'vserver' or 'directory'"]
return errors
if not self.type in ['directory', 'vserver']:
errors += ["Invalid 'type' value. It must be either 'vserver' or 'directory'"]
return errors
# Directory
if self.type == 'directory':
vserver_num = self.params.get('vserver_num')
directory = self.params.get('web_directory')
if not vserver_num:
errors += ["Property 'vserver_num' missing"]
elif not vserver_num.isdigit():
errors += ["Invalid value of the 'vserver_num' property: it must be a number"]
if not directory:
errors += ["Property 'web_directory' missing"]
elif directory[0] != '/':
errors += ["Invalid value of the 'web_directory' property: it must be a directory path"]
# Virtual Server
elif self.type == 'vserver':
vserver_nick = self.params.get('vserver_nick')
vserver_num = self.params.get('vserver_num')
if not vserver_nick:
errors += ["Property 'vserver_nick' missing"]
if vserver_num and CTK.cfg['vserver!%s'%(vserver_num)]:
errors += ["Invalid 'vserver_num'. Virtual server ID already in use."]
else:
self.params['vserver_num'] = CTK.cfg.get_next_entry_prefix ('vserver').split ('!')[-1]
return errors
def _Check_Software_Location (self):
errors = []
if self.app_fetch:
return []
if not self.app_dir:
errors += ["The 'app_dir' property must be provided"]
return errors
def _Handle_Download (self, tarball=None):
errors = []
url = None
pkg_path = None
# In case of running from the GUI, the file is downloaded by
# now, so the only thing left to do is to update the params.
if self.app_fetch and self.app_fetch.startswith ("http:"):
if CTK.DownloadEntry_Exists (self.app_fetch):
down_entry = CTK.DownloadEntry_Factory (self.app_fetch)
self.targz_path = down_entry.target_path
return []
# Auto
if not self.app_fetch or self.app_fetch == 'auto':
url = tarball
# Static file
elif self.app_fetch[0] == '/':
if not os.path.exists (self.app_fetch):
errors += [_("File or Directory not found: %(app_path)s") %({'app_path': self.app_fetch})]
return errors
if os.path.isdir(self.app_fetch):
self.targz_path = self.app_dir
else:
self.targz_path = self.app_fetch
# Download the software
else:
url = self.app_fetch
if url:
Install_Log.log ("Downloading: %s" %(url))
self.downloader = CTK.DownloadEntry_Factory (url)
self.downloader.start()
while self.downloader.isAlive():
self.downloader.join(1)
if self.downloader.size <= 0:
Install_Log.log ("Downloaded %d bytes..." %(self.downloader.downloaded))
else:
Install_Log.log ("Downloaded %d / %d (%d%%)..." %(self.downloader.downloaded, self.downloader.size, (self.downloader.downloaded * 100 / self.downloader.size)))
self.targz_path = self.downloader.target_path
Install_Log.log ("Download completed: %s" %(self.targz_path))
return []
def _Handle_Unpacking (self):
if not self.targz_path:
return []
assert self.app_dir
# Create the app directory
if not os.path.exists(self.app_dir):
os.makedirs (self.app_dir)
# Unpack
command = "gzip -dc '%s' | tar xfv -" %(self.targz_path)
Install_Log.log ("(cd: %s): %s" %(self.app_dir, command))
ret = popen.popen_sync (command, cd=self.app_dir)
Install_Log.log (ret['stdout'])
Install_Log.log (ret['stderr'])
return []
def _Update_app_dir (self, re_filter=None):
assert self.app_dir
for f in os.listdir (self.app_dir):
fp = os.path.join (self.app_dir, f)
if re_filter:
if not re.match (re_filter, f):
continue
Install_Log.log ("Updating app_dir: %s to %s" %(self.app_dir, fp))
self.app_dir = fp
return []
Install_Log.log ("app_dir was not updated: %s" %(self.app_dir))
return []
def _Handle_Log_VServer (self):
vserver_num = self.params.get('vserver_num')
cp_vsrv_log = self.params.get('cp_vsrv_log')
# For VServers only
if not self.type == 'vserver':
return []
# Log config cloning
if cp_vsrv_log:
vsrv_1st = CTK.cfg.get_lowest_entry ('vserver')
if CTK.cfg['vserver!%s!logger' %(vsrv_1st)]:
CTK.cfg.clone ('vserver!%s!logger' %(vsrv_1st), 'vserver!%s!logger' %(vserver_num))
if CTK.cfg['vserver!%s!error_writer'%(vsrv_1st)]:
CTK.cfg.clone ('vserver!%s!error_writer' %(vsrv_1st), 'vserver!%s!error_writer' %(vserver_num))
return []
#
# Checks
#
def _Check_File_Exists (self, filename):
assert self.app_dir
assert self.app_info and self.app_info.has_key('name')
errors = []
fpath = os.path.join (self.app_dir, filename)
if not os.path.exists (fpath):
errors += [_("The '%(app_dir)s' directory does not look like a %(name)s directory: The %(filename)s file is missing.")
%({'app_dir': self.app_dir, 'name': self.app_info['name'], 'filename': filename})]
return errors
_modules = {}
def Load_Module (path):
global _modules
if not path in _modules:
tmp = path.split('/')
dsc = tmp[-1].replace('.py','')
php_mod_path = os.path.realpath (__file__ + '/../wizards/' + path)
_modules[path] = CTK.load_module_pyc (php_mod_path, dsc)
return _modules[path]
_templates = {}
def Load_Template (path):
global _templates
if not path in _templates:
tmp = path.split('/')
dsc = tmp[-1].replace('.py','')
php_mod_path = os.path.realpath (__file__ + '/../templates/' + path)
_templates[path] = CTK.load_module_pyc (php_mod_path, dsc)
return _templates[path]