def __init__(self): self._props = PropertiesObject('Properties.py') self._props['dirname'] = '.' self._comps = [] self._htHeader, self._htFooter = self.htHeaderAndFooter() from DocSupport.pytp import PyTP self._pytp = PyTP() from DocSupport.autotoc import AutoToC self._autotoc = AutoToC()
class Installer(object): """Install Webware. The _comps attribute is a list of components, each of which is an instance of MiscUtils.PropertiesObject. """ ## Init ## def __init__(self): self._props = PropertiesObject('Properties.py') self._props['dirname'] = '.' self._comps = [] self._htHeader, self._htFooter = self.htHeaderAndFooter() from DocSupport.pytp import PyTP self._pytp = PyTP() from DocSupport.autotoc import AutoToC self._autotoc = AutoToC() ## Debug printing facility ## def printMsg(self, msg): if self._verbose: print ' ' + msg ## Running the installation ## def run(self, verbose=False, passprompt=True, defaultpass='', keepdocs=False): self._verbose = verbose log = [] stdout, stderr = sys.stdout, sys.stderr try: sys.stdout = OutputCatcher(sys.stdout, log) sys.stderr = OutputCatcher(sys.stderr, log) self.printHello() self.clearLogFile() if not self.checkPyVersion() or not self.checkThreading(): return self.detectComponents() self.installDocs(keepdocs) self.backupConfigs() self.copyStartScript() self.compileModules() self.fixPermissions() self.setupWebKitPassword(passprompt, defaultpass) self.printGoodbye() self.writeLogFile(log) finally: sys.stdout, sys.stderr = stdout, stderr def clearLogFile(self): """Remove the install.log file. This file with the logged output will get created at the very end of the installation, provided there are no errors. """ if os.path.exists('install.log'): print print 'Removing log from last installation...' os.remove('install.log') def printHello(self): from time import time, localtime, asctime print print '%(name)s %(versionString)s' % self._props print 'Installer' print self.printKeyValue('Cur Date', asctime(localtime(time()))) self.printKeyValue('Python', sys.version.replace(') [', ')\n[')) self.printKeyValue('Op Sys', os.name) self.printKeyValue('Platform', sys.platform) self.printKeyValue('Cur Dir', os.getcwd()) def checkPyVersion(self, minver=(2, 6)): """Check for minimum required Python version.""" try: ver = sys.version_info[:len(minver)] except AttributeError: ver = (1, ) if ver < minver: print( 'This Release of Webware requires Python %s.\n' 'Your currently used version is Python %s.\n' 'You can download a newer version at: http://www.python.org\n' % ('.'.join(map(str, minver)), '.'.join(map(str, ver)))) response = raw_input( 'You may continue to install, ' 'but Webware may not perform as expected.\n' 'Do you wish to continue with the installation? [yes/no] ') return response[:1].upper() == "Y" return True def checkThreading(self): try: import threading except ImportError: print( 'Webware requires that Python be compiled with threading support.\n' 'This version of Python does not appear to support threading.\n' ) response = raw_input( 'You may continue, ' 'but you will have to run the AppServer with a Python\n' 'interpreter that has threading enabled.\n' 'Do you wish to continue with the installation? [yes/no] ') return response[:1].upper() == "Y" return True def detectComponents(self): print print 'Scanning for components...' dirNames = [ dir for dir in os.listdir(os.curdir) if not dir.startswith('.') and os.path.isdir(dir) ] self._maxCompLen = max(map(len, dirNames)) oldPyVersion = False column = 0 for dirName in sorted(dirNames): propName = dirName + '/Properties.py' try: print dirName.ljust(self._maxCompLen, '.'), except TypeError: print dirName.ljust(self._maxCompLen), if os.path.exists(propName): comp = PropertiesObject(propName) comp['dirname'] = dirName for key in self._props: if key not in comp: comp[key] = self._props[key] if sys.version_info[:3] < comp['requiredPyVersion']: oldPyVersion = True print 'no*', else: self._comps.append(comp) print 'yes', else: print 'no ', if column < 2 and not self._verbose: print ' ', column += 1 else: print column = 0 if column: print if oldPyVersion: print "* some components require a newer Python version" self._comps.sort(key=itemgetter('name')) def setupWebKitPassword(self, prompt, defpass): """Setup a password for WebKit Application server.""" print 'Setting the WebKit password...' print if prompt: print 'Choose a password for the WebKit Application Server.' print 'If you will just press enter without entering anything,' if defpass is None: print 'a password will be automatically generated.' else: print 'the password specified on the command-line will be used.' from getpass import getpass password = getpass() else: if defpass is None: print 'A password will be automatically generated.' else: print 'A password was specified on the command-line.' password = None print 'You can check the password after installation at:' appConfig = 'WebKit/Configs/Application.config' print os.path.normpath(appConfig) if not password: if defpass is None: from string import letters, digits from random import choice password = ''.join(map(choice, [letters + digits] * 8)) else: password = defpass try: # read config file data = open(appConfig).read() except IOError: print 'Error reading Application.config file.' print 'Password not replaced, make sure to edit it by hand.' return # This will search for the construct "'AdminPassword': '******'" # and replace '...' with the content of the 'password' variable: if data.lstrip().startswith('{'): pattern = "('AdminPassword'\s*:)\s*'.*?'" else: # keyword arguments style pattern = "(AdminPassword\\s*=)\\s*['\"].*?['\"]" repl = "\g<1> '%s'" % password.replace( # escape critical characters '\\', '\\\\\\\\').replace("'", "\\\\'").replace('%', '\\\\045') from re import subn data, count = subn(pattern, repl, data) if count != 1: print "Warning:", if count > 1: print "More than one 'AdminPassword' in config file." else: print "'AdminPassword' not found in config file." return try: # write back config file open(appConfig, 'w').write(data) except IOError: print 'Error writing Application.config (probably no permission).' print 'Password not replaced, make sure to edit it by hand.' return print 'Password replaced successfully.' def installDocs(self, keep): self.processHtmlDocFiles() self.processPyTemplateFiles(keep) self.createBrowsableSource() self.createComponentIndex() self.createComponentIndexes(keep) self.createDocContexts() def processHtmlDocFiles(self): print print 'Processing html doc files...' for htmlFile in glob('Docs/*.html'): self.processHtmlDocFile(htmlFile) for comp in self._comps: dir = comp['dirname'] for htmlFile in glob(dir + '/Docs/*.html'): self.processHtmlDocFile(htmlFile) def processPyTemplateFiles(self, keep): print print 'Processing phtml doc files...' if keep: print 'The templates will not be removed.' else: print 'The templates will be removed afterwards.' for inFile in glob('Docs/*.phtml'): if not os.path.splitext(inFile)[0].endswith('OfComponent'): self.processPyTemplateFile(inFile, self._props, keep) for comp in self._comps: dir = comp['dirname'] for inFile in glob(dir + '/Docs/*.phtml'): self.processPyTemplateFile(inFile, comp, keep) def createBrowsableSource(self): """Create HTML docs for class hierarchies, summaries, sources etc.""" print print 'Creating html source, summaries and doc files...' column = 0 for comp in self._comps: dir = comp['dirname'] if self._verbose: print dir, '...' else: try: print dir.ljust(self._maxCompLen, '.'), except TypeError: print dir.ljust(self._maxCompLen), sourceDir = dir + '/Docs/Source' self.makeDir(sourceDir) filesDir = sourceDir + '/Files' self.makeDir(filesDir) summariesDir = sourceDir + '/Summaries' self.makeDir(summariesDir) docsDir = sourceDir + '/Docs' self.makeDir(docsDir) if dir == 'MiddleKit': dir += '/Core' for pyFilename in glob(dir + '/*.py'): self.createHighlightedSource(pyFilename, filesDir) self.createPySummary(pyFilename, summariesDir) self.createPyDocs(pyFilename, docsDir) self.createPyDocs(dir, docsDir) self.createFileList(dir, sourceDir) self.createClassList(dir, sourceDir) if not self._verbose: print "ok", if column < 2: print ' ', column += 1 else: print column = 0 if column: print def createHighlightedSource(self, filename, dir): """Create highlighted HTML source code using py2html.""" from DocSupport import py2html module = os.path.splitext(os.path.basename(filename))[0] targetName = '%s/%s.html' % (dir, module) self.printMsg('Creating %s...' % targetName) stdout = sys.stdout sys.stdout = StringIO() try: py2html.main((None, '-stdout', '-files', filename)) result = sys.stdout.getvalue() finally: sys.stdout = stdout open(targetName, 'w').write(result) def createPySummary(self, filename, dir): """Create an HTML module summary.""" from DocSupport.PySummary import PySummary module = os.path.splitext(os.path.basename(filename))[0] targetName = '%s/%s.html' % (dir, module) self.printMsg('Creating %s...' % targetName) sum = PySummary() sum.readConfig('DocSupport/PySummary.config') sum.readFileNamed(filename) html = sum.html() open(targetName, 'w').write(html) def createPyDocs(self, filename, dir): """Create an HTML module documentation using pydoc.""" import pydoc package, module = os.path.split(filename) module = os.path.splitext(module)[0] if package: module = package.replace('/', '.') + '.' + module targetName = '%s/%s.html' % (dir, module) self.printMsg('Creating %s...' % targetName) saveDir = os.getcwd() os.chdir(dir) try: stdout = sys.stdout sys.stdout = StringIO() try: try: pydoc.writedoc(module) except Exception: pass msg = sys.stdout.getvalue() finally: sys.stdout = stdout finally: os.chdir(saveDir) if msg: self.printMsg(msg) def createFileList(self, filesDir, docsDir): """Create an HTML list of the source files.""" from DocSupport.FileList import FileList name = filesDir.replace('/', '.') self.printMsg('Creating file list of %s...' % name) filelist = FileList(name) filesDir, subDir = (filesDir + '/').split('/', 1) saveDir = os.getcwd() os.chdir(filesDir) try: filelist.readFiles(subDir + '*.py') targetName = docsDir + '/FileList.html' self.printMsg('Creating %s...' % targetName) filelist.printForWeb('../' + targetName) finally: os.chdir(saveDir) def createClassList(self, filesDir, docsDir): """Create an HTML class hierarchy listing of the source files.""" from DocSupport.ClassList import ClassList name = filesDir.replace('/', '.') self.printMsg('Creating class list of %s...' % name) classlist = ClassList(name) filesDir, subDir = (filesDir + '/').split('/', 1) saveDir = os.getcwd() os.chdir(filesDir) try: classlist.readFiles(subDir + '*.py') targetName = docsDir + '/ClassList.html' self.printMsg('Creating %s...' % targetName) classlist.printForWeb(False, '../' + targetName) targetName = docsDir + '/ClassHierarchy.html' self.printMsg('Creating %s...' % targetName) classlist.printForWeb(True, '../' + targetName) finally: os.chdir(saveDir) def createComponentIndex(self): """Create an HTML component index of Webware itself.""" print 'Creating ComponentIndex.html...' ht = [ "<% header('Webware Documentation', 'titlebar'," " 'ComponentIndex.css') %>" ] wr = ht.append wr('<p>Don\'t know where to start? ' 'Try <a href="../WebKit/Docs/index.html">WebKit</a>.</p>') wr('<table class="doc">') wr('<tr class="ComponentHeadings">' '<th>Component</th><th>Status</th><th>Ver</th>' '<th>Py</th><th>Summary</th></tr>') row = 0 for comp in self._comps: comp['nameAsLink'] = ( '<a href=' '"../%(dirname)s/Docs/index.html">%(name)s</a>' % comp) comp['indexRow'] = row + 1 wr('<tr class="ComponentRow%(indexRow)i top">' '<td class="NameVersionCell">' '<span class="Name">%(nameAsLink)s</span></td>' '<td>%(status)s</td>' '<td><span class="Version">%(versionString)s</span></td>' '<td>%(requiredPyVersionString)s</td>' '<td>%(synopsis)s</td></tr>' % comp) row = 1 - row wr('</table>') wr("<% footer() %>") ht = '\n'.join(ht) ht = self.processPyTemplate(ht, self._props) open('Docs/ComponentIndex.html', 'w').write(ht) def createComponentIndexes(self, keep): """Create start page for all components.""" indexfile = 'Docs/indexOfComponent.phtml' if not os.path.exists(indexfile): return print print "Creating index.html for all components..." index = open(indexfile).read() link = '<p><a href="%s">%s</a></p>' for comp in self._comps: comp['webwareVersion'] = self._props['version'] comp['webwareVersionString'] = self._props['versionString'] # Create 'htDocs' as an HTML fragment corresponding to comp['docs'] ht = [] for doc in comp['docs']: ht.append(link % (doc['file'], doc['name'])) ht = ''.join(ht) comp['htDocs'] = ht # Set up release notes ht = [] files = glob(comp['dirname'] + '/Docs/RelNotes-*.html') if files: releaseNotes = [] for filename in files: item = dict(dirname=os.path.basename(filename)) filename = item['dirname'] ver = filename[filename.rfind('-') + 1:filename.rfind('.')] item['name'] = ver if ver == 'X.Y': item['ver'] = ver.split('.') else: i = 0 while i < len(ver) and ver[i] in '.0123456789': i += 1 if i: item['ver'] = map(int, ver[:i].split('.')) releaseNotes.append(item) releaseNotes.sort(key=itemgetter('ver'), reverse=True) for item in releaseNotes: ht.append(link % (item['dirname'], item['name'])) else: ht.append('<p>None</p>') ht = '\n'.join(ht) comp['htReleaseNotes'] = ht # Write file filename = comp['dirname'] + '/Docs/index.html' ht = self.processPyTemplate(index, comp) open(filename, 'w').write(ht) if not keep: os.remove(indexfile) def createDocContexts(self): """Create a WebKit context for every Docs directory.""" print print 'Making all Docs directories browsable via WebKit...' # Place an __init__.py file in every Docs directory docsDirs = ['Docs'] for comp in self._comps: if comp.get('docs'): docsDir = comp['dirname'] + '/Docs' if os.path.isdir(docsDir): docsDirs.append(docsDir) for docsDir in docsDirs: initFile = docsDir + '/__init__.py' if not os.path.exists(initFile): open(initFile, 'w').write('# this can be browsed as a Webware context\n') # Copy favicon to the default context open('WebKit/Examples/favicon.ico', 'wb').write(open('Docs/favicon.ico', 'rb').read()) def backupConfigs(self): """Copy *.config to *.config.default, if they don't already exist. This allows the user to always go back to the default config file if needed (for troubleshooting for example). """ print print 'Creating backups of original config files...' self._backupConfigs(os.curdir) def _backupConfigs(self, dir): for filename in os.listdir(dir): fullPath = os.path.join(dir, filename) if os.path.isdir(fullPath): self._backupConfigs(fullPath) elif (not filename.startswith('.') and os.path.splitext(filename)[1] == '.config'): self.printMsg(fullPath) backupPath = fullPath + '.default' if not os.path.exists(backupPath): open(backupPath, 'wb').write(open(fullPath, 'rb').read()) def copyStartScript(self): """Copy the most appropriate start script to WebKit/webkit.""" if os.name == 'posix': print print 'Copying start script...', ex = os.path.exists if (ex('/etc/rc.status') and ex('/sbin/startproc') and ex('/sbin/killproc')): s = 'SUSE' elif (ex('/etc/init.d/functions') or ex('/etc/rc.d/init.d/functions')): s = 'RedHat' elif ex('/sbin/start-stop-daemon'): s = 'Debian' elif ex('/etc/rc.subr'): s = 'NetBSD' else: s = 'Generic' print s # Copy start script: s = 'WebKit/StartScripts/SysV/' + s t = 'WebKit/webkit' open(t, 'wb').write(open(s, 'rb').read()) def compileModules(self, force=True): """Compile modules in all installed componentes.""" from compileall import compile_dir print print 'Byte compiling all modules...' for comp in self._comps: dir = os.path.abspath(comp['dirname']) compile_dir(dir, force=force, quiet=True) def fixPermissions(self): if os.name == 'posix': print print 'Setting permissions on CGI scripts...' for comp in self._comps: for filename in glob(comp['dirname'] + '/*.cgi'): cmd = 'chmod a+rx ' + filename self.printMsg(cmd) os.system(cmd) print 'Setting permission on start script...' cmd = 'chmod a+rx WebKit/webkit' self.printMsg(cmd) os.system(cmd) def printGoodbye(self): print ''' Installation looks successful. Welcome to Webware! You can already try out the WebKit application server. Start it with "WebKit%sAppServer" and point your browser to "http://localhost:8080". Browsable documentation is available in the Docs folders. You can use "Docs%sindex.html" as the main entry point. Installation is finished.''' % ((os.sep, ) * 2) def writeLogFile(self, log): """Write the logged output to the install.log file.""" open('install.log', 'w').write(''.join(log)) ## Self utility ## def printKeyValue(self, key, value): """Print a key/value pair.""" value = value.splitlines() v = value.pop(0) print '%12s: %s' % (key, v) for v in value: print '%14s%s' % ('', v) def makeDir(self, dirName): """Create a directory.""" if not os.path.exists(dirName): self.printMsg('Making %s...' % dirName) os.makedirs(dirName) def htHeaderAndFooter(self): """Return header and footer from HTML template.""" template = open('Docs/Template.html').read() return template.split('\n<!-- page content -->\n', 1) def processHtmlDocFile(self, htmlFile): """Process an HTML file.""" txtFile = os.path.splitext(htmlFile)[0] + '.txt' if os.path.exists(txtFile): # A text file with the same name exists: page = open(htmlFile).read() if ('<meta name="generator" content="Docutils' in page and '<h1 class="title">' in page): # This has obvisouly been created with Docutils; modify it # to match style, header and footer of all the other docs. page = page.replace('<h1 class="title">', '<h1 class="header">') page = page.replace('</body>\n</html>', self._htFooter) self.printMsg('Modifying %s...' % htmlFile) open(htmlFile, 'w').write(page) def processPyTemplateFile(self, inFile, props, keep): """Process a Python template file.""" page = open(inFile).read() page = self.processPyTemplate(page, props) outFile = os.path.splitext(inFile)[0] + '.html' self.printMsg('Creating %s...' % outFile) open(outFile, 'w').write(page) if not keep: os.remove(inFile) # remove template def processPyTemplate(self, input, props): """Process a Python template.""" global scope def header(title, titleclass=None, style=None): """Get the header of a document.""" if not titleclass: titleclass = 'header' titleclass = ' class="%s"' % titleclass link = '<link rel="stylesheet" href="%s" type="text/css">' stylesheets = ['Doc.css'] if style and style.endswith('.css'): stylesheets.append(style) style = None css = [] for s in stylesheets: if not scope['dirname'].startswith('.'): s = '../../Docs/' + s s = link % s css.append(s) if style: css.extend(('<style type="text/css">', '<!--', style, '-->', '</style>')) css = '\n'.join(css) return scope['htHeader'] % locals() def footer(): """Get the footer of a document.""" return scope['htFooter'] scope = props.copy() scope.update(header=header, htHeader=self._htHeader, footer=footer, htFooter=self._htFooter) return self._autotoc.process(self._pytp.process(input, scope))
class Installer(object): """Install Webware. The _comps attribute is a list of components, each of which is an instance of MiscUtils.PropertiesObject. """ ## Init ## def __init__(self): self._props = PropertiesObject('Properties.py') self._props['dirname'] = '.' self._comps = [] self._htHeader, self._htFooter = self.htHeaderAndFooter() from DocSupport.pytp import PyTP self._pytp = PyTP() from DocSupport.autotoc import AutoToC self._autotoc = AutoToC() ## Debug printing facility ## def _nop (self, msg): pass def _printMsg (self, msg): print ' ' + msg ## Running the installation ## def run(self, verbose=False, passprompt=True, defaultpass='', keepdocs=False): self._verbose = verbose self.printMsg = verbose and self._printMsg or self._nop log = [] stdout, stderr = sys.stdout, sys.stderr try: sys.stdout = OutputCatcher(sys.stdout, log) sys.stderr = OutputCatcher(sys.stderr, log) self.printHello() self.clearLogFile() if not self.checkPyVersion() or not self.checkThreading(): return self.detectComponents() self.installDocs(keepdocs) self.backupConfigs() self.copyStartScript() self.compileModules() self.fixPermissions() self.setupWebKitPassword(passprompt, defaultpass) self.printGoodbye() self.writeLogFile(log) finally: sys.stdout, sys.stderr = stdout, stderr def clearLogFile(self): """Remove the install.log file. This file with the logged output will get created at the very end of the installation, provided there are no errors. """ if os.path.exists('install.log'): print print 'Removing log from last installation...' os.remove('install.log') def printHello(self): from time import time, localtime, asctime print print '%(name)s %(versionString)s' % self._props print 'Installer' print self.printKeyValue('Cur Date', asctime(localtime(time()))) self.printKeyValue('Python', sys.version.replace(') [', ')\n[')) self.printKeyValue('Op Sys', os.name) self.printKeyValue('Platform', sys.platform) self.printKeyValue('Cur Dir', os.getcwd()) def checkPyVersion(self, minver=(2, 4)): """Check for minimum required Python version.""" try: ver = sys.version_info[:len(minver)] except AttributeError: ver = (1,) if ver < minver: print ('This Release of Webware requires Python %s.\n' 'Your currently used version is Python %s.\n' 'You can download a newer version at: http://www.python.org\n' % ('.'.join(map(str, minver)), '.'.join(map(str, ver)))) response = raw_input('You may continue to install, ' 'but Webware may not perform as expected.\n' 'Do you wish to continue with the installation? [yes/no] ') return response[:1].upper() == "Y" return True def checkThreading(self): try: import threading except ImportError: print ('Webware requires that Python be compiled with threading support.\n' 'This version of Python does not appear to support threading.\n') response = raw_input('You may continue, ' 'but you will have to run the AppServer with a Python\n' 'interpreter that has threading enabled.\n' 'Do you wish to continue with the installation? [yes/no] ') return response[:1].upper() == "Y" return True def detectComponents(self): print print 'Scanning for components...' dirNames = [dir for dir in os.listdir(os.curdir) if not dir.startswith('.') and os.path.isdir(dir)] self._maxCompLen = max(map(len, dirNames)) oldPyVersion = False column = 0 for dirName in sorted(dirNames): propName = dirName + '/Properties.py' try: print dirName.ljust(self._maxCompLen, '.'), except TypeError: print dirName.ljust(self._maxCompLen), if os.path.exists(propName): comp = PropertiesObject(propName) comp['dirname'] = dirName for key in self._props: if key not in comp: comp[key] = self._props[key] if sys.version_info[:3] < comp['requiredPyVersion']: oldPyVersion = True print 'no*', else: self._comps.append(comp) print 'yes', else: print 'no ', if column < 2 and not self._verbose: print ' ', column += 1 else: print column = 0 if column: print if oldPyVersion: print "* some components require a newer Python version" self._comps.sort(key=itemgetter('name')) def setupWebKitPassword(self, prompt, defpass): """Setup a password for WebKit Application server.""" print 'Setting the WebKit password...' print if prompt: print 'Choose a password for the WebKit Application Server.' print 'If you will just press enter without entering anything,' if defpass is None: print 'a password will be automatically generated.' else: print 'the password specified on the command-line will be used.' from getpass import getpass password = getpass() else: if defpass is None: print 'A password will be automatically generated.' else: print 'A password was specified on the command-line.' password = None print 'You can check the password after installation at:' appConfig = 'WebKit/Configs/Application.config' print os.path.normpath(appConfig) if not password: if defpass is None: from string import letters, digits from random import choice password = ''.join(map(choice, [letters + digits]*8)) else: password = defpass try: # read config file data = open(appConfig).read() except IOError: print 'Error reading Application.config file.' print 'Password not replaced, make sure to edit it by hand.' return # This will search for the construct "'AdminPassword': '******'" # and replace '...' with the content of the 'password' variable: if data.lstrip().startswith('{'): pattern = "('AdminPassword'\s*:)\s*'.*?'" else: # keyword arguments style pattern = "(AdminPassword\\s*=)\\s*['\"].*?['\"]" repl = "\g<1> '%s'" % password.replace( # escape critical characters '\\', '\\\\\\\\').replace("'", "\\\\'").replace('%', '\\\\045') from re import subn data, count = subn(pattern, repl, data) if count != 1: print "Warning:", if count > 1: print "More than one 'AdminPassword' in config file." else: print "'AdminPassword' not found in config file." return try: # write back config file open(appConfig, 'w').write(data) except IOError: print 'Error writing Application.config (probably no permission).' print 'Password not replaced, make sure to edit it by hand.' return print 'Password replaced successfully.' def installDocs(self, keep): self.processHtmlDocFiles() self.processPyTemplateFiles(keep) self.createBrowsableSource() self.createComponentIndex() self.createComponentIndexes(keep) self.createDocContexts() def processHtmlDocFiles(self): print print 'Processing html doc files...' for htmlFile in glob('Docs/*.html'): self.processHtmlDocFile(htmlFile) for comp in self._comps: dir = comp['dirname'] for htmlFile in glob(dir + '/Docs/*.html'): self.processHtmlDocFile(htmlFile) def processPyTemplateFiles(self, keep): print print 'Processing phtml doc files...' if keep: print 'The templates will not be removed.' else: print 'The templates will be removed afterwards.' for inFile in glob('Docs/*.phtml'): if not os.path.splitext(inFile)[0].endswith('OfComponent'): self.processPyTemplateFile(inFile, self._props, keep) for comp in self._comps: dir = comp['dirname'] for inFile in glob(dir + '/Docs/*.phtml'): self.processPyTemplateFile(inFile, comp, keep) def createBrowsableSource(self): """Create HTML docs for class hierarchies, summaries, sources etc.""" print print 'Creating html source, summaries and doc files...' column = 0 for comp in self._comps: dir = comp['dirname'] if self._verbose: print dir, '...' else: try: print dir.ljust(self._maxCompLen, '.'), except TypeError: print dir.ljust(self._maxCompLen), sourceDir = dir + '/Docs/Source' self.makeDir(sourceDir) filesDir = sourceDir + '/Files' self.makeDir(filesDir) summariesDir = sourceDir + '/Summaries' self.makeDir(summariesDir) docsDir = sourceDir + '/Docs' self.makeDir(docsDir) if dir == 'MiddleKit': dir += '/Core' for pyFilename in glob(dir + '/*.py'): self.createHighlightedSource(pyFilename, filesDir) self.createPySummary(pyFilename, summariesDir) self.createPyDocs(pyFilename, docsDir) self.createPyDocs(dir, docsDir) self.createFileList(dir, sourceDir) self.createClassList(dir, sourceDir) if not self._verbose: print "ok", if column < 2: print ' ', column += 1 else: print column = 0 if column: print def createHighlightedSource(self, filename, dir): """Create highlighted HTML source code using py2html.""" from DocSupport import py2html module = os.path.splitext(os.path.basename(filename))[0] targetName = '%s/%s.html' % (dir, module) self.printMsg('Creating %s...' % targetName) stdout = sys.stdout sys.stdout = StringIO() try: py2html.main((None, '-stdout', '-files', filename)) result = sys.stdout.getvalue() finally: sys.stdout = stdout open(targetName, 'w').write(result) def createPySummary(self, filename, dir): """Create an HTML module summary.""" from DocSupport.PySummary import PySummary module = os.path.splitext(os.path.basename(filename))[0] targetName = '%s/%s.html' % (dir, module) self.printMsg('Creating %s...' % targetName) sum = PySummary() sum.readConfig('DocSupport/PySummary.config') sum.readFileNamed(filename) html = sum.html() open(targetName, 'w').write(html) def createPyDocs(self, filename, dir): """Create an HTML module documentation using pydoc.""" import pydoc package, module = os.path.split(filename) module = os.path.splitext(module)[0] if package: module = package.replace('/', '.') + '.' + module targetName = '%s/%s.html' % (dir, module) self.printMsg('Creating %s...' % targetName) saveDir = os.getcwd() os.chdir(dir) try: stdout = sys.stdout sys.stdout = StringIO() try: try: pydoc.writedoc(module) except Exception: pass msg = sys.stdout.getvalue() finally: sys.stdout = stdout finally: os.chdir(saveDir) if msg: self.printMsg(msg) def createFileList(self, filesDir, docsDir): """Create an HTML list of the source files.""" from DocSupport.FileList import FileList name = filesDir.replace('/', '.') self.printMsg('Creating file list of %s...' % name) filelist = FileList(name) filesDir, subDir = (filesDir + '/').split('/', 1) saveDir = os.getcwd() os.chdir(filesDir) try: filelist.readFiles(subDir + '*.py') targetName = docsDir + '/FileList.html' self.printMsg('Creating %s...' % targetName) filelist.printForWeb('../' + targetName) finally: os.chdir(saveDir) def createClassList(self, filesDir, docsDir): """Create an HTML class hierarchy listing of the source files.""" from DocSupport.ClassList import ClassList name = filesDir.replace('/', '.') self.printMsg('Creating class list of %s...' % name) classlist = ClassList(name) filesDir, subDir = (filesDir + '/').split('/', 1) saveDir = os.getcwd() os.chdir(filesDir) try: classlist.readFiles(subDir + '*.py') targetName = docsDir + '/ClassList.html' self.printMsg('Creating %s...' % targetName) classlist.printForWeb(False, '../' + targetName) targetName = docsDir + '/ClassHierarchy.html' self.printMsg('Creating %s...' % targetName) classlist.printForWeb(True, '../' + targetName) finally: os.chdir(saveDir) def createComponentIndex(self): """Create an HTML component index of Webware itself.""" print 'Creating ComponentIndex.html...' ht = ["<% header('Webware Documentation', 'titlebar'," " 'ComponentIndex.css') %>"] wr = ht.append wr('<p>Don\'t know where to start? ' 'Try <a href="../WebKit/Docs/index.html">WebKit</a>.</p>') wr('<table align="center" border="0" ' 'cellpadding="2" cellspacing="2" width="100%">') wr('<tr class="ComponentHeadings">' '<th>Component</th><th>Status</th><th>Ver</th>' '<th>Py</th><th>Summary</th></tr>') row = 0 for comp in self._comps: comp['nameAsLink'] = ('<a href=' '"../%(dirname)s/Docs/index.html">%(name)s</a>' % comp) comp['indexRow'] = row + 1 wr('<tr valign="top" class="ComponentRow%(indexRow)i">' '<td class="NameVersionCell">' '<span class="Name">%(nameAsLink)s</span></td>' '<td>%(status)s</td>' '<td><span class="Version">%(versionString)s</span></td>' '<td>%(requiredPyVersionString)s</td>' '<td>%(synopsis)s</td></tr>' % comp) row = 1 - row wr('</table>') wr("<% footer() %>") ht = '\n'.join(ht) ht = self.processPyTemplate(ht, self._props) open('Docs/ComponentIndex.html', 'w').write(ht) def createComponentIndexes(self, keep): """Create start page for all components.""" indexfile = 'Docs/indexOfComponent.phtml' if not os.path.exists(indexfile): return print print "Creating index.html for all components..." index = open(indexfile).read() link = '<p><a href="%s">%s</a></p>' for comp in self._comps: comp['webwareVersion'] = self._props['version'] comp['webwareVersionString'] = self._props['versionString'] # Create 'htDocs' as an HTML fragment corresponding to comp['docs'] ht = [] for doc in comp['docs']: ht.append(link % (doc['file'], doc['name'])) ht = ''.join(ht) comp['htDocs'] = ht # Set up release notes ht = [] files = glob(comp['dirname'] + '/Docs/RelNotes-*.html') if files: releaseNotes = [] for filename in files: item = dict(dirname=os.path.basename(filename)) filename = item['dirname'] ver = filename[ filename.rfind('-') + 1 : filename.rfind('.')] item['name'] = ver if ver == 'X.Y': item['ver'] = ver.split('.') else: i = 0 while i < len(ver) and ver[i] in '.0123456789': i += 1 if i: item['ver'] = map(int, ver[:i].split('.')) releaseNotes.append(item) releaseNotes.sort(key=itemgetter('ver'), reverse=True) for item in releaseNotes: ht.append(link % (item['dirname'], item['name'])) else: ht.append('<p>None</p>') ht = '\n'.join(ht) comp['htReleaseNotes'] = ht # Write file filename = comp['dirname'] + '/Docs/index.html' ht = self.processPyTemplate(index, comp) open(filename, 'w').write(ht) if not keep: os.remove(indexfile) def createDocContexts(self): """Create a WebKit context for every Docs directory.""" print print 'Making all Docs directories browsable via WebKit...' # Place an __init__.py file in every Docs directory docsDirs = ['Docs'] for comp in self._comps: if comp.get('docs'): docsDir = comp['dirname'] + '/Docs' if os.path.isdir(docsDir): docsDirs.append(docsDir) for docsDir in docsDirs: initFile = docsDir + '/__init__.py' if not os.path.exists(initFile): open(initFile, 'w').write( '# this can be browsed as a Webware context\n') # Copy favicon to the default context open('WebKit/Examples/favicon.ico', 'wb').write( open('Docs/favicon.ico', 'rb').read()) def backupConfigs(self): """Copy *.config to *.config.default, if they don't already exist. This allows the user to always go back to the default config file if needed (for troubleshooting for example). """ print print 'Creating backups of original config files...' self._backupConfigs(os.curdir) def _backupConfigs(self, dir): for filename in os.listdir(dir): fullPath = os.path.join(dir, filename) if os.path.isdir(fullPath): self._backupConfigs(fullPath) elif (not filename.startswith('.') and os.path.splitext(filename)[1] == '.config'): self.printMsg(fullPath) backupPath = fullPath + '.default' if not os.path.exists(backupPath): open(backupPath, 'wb').write(open(fullPath, 'rb').read()) def copyStartScript(self): """Copy the most appropriate start script to WebKit/webkit.""" if os.name == 'posix': print print 'Copying start script...', ex = os.path.exists if (ex('/etc/rc.status') and ex('/sbin/startproc') and ex('/sbin/killproc')): s = 'SUSE' elif (ex('/etc/init.d/functions') or ex('/etc/rc.d/init.d/functions')): s = 'RedHat' elif ex('/sbin/start-stop-daemon'): s = 'Debian' elif ex('/etc/rc.subr'): s = 'NetBSD' else: s = 'Generic' print s # Copy start script: s = 'WebKit/StartScripts/' + s t = 'WebKit/webkit' open(t, 'wb').write(open(s, 'rb').read()) def compileModules(self, force=True): """Compile modules in all installed componentes.""" from compileall import compile_dir print print 'Byte compiling all modules...' for comp in self._comps: dir = os.path.abspath(comp['dirname']) compile_dir(dir, force=force, quiet=True) def fixPermissions(self): if os.name == 'posix': print print 'Setting permissions on CGI scripts...' for comp in self._comps: for filename in glob(comp['dirname'] + '/*.cgi'): cmd = 'chmod a+rx ' + filename self.printMsg(cmd) os.system(cmd) print 'Setting permission on start script...' cmd = 'chmod a+rx WebKit/webkit' self.printMsg(cmd) os.system(cmd) def printGoodbye(self): print ''' Installation looks successful. Welcome to Webware! You can already try out the WebKit application server. Start it with "WebKit%sAppServer" and point your browser to "http://localhost:8080". Browsable documentation is available in the Docs folders. You can use "Docs%sindex.html" as the main entry point. Installation is finished.''' % ((os.sep,)*2) def writeLogFile(self, log): """Write the logged output to the install.log file.""" open('install.log', 'w').write(''.join(log)) ## Self utility ## def printKeyValue(self, key, value): """Print a key/value pair.""" value = value.splitlines() v = value.pop(0) print '%12s: %s' % (key, v) for v in value: print '%14s%s' % ('', v) def makeDir(self, dirName): """Create a directory.""" if not os.path.exists(dirName): self.printMsg('Making %s...' % dirName) os.makedirs(dirName) def htHeaderAndFooter(self): """Return header and footer from HTML template.""" template = open('Docs/Template.html').read() return template.split('\n<!-- page content -->\n', 1) def processHtmlDocFile(self, htmlFile): """Process an HTML file.""" txtFile = os.path.splitext(htmlFile)[0] + '.txt' if os.path.exists(txtFile): # A text file with the same name exists: page = open(htmlFile).read() if ('<meta name="generator" content="Docutils' in page and '<h1 class="title">' in page): # This has obvisouly been created with Docutils; modify it # to match style, header and footer of all the other docs. page = page.replace('<h1 class="title">', '<h1 class="header">') page = page.replace('</body>\n</html>', self._htFooter) self.printMsg('Modifying %s...' % htmlFile) open(htmlFile, 'w').write(page) def processPyTemplateFile(self, inFile, props, keep): """Process a Python template file.""" page = open(inFile).read() page = self.processPyTemplate(page, props) outFile = os.path.splitext(inFile)[0] + '.html' self.printMsg('Creating %s...' % outFile) open(outFile, 'w').write(page) if not keep: os.remove(inFile) # remove template def processPyTemplate(self, input, props): """Process a Python template.""" global scope def header(title, titleclass=None, style=None): """Get the header of a document.""" if not titleclass: titleclass = 'header' titleclass = ' class="%s"' % titleclass link = '<link rel="stylesheet" href="%s" type="text/css">' stylesheets = ['Doc.css'] if style and style.endswith('.css'): stylesheets.append(style) style = None css = [] for s in stylesheets: if not scope['dirname'].startswith('.'): s = '../../Docs/' + s s = link % s css.append(s) if style: css.extend(('<style type="text/css">', '<!--', style, '-->', '</style>')) css = '\n'.join(css) return scope['htHeader'] % locals() def footer(): """Get the footer of a document.""" return scope['htFooter'] scope = props.copy() scope.update(header=header, htHeader=self._htHeader, footer=footer, htFooter=self._htFooter) return self._autotoc.process(self._pytp.process(input, scope))