1
Fork 0
mirror of https://github.com/RGBCube/serenity synced 2025-10-25 19:02:33 +00:00
serenity/Userland/Libraries/LibChess/UCIEndpoint.cpp
Tim Ledbetter 56dde3df54 LibChess+ChessEngine: Don't crash on error when reading UCI commands
ChessEngine and the chess GUI will no longer crash on error while
reading UCI commands. ChessEngine will print a message to the standard
output, while the GUI will ignore unknown commands. Both will print
the error to the debug log if the UCI_DEBUG flag is enabled.

Trailing and preceding whitespace is now stripped from commands before
they are parsed. Commands which are just whitespace no longer produce
errors.
2023-05-03 08:31:34 +01:00

140 lines
4.8 KiB
C++

/*
* Copyright (c) 2020, the SerenityOS developers.
* Copyright (c) 2023, Tim Ledbetter <timledbetter@gmail.com>
*
* SPDX-License-Identifier: BSD-2-Clause
*/
#include "UCIEndpoint.h"
#include <AK/ByteBuffer.h>
#include <AK/Debug.h>
#include <AK/DeprecatedString.h>
#include <LibCore/EventLoop.h>
namespace Chess::UCI {
Endpoint::Endpoint(NonnullRefPtr<Core::IODevice> in, NonnullRefPtr<Core::IODevice> out)
: m_in(in)
, m_out(out)
, m_in_notifier(Core::Notifier::construct(in->fd(), Core::Notifier::Type::Read))
{
set_in_notifier();
}
void Endpoint::send_command(Command const& command)
{
auto command_string = command.to_string().release_value_but_fixme_should_propagate_errors();
dbgln_if(UCI_DEBUG, "{} Sent UCI Command: {}", class_name(), command_string);
m_out->write(command_string);
}
void Endpoint::event(Core::Event& event)
{
switch (static_cast<Command::Type>(event.type())) {
case Command::Type::UCI:
return handle_uci();
case Command::Type::Debug:
return handle_debug(static_cast<DebugCommand const&>(event));
case Command::Type::IsReady:
return handle_uci();
case Command::Type::SetOption:
return handle_setoption(static_cast<SetOptionCommand const&>(event));
case Command::Type::Position:
return handle_position(static_cast<PositionCommand const&>(event));
case Command::Type::Go:
return handle_go(static_cast<GoCommand const&>(event));
case Command::Type::Stop:
return handle_stop();
case Command::Type::Id:
return handle_id(static_cast<IdCommand const&>(event));
case Command::Type::UCIOk:
return handle_uciok();
case Command::Type::ReadyOk:
return handle_readyok();
case Command::Type::BestMove:
return handle_bestmove(static_cast<BestMoveCommand const&>(event));
case Command::Type::Info:
return handle_info(static_cast<InfoCommand const&>(event));
case Command::Type::Quit:
return handle_quit();
case Command::Type::UCINewGame:
return handle_ucinewgame();
default:
Object::event(event);
break;
}
}
void Endpoint::custom_event(Core::CustomEvent& custom_event)
{
if (custom_event.custom_type() == EndpointEventType::UnexpectedEof)
handle_unexpected_eof();
}
void Endpoint::set_in_notifier()
{
m_in_notifier = Core::Notifier::construct(m_in->fd(), Core::Notifier::Type::Read);
m_in_notifier->on_activation = [this] {
if (!m_in->can_read_line()) {
Core::EventLoop::current().post_event(*this, make<Core::CustomEvent>(EndpointEventType::UnexpectedEof));
m_in_notifier->set_enabled(false);
return;
}
while (m_in->can_read_line()) {
auto line = m_in->read_line(4096).trim_whitespace();
if (line.is_empty())
continue;
auto maybe_command = read_command(line);
if (maybe_command.is_error()) {
dbgln_if(UCI_DEBUG, "{} Error while parsing UCI command: {}, error: {}", class_name(), maybe_command.error(), line);
if (on_command_read_error)
on_command_read_error(move(line), maybe_command.release_error());
continue;
}
Core::EventLoop::current().post_event(*this, maybe_command.release_value());
}
};
}
ErrorOr<NonnullOwnPtr<Command>> Endpoint::read_command(StringView line) const
{
dbgln_if(UCI_DEBUG, "{} Received UCI Command: {}", class_name(), line);
if (line == "uci") {
return UCICommand::from_string(line);
} else if (line.starts_with("debug"sv)) {
return DebugCommand::from_string(line);
} else if (line.starts_with("isready"sv)) {
return IsReadyCommand::from_string(line);
} else if (line.starts_with("setoption"sv)) {
return SetOptionCommand::from_string(line);
} else if (line.starts_with("position"sv)) {
return PositionCommand::from_string(line);
} else if (line.starts_with("go"sv)) {
return GoCommand::from_string(line);
} else if (line.starts_with("stop"sv)) {
return StopCommand::from_string(line);
} else if (line.starts_with("id"sv)) {
return IdCommand::from_string(line);
} else if (line.starts_with("uciok"sv)) {
return UCIOkCommand::from_string(line);
} else if (line.starts_with("readyok"sv)) {
return ReadyOkCommand::from_string(line);
} else if (line.starts_with("bestmove"sv)) {
return BestMoveCommand::from_string(line);
} else if (line.starts_with("info"sv)) {
return InfoCommand::from_string(line);
} else if (line.starts_with("quit"sv)) {
return QuitCommand::from_string(line);
} else if (line.starts_with("ucinewgame"sv)) {
return UCINewGameCommand::from_string(line);
}
return Error::from_string_literal("Unknown command");
}
};