UCICommand.cpp 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364
  1. /*
  2. * Copyright (c) 2020, the SerenityOS developers.
  3. * Copyright (c) 2023, Sam Atkins <atkinssj@serenityos.org>
  4. *
  5. * SPDX-License-Identifier: BSD-2-Clause
  6. */
  7. #include "UCICommand.h"
  8. #include <AK/StringBuilder.h>
  9. namespace Chess::UCI {
  10. ErrorOr<NonnullOwnPtr<UCICommand>> UCICommand::from_string(StringView command)
  11. {
  12. auto tokens = command.split_view(' ');
  13. VERIFY(tokens[0] == "uci");
  14. VERIFY(tokens.size() == 1);
  15. return adopt_nonnull_own_or_enomem(new (nothrow) UCICommand);
  16. }
  17. ErrorOr<String> UCICommand::to_string() const
  18. {
  19. return "uci\n"_short_string;
  20. }
  21. ErrorOr<NonnullOwnPtr<DebugCommand>> DebugCommand::from_string(StringView command)
  22. {
  23. auto tokens = command.split_view(' ');
  24. VERIFY(tokens[0] == "debug");
  25. VERIFY(tokens.size() == 2);
  26. if (tokens[1] == "on")
  27. return adopt_nonnull_own_or_enomem(new (nothrow) DebugCommand(Flag::On));
  28. if (tokens[1] == "off")
  29. return adopt_nonnull_own_or_enomem(new (nothrow) DebugCommand(Flag::Off));
  30. VERIFY_NOT_REACHED();
  31. }
  32. ErrorOr<String> DebugCommand::to_string() const
  33. {
  34. if (flag() == Flag::On) {
  35. return "debug on\n"_string;
  36. } else {
  37. return "debug off\n"_string;
  38. }
  39. }
  40. ErrorOr<NonnullOwnPtr<IsReadyCommand>> IsReadyCommand::from_string(StringView command)
  41. {
  42. auto tokens = command.split_view(' ');
  43. VERIFY(tokens[0] == "isready");
  44. VERIFY(tokens.size() == 1);
  45. return adopt_nonnull_own_or_enomem(new (nothrow) IsReadyCommand);
  46. }
  47. ErrorOr<String> IsReadyCommand::to_string() const
  48. {
  49. return "isready\n"_string;
  50. }
  51. ErrorOr<NonnullOwnPtr<SetOptionCommand>> SetOptionCommand::from_string(StringView command)
  52. {
  53. auto tokens = command.split_view(' ');
  54. VERIFY(tokens[0] == "setoption");
  55. VERIFY(tokens[1] == "name");
  56. VERIFY(tokens.size() > 2);
  57. StringBuilder name;
  58. StringBuilder value;
  59. bool in_name = false;
  60. bool in_value = false;
  61. for (auto& part : tokens) {
  62. if (in_name) {
  63. if (part == "value") {
  64. in_name = false;
  65. in_value = true;
  66. continue;
  67. }
  68. TRY(name.try_append(part));
  69. TRY(name.try_append(' '));
  70. continue;
  71. }
  72. if (in_value) {
  73. TRY(value.try_append(part));
  74. TRY(value.try_append(' '));
  75. continue;
  76. }
  77. if (part == "name") {
  78. in_name = true;
  79. continue;
  80. }
  81. }
  82. VERIFY(!name.is_empty());
  83. return adopt_nonnull_own_or_enomem(new (nothrow) SetOptionCommand(
  84. TRY(String::from_utf8(name.string_view().trim_whitespace())),
  85. TRY(String::from_utf8(value.string_view().trim_whitespace()))));
  86. }
  87. ErrorOr<String> SetOptionCommand::to_string() const
  88. {
  89. StringBuilder builder;
  90. TRY(builder.try_append("setoption name "sv));
  91. TRY(builder.try_append(name()));
  92. if (value().has_value()) {
  93. TRY(builder.try_append(" value "sv));
  94. TRY(builder.try_append(value().value()));
  95. }
  96. TRY(builder.try_append('\n'));
  97. return builder.to_string();
  98. }
  99. ErrorOr<NonnullOwnPtr<PositionCommand>> PositionCommand::from_string(StringView command)
  100. {
  101. auto tokens = command.split_view(' ');
  102. VERIFY(tokens.size() >= 3);
  103. VERIFY(tokens[0] == "position");
  104. VERIFY(tokens[2] == "moves");
  105. Optional<String> fen;
  106. if (tokens[1] != "startpos")
  107. fen = TRY(String::from_utf8(tokens[1]));
  108. Vector<Move> moves;
  109. for (size_t i = 3; i < tokens.size(); ++i) {
  110. TRY(moves.try_append(Move(tokens[i])));
  111. }
  112. return adopt_nonnull_own_or_enomem(new (nothrow) PositionCommand(move(fen), move(moves)));
  113. }
  114. ErrorOr<String> PositionCommand::to_string() const
  115. {
  116. StringBuilder builder;
  117. TRY(builder.try_append("position "sv));
  118. if (fen().has_value()) {
  119. TRY(builder.try_append(fen().value()));
  120. } else {
  121. TRY(builder.try_append("startpos "sv));
  122. }
  123. TRY(builder.try_append("moves"sv));
  124. for (auto& move : moves()) {
  125. TRY(builder.try_append(' '));
  126. TRY(builder.try_append(TRY(move.to_long_algebraic())));
  127. }
  128. TRY(builder.try_append('\n'));
  129. return builder.to_string();
  130. }
  131. ErrorOr<NonnullOwnPtr<GoCommand>> GoCommand::from_string(StringView command)
  132. {
  133. auto tokens = command.split_view(' ');
  134. VERIFY(tokens[0] == "go");
  135. auto go_command = TRY(adopt_nonnull_own_or_enomem(new (nothrow) GoCommand));
  136. for (size_t i = 1; i < tokens.size(); ++i) {
  137. if (tokens[i] == "searchmoves") {
  138. VERIFY_NOT_REACHED();
  139. } else if (tokens[i] == "ponder") {
  140. go_command->ponder = true;
  141. } else if (tokens[i] == "wtime") {
  142. VERIFY(i++ < tokens.size());
  143. go_command->wtime = tokens[i].to_int().value();
  144. } else if (tokens[i] == "btime") {
  145. VERIFY(i++ < tokens.size());
  146. go_command->btime = tokens[i].to_int().value();
  147. } else if (tokens[i] == "winc") {
  148. VERIFY(i++ < tokens.size());
  149. go_command->winc = tokens[i].to_int().value();
  150. } else if (tokens[i] == "binc") {
  151. VERIFY(i++ < tokens.size());
  152. go_command->binc = tokens[i].to_int().value();
  153. } else if (tokens[i] == "movestogo") {
  154. VERIFY(i++ < tokens.size());
  155. go_command->movestogo = tokens[i].to_int().value();
  156. } else if (tokens[i] == "depth") {
  157. VERIFY(i++ < tokens.size());
  158. go_command->depth = tokens[i].to_int().value();
  159. } else if (tokens[i] == "nodes") {
  160. VERIFY(i++ < tokens.size());
  161. go_command->nodes = tokens[i].to_int().value();
  162. } else if (tokens[i] == "mate") {
  163. VERIFY(i++ < tokens.size());
  164. go_command->mate = tokens[i].to_int().value();
  165. } else if (tokens[i] == "movetime") {
  166. VERIFY(i++ < tokens.size());
  167. go_command->movetime = tokens[i].to_int().value();
  168. } else if (tokens[i] == "infinite") {
  169. go_command->infinite = true;
  170. }
  171. }
  172. return go_command;
  173. }
  174. ErrorOr<String> GoCommand::to_string() const
  175. {
  176. StringBuilder builder;
  177. TRY(builder.try_append("go"sv));
  178. if (searchmoves.has_value()) {
  179. TRY(builder.try_append(" searchmoves"sv));
  180. for (auto& move : searchmoves.value()) {
  181. TRY(builder.try_append(' '));
  182. TRY(builder.try_append(TRY(move.to_long_algebraic())));
  183. }
  184. }
  185. if (ponder)
  186. TRY(builder.try_append(" ponder"sv));
  187. if (wtime.has_value())
  188. TRY(builder.try_appendff(" wtime {}", wtime.value()));
  189. if (btime.has_value())
  190. TRY(builder.try_appendff(" btime {}", btime.value()));
  191. if (winc.has_value())
  192. TRY(builder.try_appendff(" winc {}", winc.value()));
  193. if (binc.has_value())
  194. TRY(builder.try_appendff(" binc {}", binc.value()));
  195. if (movestogo.has_value())
  196. TRY(builder.try_appendff(" movestogo {}", movestogo.value()));
  197. if (depth.has_value())
  198. TRY(builder.try_appendff(" depth {}", depth.value()));
  199. if (nodes.has_value())
  200. TRY(builder.try_appendff(" nodes {}", nodes.value()));
  201. if (mate.has_value())
  202. TRY(builder.try_appendff(" mate {}", mate.value()));
  203. if (movetime.has_value())
  204. TRY(builder.try_appendff(" movetime {}", movetime.value()));
  205. if (infinite)
  206. TRY(builder.try_append(" infinite"sv));
  207. TRY(builder.try_append('\n'));
  208. return builder.to_string();
  209. }
  210. ErrorOr<NonnullOwnPtr<StopCommand>> StopCommand::from_string(StringView command)
  211. {
  212. auto tokens = command.split_view(' ');
  213. VERIFY(tokens[0] == "stop");
  214. VERIFY(tokens.size() == 1);
  215. return adopt_nonnull_own_or_enomem(new (nothrow) StopCommand);
  216. }
  217. ErrorOr<String> StopCommand::to_string() const
  218. {
  219. return "stop\n"_short_string;
  220. }
  221. ErrorOr<NonnullOwnPtr<IdCommand>> IdCommand::from_string(StringView command)
  222. {
  223. auto tokens = command.split_view(' ');
  224. VERIFY(tokens[0] == "id");
  225. StringBuilder value;
  226. for (size_t i = 2; i < tokens.size(); ++i) {
  227. if (i != 2)
  228. TRY(value.try_append(' '));
  229. TRY(value.try_append(tokens[i]));
  230. }
  231. if (tokens[1] == "name") {
  232. return adopt_nonnull_own_or_enomem(new (nothrow) IdCommand(Type::Name, TRY(value.to_string())));
  233. } else if (tokens[1] == "author") {
  234. return adopt_nonnull_own_or_enomem(new (nothrow) IdCommand(Type::Author, TRY(value.to_string())));
  235. }
  236. VERIFY_NOT_REACHED();
  237. }
  238. ErrorOr<String> IdCommand::to_string() const
  239. {
  240. StringBuilder builder;
  241. TRY(builder.try_append("id "sv));
  242. if (field_type() == Type::Name) {
  243. TRY(builder.try_append("name "sv));
  244. } else {
  245. TRY(builder.try_append("author "sv));
  246. }
  247. TRY(builder.try_append(value()));
  248. TRY(builder.try_append('\n'));
  249. return builder.to_string();
  250. }
  251. ErrorOr<NonnullOwnPtr<UCIOkCommand>> UCIOkCommand::from_string(StringView command)
  252. {
  253. auto tokens = command.split_view(' ');
  254. VERIFY(tokens[0] == "uciok");
  255. VERIFY(tokens.size() == 1);
  256. return adopt_nonnull_own_or_enomem(new (nothrow) UCIOkCommand);
  257. }
  258. ErrorOr<String> UCIOkCommand::to_string() const
  259. {
  260. return "uciok\n"_short_string;
  261. }
  262. ErrorOr<NonnullOwnPtr<ReadyOkCommand>> ReadyOkCommand::from_string(StringView command)
  263. {
  264. auto tokens = command.split_view(' ');
  265. VERIFY(tokens[0] == "readyok");
  266. VERIFY(tokens.size() == 1);
  267. return adopt_nonnull_own_or_enomem(new (nothrow) ReadyOkCommand);
  268. }
  269. ErrorOr<String> ReadyOkCommand::to_string() const
  270. {
  271. return "readyok\n"_string;
  272. }
  273. ErrorOr<NonnullOwnPtr<BestMoveCommand>> BestMoveCommand::from_string(StringView command)
  274. {
  275. auto tokens = command.split_view(' ');
  276. VERIFY(tokens[0] == "bestmove");
  277. VERIFY(tokens.size() == 2);
  278. return adopt_nonnull_own_or_enomem(new (nothrow) BestMoveCommand(Move(tokens[1])));
  279. }
  280. ErrorOr<String> BestMoveCommand::to_string() const
  281. {
  282. StringBuilder builder;
  283. TRY(builder.try_append("bestmove "sv));
  284. TRY(builder.try_append(TRY(move().to_long_algebraic())));
  285. TRY(builder.try_append('\n'));
  286. return builder.to_string();
  287. }
  288. ErrorOr<NonnullOwnPtr<InfoCommand>> InfoCommand::from_string([[maybe_unused]] StringView command)
  289. {
  290. // FIXME: Implement this.
  291. VERIFY_NOT_REACHED();
  292. }
  293. ErrorOr<String> InfoCommand::to_string() const
  294. {
  295. // FIXME: Implement this.
  296. VERIFY_NOT_REACHED();
  297. }
  298. ErrorOr<NonnullOwnPtr<QuitCommand>> QuitCommand::from_string(StringView command)
  299. {
  300. auto tokens = command.split_view(' ');
  301. VERIFY(tokens[0] == "quit");
  302. VERIFY(tokens.size() == 1);
  303. return adopt_nonnull_own_or_enomem(new (nothrow) QuitCommand);
  304. }
  305. ErrorOr<String> QuitCommand::to_string() const
  306. {
  307. return "quit\n"_short_string;
  308. }
  309. ErrorOr<NonnullOwnPtr<UCINewGameCommand>> UCINewGameCommand::from_string(StringView command)
  310. {
  311. auto tokens = command.split_view(' ');
  312. VERIFY(tokens[0] == "ucinewgame");
  313. VERIFY(tokens.size() == 1);
  314. return adopt_nonnull_own_or_enomem(new (nothrow) UCINewGameCommand);
  315. }
  316. ErrorOr<String> UCINewGameCommand::to_string() const
  317. {
  318. return "ucinewgame\n"_string;
  319. }
  320. }