Hi! This is the homepage of drawnneal, a piece of free software that can help you create optimized draws for your tournament.
If you are instead looking to generate schedules for appointments, have a look at the related software appoinneal.
If you only need a simple, one-venue round-robin draw - just try the online draw maker, which is also free software (available with documentation here).
Otherwise please read the following notes and contact me in case of question.
GETTING DRAWNNEAL
The most recent version of drawnneal, 0.6, is available from the Sourceforge project page
here.
WHAT DRAWNNEAL DOES
Imagine you were to create a draw for, say, a student debating tournament.
The tournament will be made up of a number of rounds which will be spread
over the year. For each round you'll possibly have a number of venues. You
want your teams not to have to travel too far, while you do need to prevent
pairs to be repeated. It'd be unfair if some teams were facing strong teams
in each and every round while others had a much less difficult time. Teams
new to the tournament might urge you to give them debates against other
newbies for the first rounds. And if somehow possible you would also like
to equally distribute sides (proposition/opposition in debates, black/white
in chess, whatever else).
The problem you will be facing is indeed a complex one. No self-respecting human could ever
try all possible options in order to optimize the draw. But that's, after
all, what we have computers for ;)
Drawnneal will create and optimize such draws using a simulated annealing
algorithm. Draws are rated according to their compliance with the criteria named above
and slight changes are applied to them, in the beginning possibly even
making them worse, then slowly but surely improving them to something close
to a global optimum. Your draw will be made in no time (well, almost) and
better than any human could ever make it.
BUILDING DRAWNNEAL
You just have to compile drawnneal.cpp using a c++ compiler. For typical
GNU/Linux systems you'll probably be using gcc/g++ and a simple
make
will suffice. If you happen to have a different compiler installed please
see the appropriate documentation for how compile 'drawnneal.cpp' into an
executable 'drawnneal'.
RUNNING DRAWNNEAL
Drawnneal, I'm afraid, has no graphical user interface. For this reason
all of the following presupposes that you are running a terminal, the
working directory being
the drawnneal source directory.
A simple
./drawnneal
will start drawnneal in interactive mode. You will be asked a couple of
things about all the days your tournament will have (days consist of a
number of rounds with fixed venues, not necessarily limited to one day),
the rounds and venues and all the teams. Rough estimations of the travel
distances should suffice. For teams that need to be at a specific venue for
a specific day just enter very high travel distances for all the other
venues on that day in order to penalize them being sent to another venue.
RUNNING DRAWNNEAL SMARTLY
As you might have noticed running drawnneal in interactive mode does not
exactly meet today's standard of user interfaces. So that's the reason I
recommend the following, even more low level method of entering your data.
It doesn't really meet today's standards either, but for me it does the trick.
Create a file named 'input.txt' and fill it with the following lines
(only numbers or strings for team / venue names!)
I) Teams and Days
01: number of teams
02: number of days (= groups of rounds with constant venues)
03: number of experience levels for fairness optimization,
1 for don't use (then skip lines concerning experience levels);
the higher the level, the more experiences a team
04: level up to which teams are considered as beginners and therefore get
easier matches in first rounds (0 for don't use)
This is the general structure of your draw. For instance, if you want 14
teams, your tournament has 3 distinct parts, you want to use two
experience levels the lower of which is for beginners you may write:
14
3
2
1
The next section deals with
II) Venues and Rounds
This section contains loops for each day, venue and round:
D01: number of venues for day i
D02: number of rounds on day i
V01: name of venue j on day i
R01: type of round k at venue j on day i;
rounds can have two different types
(e.g. in debating: prepared/impromptu)
marked by '1' or '2'; in case you don't care, just pick '1'
V02: number of teams at venue (except for last venue of the day)
So if for the three days you want to have the first will have two venues
called 'Atown' (6 teams) and 'Btown' (the rest), the second will be
entirely placed in 'Capital City' and the third one shared by 'DTown' (4
teams), 'ETown' (6 teams) and 'FTown' you will need the following
lines. Note how for each round there is a '1' or '2' saying whether the
round will be prepared or impromptu and that for the last venue of each
day the number of teams need not be specified.
2
2
Atown
1
2
6
Btown
1
2
1
5
Capital City
1
1
2
2
1
3
1
Dtown
2
4
Etown
2
6
Ftown
2
III) Teams
The third section will contain data about each team:
T01: name of the team
T02: experience level, only if limit>1 specified above
integer number between 1 and the limit specified above
T03: distance to each venue in km, in the same order as venues were entered above
T04: number of teams this team shouldn't be paired against
T05: numbers of these teams this team shouldn't be paired against,
as you entered them, between 1 and the total number of teams
So as we have 14 teams in our example and six venues, this is how the section might look:
Atown Aces
2
1
5000
200
100
150
300
0
Btown B-ginners
1
5000
1
100
130
20
200
0
Capital Captains
2
100
170
1
300
220
150
1
4
St. Elsewhere Eagles
2
120
40
100
300
450
220
0
[... and so on for the rest of the 14 teams ...]
Please note how the experience level is picked from either '1' or '2'
because that was the limit we specified.
Also note that we gave extremely high distances for the teams from Atown
or Btown to the other venues in the first round when they had a venue
in their own town (where they should by all means stay) but still
haven't used a 0 distance (you should never do that).
We have excluded the pairing of the Capital Captains against the St.
Elsewhere Eagles (therefore the 1 and 4 in the closing lines of the
Captains). Something like this might be reasonable in a couple of
situations, e.g. if you're completing on a draw that's already been
going for a few rounds or have some teams that shouldn't meet for other
reasons. Usually, you would just give a 0 on the line after all
distances and it'll all be fine.
IV) Random Seed
The output of drawnneal will only be fake-random, meaning that starting
with a certain integer and unchanged parameters the programm will always
create the same draw. The last line of your 'input.txt' will contain
that integer. You may as well simply pick '1', your birth year or the
outcome of a good old roll of the dice. For our example we took a very
meaningful number to do that job for us:
42
V) Finally running drawnneal
That's it. If you call drawnneal with the command line
./drawnneal < input.txt
your parameters will be read and the draw should be optimized. If you
want to save your output directly to a file you may as well call
./drawnneal < input.txt > output.txt
In case you want to create several different draws, make sure to for
each of the draws change the random seed in your 'input.txt' first.
HACKING DRAWNNEAL
You may change some of the internal parameters of drawnneal by editing
the source file and recompiling (with a simple 'make'). This is less
difficult than it may sound and it might help you create even better
draws.
i) annealing parameters
You can change the parameters thot (initial temperature, default: 10.0),
tfactor (factor by which temperature is multiplied with each step,
should be lower than 1.0, default: 0.999998) or freeze_timeout (number
of unsuccessful steps before draw is considered optimized, default:
10000) to change the annealing process.
ii) rating parameters
How the different criteria mentioned are judged in comparison depends
essentially on the penalties for deviation from the ideal configuration a
draw gets. You can change either of the 'malus' factors
(repeatpairmalus, unequal_p_malus for unequal side distributions,
unequal_ip_malus for unequal side distributions for the two different
kinds of rounds, newteamunfairnessmalus for hard draws for new teams in
the first rounds, generalunfairnessmalus for unequal distribution of
hard draws) and see what happens.
iii) rating procedure
Deeper changes to the rating procedure can be made by editing the function 'double rateDraw'. Take care :)
COPYING AND DISTRIBUTING DRAWNNEAL, LEGAL STUFF
This program is free software; you can redistribute it and/or modify
it under the terms of the GNU General Public License as published by
the Free Software Foundation; either version 3, or (at your option)
any later version.
This program is distributed in the hope that it will be useful, but
WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for more details.
CONTACTING THE AUTHOR
Drawnneal has been written an is maintained by Daniel Gruen.
Don't hesitate to contact me to send comments, bug reports, wishlists or even money ;)