2008-08-16 23:26:59 +08:00
|
|
|
import py
|
|
|
|
from py.__.misc.rest import convert_rest_html, strip_html_header
|
|
|
|
from py.__.misc.difftime import worded_time
|
|
|
|
|
|
|
|
html = py.xml.html
|
|
|
|
|
2009-07-21 00:54:08 +08:00
|
|
|
class css:
|
|
|
|
#pagetitle = "pagetitle"
|
|
|
|
contentspace = "contentspace"
|
|
|
|
menubar = "menubar"
|
|
|
|
navspace = "navspace"
|
|
|
|
versioninfo = "versioninfo"
|
|
|
|
|
2008-08-16 23:26:59 +08:00
|
|
|
class Page(object):
|
|
|
|
doctype = ('<!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN"'
|
|
|
|
' "http://www.w3.org/TR/xhtml1/DTD/xhtml1-transitional.dtd">\n')
|
2009-08-04 18:00:04 +08:00
|
|
|
googlefragment = """
|
|
|
|
<script type="text/javascript">
|
|
|
|
var gaJsHost = (("https:" == document.location.protocol) ? "https://ssl." : "http://www.");
|
|
|
|
document.write(unescape("%3Cscript src='" + gaJsHost + "google-analytics.com/ga.js' type='text/javascript'%3E%3C/script%3E"));
|
|
|
|
</script>
|
|
|
|
<script type="text/javascript">
|
|
|
|
try {
|
|
|
|
var pageTracker = _gat._getTracker("UA-7597274-3");
|
|
|
|
pageTracker._trackPageview();
|
|
|
|
} catch(err) {}</script>
|
|
|
|
"""
|
2008-08-16 23:26:59 +08:00
|
|
|
|
|
|
|
def __init__(self, project, title, targetpath, stylesheeturl=None,
|
|
|
|
type="text/html", encoding="ISO-8859-1"):
|
|
|
|
self.project = project
|
|
|
|
self.title = project.prefix_title + title
|
|
|
|
self.targetpath = targetpath
|
|
|
|
self.stylesheeturl = stylesheeturl
|
|
|
|
self.type = type
|
|
|
|
self.encoding = encoding
|
|
|
|
|
|
|
|
self.body = html.body()
|
|
|
|
self.head = html.head()
|
|
|
|
self._root = html.html(self.head, self.body)
|
|
|
|
self.fill()
|
|
|
|
|
2009-07-21 00:54:08 +08:00
|
|
|
def a_href(self, name, url, **kwargs):
|
|
|
|
return html.a(name, class_="menu", href=url, **kwargs)
|
2009-04-13 21:58:26 +08:00
|
|
|
|
2008-08-16 23:26:59 +08:00
|
|
|
def a_docref(self, name, relhtmlpath):
|
2009-02-27 18:18:27 +08:00
|
|
|
docpath = self.project.docpath
|
2009-08-19 01:04:57 +08:00
|
|
|
return html.div(html.a(name, class_="menu",
|
2008-08-16 23:26:59 +08:00
|
|
|
href=relpath(self.targetpath.strpath,
|
2009-08-19 01:04:57 +08:00
|
|
|
docpath.join(relhtmlpath).strpath)))
|
2008-08-16 23:26:59 +08:00
|
|
|
|
|
|
|
def a_apigenref(self, name, relhtmlpath):
|
2009-02-27 18:18:27 +08:00
|
|
|
apipath = self.project.apigenpath
|
2008-08-16 23:26:59 +08:00
|
|
|
return html.a(name, class_="menu",
|
|
|
|
href=relpath(self.targetpath.strpath,
|
|
|
|
apipath.join(relhtmlpath).strpath))
|
|
|
|
|
|
|
|
def fill_menubar(self):
|
|
|
|
items = [
|
2009-08-19 01:04:57 +08:00
|
|
|
self.a_docref("install", "download.html"),
|
|
|
|
self.a_docref("contact", "contact.html"),
|
2009-08-20 03:02:41 +08:00
|
|
|
self.a_docref("changelog", "changelog.html"),
|
2009-08-19 01:04:57 +08:00
|
|
|
self.a_docref("faq", "faq.html"),
|
|
|
|
html.div(
|
|
|
|
html.h3("py.test:"),
|
|
|
|
self.a_docref("doc index", "test/index.html"),
|
|
|
|
self.a_docref("features", "test/features.html"),
|
|
|
|
self.a_docref("quickstart", "test/quickstart.html"),
|
|
|
|
self.a_docref("tutorials", "test/talks.html"),
|
|
|
|
self.a_docref("plugins", "test/plugin/index.html"),
|
|
|
|
self.a_docref("funcargs", "test/funcargs.html"),
|
|
|
|
self.a_docref("customize", "test/customize.html"),
|
|
|
|
),
|
|
|
|
html.div(
|
|
|
|
html.h3("supporting APIs:"),
|
|
|
|
self.a_docref("pylib index", "index.html"),
|
|
|
|
self.a_docref("py.execnet", "execnet.html"),
|
|
|
|
self.a_docref("py.path", "path.html"),
|
|
|
|
self.a_docref("py.code", "code.html"),
|
|
|
|
)
|
2009-07-21 00:54:08 +08:00
|
|
|
#self.a_docref("py.code", "code.html"),
|
2009-03-23 22:01:13 +08:00
|
|
|
#self.a_apigenref("api", "api/index.html"),
|
|
|
|
#self.a_apigenref("source", "source/index.html"),
|
2009-04-13 21:58:26 +08:00
|
|
|
#self.a_href("source", "http://bitbucket.org/hpk42/py-trunk/src/"),
|
2008-08-16 23:26:59 +08:00
|
|
|
]
|
2009-07-21 00:54:08 +08:00
|
|
|
self.menubar = html.div(id=css.menubar, *[
|
|
|
|
html.div(item) for item in items])
|
|
|
|
version = py.version
|
|
|
|
self.menubar.insert(0,
|
|
|
|
html.div("%s" % (py.version), style="font-style: italic;")
|
|
|
|
)
|
|
|
|
#self.a_href("%s-%s" % (self.title, py.version),
|
|
|
|
# "http://pypi.python.org/pypi/py/%s" % version,
|
|
|
|
#id="versioninfo",
|
2008-08-16 23:26:59 +08:00
|
|
|
|
|
|
|
def fill(self):
|
|
|
|
content_type = "%s;charset=%s" %(self.type, self.encoding)
|
|
|
|
self.head.append(html.title(self.title))
|
|
|
|
self.head.append(html.meta(name="Content-Type", content=content_type))
|
|
|
|
if self.stylesheeturl:
|
|
|
|
self.head.append(
|
|
|
|
html.link(href=self.stylesheeturl,
|
|
|
|
media="screen", rel="stylesheet",
|
|
|
|
type="text/css"))
|
|
|
|
self.fill_menubar()
|
|
|
|
|
2009-07-21 00:54:08 +08:00
|
|
|
self.body.append(html.div(
|
|
|
|
self.project.logo,
|
|
|
|
self.menubar,
|
|
|
|
id=css.navspace,
|
|
|
|
))
|
|
|
|
|
|
|
|
#self.body.append(html.div(self.title, id=css.pagetitle))
|
|
|
|
self.contentspace = html.div(id=css.contentspace)
|
2008-08-16 23:26:59 +08:00
|
|
|
self.body.append(self.contentspace)
|
|
|
|
|
|
|
|
def unicode(self, doctype=True):
|
|
|
|
page = self._root.unicode()
|
2009-08-04 18:00:04 +08:00
|
|
|
page = page.replace("</body>", self.googlefragment + "</body>")
|
2008-08-16 23:26:59 +08:00
|
|
|
if doctype:
|
|
|
|
return self.doctype + page
|
|
|
|
else:
|
|
|
|
return page
|
|
|
|
|
|
|
|
class PyPage(Page):
|
|
|
|
def get_menubar(self):
|
|
|
|
menubar = super(PyPage, self).get_menubar()
|
|
|
|
# base layout
|
|
|
|
menubar.append(
|
|
|
|
html.a("issue", href="https://codespeak.net/issue/py-dev/",
|
|
|
|
class_="menu"),
|
|
|
|
)
|
|
|
|
return menubar
|
|
|
|
|
|
|
|
|
|
|
|
def getrealname(username):
|
|
|
|
try:
|
|
|
|
import uconf
|
|
|
|
except ImportError:
|
|
|
|
return username
|
|
|
|
try:
|
|
|
|
user = uconf.system.User(username)
|
|
|
|
except KeyboardInterrupt:
|
|
|
|
raise
|
|
|
|
try:
|
|
|
|
return user.realname or username
|
|
|
|
except KeyError:
|
|
|
|
return username
|
|
|
|
|
|
|
|
|
|
|
|
class Project:
|
|
|
|
mydir = py.magic.autopath().dirpath()
|
|
|
|
title = "py lib"
|
|
|
|
prefix_title = "" # we have a logo already containing "py lib"
|
|
|
|
encoding = 'latin1'
|
|
|
|
logo = html.div(
|
|
|
|
html.a(
|
2009-07-21 00:54:08 +08:00
|
|
|
html.img(alt="py lib", id='pyimg', height=114/2, width=154/2,
|
2008-08-16 23:26:59 +08:00
|
|
|
src="http://codespeak.net/img/pylib.png"),
|
2009-07-21 00:54:08 +08:00
|
|
|
href="http://pylib.org"))
|
2008-08-16 23:26:59 +08:00
|
|
|
Page = PyPage
|
|
|
|
|
2009-02-27 18:18:27 +08:00
|
|
|
def __init__(self, sourcepath=None):
|
|
|
|
if sourcepath is None:
|
|
|
|
sourcepath = self.mydir
|
|
|
|
self.setpath(sourcepath)
|
|
|
|
|
|
|
|
def setpath(self, sourcepath, docpath=None,
|
|
|
|
apigenpath=None, stylesheet=None):
|
|
|
|
self.sourcepath = sourcepath
|
|
|
|
if docpath is None:
|
|
|
|
docpath = sourcepath
|
|
|
|
self.docpath = docpath
|
|
|
|
if apigenpath is None:
|
|
|
|
apigenpath = docpath
|
|
|
|
self.apigenpath = apigenpath
|
|
|
|
if stylesheet is None:
|
|
|
|
p = sourcepath.join("style.css")
|
|
|
|
if p.check():
|
|
|
|
self.stylesheet = p
|
|
|
|
else:
|
|
|
|
self.stylesheet = None
|
|
|
|
else:
|
2009-03-06 18:59:02 +08:00
|
|
|
p = sourcepath.join(stylesheet)
|
2009-02-27 18:18:27 +08:00
|
|
|
if p.check():
|
|
|
|
stylesheet = p
|
|
|
|
self.stylesheet = stylesheet
|
2009-03-06 18:59:02 +08:00
|
|
|
#assert self.stylesheet
|
2009-02-27 18:18:27 +08:00
|
|
|
self.apigen_relpath = relpath(
|
|
|
|
self.docpath.strpath + '/', self.apigenpath.strpath + '/')
|
2008-08-16 23:26:59 +08:00
|
|
|
|
|
|
|
def get_content(self, txtpath, encoding):
|
|
|
|
return unicode(txtpath.read(), encoding)
|
|
|
|
|
|
|
|
def get_htmloutputpath(self, txtpath):
|
2009-02-27 18:18:27 +08:00
|
|
|
reloutputpath = txtpath.new(ext='.html').relto(self.sourcepath)
|
|
|
|
return self.docpath.join(reloutputpath)
|
2008-08-16 23:26:59 +08:00
|
|
|
|
|
|
|
def process(self, txtpath):
|
|
|
|
encoding = self.encoding
|
|
|
|
content = self.get_content(txtpath, encoding)
|
|
|
|
outputpath = self.get_htmloutputpath(txtpath)
|
|
|
|
|
|
|
|
stylesheet = self.stylesheet
|
2009-02-27 18:18:27 +08:00
|
|
|
if isinstance(stylesheet, py.path.local):
|
|
|
|
if not self.docpath.join(stylesheet.basename).check():
|
2008-08-16 23:26:59 +08:00
|
|
|
docpath.ensure(dir=True)
|
|
|
|
stylesheet.copy(docpath)
|
|
|
|
stylesheet = relpath(outputpath.strpath,
|
2009-02-27 18:18:27 +08:00
|
|
|
self.docpath.join(stylesheet.basename).strpath)
|
2008-08-16 23:26:59 +08:00
|
|
|
|
|
|
|
content = convert_rest_html(content, txtpath,
|
|
|
|
stylesheet=stylesheet, encoding=encoding)
|
|
|
|
content = strip_html_header(content, encoding=encoding)
|
|
|
|
|
2009-07-21 00:54:08 +08:00
|
|
|
title = txtpath.purebasename
|
|
|
|
if txtpath.dirpath().basename == "test":
|
|
|
|
title = "py.test " + title
|
|
|
|
# title = "[%s] %s" % (txtpath.purebasename, py.version)
|
2009-07-15 03:17:13 +08:00
|
|
|
page = self.Page(self, title,
|
2008-08-16 23:26:59 +08:00
|
|
|
outputpath, stylesheeturl=stylesheet)
|
|
|
|
|
|
|
|
try:
|
2009-07-15 03:17:13 +08:00
|
|
|
modified = py.process.cmdexec(
|
2009-07-21 00:54:08 +08:00
|
|
|
"hg tip --template 'modified {date|shortdate}'"
|
2009-07-15 03:17:13 +08:00
|
|
|
)
|
|
|
|
except py.process.cmdexec.Error:
|
2008-08-16 23:26:59 +08:00
|
|
|
modified = " "
|
|
|
|
|
2009-07-21 00:54:08 +08:00
|
|
|
#page.body.append(html.div(modified, id="docinfoline"))
|
2008-08-16 23:26:59 +08:00
|
|
|
|
|
|
|
page.contentspace.append(py.xml.raw(content))
|
|
|
|
outputpath.ensure().write(page.unicode().encode(encoding))
|
|
|
|
|
2009-03-04 02:06:16 +08:00
|
|
|
# XXX this function comes from apigen/linker.py, put it
|
|
|
|
# somewhere in py lib
|
|
|
|
import os
|
|
|
|
def relpath(p1, p2, sep=os.path.sep, back='..', normalize=True):
|
|
|
|
""" create a relative path from p1 to p2
|
|
|
|
|
|
|
|
sep is the seperator used for input and (depending
|
|
|
|
on the setting of 'normalize', see below) output
|
|
|
|
|
|
|
|
back is the string used to indicate the parent directory
|
|
|
|
|
|
|
|
when 'normalize' is True, any backslashes (\) in the path
|
|
|
|
will be replaced with forward slashes, resulting in a consistent
|
|
|
|
output on Windows and the rest of the world
|
|
|
|
|
|
|
|
paths to directories must end on a / (URL style)
|
|
|
|
"""
|
|
|
|
if normalize:
|
|
|
|
p1 = p1.replace(sep, '/')
|
|
|
|
p2 = p2.replace(sep, '/')
|
|
|
|
sep = '/'
|
|
|
|
# XXX would be cool to be able to do long filename
|
|
|
|
# expansion and drive
|
|
|
|
# letter fixes here, and such... iow: windows sucks :(
|
|
|
|
if (p1.startswith(sep) ^ p2.startswith(sep)):
|
|
|
|
raise ValueError("mixed absolute relative path: %r -> %r" %(p1, p2))
|
|
|
|
fromlist = p1.split(sep)
|
|
|
|
tolist = p2.split(sep)
|
|
|
|
|
|
|
|
# AA
|
|
|
|
# AA BB -> AA/BB
|
|
|
|
#
|
|
|
|
# AA BB
|
|
|
|
# AA CC -> CC
|
|
|
|
#
|
|
|
|
# AA BB
|
|
|
|
# AA -> ../AA
|
|
|
|
|
|
|
|
diffindex = 0
|
|
|
|
for x1, x2 in zip(fromlist, tolist):
|
|
|
|
if x1 != x2:
|
|
|
|
break
|
|
|
|
diffindex += 1
|
|
|
|
commonindex = diffindex - 1
|
|
|
|
|
|
|
|
fromlist_diff = fromlist[diffindex:]
|
|
|
|
tolist_diff = tolist[diffindex:]
|
|
|
|
|
|
|
|
if not fromlist_diff:
|
|
|
|
return sep.join(tolist[commonindex:])
|
|
|
|
backcount = len(fromlist_diff)
|
|
|
|
if tolist_diff:
|
|
|
|
return sep.join([back,]*(backcount-1) + tolist_diff)
|
|
|
|
return sep.join([back,]*(backcount) + tolist[commonindex:])
|
|
|
|
|
|
|
|
|