UCICommand.cpp 18 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539
  1. /*
  2. * Copyright (c) 2020, the SerenityOS developers.
  3. * Copyright (c) 2023, Sam Atkins <atkinssj@serenityos.org>
  4. * Copyright (c) 2023, Tim Ledbetter <timledbetter@gmail.com>
  5. *
  6. * SPDX-License-Identifier: BSD-2-Clause
  7. */
  8. #include "UCICommand.h"
  9. #include <AK/StringBuilder.h>
  10. namespace Chess::UCI {
  11. ErrorOr<NonnullOwnPtr<UCICommand>> UCICommand::from_string(StringView command)
  12. {
  13. auto tokens = command.split_view(' ');
  14. VERIFY(tokens[0] == "uci");
  15. VERIFY(tokens.size() == 1);
  16. return adopt_nonnull_own_or_enomem(new (nothrow) UCICommand);
  17. }
  18. ErrorOr<String> UCICommand::to_string() const
  19. {
  20. return "uci\n"_string;
  21. }
  22. ErrorOr<NonnullOwnPtr<DebugCommand>> DebugCommand::from_string(StringView command)
  23. {
  24. auto tokens = command.split_view(' ');
  25. VERIFY(tokens[0] == "debug");
  26. VERIFY(tokens.size() == 2);
  27. if (tokens[1] == "on")
  28. return adopt_nonnull_own_or_enomem(new (nothrow) DebugCommand(Flag::On));
  29. if (tokens[1] == "off")
  30. return adopt_nonnull_own_or_enomem(new (nothrow) DebugCommand(Flag::Off));
  31. VERIFY_NOT_REACHED();
  32. }
  33. ErrorOr<String> DebugCommand::to_string() const
  34. {
  35. if (flag() == Flag::On) {
  36. return "debug on\n"_string;
  37. } else {
  38. return "debug off\n"_string;
  39. }
  40. }
  41. ErrorOr<NonnullOwnPtr<IsReadyCommand>> IsReadyCommand::from_string(StringView command)
  42. {
  43. auto tokens = command.split_view(' ');
  44. VERIFY(tokens[0] == "isready");
  45. VERIFY(tokens.size() == 1);
  46. return adopt_nonnull_own_or_enomem(new (nothrow) IsReadyCommand);
  47. }
  48. ErrorOr<String> IsReadyCommand::to_string() const
  49. {
  50. return "isready\n"_string;
  51. }
  52. ErrorOr<NonnullOwnPtr<SetOptionCommand>> SetOptionCommand::from_string(StringView command)
  53. {
  54. auto tokens = command.split_view(' ');
  55. VERIFY(tokens[0] == "setoption");
  56. VERIFY(tokens[1] == "name");
  57. VERIFY(tokens.size() > 2);
  58. StringBuilder name;
  59. StringBuilder value;
  60. bool in_name = false;
  61. bool in_value = false;
  62. for (auto& part : tokens) {
  63. if (in_name) {
  64. if (part == "value") {
  65. in_name = false;
  66. in_value = true;
  67. continue;
  68. }
  69. TRY(name.try_append(part));
  70. TRY(name.try_append(' '));
  71. continue;
  72. }
  73. if (in_value) {
  74. TRY(value.try_append(part));
  75. TRY(value.try_append(' '));
  76. continue;
  77. }
  78. if (part == "name") {
  79. in_name = true;
  80. continue;
  81. }
  82. }
  83. VERIFY(!name.is_empty());
  84. auto name_string = TRY(String::from_utf8(name.string_view().trim_whitespace()));
  85. auto value_string = TRY(String::from_utf8(value.string_view().trim_whitespace()));
  86. return adopt_nonnull_own_or_enomem(new (nothrow) SetOptionCommand(name_string, value_string));
  87. }
  88. ErrorOr<String> SetOptionCommand::to_string() const
  89. {
  90. StringBuilder builder;
  91. TRY(builder.try_append("setoption name "sv));
  92. TRY(builder.try_append(name()));
  93. if (value().has_value()) {
  94. TRY(builder.try_append(" value "sv));
  95. TRY(builder.try_append(value().value()));
  96. }
  97. TRY(builder.try_append('\n'));
  98. return builder.to_string();
  99. }
  100. ErrorOr<NonnullOwnPtr<PositionCommand>> PositionCommand::from_string(StringView command)
  101. {
  102. auto tokens = command.split_view(' ');
  103. VERIFY(tokens.size() >= 3);
  104. VERIFY(tokens[0] == "position");
  105. VERIFY(tokens[2] == "moves");
  106. Optional<String> fen;
  107. if (tokens[1] != "startpos")
  108. fen = TRY(String::from_utf8(tokens[1]));
  109. Vector<Move> moves;
  110. for (size_t i = 3; i < tokens.size(); ++i) {
  111. TRY(moves.try_append(Move(tokens[i])));
  112. }
  113. return adopt_nonnull_own_or_enomem(new (nothrow) PositionCommand(move(fen), move(moves)));
  114. }
  115. ErrorOr<String> PositionCommand::to_string() const
  116. {
  117. StringBuilder builder;
  118. TRY(builder.try_append("position "sv));
  119. if (fen().has_value()) {
  120. TRY(builder.try_append(fen().value()));
  121. } else {
  122. TRY(builder.try_append("startpos "sv));
  123. }
  124. TRY(builder.try_append("moves"sv));
  125. for (auto& move : moves()) {
  126. TRY(builder.try_append(' '));
  127. TRY(builder.try_append(TRY(move.to_long_algebraic())));
  128. }
  129. TRY(builder.try_append('\n'));
  130. return builder.to_string();
  131. }
  132. ErrorOr<NonnullOwnPtr<GoCommand>> GoCommand::from_string(StringView command)
  133. {
  134. auto tokens = command.split_view(' ');
  135. VERIFY(tokens[0] == "go");
  136. auto go_command = TRY(adopt_nonnull_own_or_enomem(new (nothrow) GoCommand));
  137. for (size_t i = 1; i < tokens.size(); ++i) {
  138. if (tokens[i] == "searchmoves") {
  139. VERIFY_NOT_REACHED();
  140. } else if (tokens[i] == "ponder") {
  141. go_command->ponder = true;
  142. } else if (tokens[i] == "wtime") {
  143. VERIFY(i++ < tokens.size());
  144. go_command->wtime = tokens[i].to_number<int>().value();
  145. } else if (tokens[i] == "btime") {
  146. VERIFY(i++ < tokens.size());
  147. go_command->btime = tokens[i].to_number<int>().value();
  148. } else if (tokens[i] == "winc") {
  149. VERIFY(i++ < tokens.size());
  150. go_command->winc = tokens[i].to_number<int>().value();
  151. } else if (tokens[i] == "binc") {
  152. VERIFY(i++ < tokens.size());
  153. go_command->binc = tokens[i].to_number<int>().value();
  154. } else if (tokens[i] == "movestogo") {
  155. VERIFY(i++ < tokens.size());
  156. go_command->movestogo = tokens[i].to_number<int>().value();
  157. } else if (tokens[i] == "depth") {
  158. VERIFY(i++ < tokens.size());
  159. go_command->depth = tokens[i].to_number<int>().value();
  160. } else if (tokens[i] == "nodes") {
  161. VERIFY(i++ < tokens.size());
  162. go_command->nodes = tokens[i].to_number<int>().value();
  163. } else if (tokens[i] == "mate") {
  164. VERIFY(i++ < tokens.size());
  165. go_command->mate = tokens[i].to_number<int>().value();
  166. } else if (tokens[i] == "movetime") {
  167. VERIFY(i++ < tokens.size());
  168. go_command->movetime = tokens[i].to_number<int>().value();
  169. } else if (tokens[i] == "infinite") {
  170. go_command->infinite = true;
  171. }
  172. }
  173. return go_command;
  174. }
  175. ErrorOr<String> GoCommand::to_string() const
  176. {
  177. StringBuilder builder;
  178. TRY(builder.try_append("go"sv));
  179. if (searchmoves.has_value()) {
  180. TRY(builder.try_append(" searchmoves"sv));
  181. for (auto& move : searchmoves.value()) {
  182. TRY(builder.try_append(' '));
  183. TRY(builder.try_append(TRY(move.to_long_algebraic())));
  184. }
  185. }
  186. if (ponder)
  187. TRY(builder.try_append(" ponder"sv));
  188. if (wtime.has_value())
  189. TRY(builder.try_appendff(" wtime {}", wtime.value()));
  190. if (btime.has_value())
  191. TRY(builder.try_appendff(" btime {}", btime.value()));
  192. if (winc.has_value())
  193. TRY(builder.try_appendff(" winc {}", winc.value()));
  194. if (binc.has_value())
  195. TRY(builder.try_appendff(" binc {}", binc.value()));
  196. if (movestogo.has_value())
  197. TRY(builder.try_appendff(" movestogo {}", movestogo.value()));
  198. if (depth.has_value())
  199. TRY(builder.try_appendff(" depth {}", depth.value()));
  200. if (nodes.has_value())
  201. TRY(builder.try_appendff(" nodes {}", nodes.value()));
  202. if (mate.has_value())
  203. TRY(builder.try_appendff(" mate {}", mate.value()));
  204. if (movetime.has_value())
  205. TRY(builder.try_appendff(" movetime {}", movetime.value()));
  206. if (infinite)
  207. TRY(builder.try_append(" infinite"sv));
  208. TRY(builder.try_append('\n'));
  209. return builder.to_string();
  210. }
  211. ErrorOr<NonnullOwnPtr<StopCommand>> StopCommand::from_string(StringView command)
  212. {
  213. auto tokens = command.split_view(' ');
  214. VERIFY(tokens[0] == "stop");
  215. VERIFY(tokens.size() == 1);
  216. return adopt_nonnull_own_or_enomem(new (nothrow) StopCommand);
  217. }
  218. ErrorOr<String> StopCommand::to_string() const
  219. {
  220. return "stop\n"_string;
  221. }
  222. ErrorOr<NonnullOwnPtr<IdCommand>> IdCommand::from_string(StringView command)
  223. {
  224. auto tokens = command.split_view(' ');
  225. VERIFY(tokens[0] == "id");
  226. StringBuilder value;
  227. for (size_t i = 2; i < tokens.size(); ++i) {
  228. if (i != 2)
  229. TRY(value.try_append(' '));
  230. TRY(value.try_append(tokens[i]));
  231. }
  232. auto value_string = TRY(value.to_string());
  233. if (tokens[1] == "name") {
  234. return adopt_nonnull_own_or_enomem(new (nothrow) IdCommand(Type::Name, value_string));
  235. } else if (tokens[1] == "author") {
  236. return adopt_nonnull_own_or_enomem(new (nothrow) IdCommand(Type::Author, value_string));
  237. }
  238. VERIFY_NOT_REACHED();
  239. }
  240. ErrorOr<String> IdCommand::to_string() const
  241. {
  242. StringBuilder builder;
  243. TRY(builder.try_append("id "sv));
  244. if (field_type() == Type::Name) {
  245. TRY(builder.try_append("name "sv));
  246. } else {
  247. TRY(builder.try_append("author "sv));
  248. }
  249. TRY(builder.try_append(value()));
  250. TRY(builder.try_append('\n'));
  251. return builder.to_string();
  252. }
  253. ErrorOr<NonnullOwnPtr<UCIOkCommand>> UCIOkCommand::from_string(StringView command)
  254. {
  255. auto tokens = command.split_view(' ');
  256. VERIFY(tokens[0] == "uciok");
  257. VERIFY(tokens.size() == 1);
  258. return adopt_nonnull_own_or_enomem(new (nothrow) UCIOkCommand);
  259. }
  260. ErrorOr<String> UCIOkCommand::to_string() const
  261. {
  262. return "uciok\n"_string;
  263. }
  264. ErrorOr<NonnullOwnPtr<ReadyOkCommand>> ReadyOkCommand::from_string(StringView command)
  265. {
  266. auto tokens = command.split_view(' ');
  267. VERIFY(tokens[0] == "readyok");
  268. VERIFY(tokens.size() == 1);
  269. return adopt_nonnull_own_or_enomem(new (nothrow) ReadyOkCommand);
  270. }
  271. ErrorOr<String> ReadyOkCommand::to_string() const
  272. {
  273. return "readyok\n"_string;
  274. }
  275. ErrorOr<NonnullOwnPtr<BestMoveCommand>> BestMoveCommand::from_string(StringView command)
  276. {
  277. auto tokens = command.split_view(' ');
  278. VERIFY(tokens[0] == "bestmove");
  279. VERIFY(tokens.size() == 2 || tokens.size() == 4);
  280. auto best_move = Move(tokens[1]);
  281. Optional<Move> move_to_ponder;
  282. if (tokens.size() == 4) {
  283. VERIFY(tokens[2] == "ponder");
  284. move_to_ponder = Move(tokens[3]);
  285. }
  286. return adopt_nonnull_own_or_enomem(new (nothrow) BestMoveCommand(best_move, move_to_ponder));
  287. }
  288. ErrorOr<String> BestMoveCommand::to_string() const
  289. {
  290. StringBuilder builder;
  291. TRY(builder.try_append("bestmove "sv));
  292. TRY(builder.try_append(TRY(move().to_long_algebraic())));
  293. if (move_to_ponder().has_value()) {
  294. TRY(builder.try_append(" ponder "sv));
  295. TRY(builder.try_append(TRY(move_to_ponder()->to_long_algebraic())));
  296. }
  297. TRY(builder.try_append('\n'));
  298. return builder.to_string();
  299. }
  300. ErrorOr<NonnullOwnPtr<InfoCommand>> InfoCommand::from_string(StringView command)
  301. {
  302. auto tokens = command.split_view(' ');
  303. VERIFY(tokens[0] == "info");
  304. auto info_command = TRY(try_make<InfoCommand>());
  305. auto parse_integer_token = [](StringView value_token) -> ErrorOr<int> {
  306. auto value_as_integer = value_token.to_number<int>();
  307. if (!value_as_integer.has_value())
  308. return Error::from_string_literal("Expected integer token");
  309. return value_as_integer.release_value();
  310. };
  311. auto parse_line = [](auto const& move_tokens) -> ErrorOr<Vector<Chess::Move>> {
  312. Vector<Chess::Move> moves;
  313. TRY(moves.try_ensure_capacity(move_tokens.size()));
  314. for (auto move_token : move_tokens)
  315. moves.unchecked_append({ move_token });
  316. return moves;
  317. };
  318. size_t i = 1;
  319. while (i < tokens.size()) {
  320. auto name = tokens[i++];
  321. if (name == "depth"sv) {
  322. info_command->m_depth = TRY(parse_integer_token(tokens[i++]));
  323. } else if (name == "seldepth"sv) {
  324. info_command->m_seldepth = TRY(parse_integer_token(tokens[i++]));
  325. } else if (name == "time"sv) {
  326. info_command->m_time = TRY(parse_integer_token(tokens[i++]));
  327. } else if (name == "nodes"sv) {
  328. info_command->m_nodes = TRY(parse_integer_token(tokens[i++]));
  329. } else if (name == "multipv"sv) {
  330. info_command->m_multipv = TRY(parse_integer_token(tokens[i++]));
  331. } else if (name == "score"sv) {
  332. auto score_type_string = tokens[i++];
  333. ScoreType score_type;
  334. if (score_type_string == "cp"sv) {
  335. score_type = ScoreType::Centipawns;
  336. } else if (score_type_string == "mate"sv) {
  337. score_type = ScoreType::Mate;
  338. } else {
  339. return Error::from_string_literal("Invalid score type");
  340. }
  341. auto score_value = TRY(parse_integer_token(tokens[i++]));
  342. auto maybe_score_bound_string = tokens[i];
  343. auto score_bound = ScoreBound::None;
  344. if (maybe_score_bound_string == "upperbound"sv)
  345. score_bound = ScoreBound::Upper;
  346. else if (maybe_score_bound_string == "lowerbound"sv)
  347. score_bound = ScoreBound::Lower;
  348. if (score_bound != ScoreBound::None)
  349. i++;
  350. info_command->m_score = Score { score_type, score_value, score_bound };
  351. } else if (name == "currmove"sv) {
  352. info_command->m_currmove = Chess::Move { tokens[i++] };
  353. } else if (name == "currmovenumber"sv) {
  354. info_command->m_currmovenumber = TRY(parse_integer_token(tokens[i++]));
  355. } else if (name == "hashfull"sv) {
  356. info_command->m_hashfull = TRY(parse_integer_token(tokens[i++]));
  357. } else if (name == "nps"sv) {
  358. info_command->m_nps = TRY(parse_integer_token(tokens[i++]));
  359. } else if (name == "tbhits"sv) {
  360. info_command->m_tbhits = TRY(parse_integer_token(tokens[i++]));
  361. } else if (name == "cpuload"sv) {
  362. info_command->m_cpuload = TRY(parse_integer_token(tokens[i++]));
  363. }
  364. // We assume the info types: pv, string, refutation, and currline, are the final info type in a command.
  365. else if (name == "pv"sv) {
  366. info_command->m_pv = TRY(parse_line(tokens.span().slice(i)));
  367. break;
  368. } else if (name == "string"sv) {
  369. info_command->m_string = TRY(String::join(' ', tokens.span().slice(i)));
  370. break;
  371. } else if (name == "refutation"sv) {
  372. info_command->m_refutation = TRY(parse_line(tokens.span().slice(i)));
  373. break;
  374. } else if (name == "currline"sv) {
  375. info_command->m_currline = TRY(parse_line(tokens.span().slice(i)));
  376. break;
  377. } else {
  378. return Error::from_string_literal("Unknown info type");
  379. }
  380. }
  381. return info_command;
  382. }
  383. ErrorOr<String> InfoCommand::to_string() const
  384. {
  385. StringBuilder builder;
  386. auto append_moves = [&](Vector<Chess::Move> const& moves) -> ErrorOr<void> {
  387. bool first = true;
  388. for (auto const& move : moves) {
  389. if (!first)
  390. TRY(builder.try_append(' '));
  391. first = false;
  392. TRY(builder.try_append(TRY(move.to_long_algebraic())));
  393. }
  394. return {};
  395. };
  396. TRY(builder.try_append("info"sv));
  397. if (m_depth.has_value())
  398. TRY(builder.try_appendff(" depth {}", m_depth.value()));
  399. if (m_seldepth.has_value())
  400. TRY(builder.try_appendff(" seldepth {}", m_seldepth.value()));
  401. if (m_time.has_value())
  402. TRY(builder.try_appendff(" time {}", m_time.value()));
  403. if (m_nodes.has_value())
  404. TRY(builder.try_appendff(" nodes {}", m_nodes.value()));
  405. if (m_multipv.has_value())
  406. TRY(builder.try_appendff(" multipv {}", m_multipv.value()));
  407. if (m_score.has_value()) {
  408. TRY(builder.try_append(" score"sv));
  409. switch (m_score->type) {
  410. case ScoreType::Centipawns:
  411. TRY(builder.try_append(" cp"sv));
  412. break;
  413. case ScoreType::Mate:
  414. TRY(builder.try_append(" mate"sv));
  415. break;
  416. }
  417. TRY(builder.try_appendff(" {}", m_score->value));
  418. switch (m_score->bound) {
  419. case ScoreBound::None:
  420. break;
  421. case ScoreBound::Lower:
  422. TRY(builder.try_append(" lowerbound"sv));
  423. break;
  424. case ScoreBound::Upper:
  425. TRY(builder.try_append(" upperbound"sv));
  426. break;
  427. }
  428. }
  429. if (m_currmove.has_value())
  430. TRY(builder.try_appendff(" currmove {}", TRY(m_currmove->to_long_algebraic())));
  431. if (m_currmovenumber.has_value())
  432. TRY(builder.try_appendff(" currmovenumber {}", m_currmovenumber.value()));
  433. if (m_hashfull.has_value())
  434. TRY(builder.try_appendff(" hashfull {}", m_hashfull.value()));
  435. if (m_nps.has_value())
  436. TRY(builder.try_appendff(" nps {}", m_nps.value()));
  437. if (m_tbhits.has_value())
  438. TRY(builder.try_appendff(" tbhits {}", m_tbhits.value()));
  439. if (m_cpuload.has_value())
  440. TRY(builder.try_appendff(" cpuload {}", m_cpuload.value()));
  441. if (m_string.has_value())
  442. TRY(builder.try_appendff(" string {}", m_string.value()));
  443. if (m_pv.has_value()) {
  444. TRY(builder.try_append(" pv "sv));
  445. TRY(append_moves(m_pv.value()));
  446. }
  447. if (m_refutation.has_value()) {
  448. TRY(builder.try_append(" refutation "sv));
  449. TRY(append_moves(m_refutation.value()));
  450. }
  451. if (m_currline.has_value()) {
  452. TRY(builder.try_append(" currline "sv));
  453. TRY(append_moves(m_currline.value()));
  454. }
  455. return builder.to_string();
  456. }
  457. ErrorOr<NonnullOwnPtr<QuitCommand>> QuitCommand::from_string(StringView command)
  458. {
  459. auto tokens = command.split_view(' ');
  460. VERIFY(tokens[0] == "quit");
  461. VERIFY(tokens.size() == 1);
  462. return adopt_nonnull_own_or_enomem(new (nothrow) QuitCommand);
  463. }
  464. ErrorOr<String> QuitCommand::to_string() const
  465. {
  466. return "quit\n"_string;
  467. }
  468. ErrorOr<NonnullOwnPtr<UCINewGameCommand>> UCINewGameCommand::from_string(StringView command)
  469. {
  470. auto tokens = command.split_view(' ');
  471. VERIFY(tokens[0] == "ucinewgame");
  472. VERIFY(tokens.size() == 1);
  473. return adopt_nonnull_own_or_enomem(new (nothrow) UCINewGameCommand);
  474. }
  475. ErrorOr<String> UCINewGameCommand::to_string() const
  476. {
  477. return "ucinewgame\n"_string;
  478. }
  479. }