mirror of
https://gitlab.com/manzerbredes/pgnp.git
synced 2025-04-06 10:06:25 +02:00
- Debug parser (carriage returns)
- Improve test framework
This commit is contained in:
parent
bb914f047b
commit
f4f436870f
11 changed files with 1577 additions and 32 deletions
|
@ -3,4 +3,4 @@ archlinux:
|
||||||
before_script:
|
before_script:
|
||||||
- pacman -Sy base-devel cmake --noconfirm --needed
|
- pacman -Sy base-devel cmake --noconfirm --needed
|
||||||
script:
|
script:
|
||||||
- mkdir build && cd build && cmake ../ && make && cd tests && ./pgnp_tests
|
- mkdir build && cd build && cmake ../ && make && ctest
|
||||||
|
|
|
@ -15,4 +15,5 @@ configure_file(src/LargeFileStream.hpp ${PGNP_INCLUDE_DIR} COPYONLY)
|
||||||
include_directories(${PGNP_INCLUDE_DIR})
|
include_directories(${PGNP_INCLUDE_DIR})
|
||||||
|
|
||||||
# Unit tests
|
# Unit tests
|
||||||
|
enable_testing()
|
||||||
add_subdirectory(./tests)
|
add_subdirectory(./tests)
|
||||||
|
|
|
@ -58,6 +58,6 @@ char LargeFileStream::operator[](long loc) {
|
||||||
return buffer[offset];
|
return buffer[offset];
|
||||||
}
|
}
|
||||||
|
|
||||||
bool LargeFileStream::IsEOF(long loc) { return (eof); }
|
bool LargeFileStream::IsEOF() { return (eof); }
|
||||||
|
|
||||||
} // namespace pgnp
|
} // namespace pgnp
|
|
@ -36,7 +36,7 @@ public:
|
||||||
/// @brief Allow array like access to the file
|
/// @brief Allow array like access to the file
|
||||||
char operator[](long loc);
|
char operator[](long loc);
|
||||||
/// @brief Check if we reach the EOF
|
/// @brief Check if we reach the EOF
|
||||||
bool IsEOF(long loc);
|
bool IsEOF();
|
||||||
|
|
||||||
// Various Exceptions
|
// Various Exceptions
|
||||||
struct BackwardRead : public std::exception {
|
struct BackwardRead : public std::exception {
|
||||||
|
|
30
src/PGN.cpp
30
src/PGN.cpp
|
@ -3,14 +3,14 @@
|
||||||
#include <iostream>
|
#include <iostream>
|
||||||
#include <string>
|
#include <string>
|
||||||
|
|
||||||
#define IS_BLANK(c) (c == ' ' || c == '\n' || c == '\t')
|
#define IS_BLANK(c) (c == ' ' || c == '\n' || c == '\t' || c == '\r')
|
||||||
#define IS_DIGIT(c) \
|
#define IS_DIGIT(c) \
|
||||||
(c == '0' || c == '1' || c == '2' || c == '3' || c == '4' || c == '5' || \
|
(c == '0' || c == '1' || c == '2' || c == '3' || c == '4' || c == '5' || \
|
||||||
c == '6' || c == '7' || c == '8' || c == '9')
|
c == '6' || c == '7' || c == '8' || c == '9')
|
||||||
#define IS_EOF(loc) (pgn_content.IsEOF(loc))
|
#define IS_EOF (pgn_content.IsEOF())
|
||||||
#define EOF_CHECK(loc) \
|
#define EOF_CHECK(loc) \
|
||||||
{ \
|
{ \
|
||||||
if (IS_EOF(loc)) \
|
if (IS_EOF) \
|
||||||
throw UnexpectedEOF(); \
|
throw UnexpectedEOF(); \
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -42,17 +42,16 @@ void PGN::ParseNextGame() {
|
||||||
moves = new HalfMove();
|
moves = new HalfMove();
|
||||||
|
|
||||||
// Search for new game
|
// Search for new game
|
||||||
if (IS_EOF(LastGameEndLoc)) {
|
if (IS_EOF) {
|
||||||
throw NoGameFound();
|
throw NoGameFound();
|
||||||
}
|
}
|
||||||
int loc = NextNonBlank(LastGameEndLoc);
|
int loc = NextNonBlank(LastGameEndLoc);
|
||||||
|
if (IS_EOF) {
|
||||||
if (IS_EOF(loc)) {
|
|
||||||
throw NoGameFound();
|
throw NoGameFound();
|
||||||
}
|
}
|
||||||
|
|
||||||
// Parse game
|
// Parse game
|
||||||
while (!IS_EOF(loc)) {
|
while (!IS_EOF) {
|
||||||
char c = pgn_content[loc];
|
char c = pgn_content[loc];
|
||||||
if (!IS_BLANK(c)) {
|
if (!IS_BLANK(c)) {
|
||||||
if (c == '[') {
|
if (c == '[') {
|
||||||
|
@ -108,7 +107,8 @@ long PGN::ParseComment(long loc, HalfMove *hm) {
|
||||||
EOF_CHECK(loc);
|
EOF_CHECK(loc);
|
||||||
char c = pgn_content[loc];
|
char c = pgn_content[loc];
|
||||||
|
|
||||||
if (c == '{') {
|
// Parse a sequence of comment
|
||||||
|
while (!IS_EOF && c == '{') {
|
||||||
loc++;
|
loc++;
|
||||||
EOF_CHECK(loc);
|
EOF_CHECK(loc);
|
||||||
c = pgn_content[loc];
|
c = pgn_content[loc];
|
||||||
|
@ -119,6 +119,12 @@ long PGN::ParseComment(long loc, HalfMove *hm) {
|
||||||
c = pgn_content[loc];
|
c = pgn_content[loc];
|
||||||
}
|
}
|
||||||
loc++; // Skip '}'
|
loc++; // Skip '}'
|
||||||
|
|
||||||
|
// Goto next non blank to look for another comment token
|
||||||
|
loc = NextNonBlank(loc);
|
||||||
|
if(!IS_EOF){
|
||||||
|
c = pgn_content[loc];
|
||||||
|
}
|
||||||
}
|
}
|
||||||
return (loc);
|
return (loc);
|
||||||
}
|
}
|
||||||
|
@ -204,7 +210,7 @@ long PGN::ParseHalfMove(long loc, HalfMove *hm) {
|
||||||
|
|
||||||
// Check for variations
|
// Check for variations
|
||||||
loc = NextNonBlank(loc);
|
loc = NextNonBlank(loc);
|
||||||
while (!IS_EOF(loc) && pgn_content[loc] == '(') {
|
while (!IS_EOF && pgn_content[loc] == '(') {
|
||||||
loc++; // Skip '('
|
loc++; // Skip '('
|
||||||
HalfMove *var = new HalfMove;
|
HalfMove *var = new HalfMove;
|
||||||
loc = ParseHalfMove(loc, var);
|
loc = ParseHalfMove(loc, var);
|
||||||
|
@ -226,7 +232,7 @@ long PGN::ParseHalfMove(long loc, HalfMove *hm) {
|
||||||
|
|
||||||
// Parse next HalfMove
|
// Parse next HalfMove
|
||||||
loc = NextNonBlank(loc);
|
loc = NextNonBlank(loc);
|
||||||
if (!IS_EOF(loc)) {
|
if (!IS_EOF) {
|
||||||
HalfMove *next_hm = new HalfMove;
|
HalfMove *next_hm = new HalfMove;
|
||||||
next_hm->count = hm->count;
|
next_hm->count = hm->count;
|
||||||
loc = ParseHalfMove(loc, next_hm);
|
loc = ParseHalfMove(loc, next_hm);
|
||||||
|
@ -259,7 +265,7 @@ long PGN::ParseNextTag(long start_loc) {
|
||||||
long valueloc = NextNonBlank(keyloc) + 1;
|
long valueloc = NextNonBlank(keyloc) + 1;
|
||||||
EOF_CHECK(keyloc);
|
EOF_CHECK(keyloc);
|
||||||
c = pgn_content[valueloc];
|
c = pgn_content[valueloc];
|
||||||
while (c != '"' or IS_EOF(valueloc)) {
|
while (c != '"' or IS_EOF) {
|
||||||
value += c;
|
value += c;
|
||||||
valueloc++;
|
valueloc++;
|
||||||
EOF_CHECK(keyloc);
|
EOF_CHECK(keyloc);
|
||||||
|
@ -308,7 +314,7 @@ long PGN::NextNonBlank(long loc) {
|
||||||
char c = pgn_content[loc];
|
char c = pgn_content[loc];
|
||||||
while (IS_BLANK(c)) {
|
while (IS_BLANK(c)) {
|
||||||
loc++;
|
loc++;
|
||||||
if (IS_EOF(loc)) {
|
if (IS_EOF) {
|
||||||
return (loc);
|
return (loc);
|
||||||
}
|
}
|
||||||
c = pgn_content[loc];
|
c = pgn_content[loc];
|
||||||
|
|
|
@ -58,6 +58,7 @@ private:
|
||||||
long NextNonBlank(long);
|
long NextNonBlank(long);
|
||||||
/// @brief Parse a HalfMove at a specific location into @a pgn_content
|
/// @brief Parse a HalfMove at a specific location into @a pgn_content
|
||||||
long ParseHalfMove(long, HalfMove *);
|
long ParseHalfMove(long, HalfMove *);
|
||||||
|
/// @brief Parse a consecutive sequence of comment
|
||||||
long ParseComment(long, HalfMove *);
|
long ParseComment(long, HalfMove *);
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
|
@ -1,8 +1,20 @@
|
||||||
include_directories(./catch3/)
|
|
||||||
|
|
||||||
# Copy asset files
|
# Copy asset files
|
||||||
file(COPY pgn_files DESTINATION ${CMAKE_CURRENT_BINARY_DIR}/)
|
file(COPY pgn_files DESTINATION ${CMAKE_CURRENT_BINARY_DIR}/)
|
||||||
|
|
||||||
# Run tests
|
# Configure catch3
|
||||||
add_executable(pgnp_tests tests.cpp ./catch3/catch_amalgamated.cpp)
|
include_directories(./catch3/)
|
||||||
target_link_libraries(pgnp_tests pgnp)
|
add_library(catch3 SHARED ./catch3/catch_amalgamated.cpp)
|
||||||
|
|
||||||
|
# Add tests
|
||||||
|
add_executable(pgnp_valid valid.cpp)
|
||||||
|
target_link_libraries(pgnp_valid pgnp catch3)
|
||||||
|
add_test(PGNP_Valid_PGN_Set pgnp_valid)
|
||||||
|
|
||||||
|
add_executable(pgnp_str str.cpp)
|
||||||
|
target_link_libraries(pgnp_str pgnp catch3)
|
||||||
|
add_test(PGNP_STR_Compliant_Set pgnp_str)
|
||||||
|
|
||||||
|
add_executable(pgnp_combined combined.cpp)
|
||||||
|
target_link_libraries(pgnp_combined pgnp catch3)
|
||||||
|
add_test(PGNP_Combined_Set pgnp_combined)
|
43
tests/combined.cpp
Normal file
43
tests/combined.cpp
Normal file
|
@ -0,0 +1,43 @@
|
||||||
|
#include "pgnp.hpp"
|
||||||
|
#include <catch_amalgamated.hpp>
|
||||||
|
|
||||||
|
using namespace pgnp;
|
||||||
|
|
||||||
|
TEST_CASE("Hartwig PGN", "[combined/hartwig]") {
|
||||||
|
// PGN source: https://www.angelfire.com/games3/smartbridge/
|
||||||
|
|
||||||
|
pgnp::PGN pgn;
|
||||||
|
pgn.FromFile("pgn_files/combined/hartwig.pgn");
|
||||||
|
|
||||||
|
// Count games
|
||||||
|
REQUIRE_NOTHROW([&]() {
|
||||||
|
char i = 0;
|
||||||
|
try {
|
||||||
|
while (true) {
|
||||||
|
pgn.ParseNextGame();
|
||||||
|
i++;
|
||||||
|
}
|
||||||
|
} catch (const NoGameFound &e) {
|
||||||
|
CHECK(i == 29);
|
||||||
|
}
|
||||||
|
}());
|
||||||
|
|
||||||
|
SECTION("Check comments of a game") {
|
||||||
|
pgnp::PGN pgn;
|
||||||
|
pgn.FromFile("pgn_files/combined/hartwig.pgn");
|
||||||
|
pgn.ParseNextGame();
|
||||||
|
pgn.ParseNextGame();
|
||||||
|
pgn.ParseNextGame();
|
||||||
|
pgn.ParseNextGame();
|
||||||
|
pgn.ParseNextGame(); // Load game 5
|
||||||
|
|
||||||
|
HalfMove *m = new HalfMove();
|
||||||
|
pgn.GetMoves(m);
|
||||||
|
std::cout << m->comment;
|
||||||
|
CHECK(m->comment ==
|
||||||
|
"I had actually prepared 1.d4 for the tournament, but I backed out "
|
||||||
|
"in every (!) game for various different reasons. In this case it "
|
||||||
|
"was because things were in such a rut I would only be cheered by "
|
||||||
|
"winning in crushing style. Thankfully it worked!");
|
||||||
|
}
|
||||||
|
}
|
1478
tests/pgn_files/combined/hartwig.pgn
Normal file
1478
tests/pgn_files/combined/hartwig.pgn
Normal file
File diff suppressed because it is too large
Load diff
17
tests/str.cpp
Normal file
17
tests/str.cpp
Normal file
|
@ -0,0 +1,17 @@
|
||||||
|
#include "pgnp.hpp"
|
||||||
|
#include <catch_amalgamated.hpp>
|
||||||
|
|
||||||
|
using namespace pgnp;
|
||||||
|
|
||||||
|
TEST_CASE("Seven Tag Roster", "[std/pgn1]") {
|
||||||
|
PGN pgn;
|
||||||
|
REQUIRE_NOTHROW(pgn.FromFile("pgn_files/str/pgn1.pgn"));
|
||||||
|
REQUIRE_NOTHROW(pgn.ParseNextGame());
|
||||||
|
|
||||||
|
REQUIRE_NOTHROW(pgn.STRCheck());
|
||||||
|
HalfMove *m = new HalfMove();
|
||||||
|
pgn.GetMoves(m);
|
||||||
|
REQUIRE(m->GetLength() == 85);
|
||||||
|
CHECK(pgn.GetResult() == "1/2-1/2");
|
||||||
|
REQUIRE_THROWS_AS(pgn.ParseNextGame(),NoGameFound);
|
||||||
|
}
|
|
@ -87,16 +87,3 @@ TEST_CASE("Valid PGN", "[valid/pgn2]") {
|
||||||
}
|
}
|
||||||
REQUIRE_THROWS_AS(pgn.ParseNextGame(),NoGameFound);
|
REQUIRE_THROWS_AS(pgn.ParseNextGame(),NoGameFound);
|
||||||
}
|
}
|
||||||
|
|
||||||
TEST_CASE("Seven Tag Roster", "[std/pgn1]") {
|
|
||||||
PGN pgn;
|
|
||||||
REQUIRE_NOTHROW(pgn.FromFile("pgn_files/str/pgn1.pgn"));
|
|
||||||
REQUIRE_NOTHROW(pgn.ParseNextGame());
|
|
||||||
|
|
||||||
REQUIRE_NOTHROW(pgn.STRCheck());
|
|
||||||
HalfMove *m = new HalfMove();
|
|
||||||
pgn.GetMoves(m);
|
|
||||||
REQUIRE(m->GetLength() == 85);
|
|
||||||
CHECK(pgn.GetResult() == "1/2-1/2");
|
|
||||||
REQUIRE_THROWS_AS(pgn.ParseNextGame(),NoGameFound);
|
|
||||||
}
|
|
Loading…
Add table
Reference in a new issue