diff options
Diffstat (limited to 'data/rbot/plugins/shiritori.rb')
-rw-r--r-- | data/rbot/plugins/shiritori.rb | 448 |
1 files changed, 0 insertions, 448 deletions
diff --git a/data/rbot/plugins/shiritori.rb b/data/rbot/plugins/shiritori.rb deleted file mode 100644 index f13afeb8..00000000 --- a/data/rbot/plugins/shiritori.rb +++ /dev/null @@ -1,448 +0,0 @@ -#-- vim:sw=2:et -#kate: indent-width 2 -#++ -# -# :title: Shiritori Plugin for RBot -# -# Author:: Yaohan Chen <yaohan.chen@gmail.com> -# Copyright:: (c) 2007 Yaohan Chen -# License:: GNU Public License -# -# -# Shiritori is a word game where a few people take turns to continue a chain of words. -# To continue a word, the next word must start with the ending of the previous word, -# usually defined as the one to few letters/characters at the end. This plugin allows -# playing several games, each per channel. A game can be turn-based, where only new -# players can interrupt a turn to join, or a free mode where anyone can speak at any -# time. -# -# TODO -# * a system to describe settings, so they can be displayed, changed and saved -# * adjust settings during game -# * allow other definitions of continues? -# * read default settings from configuration -# * keep statistics -# * other forms of dictionaries - - -# Abstract class representing a dictionary used by Shiritori -class Dictionary - # whether string s is a word - def has_word?(s) - raise NotImplementedError - end - - # whether any word starts with prefix, excluding words in excludes. This can be - # possible with non-enumerable dictionaries since some dictionary engines provide - # prefix searching. - def any_word_starting?(prefix, excludes) - raise NotImplementedError - end -end - -# A Dictionary that uses a enumrable word list. -class WordlistDictionary < Dictionary - def initialize(words) - super() - @words = words - debug "Created dictionary with #{@words.length} words" - end - - # whether string s is a word - def has_word?(s) - @words.include? s - end - - # whether any word starts with prefix, excluding words in excludes - def any_word_starting?(prefix, excludes) - # (@words - except).any? {|w| w =~ /\A#{prefix}.+/} - # this seems to be faster: - !(@words.grep(/\A#{prefix}.+/) - excludes).empty? - end -end - -# Logic of shiritori game, deals with checking whether words continue the chain, and -# whether it's possible to continue a word -class Shiritori - attr_reader :used_words - - # dictionary:: a Dictionary object - # overlap_lengths:: a Range for allowed lengths to overlap when continuing words - # check_continuable:: whether all words are checked whether they're continuable, - # before being commited - # allow_reuse:: whether words are allowed to be used again - def initialize(dictionary, overlap_lengths, check_continuable, allow_reuse) - @dictionary = dictionary - @overlap_lengths = overlap_lengths - @check_continuable = check_continuable - @allow_reuse = allow_reuse - @used_words = [] - end - - # Prefix of s with length n - def head_of(s, n) - # TODO ruby2 unicode - s.split(//u)[0, n].join - end - # Suffix of s with length n - def tail_of(s, n) - # TODO ruby2 unicode - s.split(//u)[-n, n].join - end - # Number of unicode characters in string - def len(s) - # TODO ruby2 unicode - s.split(//u).length - end - # return subrange of range r that's under n - def range_under(r, n) - r.begin .. [r.end, n-1].min - end - - # TODO allow the ruleset to customize this - def continues?(w2, w1) - # this uses the definition w1[-n,n] == w2[0,n] && n < [w1.length, w2.length].min - # TODO it might be worth allowing <= for the second clause - range_under(@overlap_lengths, [len(w1), len(w2)].min).any? {|n| - tail_of(w1, n)== head_of(w2, n)} - end - - # Checks whether *any* unused word in the dictionary completes the word - # This has the limitation that it can't detect when a word is continuable, but the - # only continuers aren't continuable - def continuable_from?(s) - range_under(@overlap_lengths, len(s)).any? {|n| - @dictionary.any_word_starting?(tail_of(s, n), @used_words) } - end - - # Given a string, give a verdict based on current shiritori state and dictionary - def process(s) - # TODO optionally allow used words - # TODO ruby2 unicode - if len(s) < @overlap_lengths.min || !@dictionary.has_word?(s) - debug "#{s} is too short or not in dictionary" - :ignore - elsif @used_words.empty? - if !@check_continuable || continuable_from?(s) - @used_words << s - :start - else - :start_end - end - elsif continues?(s, @used_words.last) - if !@allow_reuse && @used_words.include?(s) - :used - elsif !@check_continuable || continuable_from?(s) - @used_words << s - :next - else - :end - end - else - :ignore - end - end -end - -# A shiritori game on a channel. keeps track of rules related to timing and turns, -# and interacts with players -class ShiritoriGame - # timer:: the bot.timer object - # say:: a Proc which says the given message on the channel - # when_die:: a Proc that removes the game from plugin's list of games - def initialize(channel, ruleset, timer, say, when_die) - raise ArgumentError unless [:words, :overlap_lengths, :check_continuable, - :end_when_uncontinuable, :allow_reuse, :listen, :normalize, :time_limit, - :lose_when_timeout].all? {|r| ruleset.has_key?(r)} - @last_word = nil - @players = [] - @booted_players = [] - @ruleset = ruleset - @channel = channel - @timer = timer - @timer_handle = nil - @say = say - @when_die = when_die - - # TODO allow other forms of dictionaries - dictionary = WordlistDictionary.new(@ruleset[:words]) - @game = Shiritori.new(dictionary, @ruleset[:overlap_lengths], - @ruleset[:check_continuable], - @ruleset[:allow_reuse]) - end - - # Whether the players must take turns - # * when there is only one player, turns are not enforced - # * when time_limit > 0, new players can join at any time, but existing players must - # take turns, each of which expires after time_limit - # * when time_imit is 0, anyone can speak in the game at any time - def take_turns? - @players.length > 1 && @ruleset[:time_limit] > 0 - end - - # the player who has the current turn - def current_player - @players.first - end - # the word to continue in the current turn - def current_word - @game.used_words[-1] - end - # the word in the chain before current_word - def previous_word - @game.used_words[-2] - end - - # announce the current word, and player if take_turns? - def announce - if take_turns? - @say.call "#{current_player}, it's your turn. #{previous_word} -> #{current_word}" - elsif @players.empty? - @say.call "No one has given the first word yet. Say the first word to start." - else - @say.call "Poor #{current_player} is playing alone! Anyone care to join? #{previous_word} -> #{current_word}" - end - end - # create/reschedule timer - def restart_timer - # the first time the method is called, a new timer is added - @timer_handle = @timer.add(@ruleset[:time_limit]) {time_out} - # afterwards, it will reschdule the timer - instance_eval do - def restart_timer - @timer.reschedule(@timer_handle, @ruleset[:time_limit]) - end - end - end - # switch to the next player's turn if take_turns?, and announce current words - def next_player - # when there's only one player, turns and timer are meaningless - if take_turns? - # place the current player to the last position, to implement circular queue - @players << @players.shift - # stop previous timer and set time for this turn - restart_timer - end - announce - end - - # handle when turn time limit goes out - def time_out - if @ruleset[:lose_when_timeout] - @say.call "#{current_player} took too long and is out of the game. Try again next game!" - if @players.length == 2 - # 2 players before, and one should remain now - # since the game is ending, save the trouble of removing and booting the player - @say.call "#{@players[1]} is the last remaining player and the winner! Congratulations!" - die - else - @booted_players << @players.shift - announce - end - else - @say.call "#{current_player} took too long and skipped the turn." - next_player - end - end - - # change the rules, and update state when necessary - def change_rules(rules) - @ruleset.update! rules - end - - # handle a message to @channel - def handle_message(m) - message = m.message - speaker = m.sourcenick.to_s - - return unless @ruleset[:listen] =~ message - - # in take_turns mode, only new players are allowed to interrupt a turn - return if @booted_players.include? speaker || - (take_turns? && - speaker != current_player && - (@players.length > 1 && @players.include?(speaker))) - - # let Shiritori process the message, and act according to result - case @game.process @ruleset[:normalize].call(message) - when :start - @players << speaker - m.reply "#{speaker} has given the first word: #{current_word}" - when :next - if !@players.include?(speaker) - # A new player - @players.unshift speaker - m.reply "Welcome to shiritori, #{speaker}." - end - next_player - when :used - m.reply "The word #{message} has been used. Retry from #{current_word}" - when :end - # TODO respect shiritori.end_when_uncontinuable setting - if @ruleset[:end_when_uncontinuable] - m.reply "It's impossible to continue the chain from #{message}. The game has ended. Thanks a lot, #{speaker}! :(" - die - else - m.reply "It's impossible to continue the chain from #{message}. Retry from #{current_word}" - end - when :start_end - # when the first word is uncontinuable, the game doesn't stop, as presumably - # someone wanted to play - m.reply "It's impossible to continue the chain from #{message}. Start with another word." - end - end - - # end the game - def die - # redefine restart_timer to no-op - instance_eval do - def restart_timer - end - end - # remove any registered timer - @timer.remove @timer_handle unless @timer_handle.nil? - # should remove the game object from plugin's @games list - @when_die.call - end -end - -# shiritori plugin for rbot -class ShiritoriPlugin < Plugin - def help(plugin, topic="") - "A game in which each player must continue the previous player's word, by using its last one or few characters/letters of the word to start a new word. 'shiritori <ruleset>' => Play shiritori with a set of rules. Available rulesets: #{@rulesets.keys.join ', '}. 'shiritori stop' => Stop the current shiritori game." - end - - def initialize() - super - @games = {} - - # TODO make rulesets more easily customizable - # TODO initialize default ruleset from config - # Default values of rulesets - @default_ruleset = { - # the range of the length of "tail" that must be followed to continue the chain - :overlap_lengths => 1..2, - # messages cared about, pre-normalize - :listen => /\A\S+\Z/u, - # normalize messages with this function before checking with Shiritori - :normalize => lambda {|w| w}, - # number of seconds for each player's turn - :time_limit => 60, - # when the time limit is reached, the player's booted out of the game and cannot - # join until the next game - :lose_when_timeout => true, - # check whether the word is continuable before adding it into chain - :check_continuable => true, - # allow reusing used words - :allow_reuse => false, - # end the game when an uncontinuable word is said - :end_when_uncontinuable => true - } - @rulesets = { - 'english' => { - :wordlist_file => 'english', - :listen => /\A[a-zA-Z]+\Z/, - :overlap_lengths => 2..5, - :normalize => lambda {|w| w.downcase}, - :desc => 'Use English words; case insensitive; 2-6 letters at the beginning of the next word must overlap with those at the end of the previous word.' - }, - 'japanese' => { - :wordlist_file => 'japanese', - :listen => /\A\S+\Z/u, - :overlap_lengths => 1..4, - :desc => 'Use Japanese words in hiragana; 1-4 kana at the beginning of the next word must overlap with those at the end of the previous word.', - # Optionally use a module to normalize Japanese words, enabling input in multiple writing systems - } - } - @rulesets.each_value do |ruleset| - # set default values for each rule to default_ruleset's values - ruleset.replace @default_ruleset.merge(ruleset) - unless ruleset.has_key?(:words) - if ruleset.has_key?(:wordlist_file) - # TODO read words only when rule is used - # read words separated by newlines from file - ruleset[:words] = - File.new("#{@bot.botclass}/shiritori/#{ruleset[:wordlist_file]}").grep( - ruleset[:listen]) {|l| ruleset[:normalize].call l.chomp} - else - raise NotImplementedError - end - end - end - end - - # start shiritori in a channel - def cmd_shiritori(m, params) - if @games.has_key?( m.channel ) - m.reply "Already playing shiritori here" - @games[m.channel].announce - else - if @rulesets.has_key? params[:ruleset] - @games[m.channel] = ShiritoriGame.new( - m.channel, @rulesets[params[:ruleset]], - @bot.timer, - lambda {|msg| m.reply msg}, - lambda {remove_game m.channel} ) - m.reply "Shiritori has started. Please say the first word" - else - m.reply "There is no defined ruleset named #{params[:ruleset]}" - end - end - end - - # change rules for current game - def cmd_set(m, params) - require 'enumerator' - new_rules = {} - params[:rules].each_slice(2) {|opt, value| new_rules[opt] = value} - raise NotImplementedError - end - - # stop the current game - def cmd_stop(m, params) - if @games.has_key? m.channel - # TODO display statistics - @games[m.channel].die - m.reply "Shiritori has stopped. Hope you had fun!" - else - # TODO display statistics - m.reply "No game to stop here, because no game is being played." - end - end - - # remove the game, so channel messages are no longer processed, and timer removed - def remove_game(channel) - @games.delete channel - end - - # all messages from a channel is sent to its shiritori game if any - def listen(m) - return unless m.kind_of?(PrivMessage) - return unless @games.has_key?(m.channel) - # send the message to the game in the channel to handle it - @games[m.channel].handle_message m - end - - # remove all games - def cleanup - @games.each_key {|g| g.die} - @games.clear - end -end - -plugin = ShiritoriPlugin.new -plugin.default_auth( 'edit', false ) - -# Normal commandsi have a stop_gamei have a stop_game -plugin.map 'shiritori stop', - :action => 'cmd_stop', - :private => false -# plugin.map 'shiritori set ', -# :action => 'cmd_set' -# :private => false -# plugin.map 'shiritori challenge', -# :action => 'cmd_challenge' -plugin.map 'shiritori [:ruleset]', - :action => 'cmd_shiritori', - :defaults => {:ruleset => 'japanese'}, - :private => false |