using System.Collections.Generic;
const int PLAYER_COUNT = 32;
const int ROUND_COUNT = 5;
const int POINTS_FOR_WIN = 15;
const int POINTS_FOR_TIE = 10;
const int POINTS_FOR_LOSS = 5;
const int MIN_BONUS = -5;
static readonly Random RANDOM = new Random();
public static void Main()
IEnumerable<Player> players = Enumerable.Range(1, PLAYER_COUNT)
.Select((id) => new Player(id)).ToList();
for(int round = 1; round <= ROUND_COUNT; ++round)
Console.WriteLine("ROUND {0}", round);
IEnumerable<Tuple<Player, Player>> pairs = PairPlayers(players);
foreach(Tuple<Player, Player> pair in pairs)
pair.Item1.Matches.Add(new Match(pair.Item1, pair.Item2));
pair.Item2.Matches.Add(new Match(pair.Item1.Matches.Last()));
foreach(Player player in players.OrderByDescending((p) => p.Score))
Match match = player.Matches.Last();
Console.WriteLine("{0,-2} {1,-8} vs {2,-2} {3,-8}: {4,-4} ({5,-2} + {6,-2}) {7}",
string.Format("({0}-{1}-{2})", player.WinCount, player.TieCount, player.LossCount),
string.Format("({0}-{1}-{2})", match.Opponent.WinCount, match.Opponent.TieCount, match.Opponent.LossCount),
public static IEnumerable<Tuple<Player, Player>> PairPlayers(IEnumerable<Player> players)
List<Player> ordered = players
.GroupBy((player) => player.Score)
.OrderBy((grp) => grp.Key)
.Select((grp) => grp.Shuffle())
.SelectMany((grp) => grp).ToList();
Player first = ordered.First();
Player second = ordered.Skip(1).First((p) => !first.Matches.Select((m) => m.Opponent).Contains(p));
yield return Tuple.Create(first, second);
public static IEnumerable<T> Shuffle<T>(this IEnumerable<T> values)
IList<T> list = values.ToList();
for(int i = 0; i < list.Count; ++i)
int j = RANDOM.Next(i, list.Count);
public readonly IList<Match> Matches;
this.Matches = new List<Match>(ROUND_COUNT);
return this.Matches.Select((match) => match.Score).Sum();
return this.Matches.Count((match) => match.Result == MatchResult.Win);
return this.Matches.Count((match) => match.Result == MatchResult.Tie);
return this.Matches.Count((match) => match.Result == MatchResult.Loss);
public readonly Player Player;
public readonly Player Opponent;
public readonly MatchResult Result;
public readonly int Bonus;
public Match(Player player, Player opponent)
this.Opponent = opponent;
this.Result = MatchResult.Random();
this.Bonus = RANDOM.Next(MIN_BONUS, MAX_BONUS);
public Match(Match match)
this.Player = match.Opponent;
this.Opponent = match.Player;
this.Result = match.Result.Reverse;
this.Bonus = RANDOM.Next(MIN_BONUS, MAX_BONUS);
return this.Result.Score + this.Bonus;
public static MatchResult Win = new MatchResult("Win", POINTS_FOR_WIN);
public static MatchResult Tie = new MatchResult("Tie", POINTS_FOR_TIE);
public static MatchResult Loss = new MatchResult("Loss", POINTS_FOR_LOSS);
public readonly string Name;
public readonly int Score;
private MatchResult(string name, int score)
public MatchResult Reverse
if (object.ReferenceEquals(this, Win))
else if (object.ReferenceEquals(this, Tie))
else if (object.ReferenceEquals(this, Loss))
throw new ArgumentException("Unknown MatchResult.");
public override string ToString()
public static MatchResult Random()
int value = RANDOM.Next(0, 3);