3 ## Copyright (C) 2004 NABEYA Kenichi (aka nanami@2ch)
4 ## Copyright (C) 2007-2008 Daigo Moriwaki (daigo at debian dot org)
6 ## This program is free software; you can redistribute it and/or modify
7 ## it under the terms of the GNU General Public License as published by
8 ## the Free Software Foundation; either version 2 of the License, or
9 ## (at your option) any later version.
11 ## This program is distributed in the hope that it will be useful,
12 ## but WITHOUT ANY WARRANTY; without even the implied warranty of
13 ## MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
14 ## GNU General Public License for more details.
16 ## You should have received a copy of the GNU General Public License
17 ## along with this program; if not, write to the Free Software
18 ## Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
20 require 'shogi_server/util'
28 return sort_by_rate_with_randomness
31 def sort_by_rate_with_randomness
32 return [ExcludeSacrificeGps500.new,
34 SortByRateWithRandomness.new(1200, 2400),
39 return [ExcludeSacrificeGps500.new,
46 logics = default_factory
47 logics.inject(players) do |result, item|
58 def include_newbie?(players)
59 return players.find{|a| a.rate == 0} == nil ? false : true
62 def less_than_one?(players)
64 log_warning("Floodgate: There should be at least one player.")
71 def log_players(players)
72 str_array = players.map do |one|
80 log_message("Floodgate: [Players] None is here.")
82 log_message("Floodgate: [Players] %s." % [str_array.join(", ")])
87 class StartGame < Pairing
91 log_warning("Floodgate: There should be more than one player: %d" % [players.size])
95 log_warning("Floodgate: There are odd players: %d. %s will not be matched." %
96 [players.size, players.last.name])
100 while (players.size >= 2) do
101 pair = players.shift(2)
103 start_game(pair.first, pair.last)
108 log_message("Floodgate: BLACK %s; WHITE %s" % [p1.name, p2.name])
111 Game.new(p1.game_name, p1, p2)
115 class Randomize < Pairing
118 log_message("Floodgate: Randomize... before")
121 log_message("Floodgate: Randomized after")
126 class SortByRate < Pairing
129 log_message("Floodgate: Ordered by rate")
130 players.sort! {|a,b| a.rate <=> b.rate} # decendent order
135 class SortByRateWithRandomness < Pairing
136 def initialize(rand1, rand2)
138 @rand1, @rand2 = rand1, rand2
144 players.each{|a| cur_rate[a] = a.rate ? a.rate + rand(@rand1) : rand(@rand2)}
145 players.sort!{|a,b| cur_rate[a] <=> cur_rate[b]}
146 log_players(players) do |one|
147 "%s %d (randomness %d)" % [one.name, one.rate, cur_rate[one] - one.rate]
152 class DeletePlayerAtRandom < Pairing
155 return if less_than_one?(players)
157 log_message("Floodgate: Deleted %s at random" % [one.name])
163 class DeletePlayerAtRandomExcept < Pairing
164 def initialize(except)
171 log_message("Floodgate: Deleting a player at rondom except %s" % [@except.name])
172 players.delete(@except)
173 DeletePlayerAtRandom.new.match(players)
174 players.push(@except)
178 class DeleteMostPlayingPlayer < Pairing
181 one = players.max_by {|a| a.win + a.loss}
182 log_message("Floodgate: Deleted the most playing player: %s (%d)" % [one.name, one.win + one.loss])
188 class DeleteLeastRatePlayer < Pairing
191 one = players.min_by {|a| a.rate}
192 log_message("Floodgate: Deleted the least rate player %s (%d)" % [one.name, one.rate])
198 class ExcludeSacrifice < Pairing
199 attr_reader :sacrifice
201 # @sacrifice a player id to be eliminated
202 def initialize(sacrifice)
204 @sacrifice = sacrifice
211 players.find{|a| a.player_id == @sacrifice}
212 log_message("Floodgate: Deleting the sacrifice %s" % [@sacrifice])
213 players.delete_if{|a| a.player_id == @sacrifice}
217 end # class ExcludeSacrifice
219 class ExcludeSacrificeGps500 < ExcludeSacrifice
221 super("gps500+e293220e3f8a3e59f79f6b0efffaa931")
225 class MakeEven < Pairing
228 return if players.size.even?
229 log_message("Floodgate: there are odd players: %d. Deleting one..." %
231 DeletePlayerAtRandom.new.match(players)