require 'fastercsv'
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.
data = open('tournament.csv') { |f| f.read }
parser = ICU::Tournament::ForeignCSV.new
tournament = parser.parse(data)
If the file is correctly specified, the return value from the parse 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
A tournament can be serialized back to CSV format (the reverse of parsing) with the _serialize_ method.
csv = parser.serialize(tournament)
=end
class ForeignCSV
attr_reader :error
# 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
# 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')
FasterCSV.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
end
# Serialise a tournament back into CSV format.
def serialize(t)
return nil unless t.class == ICU::Tournament;
FasterCSV.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
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 rating and federation" unless opponent.rating && 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