# -*- python -*-
# ex: set syntax=python:

# This is a sample buildmaster config file. It must be installed as
# "master.cfg" in your buildmaster's base directory.

import re

# This is the dictionary that the buildmaster pays attention to. We
# also use a shorter alias to save typing.
c = BuildmasterConfig = {}

####### BUILDSLAVES

# The "slaves" list defines the set of recognized buildslaves. Each
# element is a BuildSlave object, specifying a unique slave name and
# password.  The same slave name and password must be configured on
# the slave.
from buildbot.buildslave import BuildSlave

try:
    from secret_passwords import buildslave_passwords, webauth_passwords, \
            autodoc_bot
except ImportError:
    raise SystemExit("Fatal error: Is secret_passwords.py missing?\n"
                     "See code comments.")
webauth_passwords = webauth_passwords.items()

## secret_passwords.py is a file that should not be committed to the repository
## and should look like this:
## buildslave_passwords = {
##    "bot1": "password-of-bot1",
##    "bot2": "password-of-bot2",
##    ...
##    }
## autodoc_bot = "bot2"

SLAVES = sorted(buildslave_passwords.keys())

c["slaves"] = [BuildSlave(slave, buildslave_passwords[slave]) for slave in SLAVES]


# "slavePortnum" defines the TCP port to listen on for connections from slaves.
# This must match the value configured into the buildslaves (with their
# --master option)
c["slavePortnum"] = 9989

####### CHANGESOURCES

# the "change_source" setting tells the buildmaster how it should find out
# about source code changes.

from buildbot.changes.pb import PBChangeSource
c["change_source"] = PBChangeSource(port="tcp:9987:interface=127.0.0.1")

####### SCHEDULERS

# Configure the Schedulers, which decide how to react to incoming
# changes.

from buildbot.schedulers.basic import SingleBranchScheduler, Dependent
from buildbot.schedulers.timed import Nightly
from buildbot.changes import filter

schedulers_to_slaves = {}

for scheduler in ["build-quick", "build-full", "nightly", "weekly"]:
    schedulers_to_slaves[scheduler] = SLAVES
schedulers_to_slaves["autodoc"] = [autodoc_bot]

c["schedulers"] = []

def add_scheduler(name, cls, **kwargs):
    builder_names = ["%s-%s" % (name, slave_name)
                     for slave_name in schedulers_to_slaves[name]]
    if cls is SingleBranchScheduler and 'change_filter' not in kwargs:
        kwargs['change_filter'] = filter.ChangeFilter(branch="default")
    scheduler = cls(
            name=name,
            builderNames=builder_names,
            **kwargs)
    c["schedulers"].append(scheduler)
    return scheduler

# Scheduler for quick (incremental) builds: run after every change.
build_quick = add_scheduler(name="build-quick",
                            cls=SingleBranchScheduler,
                            treeStableTimer=None)

# Scheduler for full (clean) builds: run after every change.
# Also runs a brief translator test.
add_scheduler(name="build-full",
              cls=SingleBranchScheduler,
              treeStableTimer=None)

# Scheduler for a test that runs every night at 3:00 AM.
add_scheduler(name="nightly",
              cls=Nightly,
              branch="default",
              hour=3,
              minute=0,
              onlyIfChanged=True)

# Scheduler for a test that runs every Saturday at 8:00 AM.
add_scheduler(name="weekly",
              cls=Nightly,
              branch="default",
              dayOfWeek=5,
              hour=8,
              minute=0,
              onlyIfChanged=True)

# Scheduler autodoc: run after successful quick builds.
add_scheduler(name="autodoc",
              cls=Dependent,
              upstream=build_quick)

####### BUILDERS

# The "builders" list defines the Builders, which tell Buildbot how to
# perform a build: what steps, and which slaves can execute them.
# Note that any particular build will only take place on one slave.

from buildbot.process.factory import BuildFactory
from buildbot.steps.source.mercurial import Mercurial
from buildbot.steps.shell import Compile, ShellCommand

REPO_URL = "http://hg.fast-downward.org/"

incremental_update_step = Mercurial(
    name="incremental_update",
    repourl=REPO_URL,
    mode="incremental",
    branchType="inrepo",
    haltOnFailure=True,
    warnOnWarnings=True)

clean_update_step = Mercurial(
    name="clean_update",
    repourl=REPO_URL,
    mode="full",
    method="clobber",
    branchType="inrepo",
    haltOnFailure=True,
    warnOnWarnings=True)

build_all_step = Compile(
    name="build_all",
    command=["./build_all"],
    workdir="build/src",
    haltOnFailure=True,
    warnOnWarnings=True)

nightly_planner_step = ShellCommand(
    name="nightly_planner",
    command=["./buildbot-exp.py", "--test", "nightly", "--all"],
    workdir="build/misc/buildbot",
    description=["running", "planner", "tests"],
    descriptionDone=["planner", "tests"],
    flunkOnFailure=True,
    warnOnWarnings=True)

weekly_planner_step = ShellCommand(
    name="weekly_planner",
    command=["./buildbot-exp.py", "--test", "weekly", "--all"],
    workdir="build/misc/buildbot",
    description=["running", "planner", "tests"],
    descriptionDone=["planner", "tests"],
    flunkOnFailure=True,
    warnOnWarnings=True)

small_translator_step = ShellCommand(
    name="small_translator",
    command=["./test-translator.py"],
    workdir="build/misc/buildbot",
    description=["running", "translator", "tests"],
    descriptionDone=["translator", "tests"],
    flunkOnFailure=True,
    warnOnWarnings=True)

medium_translator_step = ShellCommand(
    name="medium_translator",
    command=["./test-translator.py", "first"],
    workdir="build/misc/buildbot",
    description=["running", "translator", "tests"],
    descriptionDone=["translator", "tests"],
    flunkOnFailure=True,
    warnOnWarnings=True)

autodoc_step = ShellCommand(
    name="autodoc",
    command=["./autodoc.py"],
    workdir="build/misc/autodoc",
    description=["running", "autodoc"],
    descriptionDone=["autodoc"],
    flunkOnFailure=True,
    warnOnWarnings=True)


factory_dict = {}

factory_dict["build-quick"] = BuildFactory([
        incremental_update_step,
        build_all_step,
])
factory_dict["build-quick"].useProgress = False

factory_dict["build-full"] = BuildFactory([
        clean_update_step,
        build_all_step,
        small_translator_step,
        ])

factory_dict["nightly"] = BuildFactory([
        clean_update_step,
        build_all_step,
        nightly_planner_step,
        medium_translator_step,
        ])

factory_dict["weekly"] = BuildFactory([
        clean_update_step,
        build_all_step,
        weekly_planner_step,
        medium_translator_step,
        ])

factory_dict["autodoc"] = BuildFactory([
        incremental_update_step,
        autodoc_step,
])
factory_dict["autodoc"].useProgress = False


from buildbot.config import BuilderConfig

c["builders"] = []
for scheduler, factory in sorted(factory_dict.items()):
    for slave in schedulers_to_slaves[scheduler]:
        c["builders"].append(BuilderConfig(
                name="%s-%s" % (scheduler, slave),
                slavename=slave,
                factory=factory,
                mergeRequests=False))


####### STATUS TARGETS

# "status" is a list of Status Targets. The results of each build will be
# pushed to these targets. buildbot/status/*.py has a variety to choose from,
# including web pages, email senders, and IRC bots.

c["status"] = []

from buildbot.status.html import WebStatus
from buildbot.status.web.auth import BasicAuth
from buildbot.status.web.authz import Authz

authz_cfg = Authz(
    auth=BasicAuth(webauth_passwords),
    gracefulShutdown="auth",
    forceBuild="auth",
    forceAllBuilds="auth",
    pingBuilder="auth",
    stopBuild="auth",
    stopAllBuilds="auth",
    cancelPendingBuild="auth",
)

c["status"].append(WebStatus(
        http_port="tcp:8010:interface=127.0.0.1",
        authz=authz_cfg))


from zope.interface import implements

from buildbot import interfaces
from buildbot.status.mail import MailNotifier

class ParseEmail(object):
    implements(interfaces.IEmailLookup)
    def getAddress(self, name):
        """John Doe <john.doe@abc.com> --> john.doe@abc.com"""
        match = re.match(r".*<(.+@.+)>\s*$", name)
        if match:
            return match.group(1)
        return None

c["status"].append(MailNotifier(
        fromaddr="malte.helmert@unibas.ch",
        sendToInterestedUsers=True,
        extraRecipients=["malte.helmert@unibas.ch"],
        mode="warnings",
        lookup=ParseEmail()))

####### PROJECT IDENTITY

# the "title" string will appear at the top of this buildbot
# installation's html.WebStatus home page (linked to the
# "titleURL") and is embedded in the title of the waterfall HTML page.

c["title"] = "Fast Downward"
c["titleURL"] = "http://www.fast-downward.org/"

# the "buildbotURL" string should point to the location where the buildbot's
# internal web server (usually the html.WebStatus page) is visible. This
# typically uses the port number set in the Waterfall "status" entry, but
# with an externally-visible host name which the buildbot cannot figure out
# without some help.

c["buildbotURL"] = "http://buildbot.fast-downward.org/"

####### DB URL

# This specifies what database buildbot uses to store change and scheduler
# state.  You can leave this at its default for all but the largest
# installations.
c["db_url"] = "sqlite:///state.sqlite"
