#!/usr/bin/env python
# encoding: utf-8

"""hgsite

Create and/or upload a static copy of the repository.

The main goal is sharing Mercurial on servers with only FTP access and
statically served files, while providing the same information as hg
serve and full solutions like bitbucket and gitorious (naturally
without the interactivity).
"""

__copyright__ = """Copyright 2012 till 2014 Arne Babenhauserheide

This software may be used and distributed according to the terms of the
GNU General Public License version 2 or any later version.
"""

import os
import shutil
import re
import mercurial
import ftplib
import socket
import datetime
import codecs
from mercurial import cmdutil, util, scmutil
from mercurial import commands, dispatch
# from mercurial.i18n import _
def _(string): # avoid partial localizations
    return string
from mercurial import hg, discovery, util, extensions

_staticidentifier = ".statichgrepo"
_freenetprivkeystring = "AQECAAE/"

templates = {
    "head": """<!DOCTYPE html>
<html><head>
    <meta charset="utf-8" />
    <meta http-equiv="Content-Type" content="text/html; charset=utf-8" /> <!--duplicate for older browsers-->
    <link href='http://fonts.googleapis.com/css?family=OFL+Sorts+Mill+Goudy+TT:regular,italic' rel='stylesheet' type='text/css' />
    <link rel="stylesheet" href="{relpath}style.css" type="text/css" media="screen" />
    <link rel="stylesheet" href="{relpath}print.css" type="text/css" media="print" />
    <script data-cfasync="false" src="{relpath}jquery.js" type="text/javascript"></script>
    <script data-cfasync="false" src="{relpath}jquery.timeago.js" type="text/javascript"></script>
    <script data-cfasync="false" src="{relpath}sjl.js" type="text/javascript"></script>
    <script data-cfasync="false" src="{relpath}print.js" type="text/javascript"></script>
    <title>{title}</title>
</head>
<body>
<div class="wrap">
<div class="top group">
<header id="maintitle">{reponame}</header>
<nav>{nav}</nav>
</div>
<div class="hr"> </div>
<div id="content">
""",
    "srchead": """<!DOCTYPE html>
<html><head>
    <meta charset="utf-8" />
    <meta http-equiv="Content-Type" content="text/html; charset=utf-8" /> <!--duplicate for older browsers-->
    <link rel="stylesheet" href="../../style.css" type="text/css" media="screen" />
    <link rel="stylesheet" href="../../print.css" type="text/css" media="print" />
    <link rel="stylesheet" href="../../prettify.css" type="text/css" />
    <script type="text/javascript" src="../../prettify.js"></script>
    <title>{filetitle}</title>
</head>
<body onload="prettyPrint()">
<div class="wrap">
<div id="content">
""",
    "forkhead": """<!DOCTYPE html>
<html><head>
    <meta charset="utf-8" />
    <meta http-equiv="Content-Type" content="text/html; charset=utf-8" /> <!--duplicate for older browsers-->
    <link href='http://fonts.googleapis.com/css?family=OFL+Sorts+Mill+Goudy+TT:regular,italic' rel='stylesheet' type='text/css' />
    <link rel="stylesheet" href="{relpath}style.css" type="text/css" media="screen" />
    <link rel="stylesheet" href="{relpath}print.css" type="text/css" media="print" />
    <title>{forkname}</title>
</head>
<body>
<div class="wrap">
<div class="top group">
<header id="maintitle"><a href="../../">{reponame}</a></header>
<nav id="forknav">{forkname} <small>(fork found at {forkuri})</small></header></nav>
</div>
<div class="hr"> </div>
<div class="content">
""",
    "foot": """<div class="hrb"> </div><footer>
                <p>
                    Site made with the <a href="http://draketo.de/proj/hgsite">hg site extension</a> by Arne Babenhauserheide, 
                    Design based on the site from <a href="http://stevelosh.com">Steve Losh</a>
                </p>
            </footer></div><!-- /content --></div><!-- /wrap--></body></html>\n""",
    # this screenstyle is mostly from the wonderful site http://stevelosh.com and licensed under MIT
    "screenstyle": """ 
.bugnumbers {
    font-size: x-small;
    vertical-align: super;
}
.openbugnumber, .openbugnumber a {
    color: #f00;
    text-decoration: none;
}
.resolvedbugnumber, .resolvedbugnumber a {
    color: #00f;
    text-decoration: none;
}
/* Copyright (c) 2008-2010 Steve Losh

Permission is hereby granted, free of charge, to any person obtaining a copy
of this software and associated documentation files (the "Software"), to deal
in the Software without restriction, including without limitation the rights
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
copies of the Software, and to permit persons to whom the Software is
furnished to do so, subject to the following conditions:

The above copyright notice and this permission notice shall be included in
all copies or substantial portions of the Software.

THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
THE SOFTWARE.*/
html,body,div,span,applet,object,iframe,h1,h2,h3,h4,h5,h6,p,blockquote,pre,a,abbr,acronym,address,big,cite,code,del,dfn,em,font,img,ins,kbd,q,s,samp,small,strike,strong,sub,sup,tt,var,b,u,i,center,dl,dt,dd,ol,ul,li,fieldset,form,label,legend,table,caption,tbody,tfoot,thead,tr,th,td,header,nav,section,article,aside,footer{
    border:0;margin:0;outline:0;padding:0;background:transparent;vertical-align:baseline;
}
article,aside,details,figcaption,figure,footer,header,hgroup,menu,nav,section{
    display:block;
}
blockquote,q{
    quotes:none;
}
blockquote:before,blockquote:after,q:before,q:after{
    content:'';content:none;
}
header,nav,section,article,aside,footer{
    display:block;
}
html{
    overflow-y:scroll;
}
body{
    background:#fdfdfd;color:#353535;font:normal 18px/25px Palatino,"Palatino Linotype",serif;text-rendering:optimizeLegibility;
}
html>body{
    font-size:18px;line-height:25px;
}
img{
    display:inline-block;vertical-align:bottom;
}
h1,h2,h3,h4,h5,h6,strong,b,dt,th{
    font-weight:700;
}
address,cite,em,i,caption,dfn,var{
    font-style:italic;
}
h1{
    font-size:45px;line-height:50px;margin:25px 0;
}
h2{
    font-size:32px;line-height:50px;margin:25px 0;
}
h3{
    font-size:23px;line-height:25px;margin:25px 0;
}
h4{
    margin:0 0 22px;font-size:16px;line-height:22px;
}
h5{
    margin:0 0 22px;font-size:14px;line-height:22px;
}
h6{
    margin:0 0 22px;font-size:12px;line-height:22px;
}
p,ul,ol,dl,blockquote,pre{
    margin:0 0 25px;
}
li ul,li ol{
    margin:0;
}
ul{
    list-style:outside disc;
}
ol{
    list-style:outside decimal;
}
li{
    margin:0 0 0 44px;
}
dd{
    padding-left:25px;
}
blockquote{
    padding:0 25px;
}
a{
    text-decoration:underline;
}
a:hover{
    text-decoration:none;
}
abbr,acronym{
    border-bottom:1px dotted;cursor:help;
}
del{
    text-decoration:line-through;
}
ins{
    text-decoration:overline;
}
sub{
    font-size:14px;line-height:25px;vertical-align:sub;
}
sup{
    font-size:14px;line-height:25px;vertical-align:super;
}
tt,code,kbd,samp,pre{
    font-size:14px;line-height:25px;font-family:Menlo,Monaco,Consolas,"Courier New",monospace;
}
table{
    border-collapse:collapse;border-spacing:0;margin:0 0 1.5em;
}
caption{
    text-align:left;
}
th,td{
    padding:.25em .5em;
}
tbody td,tbody th{
    border:1px solid #222;
}
tfoot{
    font-style:italic;
}
fieldset{
    clear:both;
}
legend{
    padding:0 0 1.286em;font-size:1.167em;font-weight:700;
}
fieldset fieldset legend{
    padding:0 0 1.5em;font-size:1em;
}
* html legend{
    margin-left:-7px;
}
*+html legend{
    margin-left:-7px;
}
form .field,form .buttons{
    clear:both;margin:0 0 1.5em;
}
form .field label{
    display:block;
}
form ul.fields li{
    list-style-type:none;margin:0;
}
form ul.inline li,form ul.inline label{
    display:inline;
}
form ul.inline li{
    padding:0 .75em 0 0;
}
input.radio,input.checkbox{
    vertical-align:top;
}
label,button,input.submit,input.image{
    cursor:pointer;
}
* html input.radio,* html input.checkbox{
    vertical-align:middle;
}
*+html input.radio,*+html input.checkbox{
    vertical-align:middle;
}
textarea{
    overflow:auto;
}
input.text,input.password,textarea,select{
    margin:0;font:1em/1.3 Helvetica,Arial,"Liberation Sans","Bitstream Vera Sans",sans-serif;vertical-align:baseline;
}
input.text,input.password,textarea{
    border:1px solid #444;border-bottom-color:#666;border-right-color:#666;padding:2px;
}
* html button{
    margin:0 .34em 0 0;
}
*+html button{
    margin:0 .34em 0 0;
}
form.horizontal .field{
    padding-left:150px;
}
form.horizontal .field label{
    display:inline;float:left;width:140px;margin-left:-150px;
}
img.left{
    display:inline;float:left;margin:0 1.5em .75em 0;
}
img.right{
    display:inline;float:right;margin:0 0 .75em .75em;
}
.group:after{
    content:".";display:block;height:0;clear:both;visibility:hidden;
}

html{
    overflow-y:scroll;
}
body{
    text-rendering:optimizeLegibility;color:#222222;position:relative;
}
body a{
    text-decoration:none;color:#e50053;
}
body a:hover{
    text-decoration:underline;
}
body h1,body h2,body h3,body h4,body h5,body h6{
    font-family:'OFL Sorts Mill Goudy','OFL Sorts Mill Goudy TT',serif;font-weight:normal;
}
body h1 a,body h2 a,body h3 a,body h4 a,body h5 a,body h6 a{
    color:#222222;
}
body h1 a:hover,body h2 a:hover,body h3 a:hover,body h4 a:hover,body h5 a:hover,body h6 a:hover{
    color:#e50053;text-decoration:none;
}
body h1 .amp,body h2 .amp,body h3 .amp,body h4 .amp,body h5 .amp,body h6 .amp{
    font-style:italic;
}
body h1{
    font-size:45px;line-height:50px;margin:25px 0;
}
body h2{
    font-size:32px;line-height:50px;margin:25px 0;
}
body h3{
    font-size:23px;line-height:25px;margin:25px 0;
}
body h4{
    font-size:18px;line-height:25px;margin:25px 0;
}
body code,body pre{
    font-family:Consolas,Menlo,"Courier New",monospace;font-size:14px;
}
body .codehilite code,body .codehilite pre{
    font-family:Consolas,Menlo,"Courier New",monospace;font-size:16px;line-height:25px;overflow-x:auto;border:1px solid #d5d5d5;border-left:10px solid #d5d5d5;background-color:#fafafa;padding:11px 15px 12px;margin-left:-25px;
}
body .codehilite pre::-webkit-scrollbar{
    height:25px;
}
body .codehilite pre::-webkit-scrollbar-button:start,body .codehilite pre::-webkit-scrollbar-button:end{
    display:none;
}
body .codehilite pre::-webkit-scrollbar-track-piece{
    background-color:#eee;
}
body .codehilite pre::-webkit-scrollbar-thumb{
    background-color:#bbb;border:7px solid #eee;-webkit-background-clip:padding-box;-webkit-border-radius:12px;
}
body p code,body li code,body table code{
    border:1px solid #ccc;background-color:#fafafa;font-size:13px;padding:1px 3px;line-height:20px;margin:0;white-space:nowrap;
}
body .wrap{
    width:612px;margin:0 auto;margin-bottom:5em;
}
body .wrap .top header{
    float:left;font-family:'OFL Sorts Mill Goudy','OFL Sorts Mill Goudy TT',serif;font-size:23px;line-height:50px;text-transform:lowercase;padding-left:2px;width:210px;
}
body .wrap .top.fullwidth header{
    float:left;font-family:'OFL Sorts Mill Goudy','OFL Sorts Mill Goudy TT',serif;font-size:23px;line-height:50px;text-transform:lowercase;padding-left:2px;width:610px;
}
body .wrap .top header .amp{
    font-style:italic;
}
body .wrap .top header a{
    color:#222222;
}
body .wrap .top header a:hover{
    color:#e50053;
}
body .wrap .top nav{
    /* reduced font size and/or line height compared to steve losh (18px/50px), 
    because we can have a long lis tof forks and use <sup> elements for the bugs.*/
    font:normal 18px/35px 'OFL Sorts Mill Goudy','OFL Sorts Mill Goudy TT',serif;text-align:right;text-transform:lowercase;padding-right:2px;width:398px;float:left;
}
body .wrap .top nav#forknav{
    font:normal 18px/50px 'OFL Sorts Mill Goudy','OFL Sorts Mill Goudy TT',serif;text-align:right;text-transform:lowercase;padding-right:2px;width:398px;float:left;
}
body .wrap .top nav .sep{
    padding:0 4px;color:#666;
}
body .wrap .hr{
    margin-top:-12px;margin-bottom:37px;height:25px;background:transparent url('')/*../images/hr.png*/ top left no-repeat;
}
body .wrap .hrb{
    height:25px;margin-top:50px;background:transparent url('')/*../images/hrb.png*/ top left no-repeat;
}
body .wrap footer{
    text-align:center;
}
body .wrap footer p{
    font-size:14px;font-style:italic;line-height:50px;margin-bottom:0px;
}
body .wrap footer .rochester-made img{
    opacity:0.7;padding:5px 20px;
}
hr{
    border:none;background:#ccc;height:1px;margin-bottom:24px;
}
.splash{
    color:#454545;text-align:center;font:normal 27px/32px 'OFL Sorts Mill Goudy','OFL Sorts Mill Goudy TT',serif;margin-bottom:-1px;padding-top:0;
}
.splash p{
    margin-bottom:24px;
}
.splash .amp{
    font-style:italic;
}
.splash .fn{
    color:#454545;text-decoration:none;
}
.splash .fn:hover{
    color:#e50053;text-decoration:none;
}
.splash .fn .last-name{
    display:none;
}
.section-listing{
    margin-bottom:-50px;
}
.section-listing ol{
    list-style-type:none;
}
.section-listing ol li{
    float:left;width:281px;height:150px;margin:0;text-align:center;
}
.section-listing ol li a{
    font:normal 23px/32px 'OFL Sorts Mill Goudy','OFL Sorts Mill Goudy TT',serif;color:#222222;display:block;
}
.section-listing ol li a:hover{
    color:#e50053;text-decoration:none;
}
.section-listing ol li span.snip{
    font-size:18px;color:#333;font-family:'OFL Sorts Mill Goudy','OFL Sorts Mill Goudy TT',serif;line-height:25px;font-style:italic;
}
.section-listing ol li .amp{
    font-style:italic;
}
.section-listing ol li:nth-child(odd){
    margin-right:25px;
}
.section-listing ol li:nth-child(even){
    margin-left:25px;
}
blockquote{
    border:1px solid #ccc;background-color:#fafafa;padding:11px 15px 12px;margin-left:2em;overflow:auto;
}
blockquote p:last-child{
    margin-bottom:0;
}
span.dquo{
    margin-left:-0.23em;
}
#leaf-stats p{
    color:#666;margin-top:-22px;margin-bottom:22px;
}
#leaf-content img{
    display:block;margin:25px auto 26px;border:11px solid #e5e5e5;padding:1px;background:black;max-width:590px;
}
#leaf-content img.left,#leaf-content img.right{
    border:none;background:none;padding:none;
}
#leaf-content img.left{
    margin:0 1.5em 1em 0;
}
#leaf-content img.right{
    margin:0 0 .75em 1em;
}
#leaf-content .gallery img{
    background:none;padding:0;border:none;display:inline;margin-bottom:25px;margin-right:25px;
}
#leaf-content .toc ul{
    list-style:none;
}
#leaf-content .toc ul:first-child>li{
    margin-left:0em;
}
#leaf-content table{
    padding:0px;margin-top:-8px;margin-bottom:25px;
}
#leaf-content table tr{
    margin:0px;padding:0px;
}
#leaf-content table tr td,#leaf-content table tr th{
    margin:0px;padding:5px 5px;line-height:23px;
}
#leaf-content table tr td{
    border:1px solid #666;
}
#leaf-content.with-diagrams img{
    display:block;margin:25px auto;padding:0;background:none;border:none;
}
.print-links{
    display:none;
}
img.self{
    border:none;padding:0;margin:0;margin-right:-108px;margin-top:-15px;margin-left:30px;margin-bottom:20px;
}
div#cboxCurrent{
    bottom:-30px;font-size:17px;font-weight:normal;left:60px;
}
div.screenshots img{
    max-width:580px;
}
#scrolling-header{
    color:#999;font-size:23px;font-style:italic;font:italic 23px 'OFL Sorts Mill Goudy','OFL Sorts Mill Goudy TT',serif;line-height:30px;position:fixed;top:75px;text-align:right;width:180px;
}
.flattr{
    float:right;padding-top:1px;
}
#shortlog a, #branches a, #tags a, #bookmarks a, #incoming a, #outgoing a, #log a{
    text-decoration:none;color:gray;
}
#shortlog a:hover, #branches a:hover, #tags a:hover, #bookmarks a:hover, #incoming a:hover, #outgoing a:hover, #log a{
    text-decoration:underline;
}
#intro{
    color:#353535;font:normal 23px/25px Palatino,"Palatino Linotype",serif;text-rendering:optimizeLegibility;font-style:italic;padding-left:50px;padding-right:50px;padding-bottom:20px;padding-top:15px;/*background:#eeeeee;border:solid thin lightgray;border-radius:25px;*/
}
#introtoreadmelink{
    color: lightgray;
}
#readme>pre{
    background:#fdfdfd;color:#353535;font:normal 18px/25px Palatino,"Palatino Linotype",serif;text-rendering:optimizeLegibility;
/*font-family:Consolas,Menlo,"Courier New",monospace;*/
}
""",
    # this printstyle is from the wonderful site http://stevelosh.com and licensed under MIT
    "printstyle": """ 
/*Copyright (c) 2008-2010 Steve Losh

Permission is hereby granted, free of charge, to any person obtaining a copy
of this software and associated documentation files (the "Software"), to deal
in the Software without restriction, including without limitation the rights
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
copies of the Software, and to permit persons to whom the Software is
furnished to do so, subject to the following conditions:

The above copyright notice and this permission notice shall be included in
all copies or substantial portions of the Software.

THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
THE SOFTWARE.*/

body{
font:normal 10pt/1.25 Palatino,"Palatino Linotype";text-rendering:optimizeLegibility;
}
nav,header,.toc{
display:none;
}
#leaf-title a{
text-decoration:none;color:black;
}
#leaf-content ul.print-links{
display:block;font-size:1em;list-style-type:none;margin-left:0em;
}
#leaf-content ul.print-links a{
text-decoration:none;
}
#leaf-content code,#leaf-content pre{
font:normal 9pt Menlo,Monaco,Consolas,"Courier New",Courier,monospace;
}
#leaf-content img{
display:block;margin-left:auto;margin-right:auto;border:1.43em solid #e5e5e5;padding:1px;background:black;
}
#leaf-content img.left,#leaf-content img.right{
border:none;background:none;padding:none;
}
#leaf-content img.left{
margin:0 1.5em 1em 0;float:left;
}
#leaf-content img.right{
margin:0 0 .75em 1em;float:right;
}
div#leaf-content.with-diagrams img{
display:block;margin-left:auto;margin-right:auto;background:none;border:none;
}
span.amp{
font-family:"Palatino","Constantia","Palatino Linotype",serif;font-style:italic;
}
.flattr{
display:none;
}
a{
text-decoration:underline;color:#c06;
}
footer{
display:none;
}
""",
    "manifesthead": """<h2>""" + _("Commit (click to see the diff)")+""": <a href='../../commit/{hex}.html'>{hex}</a></h2>
<p>{desc}</p><p>{user}</p>
    <h2>""" + _("Diffstat") + """</h2>
<pre>{diffstat}</pre>
    <h2>""" + _("Files in this revision") + "</h2>", 
    "commitlog": """\n<div style='float: right; padding-left: 0.5em'><em>({author|person})</em></div><strong> {date|shortdate}: <a href='{relativepath}src/{node}/index.html'>{desc|strip|fill68|firstline}</a></strong> <span style='font-size: xx-small'>{branches} {tags} {bookmarks}</span><p>{desc|escape}</p>\n""",
}

_indexregexp = re.compile("^\\.*index.html$")


def samefilecontent(filepath1, filepath2): 
    """Check if the content of the two referenced files is equal."""
    try: 
        with open(filepath1) as f1: 
            with open(filepath2) as f2: 
                return f1.read() == f2.read()
    except OSError: return False

def contentequals(filepath, content): 
    """Check if the files content is content."""
    try: 
        with open(filepath) as f: 
            return f.read() == content
    except OSError: return not content # equal, if the content is empty
    except IOError: return False # file does not exist. Empty != not existing.
    # TODO: check: return True if content is None?

def bisenabled():
    """Check if the b extension is enabled to decide if we want to add
    a bug listing."""
    enabled = extensions.enabled()
    if "b" in enabled:
        return True

def splitbugline(line):
    """Split a b extension bug line into the ID and the description."""
    try:
        bugid = line.split("-")[0].strip()
    except IndexError:
        return "", line
    description = "".join(line.split("-")[1:]).lstrip()
    return bugid, description

def getbugdetails(ui, repo, bugid):
    """Get the details for a bug."""
    # first get the details
    ui.pushbuffer()
    req = dispatch.request(["b", "details", bugid], ui=ui, repo=repo)
    dispatch.dispatch(req)
    return ui.popbuffer()

def getbugfullid(details, bugid):
    """Get the real ID of a bug from its detailed info. If it’s not available, just give the short bugid"""
    try:
        idline = [i for i in details.splitlines() if i.startswith("ID: ")][0]
    except IndexError: # no id line
        return bugid
    realid = idline[4:].strip()
    return realid

class BBug(object):
    """A b-extension bug."""
    def __init__(self, shortid, fullid, description, state, details=""):
        self.shortid, self.fullid, self.description, self.state, self.details = shortid, fullid, description, state, details

def getbuginfo(ui, repo, bugline):
    """Get information about a bug from its bugline."""
    shortid, description = splitbugline(bugline)
    details = getbugdetails(ui, repo, shortid)
    fullid = getbugfullid(details, shortid)
    return shortid, fullid, description, details

def getbugs(ui, repo):
    """Get all bugs."""
    if not bisenabled():
        return [], []
    # run the b command to get all open bugs
    ui.pushbuffer()
    req = dispatch.request(["b"], ui=ui, repo=repo)
    dispatch.dispatch(req)
    openbuglines = [line for line in ui.popbuffer().splitlines() if "-" in line]
    # similarly get all resolved bugs
    ui.pushbuffer()
    req = dispatch.request(["b", "list", "-r"], ui=ui, repo=repo)
    dispatch.dispatch(req)
    resolvedbuglines = [line for line in ui.popbuffer().splitlines() if "-" in line]
    # now turn them into a list of bugs
    openbugs = []
    for bugline in openbuglines:
        bugid, fullid, description, details = getbuginfo(ui, repo, bugline)
        openbugs.append(BBug(bugid, fullid, description, "open", details))
    resolvedbugs = []
    for bugline in resolvedbuglines:
        bugid, fullid, description, details = getbuginfo(ui, repo, bugline)
        resolvedbugs.append(BBug(bugid, fullid, description, "resolved", details))
    return openbugs, resolvedbugs

def parsereadme(filepath, truncated=False):
    """Parse the readme file"""
    with open(filepath) as r:
        readme = r.read()
    # TODO: Markdown parsing belongs here.
    if truncated:
        text = "\n".join(readme.splitlines()[:6])
    else:
        text = readme
    try: # to use markdown if we can
        import markdown
        html = markdown.markdown(
            unicode(text, encoding="utf-8"), 
            extensions=["markdown.extensions.extra"]).encode("utf-8")
        return html
    except ImportError:
        return "<pre markdown=\"1\" style=\"white-space: pre-wrap; white-space: -moz-pre-wrap; white-space: -pre-wrap; white-space: -o-pre-wrap; word-wrap: break-word;\">" + text + "</pre>"

def overviewlogstring(ui, repo, revs, template=templates["commitlog"]): 
    """Get the string for a log of the given revisions for the overview page."""
    ui.pushbuffer()
    t = cmdutil.changeset_templater(ui, repo, patch=False, diffopts=None, mapfile=None, buffered=False)
    t.use_template(template.replace("{relativepath}", ""))
    for c in revs:
        ctx = repo.changectx(c)
        t.show(ctx)
    return ui.popbuffer()


def sortedtagnames(repo, tags):
    """Get the names of tags from a tags dictionary.

    :param tags: {"tagname": rev}, example: {'v0.5.3': '\xdfd\xe8.\t \xbd\xd0y\xd7|\xd4\x82\x9b\xac\x9d0F\x1a%'}
    :return: [first, second, third, fourth]
    """
    rev2tag = {}
    for tag in tags:
        rev2tag[repo[tag].rev()] = tag
    return [rev2tag[i] for i in sorted(rev2tag.keys())]


def writeoverview(ui, repo, target, name):
    """Create the overview page"""
    ui.debug("[staticsite] writeoverview: header\n")
    overview = ""
    # start with the nav
    nav = ""
    # now the links to the log and the files.
    nav += "<a href='commits'>" + _("changelog") + "</a> - <a href='src/" + repo["tip"].hex() + "/'>" + _("files") + "</a>"
    # and the bugs
    openbugs, resolvedbugs = getbugs(ui, repo)
    if openbugs or resolvedbugs:
        nav += " - <a href=\"bugs\">" + _("bugs") + "</a>"
        if openbugs:
            nav += " <span class=\"bugnumbers\">(<span class=\"openbugnumber\"><a href=\"bugs#open\">" + str(len(openbugs)) + "!</a></span> "
        else:
            nav += " <span class=\"bugnumber openbugnumberzero\">0</span>"
        nav += "<span class=\"bugnumber resolvedbugnumber\"><a href=\"bugs#resolved\">" + str(len(resolvedbugs)) + "√</a></span>)</span>"
    
    # and the forks
    ui.debug("[staticsite] writenav: header: forks\n")
    forks = getforkinfo(ui, target)
    if forks:
        nav += " - " + _("forks: ")
        for forkname, forkuri in forks.items():
            ui.debug("[staticsite] writenav: fork: " + forkname + ": " + forkuri + "\n")
            ui.debug("[staticsite] writenav: forks: getforkdir\n")
            nav += "<a href='" + getforkdir(target, forkname) + "'>" + forkname + "</a> "
            ui.debug("[staticsite] writenav: forks: getincoming\n")
            incoming, fn, localother = getincoming(ui, repo, otheruri=forkuri, othername=forkname)
            nav += "<small>(" + str(len(incoming))
            ui.debug("[staticsite] writenav: forks: getoutgoing\n")
            outgoing, fn, localother = getoutgoing(ui, repo, otheruri=forkuri, othername=forkname)
            nav += "<small>↓↑</small>" + str(len(outgoing)) + ")</small> "
            
    
    # embed in the overview via the template
    overview += templates["head"].replace("{reponame}", name).replace("{title}", name).replace("{nav}", nav).replace("{relpath}", "./")
    # add a short identifier from the first line of the readme, if it
    # exists # TODO: Parse different types of readme files
    readme = name
    for f in os.listdir(repo.root):
        if f.lower().startswith("readme"):
            readme = parsereadme(os.path.join(repo.root, f))
            readme_intro = parsereadme(os.path.join(repo.root, f), truncated=True)
            overview += "<div id='intro'>"
            overview += readme_intro
            overview += """<p id="introtoreadmelink">(<a href="#readme" title="Show me the full Readme">...</a>)</p>"""
            overview += "</div>"
            break
    
    ui.debug("[staticsite] writeoverview: shortlog\n")
    # now add the 5 most recent log entries
    # divert all following ui output to a string, so we can just use standard functions
    overview += "\n<div id='shortlog'><h2>Changes (<a href='commits'>full changelog</a>)</h2>\n"
    ui.pushbuffer()
    t = cmdutil.changeset_templater(ui, repo, patch=False, diffopts=None, mapfile=None, buffered=False)
    t.use_template(templates["commitlog"].replace("{relativepath}", ""))
    for c in range(1, min(len(repo.changelog), 5)):
        ctx = repo.changectx(str(-c))
        t.show(ctx)
    overview += ui.popbuffer()
    overview += "</div>"
    
    # Add branch, bookmark and tag information, if they exist.
    ui.debug("[staticsite] writeoverview: branches, tags and bookmarks\n")
    branches = []
    for branch, heads in repo.branchmap().items(): 
        if branch and branch != "default": # not default
            branches.extend([h for h in heads if not repo[h].closesbranch()])
    # if other branches exist, add default, too.
    if branches:
        branches.extend(repo.branchmap()["default"])
            
    try: 
        tags = repo.tags()
    except AttributeError: 
        try: 
            # FIXME: For some reason this does not seem to give the tags anymore.
            tags = repo._tags
        except AttributeError: 
            tags = {}
    try: 
        bookmarks = repo._bookmarks
    except AttributeError:
        bookmarks = []
    if branches: # add branches
        overview += "\n<div id='branches'><h2>Branches</h2>\n"
        overview += overviewlogstring(ui, repo, branches,
                                      template=templates["commitlog"].replace(
                "{branches}", "XXXXX").replace(
                "{date|shortdate}", "{branch}").replace(
                "XXXXX", "{date|shortdate}").replace(
                "{tags}", "XXXXX").replace(
                "{date|shortdate}", "{tags}").replace(
                "XXXXX", "{date|shortdate}"))
        overview += "</div>"
    if len(tags) > 1: 
        overview += "\n<div id='tags'><h2>Tags</h2>\n"
        overview += overviewlogstring(ui, repo, 
                                      [tags[t] 
                                       for t in reversed(sortedtagnames(
                                               repo, tags))
                                       if t != "tip"],
                                      template=templates["commitlog"].replace(
                "{tags}", "XXXXX").replace(
                "{date|shortdate}", "{tags}").replace(
                "XXXXX", "{date|shortdate}"))
        overview += "</div>"
    if len(bookmarks): 
        overview += "\n<div id='bookmarks'><h2>Bookmarks</h2>\n"
        overview += overviewlogstring(ui, repo, bookmarks.values(),
                                      template=templates["commitlog"].replace(
                "{bookmarks}", "XXXXX").replace(
                "{date|shortdate}", "{bookmarks}").replace(
                "XXXXX", "{date|shortdate}"))
        overview += "</div>"
    # add the full readme
    ui.debug("[staticsite] writeoverview: readme\n")
    overview += "<div id='readme'><h2>"+_("Readme")+"</h2>\n"
    overview += readme
    overview += "</div>"
    
    # finish the overview
    ui.debug("[staticsite] writeoverview: footer\n")
    overview += templates["foot"]
    indexfile = os.path.join(target, "index.html")
    if not contentequals(indexfile, overview): 
        with open(indexfile, "w") as f: 
            f.write(overview)

def writelog(ui, repo, target, name):
    """Write the full changelog, in steps of 100."""
    commits = os.path.join(target, "commits")
    
    # create the folders
    if not os.path.isdir(commits):
        os.makedirs(commits)
    for i in range(len(repo.changelog)/100):
        d = commits+"-"+str(i+1)+"00"
        if not os.path.isdir(d):
            os.makedirs(d)
            
    # create the log files
    t = cmdutil.changeset_templater(ui, repo, patch=False, diffopts=None, mapfile=None, buffered=False)
    t.use_template(templates["commitlog"].replace("{relativepath}", "../"))
    logs = []
    for ck in range(len(repo.changelog)/100+1):
        ui.pushbuffer()
        if ck:
            dd = d
            di = str(ck)+"00"
            d = commits+"-"+di
            logs[-1][-1] += "<p><a href=\"../commits-"+di+"\">earlier</a></p>"
            if ck>2:
                # the older log gets a reference to the newer one
                logs[-1][-1] += "<p><a href=\"../commits-"+str(ck-2)+"00"+"\">later</a></p>"
            elif ck>1:
                logs[-1][-1] += "<p><a href=\"../commits\">later</a></p>"
            logs.append([os.path.join(d, "index.html"), ""])
        else:
            di = "" # for easy later usage
            d = commits
            logs.append([os.path.join(d, "index.html"), ""])
            
        logs[-1][-1] += templates["head"].replace("{reponame}", "<a href='../'>"+name+"</a>").replace("{title}", name).replace("{nav}", "log " + di).replace("{relpath}", "../")
        logs[-1][-1] += """<div id="log">"""
        for c in range(ck*100+1, min(len(repo.changelog)+1, (ck+1)*100)):
            ctx = repo.changectx(str(-c))
            t.show(ctx)
        logs[-1][-1] += ui.popbuffer()
        logs[-1][-1] += """</div>"""
        
    for filepath,data in logs:
        data += templates["foot"].replace("{reponame}", "<a href='../'>"+name+"</a>")
        if not contentequals(filepath,data): 
            with open(filepath, "w") as f: 
                f.write(data)

def getlocalother(repo, ui, otheruri, othername):
    """Get a local clone of the repo identified by uri and name within .hg/paths.
    
    This creates that local clone!
    """
    # if we cannot get the changes via bundlerepo, we create a
    # local clone in .hg/paths/<othername>-<sha1-of-otheruri> and
    # check from there.  in case that local clone already exists,
    # we tell it to pull there.  The hash is necessary to prevent
    # collisions when the uri changes.
    if othername is None:
        othername = ""
    urihash = util.sha1(otheruri).hexdigest()
    localcopy = os.path.join(repo.root, ".hg", "paths", 
                     othername+"-"+urihash)
    # if getting remote changes directly fails, we take the
    # completely safe path: dispatch uses the only really stable
    # interface: the cli.
    if os.path.isdir(localcopy):
        req = dispatch.request(["-R", localcopy, "pull", otheruri])
    else:
        req = dispatch.request(["clone", otheruri, localcopy], ui=ui)
    dispatch.dispatch(req)
    other = hg.peer(repo, {}, localcopy)
    return other

def getincoming(ui, repo, otheruri, other=None, othername=None):
    """Get incoming changes."""
    # Note: We cannot just use getcommonincoming and I do not yet know
    # how to use its output to get good changes. TODO: do this nicer.
    def cleanupfn():
        """non-operation cleanup function (default)."""
        pass
    # cannot do that for ftp or freenet insertion uris (freenet
    # separates insertion and retrieval by private/public key)
    isftpuri = otheruri.startswith("ftp://") or otheruri.startswith("ftps://")
    isfreenetpriv = _freenetprivkeystring in otheruri
    if isftpuri or isfreenetpriv:
        chlist = []
        return chlist, cleanupfn, other
    
    if not other:
        other = hg.peer(repo, {}, otheruri)
    ui.pushbuffer() # ignore ui events
    source, branches = hg.parseurl(otheruri, None)
    revs, checkout = hg.addbranchrevs(repo, other, branches, None)
    if revs:
        revs = [other.lookup(rev) for rev in revs]
    try: # FIXME: This breaks on http repos! 
        other, chlist, cleanupfn = hg.bundlerepo.getremotechanges(ui, repo, other,
                                                               revs, False, False)
    except (AttributeError, util.Abort):
        other = getlocalother(repo, ui, otheruri, othername)
        other, chlist, cleanupfn = hg.bundlerepo.getremotechanges(ui, repo, other,
                                                               revs, False, False)
        
    ui.popbuffer()
    return chlist, cleanupfn, other

def getoutgoing(ui, repo, otheruri, other=None, othername=None):
    def cleanupfn():
        """non-operation cleanup function (default)."""
        pass
    # cannot do that for ftp or freenet insertion uris (freenet
    # separates insertion and retrieval by private/public key)
    ui.debug("[staticsite] getoutgoing: checkkeys\n")
    isftpuri = otheruri.startswith("ftp://") or otheruri.startswith("ftps://")

    isfreenetpriv = "AQECAAE/" in otheruri
    if isftpuri or isfreenetpriv:
        chlist = []
        return chlist, cleanupfn, other
    
    if not other: 
        ui.debug("[staticsite] getoutgoing: findpeer\n")
        other = hg.peer(repo, {}, otheruri)
        
    def outgoingchanges(repo, other):
        from mercurial import discovery
        fco = discovery.findcommonoutgoing
        og = fco(repo, other, force=True)
        # print dir(og), og.missingheads, og.missing
        ui.debug("[staticsite] getoutgoing: outgoingchanges: og.missing\n")
        try:
            return og.missing
        except AttributeError: # old client
            common, outheads = og
            o = repo.changelog.findmissing(common=common, heads=outheads)
            return o
        except Exception as e:
            ui.warn("Cannot parse parts of your local history. Likely your local repository is broken. You might be able to fix it by getting a fresh clone and pulling from your current repo. The Error was: " + str(e) + "\n")
    
    other.ui.pushbuffer() # ignore ui events
    
    ui.debug("[staticsite] getoutgoing: outgoingchanges\n")
    try:
        chlist = outgoingchanges(repo, other)
    except (AttributeError, util.Abort):
        other.ui.popbuffer()
        ui.debug("[staticsite] getoutgoing: local other\n")
        other = getlocalother(repo, ui, otheruri, othername)
        other.ui.pushbuffer()
        ui.debug("[staticsite] getoutgoing: outgoingchanges, local\n")
        try:
            chlist = outgoingchanges(repo, other)
        except Exception as e: # no changes
            ui.debug("[staticsite] getoutgoing: outgoingchanges, errer: " + str(e) + "\n")
            raise
            chlist = []
        
    ui.debug("[staticsite] getoutgoing: popbuffer\n")
    other.ui.popbuffer()
    ui.debug("[staticsite] getoutgoing: done\n")
    return chlist, cleanupfn, other

def getforkinfo(ui, target):
    """Name and Uri of all forks."""
    forks = dict(ui.configitems("paths"))
    forkinfo = {}
    for forkname, forkuri in forks.items():
        # ignore the static repo
        if os.path.abspath(forkuri) == os.path.abspath(target):
            continue
        forkinfo[forkname] = forkuri
    return forkinfo

def safeuri(uri):
    """Shareable uris: Hide password + hide freenet insert keys."""
    uri = util.hidepassword(uri)
    if "USK@" in uri and _freenetprivkeystring in uri:
        uri = "freenet://USK@******" + uri[uri.index(_freenetprivkeystring)+len(_freenetprivkeystring)-1:]
    return uri

def getforkdata(ui, repo, target, name, forkname, forkuri):
    """Write the site for a single fork."""
    # make sure the forkdir exists.
    other = hg.peer(repo, {}, forkuri)
    
    # incrementally build the html
    html = templates["forkhead"].replace(
            "{forkname}", forkname).replace(
        "{reponame}", name).replace(
                "{forkuri}", safeuri(forkuri)).replace(
                    "{relpath}", "../../")
        
    # prepare the log templater
    t = cmdutil.changeset_templater(ui, repo, patch=False, diffopts=None, mapfile=None, buffered=False)
    t.use_template(templates["commitlog"].replace(
            "{relativepath}", "../../"))
    
    # Add incoming commits
    html += "<div id='incoming'><h2>Incoming commits</h2>"
    chlist, cleanupfn, localother = getincoming(ui, repo, otheruri=forkuri, other=other, othername=forkname)

    # write all missing incoming commits directly from the incoming repo
    if chlist:
        try:
            writecommitsforchlist(ui, localother, target, name, chlist)
            writesourcetreeforchlist(ui, localother, target, name, chlist, force=False)
        except AttributeError:
            if not hasattr(localother, "changelog"):
                print "Cannot write commits from fork", forkname, "because the repository type does not support getting the changelog."
            else:
                raise
    
    ui.pushbuffer()
    for ch in chlist:
        ctx = localother.changectx(ch)
        t.show(ctx)
    html += ui.popbuffer()
    cleanupfn()
    html += "</div><!-- /incoming-->"
        
    # add outgoing commits
    html += "<div id='outgoing'><h2>Outgoing commits</h2>"
    chlist, cleanupfn, localother = getoutgoing(ui, repo, forkuri, other=other, othername=forkname)
    
    ui.pushbuffer()
    for ch in chlist:
        ctx = repo.changectx(ch)
        t.show(ctx)
    html += ui.popbuffer()
    cleanupfn()
    
    html += "</div><!-- /outgoing-->"
    html += templates["foot"]
    return html

def getforkdir(target, forkname):
    return os.path.join("forks", forkname)

def writeforks(ui, repo, target, name):
    """Write an info-page for each fork, defined in hg paths.
    
    relevant data: incoming commits, outgoing commits, branches and bookmarks not in fork or not in repo. Short: incoming (commits, branches, bookmarks), outgoing (incoming first means, we consider this repo to be the main repo).
    """
    forkinfo = getforkinfo(ui, target)
    for forkname, forkuri in forkinfo.items():
        # ignore the static repo itself
        if os.path.abspath(forkuri) == os.path.abspath(target):
            continue
        forkdir = getforkdir(target, forkname)
        if not os.path.isdir(os.path.join(target, forkdir)):
            os.makedirs(os.path.join(target, forkdir))
        forkindex = os.path.join(target, forkdir, "index.html")
        forkcontent = getforkdata(ui, repo, target, name, forkname, forkuri)
        if not contentequals(forkindex, forkcontent):
            with open(forkindex, "w") as f:
                f.write(forkcontent)

def writecommitsforchlist(ui, repo, target, name, chlist, force=False):
    """Write all not yet existing commit files."""
    commit = os.path.join(target, "commit")
    
    # create the folders
    if not os.path.isdir(commit):
        os.makedirs(commit)
    
    t = cmdutil.changeset_templater(ui, repo, patch=False, diffopts=None, mapfile=None, buffered=False)
    t.use_template(templates["commitlog"].replace("{relativepath}", "../"))

    for c in chlist:
        ctx = repo.changectx(str(c))
        cpath = os.path.join(commit, ctx.hex() + ".html")
        if not force and os.path.isfile(cpath):
            continue
        with open(cpath, "w") as cf:
            cf.write(templates["head"].replace("{reponame}", "<a href='../'>"+name+"</a>").replace("{title}", name).replace("{nav}", "").replace("{relpath}", "../"))
            ui.pushbuffer()
            t.show(ctx)
            cf.write(ui.popbuffer())
            ui.pushbuffer()
            commands.diff(ui, repo, change=str(c), git=True)
            cf.write("<pre>"+ui.popbuffer().replace("<", "<")+"</pre>")
            cf.write(templates["foot"].replace("{reponame}", "<a href='../'>"+name+"</a>"))

def writecommits(ui, repo, target, name, force=False):
    """Write all not yet existing commit files."""
    chlist = range(len(repo.changelog))
    return writecommitsforchlist(ui, repo, target, name, chlist, force=force)

#: html escape codes thanks to http://wiki.python.org/moin/EscapingHtml
htmlescapetable = {
    "&": "&",
    '"': '"',
    "'": "'",
    ">": ">",
    "<": "<",
    }

def htmlescape(text):
    """Produce entities within text."""
    return "".join(htmlescapetable.get(c,c) for c in text)

def writebugs(ui, repo, target, name):
    """Write bug information, a listing and the details for each bug."""
    bugdir = os.path.join(target, "bugs")
    
    # create the bugs folder
    if not os.path.isdir(bugdir):
        os.makedirs(bugdir)
        
    # get all bugs
    openbugs, resolvedbugs = getbugs(ui, repo)
    # write the bugs list
    bugslist = os.path.join(bugdir, "index.html")
    content = "<h2 id=\"open\">Open Bugs</h2>\n<ul>"
    for bug in openbugs:
        content += "<li><a href=\"" + bug.fullid + ".html\">" + bug.shortid + "</a> - " + htmlescape(bug.description) + "</li>\n"
    content += "</ul>\n"
    content += "<h2 id=\"resolved\">Resolved Bugs</h2>\n<ul>"
    for bug in resolvedbugs:
        content += "<li><a href=\"" + bug.fullid + ".html\">" + bug.shortid + "</a> - " + htmlescape(bug.description) + "</li>\n"
    content += "</ul>\n"
    html = templates["head"].replace("{reponame}", "<a href='../'>"+name+"</a>").replace("{title}", name).replace("{nav}", "").replace("{relpath}", "../")
    html += content
    html += templates["foot"].replace("{reponame}", "<a href='../'>"+name+"</a>")
    if not contentequals(bugslist, html):
        with open(bugslist, "w") as f:
            f.write(html)
    # write all bug details
    for bug in openbugs + resolvedbugs:
        bugsfile = os.path.join(bugdir, bug.fullid + ".html")
        body = "<h2>" + bug.description + "</h2>\n"
        body += "<pre>" + bug.details + "</pre>\n"
        body += "<hr>"
        body += "- <a href=\"index.html\">" + _("all bugs") + "</a> -"
        content = templates["head"].replace("{reponame}", "<a href='../'>"+name+"</a>").replace("{title}", name).replace("{nav}", "").replace("{relpath}", "../")
        content += body
        content += templates["foot"].replace("{reponame}", "<a href='../'>"+name+"</a>")
        try:
            if not contentequals(bugsfile, content):
                raise Exception("bugfile content does not match content to write. Needs overwriting.")
        except Exception as e: # generic exception: If anything went wrong, we need to write the file.
            print "Overwriting bugs file", bugsfile, "; Reason:", e
            with open(bugsfile, "w") as bf:
                bf.write(content)

def escapename(filename):
    """escape index.html as .index.html and .ind… as ..ind… and so fort."""
    if _indexregexp.match(filename) is not None:
        return "." + filename
    else: return filename

def parsesrcdata(data):
    """Parse a src file into a html file."""
    return "<pre class=\"prettyprint linenums\">"+data.replace("<", "<")+"</pre>"

def srcpath(target, ctx, filename):
    """Get the relative path to the static sourcefile for an already escaped filename."""
    return os.path.join(target,"src",ctx.hex(),filename+".html")

def rawpath(target, ctx, filename):
    """Get the relative path to the static sourcefile for an already escaped filename."""
    return os.path.join(target,"raw",ctx.hex(),filename)

def ctxdiffstat(ui, repo, ctx):
    """Get the diffstat of a change context."""
    if "color" in extensions.enabled():
        command = "log -r " + ctx.hex() + " --stat --color=never"
    else:
        command = "log -r " + ctx.hex() + " --stat"
    req = dispatch.request(command.split(), ui=ui, repo=repo)
    ui.pushbuffer()
    dispatch.dispatch(req)
    # FIXME: remove the color in an elegant way instead of fudging like this.
    return ui.popbuffer().replace(
        "[0;33m","").replace(
        "[0;32m","").replace(
            "[0m", "").replace(
            "[0;31m", "").replace(
                "","")
    

def createindex(ui, repo, target, ctx):
    """Create an index page for the changecontext: the commit message + the user + all files in the changecontext."""
    # first the head
    index = templates["manifesthead"].replace(
        "{hex}", ctx.hex()).replace(
        "{desc}", ctx.description()).replace(
            "{user}", ctx.user()).replace(
                "{diffstat}", ctxdiffstat(ui, repo, ctx))
    # then the files
    index += "<ul>"
    for filename in ctx:
        filectx = ctx[filename]
        lasteditctx = filectx.filectx(filectx.filerev())
        index += "<li><a href='../../"+ os.path.join("src",lasteditctx.hex(), escapename(filename)+".html") + "'>" + filename + "</a>"# (<a href='../../" + os.path.join("raw",lasteditctx.hex(), filename) + "'>raw</a>)</li>"
    index += "</ul>"
    return index

def writesourcetreeforchlist(ui, repo, target, name, chlist, force=False, rawfiles=False):
    """Write manifests for all commits and websites for all files.
    
    * For each file, write sites for all revisions where the file was changed: under src/<hex>/path as html site (with linenumbers and maybe colored source), under raw/<hex>/<path> as plain files. If there is an index.html file, write it as .index.html. If there also is .index.html, turn it to ..index.html, …
    * For each commit write an index with links to the included files at their latest revisions before/at the commit.
    """
    # first write all files in all commits.
    for c in chlist:
        ctx = repo.changectx(str(c))
        for filename in ctx.files():
            try:
                filectx = ctx.filectx(filename)
            except LookupError, e:
                ui.debug("File not found, likely moved ", e, "\n")
            if rawfiles: 
                # first write the raw data
                filepath = rawpath(target,ctx,filectx.path())
                # skip already existing files
                if not force and os.path.isfile(filepath):
                    continue
                try:
                    os.makedirs(os.path.dirname(filepath))
                except OSError: pass # exists
                with open(filepath, "w") as f:
                    f.write(filectx.data())
            # then write it as html
            _filenameescaped = escapename(filectx.path())
            numberofslashes = len([i for i in _filenameescaped if i == "/"])
            filepath = srcpath(target,ctx,_filenameescaped)
            if not force and os.path.isfile(filepath):
                continue
            try:
                os.makedirs(os.path.dirname(filepath))
            except OSError: pass # exists
            with open(filepath, "w") as f:
                f.write(templates["srchead"].replace("{filetitle}", name+": " + filename).replace("href=\"../../", "href=\"../../" + "../"*numberofslashes))
                f.write(parsesrcdata(filectx.data()))
                f.write(templates["foot"].replace("{reponame}", name))
    # then write manifests for all commits
    for c in chlist:
        ctx = repo.changectx(str(c))
        filepath = os.path.join(target,"src",ctx.hex(),"index.html")
        # skip already existing files
        if not force and os.path.isfile(filepath):
            continue
        try:
            os.makedirs(os.path.dirname(filepath))
        except OSError: pass # exists
        content = templates["head"].replace("{reponame}", "<a href='../../'>"+name+"</a>").replace("{title}", name).replace("{nav}", "").replace("{relpath}", "../../")
        content += createindex(ui, repo, target, ctx)
        content += templates["foot"].replace("{reponame}", "<a href='../../'>"+name+"</a>")
        with open(filepath, "w") as f:
            f.write(content)


def writesourcetree(ui, repo, target, name, force, rawfiles=False):
    """Write manifests for all commits and websites for all files.
    
    * For each file, write sites for all revisions where the file was changed: under src/<hex>/path as html site (with linenumbers and maybe colored source), under raw/<hex>/<path> as plain files. If there is an index.html file, write it as .index.html. If there also is .index.html, turn it to ..index.html, …
    * For each commit write an index with links to the included files at their latest revisions before/at the commit.
    """
    chlist = range(len(repo.changelog))
    return writesourcetreeforchlist(ui, repo, target, name, chlist, force=force, rawfiles=rawfiles)

def parsesite(ui, repo, target, **opts):
    """Create the static folder."""
    idfile = os.path.join(target, _staticidentifier)
    if not os.path.isdir(target):
        # make sure the target exists
        os.makedirs(target)
    else: # make sure it is a staticrepo
        if not os.path.isfile(idfile):
            if not ui.prompt("The target folder " + target + " has not yet been used as static repo. Really use it? (y/N)", default="n").lower() in ["y", "yes"]:
                return
            with open(idfile, "w") as i:
                i.write("")
                
    if opts["sitename"]:
        name = opts["sitename"]
    elif target != "static": name = target
    else: name = os.path.basename(repo.root)
    
    # first the stylesheets
    screenstyle = opts["screenstyle"]
    screenfile = os.path.join(target, "style.css")
    if screenstyle and not samefilecontent(screenstyle, screenfile):
        shutil.copyfile(screenstyle, screenfile)
    elif not contentequals(screenfile,templates["screenstyle"]):
        with open(screenfile, "w") as f:
            f.write(templates["screenstyle"])
    printstyle = opts["printstyle"]
    printfile = os.path.join(target, "print.css")
    if printstyle and not samefilecontent(printstyle, printfile):
        shutil.copyfile(printstyle, printfile)
    elif not contentequals(printfile, templates["printstyle"]):
        with open(printfile, "w") as f:
            f.write(templates["printstyle"])
    # add google code prettify for source code highlighting
    for pretty in ["prettify.js", "prettify.css"]:
        highlightingsrc = os.path.join(os.path.dirname(__file__), pretty)
        highlightingfile = os.path.join(target, pretty)
        if not os.path.isfile(highlightingfile) or not samefilecontent(highlightingsrc, highlightingfile):
            shutil.copyfile(highlightingsrc, highlightingfile)
    # and add steve loshs sidebar-title
    for sjl in ["sjl.js", "jquery.js", "jquery.timeago.js", "print.js"]:
        sjlsrc = os.path.join(os.path.dirname(__file__), sjl)
        sjlfile = os.path.join(target, sjl)
        if not os.path.isfile(sjlfile) or not samefilecontent(sjlsrc, sjlfile):
            shutil.copyfile(sjlsrc, sjlfile)
    
    ui.debug("[staticsite] writeoverview\n")
    # then the overview
    writeoverview(ui, repo, target, name)
    
    ui.debug("[staticsite] writelog\n")
    # and the log
    writelog(ui, repo, target, name)
    
    ui.debug("[staticsite] writecommits\n")
    # and all commit files
    writecommits(ui, repo, target, name, force=opts["force"])
    
    ui.debug("[staticsite] writesourcetree\n")
    # and all file data
    writesourcetree(ui, repo, target, name, force=opts["force"])
    
    ui.debug("[staticsite] writeforks\n")
    # and all forks
    writeforks(ui, repo, target, name)
    
    ui.debug("[staticsite] writebugs\n")
    # and all bugs
    writebugs(ui, repo, target, name)

def addrepo(ui, repo, target, bookmarks, force):
    """Add the repo to the target and make sure it is up to date."""
    try:
        commands.init(ui, dest=target)
    except mercurial.error.RepoError, e:
        # already exists
        pass
    
    ui.pushbuffer()
    if bookmarks: 
        commands.push(ui, repo, dest=target, bookmark=repo._bookmarks, force=force)
    else: 
        commands.push(ui, repo, dest=target, force=force)
    ui.popbuffer()

def upload(ui, repo, target, ftpstring, force):
    """upload the repo to the FTP server identified by the ftp string."""
    # assume ftps if no prefix was given (security by default).
    isftps = not ftpstring.startswith("ftp://")
    if ftpstring.startswith("ftp://"):
        ftpstring = ftpstring[len("ftp://"):]
    if ftpstring.startswith("ftps://"):
        ftpstring = ftpstring[len("ftps://"):]
    try:
        user, password = ftpstring.split("@")[0].split(":")
        serverandpath = "@".join(ftpstring.split("@")[1:])
    except ValueError:
        ui.warn(_("FTP-upload: No @ in FTP-Url. We try anonymous access.\n"))
        user, password = "anonymous", ""
        serverandpath = ftpstring # no @, so we just take the whole string
    server = serverandpath.split("/")[0]
    ftppath = "/".join(serverandpath.split("/")[1:])
    timeout = 60
    # Use FTP instead of FTPS, if the uri is ftp://
    try:
        if isftps:
            ftp = ftplib.FTP_TLS(server, user, password, "", timeout=timeout)
            ftp.prot_p()
        else:
            ftp = ftplib.FTP(server, user, password, "", timeout=timeout)
    except socket.timeout:
        ui.warn(_("connection to "), server, _(" timed out after "), timeout, _(" seconds.\n"))
        return
    
    ui.status(ftp.getwelcome(), "\n")
    
    # create the target dir.
    serverdir = os.path.dirname(ftppath)
    serverdirparts = ftppath.split("/")
    sd = serverdirparts[0]
    if not sd in ftp.nlst():
        ftp.mkd(sd)
    for sdp in serverdirparts[1:]:
        sdo = sd
        sd = os.path.join(sd, sdp)
        if not sd in ftp.nlst(sdo):
            ftp.mkd(sd)
            
            
    ftp.cwd(ftppath)
    if not ftp.pwd() == "/" + ftppath:
        ui.warn(_("not in the correct ftp directory. Cowardly bailing out.\n"))
        return
    
    #ftp.dir()
    #return
    ftpfeatures = ftp.sendcmd("FEAT")
    featuremtime = " MDTM" in ftpfeatures.splitlines()
    _ftplistcache = set()
    
    for d, dirnames, filenames in os.walk(target):
        for filename in filenames:
            localfile = os.path.join(d, filename)
            serverfile = localfile[len(target)+1:]
            serverdir = os.path.dirname(serverfile)
            serverdirparts = serverdir.split("/")
            # print serverdirparts, serverfile
            with open(localfile, "rb") as f:
                sd = serverdirparts[0]
                if sd and not sd in _ftplistcache: # should happen only once per superdir
                    _ftplistcache.update(set(ftp.nlst()))
                if sd and not sd in _ftplistcache:
                    try:
                        ui.status(_("creating directory "), sd, "\n")
                        ftp.mkd(sd)
                        _ftplistcache.add(sd)
                    except ftplib.error_perm, resp:
                        ui.warn(_("could not create directory "), sd, ": " , resp, "\n")
                    else: _ftplistcache.add(sd)
                    
                for sdp in serverdirparts[1:]:
                    sdold = sd
                    sd = os.path.join(sd, sdp)
                    #print sd, sdp
                    #print ftp.nlst(sdold)
                    if sd and not sd in _ftplistcache: # should happen only once per superdir
                        _ftplistcache.update(set(ftp.nlst(sdold)))
                    if sd and not sd in _ftplistcache:
                        try:
                            ui.status(_("creating directory "), sd, "\n")
                            ftp.mkd(sd)
                            _ftplistcache.add(sd)
                        except ftplib.error_perm, resp:
                            ui.warn(_("could not create directory "),
                                    sd, ": " , resp, "\n")
                            
                if not serverfile in _ftplistcache: # should happen for existing files only once per dir.
                    _ftplistcache.update(set(ftp.nlst(serverdir)))
                if not serverfile in _ftplistcache or force:
                    if force:
                        ui.status(_("uploading "), serverfile,
                                  _(" because I am forced to.\n"))
                    else:
                        ui.status(_("uploading "), serverfile,
                                  _(" because it is not yet online.\n"))
                        
                    ftp.storbinary("STOR "+ serverfile, f)
                else:
                    # reupload the file if the file on the server is older than the local file.
                    if featuremtime:
                        ftpmtime = ftp.sendcmd("MDTM " + serverfile).split()[1]
                        localmtime = os.stat(localfile).st_mtime
                        localmtimestr = datetime.datetime.utcfromtimestamp(localmtime).strftime("%Y%m%d%H%M%S")
                        newer = int(localmtimestr) > int(ftpmtime)
                        if newer:
                            ui.status(_("uploading "), serverfile,
                                      _(" because it is newer than the file on the FTP server.\n"))
                            ftp.storbinary("STOR "+ serverfile, f)

def staticsite(ui, repo, target=None, **opts):
    """Create a static copy of the repository and/or upload it to an FTP server."""
    if repo.root == target:
        ui.warn(_("static target repo can’t be the current repo"))
        return
    if not target: target = "static"
    #print repo["."].branch()
    # add the hg repo to the static site
    # currently we need to either include all bookmarks or not, because we don’t have the remote repo when parsing the site.
    # TODO: I don’t know if that is the correct way to go. Maybe always push all.
    bookmark = opts["bookmark"]
    addrepo(ui, repo, target, bookmark, force=opts["force"])
    # first: just create the site.
    parsesite(ui, repo, target, **opts)
    if opts["upload"]:
        # upload the repo
        upload(ui, repo, target, opts["upload"], opts["force"])

cmdtable = {
    # "command-name": (function-call, options-list, help-string)
    "site": (staticsite,
                     [
                      #('r', 'rev', None, 'parse the given revision'),
                      #('a', 'all', None, 'parse all revisions (requires much space)'),
                      ('n', 'sitename', "", 'the repo name. Default: folder or last segment of the repo-path.'),
                      ('u', 'upload', "", 'upload the repo to the given ftp host. Format: user:password@host/path/to/dir'),
                      ('f', 'force', False, 'force recreating all commit files. Slow.'),
                      ('s', 'screenstyle', "", 'use a custom stylesheet for display on screen'),
                      ('p', 'printstyle', "", 'use a custom stylesheet for printing'),
                      ('B', 'bookmark', False, 'include the bookmarks')],
                     "[options] [folder]")
}

## add ftp as scheme to be handled by this plugin.

wrapcmds = { # cmd: generic, target, fixdoc, ppopts, opts
    'push': (False, None, False, False, [
        ('', 'staticsite', None, 'upload a static site'),
    ])
}

## Explicitely wrap functions to change local commands in case the remote repo is an FTP repo. See mercurial.extensions for more information.
# Get the module which holds the functions to wrap
# the new function: gets the original function as first argument and the originals args and kwds.
def findcommonoutgoing(orig, *args, **opts):
    repo = args[1]
    capable = getattr(repo, 'capable', lambda x: False)
    if capable('ftp'):
        class fakeoutgoing(object):
            def __init__(self):
                self.excluded = []
                self.missing = []
                self.commonheads = []
        return fakeoutgoing()
    else:
        return orig(*args, **opts)
# really wrap the functions
extensions.wrapfunction(discovery, 'findcommonoutgoing', findcommonoutgoing)

# explicitely wrap commands in case the remote repo is an FTP repo.
def ftppush(orig, *args, **opts):
    try:
        ui, repo, path = args
        path = ui.expandpath(path)
    except ValueError: # no ftp string
        ui, repo = args
        path = ui.expandpath('default-push', 'default')
    # only act differently, if the target is an FTP repo.
    if not path.startswith("ftp"):
        return orig(*args, **opts)
    # first create the site at ._site
    target = "._site"
    # fix the options to fit those of the site command
    opts["name"] = opts["sitename"]
    opts["upload"] = path
    staticsite(ui, repo, target, **opts)
    return 0
        
# really wrap the command
siteopts = [('', 'sitename', "", 'staticsite: the title of the site. Default: folder or last segment of the repo-path.'),
            ('', 'screenstyle', "", 'use a custom stylesheet for display on screen'),
            ('', 'printstyle', "", 'use a custom stylesheet for printing')]
entry = extensions.wrapcommand(commands.table, "push", ftppush)
entry[1].extend(siteopts)

# Starting an FTP repo. Not yet used, except for throwing errors for missing commands and faking the lock.

# TODO: repo -> peer
from mercurial import util
try:
    from mercurial.peer import peerrepository
except ImportError:
    from mercurial.repo import repository as peerrepository
try:
    from mercurial.error import RepoError
except ImportError:
    from mercurial.repo import RepoError

# TODO: repo -> peer
class FTPRepository(peerrepository):
    def __init__(self, ui, path, create):
        self.create = create
        self.ui = ui
        self.path = path
        self.capabilities = set(["ftp", "ftps"])

    def lock(self):
        """We cannot really lock FTP repos, yet.

        TODO: Implement as locking the repo in the static site folder."""
        class DummyLock:
            def release(self):
                pass
        l = DummyLock()
        return l

    def url(self):
        return self.path

    def lookup(self, key):
        return key

    def cancopy(self):
        return False

    def heads(self, *args, **opts):
        """
        Whenever this function is hit, we abort. The traceback is useful for
        figuring out where to intercept the functionality.
        """
        raise util.Abort('command heads unavailable for FTP repositories')

    def pushkey(self, namespace, key, old, new):
        return False

    def listkeys(self, namespace):
        return {}

    def push(self, remote, force=False, revs=None, newbranch=None):
        raise util.Abort('command push unavailable for FTP repositories')
    
    def pull(self, remote, heads=[], force=False):
        raise util.Abort('command pull unavailable for FTP repositories')
    
    def findoutgoing(self, remote, base=None, heads=None, force=False):
        raise util.Abort('command findoutgoing unavailable for FTP repositories')


class FTPRepoContainer(object):
    def __init__(self):
        pass

    def __repr__(self):
        return '<FTPRepository>'

    def instance(self, ui, url, create):
        # Should this use urlmod.url(), or is manual parsing better?
        #context = {}
        return FTPRepository(ui, url, create)

class FTPSRepoContainer(object):
    def __init__(self):
        pass

    def __repr__(self):
        return '<FTPRepository>'

    def instance(self, ui, url, create):
        # Should this use urlmod.url(), or is manual parsing better?
        #context = {}
        return FTPRepository(ui, url, create)

hg.schemes["ftp"] = FTPRepoContainer()
hg.schemes["ftps"] = FTPSRepoContainer()

def test():
    import subprocess as sp
    def showcall(args):
        print args
        sp.call(args)
    os.chdir(os.path.dirname(__file__))
    # just check if loading the extension works
    showcall(["hg", "--config", "extensions.site="+__file__])
    # check if I can create a site
    showcall(["hg", "--config", "extensions.site="+__file__, "site", "-f", "-B", "-n", "mysite"])
    # check if uploading works: Only a valid test, if you have a
    # post-push hook which does the uploading
    # showcall(["hg", "--config", "extensions.site="+__file__, "push"])
    # check if push directly to ftp works. Requires the path draketo
    # to be set up in .hg/hgrc as ftp://user:password/path
    # showcall(["hg", "--config", "extensions.site="+__file__, "push", "draketo", "--sitename", "hg site extension"])

if __name__ == "__main__":
    test()