module ICU
class Tournament
=begin rdoc
== Foreign CSV
This is a format ({specification}[http://www.icu.ie/articles/display.php?id=172]) used by the ICU[http://icu.ie]
for players to submit their individual results in foreign tournaments for domestic rating.
Suppose, for example, that the following data is the file tournament.csv:
Event,"Isle of Man Masters, 2007"
Start,2007-09-22
Rounds,9
Website,http://www.bcmchess.co.uk/monarch2007/
Player,456,Fox,Anthony
1,0,B,Taylor,Peter P.,2209,,ENG
2,=,W,Nadav,Egozi,2205,,ISR
3,=,B,Cafolla,Peter,2048,,IRL
4,1,W,Spanton,Tim R.,1982,,ENG
5,1,B,Grant,Alan,2223,,SCO
6,0,-
7,=,W,Walton,Alan J.,2223,,ENG
8,0,B,Bannink,Bernard,2271,FM,NED
9,=,W,Phillips,Roy,2271,,MAU
Total,4
This file can be parsed as follows.
parser = ICU::Tournament::ForeignCSV.new
tournament = parser.parse_file('tournament.csv')
If the file is correctly specified, the return value from the parse_file method is an instance of
ICU::Tournament (rather than nil, which indicates an error). In this example the file is valid, so:
tournament.name # => "Isle of Man Masters, 2007"
tournament.start # => "2007-09-22"
tournament.rounds # => 9
tournament.website # => "http://www.bcmchess.co.uk/monarch2007/"
The main player (the player whose results are being reported for rating) played 9 rounds
but only 8 other players (he had a bye in round 6), so the total number of players is 9.
tournament.players.size # => 9
Each player has a unique number for the tournament. The main player always occurs first in this type of file, so his number is 1.
player = tournament.player(1)
player.name # => "Fox, Anthony"
This player has 4 points from 9 rounds but only 8 of his results are are rateable (because of the bye).
player.points # => 4.0
player.results.size # => 9
player.results.find_all{ |r| r.rateable }.size # => 8
The other players all have numbers greater than 1.
opponents = tournamnet.players.reject { |o| o.num == 1 }
There are 8 opponents (of the main player) each with exactly one game.
opponents.size # => 8
opponents.find_all{ |o| o.results.size == 1 }.size # => 8
However, none of the opponents' results are rateable because they are foreign to the domestic rating list
to which the main player belongs. For example:
opponent = tournament.players(2)
opponent.name # => "Taylor, Peter P."
opponent.results[0].rateable # => false
If the file contains errors, then the return value from parse_file is nil and
an error message is returned by the error method of the parser object. The method
parse_file! is similar except that it raises errors, and the methods parse
and parse! are similar except their inputs are strings rather than file names.
A tournament can be serialized back to CSV format (the reverse of parsing) with the _serialize_ method
of the parser object.
csv = parser.serialize(tournament)
Or equivalently, the _serialize_ instance method of the tournament, if the appropriate parser name is supplied.
csv = tournament.serialize('ForeignCSV')
Extra condtions, over and above the normal validation rules, apply before any tournament validates or can be serialized in this format.
* the tournament must have a _site_ attribute
* there must be at least one player with an _id_ (interpreted as an ICU ID number)
* all foreign players (those without an ICU ID) must have a _fed_ attribute (federation)
* all ICU players must have a result in every round (even if it is just bye or is unrateable)
* the opponents of all ICU players must have a federation (this could include other ICU players)
If any of these are not satisfied, then the following method calls will all raise an exception:
tournament.validate!(:type => 'ForeignCSV')
tournament.serialize('ForeignCSV')
ICU::Tournament::ForeignCSV.new.serialize(tournament)
You can also build the tournament object from scratch using your own data and then serialize it.
For example, here are the commands to reproduce the example above.
t = ICU::Tournament.new("Isle of Man Masters, 2007", '2007-09-22', :rounds => 9)
t.site = 'http://www.bcmchess.co.uk/monarch2007/'
t.add_player(ICU::Player.new('Anthony', 'Fox', 1, :rating => 2100, :fed => 'IRL', :id => 456))
t.add_player(ICU::Player.new('Peter P.', 'Taylor', 2, :rating => 2209, :fed => 'ENG'))
t.add_player(ICU::Player.new('Egozi', 'Nadav', 3, :rating => 2205, :fed => 'ISR'))
t.add_player(ICU::Player.new('Peter', 'Cafolla', 4, :rating => 2048, :fed => 'IRL'))
t.add_player(ICU::Player.new('Tim R.', 'Spanton', 5, :rating => 1982, :fed => 'ENG'))
t.add_player(ICU::Player.new('Alan', 'Grant', 6, :rating => 2223, :fed => 'SCO'))
t.add_player(ICU::Player.new('Alan J.', 'Walton', 7, :rating => 2223, :fed => 'ENG'))
t.add_player(ICU::Player.new('Bernard', 'Bannink', 8, :rating => 2271, :fed => 'NED', :title => 'FM'))
t.add_player(ICU::Player.new('Roy', 'Phillips', 9, :rating => 2271, :fed => 'MAU'))
t.add_result(ICU::Result.new(1, 1, 'L', :opponent => 2, :colour => 'B'))
t.add_result(ICU::Result.new(2, 1, 'D', :opponent => 3, :colour => 'W'))
t.add_result(ICU::Result.new(3, 1, 'D', :opponent => 4, :colour => 'B'))
t.add_result(ICU::Result.new(4, 1, 'W', :opponent => 5, :colour => 'W'))
t.add_result(ICU::Result.new(5, 1, 'W', :opponent => 6, :colour => 'B'))
t.add_result(ICU::Result.new(6, 1, 'L'))
t.add_result(ICU::Result.new(7, 1, 'D', :opponent => 7, :colour => 'W'))
t.add_result(ICU::Result.new(8, 1, 'L', :opponent => 8, :colour => 'B'))
t.add_result(ICU::Result.new(9, 1, 'D', :opponent => 9, :colour => 'W'))
puts t.serialize('ForeignCSV')
=end
class ForeignCSV
attr_reader :error
# Parse CSV data returning a Tournament on success or raising an exception on error.
def parse!(csv)
@state, @line, @round, @sum, @error = 0, 0, nil, nil, nil
@tournament = Tournament.new('Dummy', '2000-01-01')
Util::CSV.parse(csv, :row_sep => :auto) do |r|
@line += 1 # increment line number
next if r.size == 0 # skip empty lines
r = r.map{|c| c.nil? ? '' : c.strip} # trim all spaces, turn nils to blanks
next if r[0] == '' # skip blanks in column 1
@r = r # remember this record for later
begin
case @state
when 0 then event
when 1 then start
when 2 then rounds
when 3 then website
when 4 then player
when 5 then result
when 6 then total
else raise "internal error - state #{@state} does not exist"
end
rescue => err
raise err.class, "line #{@line}: #{err.message}", err.backtrace unless err.message.match(/^line [1-9]/)
raise
end
end
unless @state == 4
exp = case @state
when 0 then "the event name"
when 1 then "the start date"
when 2 then "the number of rounds"
when 3 then "the website address"
when 5 then "a result for round #{@round+1}"
when 6 then "a total score"
end
raise "line #{@line}: premature termination - expected #{exp}"
end
raise "line #{@line}: no players found in file" if @tournament.players.size == 0
@tournament.validate!
@tournament
end
# Parse CSV data returning a Tournament on success or a nil on failure.
# In the case of failure, an error message can be retrived via the error method.
def parse(csv)
begin
parse!(csv)
rescue => ex
@error = ex.message
nil
end
end
# Same as parse! except the input is a file name rather than file contents.
def parse_file!(file)
csv = open(file) { |f| f.read }
parse!(csv)
end
# Same as parse except the input is a file name rather than file contents.
def parse_file(file)
begin
parse_file!(file)
rescue => ex
@error = ex.message
nil
end
end
# Serialise a tournament back into CSV format.
def serialize(t, arg={})
t.validate!(:type => self)
Util::CSV.generate do |csv|
csv << ["Event", t.name]
csv << ["Start", t.start]
csv << ["Rounds", t.rounds]
csv << ["Website", t.site]
t.players.each do |p|
next unless p.id
csv << []
csv << ["Player", p.id, p.last_name, p.first_name]
(1..t.rounds).each do |n|
data = []
data << n
r = p.find_result(n)
data << case r.score; when 'W' then '1'; when 'L' then '0'; else '='; end
if r.rateable
data << r.colour
o = t.player(r.opponent)
data << o.last_name
data << o.first_name
data << o.rating
data << o.title
data << o.fed
else
data << '-'
end
csv << data
end
csv << ["Total", p.points]
end
end
end
# Additional tournament validation rules for this specific type.
def validate!(t)
raise "missing site" unless t.site.to_s.length > 0
icu = t.players.find_all { |p| p.id }
raise "there must be at least one ICU player (with an ID number)" if icu.size == 0
foreign = t.players.find_all { |p| !p.id }
raise "all foreign players must have a federation" if foreign.detect { |f| !f.fed }
icu.each do |p|
(1..t.rounds).each do |r|
result = p.find_result(r)
raise "ICU players must have a result in every round" unless result
raise "all opponents of ICU players must have a federation" if result.opponent && !t.player(result.opponent).fed
end
end
end
private
def event
abort "the 'Event' keyword", 0 unless @r[0].match(/^(Event|Tournament)$/i)
abort "the event name", 1 unless @r.size > 1 && @r[1] != ''
@tournament.name = @r[1]
@state = 1
end
def start
abort "the 'Start' keyword", 0 unless @r[0].match(/^(Start(\s+Date)?|Date)$/i)
abort "the start date", 1 unless @r.size > 1 && @r[1] != ''
@tournament.start = @r[1]
@state = 2
end
def rounds
abort "the 'Rounds' keyword", 0 unless @r[0].match(/(Number of )?Rounds$/)
abort "the number of rounds", 1 unless @r.size > 1 && @r[1].match(/^[1-9]\d*/)
@tournament.rounds = @r[1]
@state = 3
end
def website
abort "the 'Website' keyword", 0 unless @r[0].match(/^(Web(\s?site)?|Site)$/i)
abort "the event website", 1 unless @r.size > 1 && @r[1] != ''
@tournament.site = @r[1]
@state = 4
end
def player
abort "the 'Player' keyword", 0 unless @r[0].match(/^Player$/i)
abort "a player's ICU number", 1 unless @r.size > 1 && @r[1].match(/^[1-9]/i)
abort "a player's last name", 2 unless @r.size > 2 && @r[2].match(/[a-z]/i)
abort "a player's first name", 3 unless @r.size > 3 && @r[3].match(/[a-z]/i)
@player = Player.new(@r[3], @r[2], @tournament.players.size + 1, :id => @r[1])
old_player = @tournament.find_player(@player)
if old_player
raise "two players with the same name (#{@player.name}) have conflicting details" unless old_player.eql?(@player)
raise "same player (#{@player.name}) has more than one set of results" if old_player.id
old_player.merge(@player)
@player = old_player
else
@tournament.add_player(@player)
end
@round = 0
@state = 5
end
def result
@round+= 1
abort "round number #{round}", 0 unless @r[0].to_i == @round
abort "a colour (W/B) or dash (for a bye)", 2 unless @r.size > 2 && @r[2].match(/^(W|B|-)/i)
result = Result.new(@round, @player.num, @r[1])
if @r[2] == '-'
@tournament.add_result(result)
else
result.colour = @r[2]
opponent = Player.new(@r[4], @r[3], @tournament.players.size + 1, :rating => @r[5], :title => @r[6], :fed => @r[7])
raise "opponent must have a federation" unless opponent.fed
old_player = @tournament.find_player(opponent)
if old_player
raise "two players with the same name (#{opponent.name}) have conflicting details" unless old_player.eql?(opponent)
result.opponent = old_player.num
if old_player.id
old_player.merge(opponent)
old_result = @player.find_result(@round)
raise "missing result for player (#{@player.name}) in round #{@round}" unless old_result
raise "mismatched results for player (#{old_player.name}) in round #{@round}" unless result == old_result
old_result.rateable = true
else
old_result = old_player.find_result(@round)
raise "a player (#{old_player.name}) has more than one game in the same round (#{@round})" if old_result
@tournament.add_result(result, false)
end
else
@tournament.add_player(opponent)
result.opponent = opponent.num
@tournament.add_result(result, false)
end
end
@state = 6 if @round == @tournament.rounds
end
def total
points = @player.points
abort "the 'Total' keyword", 0 unless @r[0].match(/^Total$/i)
abort "the player's (#{@player.object_id}, #{@player.results.size}) total points to be #{points}", 1 unless @r[1].to_f == points
@state = 4
end
def abort(expected, cell)
got = @r[cell]
error = "line #{@line}"
error << ", cell #{cell+1}"
error << ": expected #{expected}"
error << " but got #{got == '' ? 'a blank cell' : "'#{got}'"}"
raise error
end
end
end
end