# frozen-string-literal: true module Bioshogi module InputParser extend self concern :CLI do included do desc "input_match", "指し手の一つ分の入力の読み取り" def input_match(*argv) InputParser.check(argv.join(" ")) end end end def check(text) rows = InputParser.scan(text).collect do |str| {"入力" => str}.merge(match!(str).named_captures) end tp rows end def match!(str) md = str.to_s.match(regexp) unless md raise SyntaxDefact, "表記が間違っています : #{str.inspect}" end md end def scan(str) str.to_s.scan(regexp).collect(&:join) end def slice_one(str) scan(str.to_s).first end def regexp @regexp ||= Regexp.union(kif_or_ki2_regexp, csa_regexp, sfen_regexp) end private def kif_or_ki2_regexp ki2_location = /(?[#{Location.polygon_chars_str}])/o kif_place = /(?#{Place.regexp})/o ki2_same = /(?同)\p{blank}*/ ki2_as_it_is = /(?不成|生)/ ki2_promote_trigger = /(?成)/ kif_drop_trigger = /(?[打合])/ ki2_one_up = /(?直)/ ki2_left_right = /(?[左右])/ ki2_up_down = /(?[上行引寄])/ # 行は上のalias / #{ki2_location}? (#{kif_place}#{ki2_same}|#{ki2_same}#{kif_place}|#{kif_place}|#{ki2_same}) # 12同 or 同12 or 12 or 同 に対応 (?#{Piece.all_names.join("|")}) (#{ki2_one_up}|#{ki2_left_right}?#{ki2_up_down}?)? (#{ki2_as_it_is}|#{ki2_promote_trigger}|#{kif_drop_trigger})? (?\(\d{2}\))? # scan の結果を join したものがマッチした元の文字列と一致するように () も含める /ox end def csa_regexp csa_piece = Piece.flat_map { |e| [e.csa.basic_name, e.csa.promoted_name] }.compact.join("|") / (?[+-])? (?[0-9]{2}) # 00 = 駒台 (?[1-9]{2}) (?#{csa_piece}) /ox end def sfen_regexp sfen_drop_piece = Piece.collect(&:sfen_char).compact.join sfen_to = /[1-9][[:lower:]]/ drop_part = /(?[#{sfen_drop_piece}])(?\*)(?#{sfen_to})/o move_part = /(?#{sfen_to})(?#{sfen_to})(?\+)?/o /((#{drop_part})|(#{move_part}))/o end end end