Jelajahi Sumber

basic aggresive bot

Kajetan Johannes Hammerle 4 tahun lalu
induk
melakukan
b3b34d336c
8 mengubah file dengan 391 tambahan dan 6 penghapusan
  1. 3 3
      Makefile
  2. 247 0
      source/Game.cpp
  3. 56 0
      source/Game.h
  4. 4 3
      source/Main.cpp
  5. 42 0
      source/String.cpp
  6. 23 0
      source/String.h
  7. 6 0
      source/Types.h
  8. 10 0
      source/meson.build

+ 3 - 3
Makefile

@@ -3,10 +3,10 @@ ARGS = -o fanorona -Wall -Wextra -pedantic -Werror
 all: fanorona
 
 test: fanorona
-	./fanorona
+	socat TCP4-CONNECT:alpaga.hammerle.me:4455 EXEC:"./fanorona"
 	
-fanorona: source/Main.cpp
-	g++ source/Main.cpp ${ARGS}
+fanorona: source/Main.cpp source/Game.cpp source/Game.h source/String.cpp source/String.h
+	g++ source/Main.cpp source/Game.cpp source/String.cpp ${ARGS}
 	
 clean:
 	rm -f fanorona

+ 247 - 0
source/Game.cpp

@@ -0,0 +1,247 @@
+#include <iostream>
+
+#include "Game.h"
+#include "Types.h"
+
+Game::Game() : activeX(-1), activeY(-1), directionX(0), directionY(0), lastLocation(0) {
+}
+
+void Game::print(String& s) const {
+    s.append("\n\r  0 1 2 3 4 5 6 7 8\n\r");
+    printLine(s, 0);
+    s.append("  |\\|/|\\|/|\\|/|\\|/|\n\r");
+    printLine(s, 1);
+    s.append("  |/|\\|/|\\|/|\\|/|\\|\n\r");
+    printLine(s, 2);
+    s.append("  |\\|/|\\|/|\\|/|\\|/|\n\r");
+    printLine(s, 3);
+    s.append("  |/|\\|/|\\|/|\\|/|\\|\n\r");
+    printLine(s, 4);
+    s.append("\n\r");
+}
+
+void Game::printLine(String& s, int index) const {
+    static const char map[] = {
+        '#', 'O', '.'
+    };
+    s.append(index + '0').append(' ');
+    for(int x = 0; x < 8; x++) {
+        if(x == activeX && index == activeY) {
+            s.append("*-");
+            continue;
+        }
+        s.append(map[fields[x][index]]).append('-');
+    }
+    if(8 == activeX && index == activeY) {
+        s.append('*');
+    } else {
+        s.append(map[fields[8][index]]);
+    }
+    s.append("\n\r");
+}
+
+void Game::parse() {
+    String s;
+    while(true) {
+        char c = std::cin.get();
+        if(c == '0') {
+            parseFields();
+            continue;
+        }
+        if(c == '\n' || c == '\r') {
+            std::cerr << s << "\n";
+            s.clear();
+            continue;
+        }
+        s.append(c);
+        if(s == "select stone:") {
+            std::cerr << s;
+            s.clear();
+            selectStone();
+        } else if(s == "select location to move:") {
+            std::cerr << s;
+            s.clear();
+            selectLocationToMove();
+        } else if(s == "select stone to take:") {
+            std::cerr << s;
+            s.clear();
+            selectStoneToTake();
+        }
+    }
+}
+
+void Game::consumeLine() const {
+    while(std::cin.get() != '\n');
+    while(true) {
+        char c = std::cin.peek();
+        if(c != '\r') {
+            break;
+        }
+        std::cin.get();
+    }
+}
+
+void Game::consumeLine(uint index) {
+    std::cin.get();
+    std::cin.get();
+    for(uint i = 0; i < 9; i++) {
+        char c = std::cin.get();
+        if(c == '#') {
+            fields[i][index] = BLACK;
+        } else if(c == '.') {
+            fields[i][index] = EMPTY;
+        } else if(c == 'O') {
+            fields[i][index] = WHITE;
+        } else if(c == '*') {
+            activeX = i;
+            activeY = index;
+            fields[i][index] = WHITE;
+        } else {
+            std::cerr << "game field parsing error\n";
+        }
+        std::cin.get();
+    }
+}
+
+void Game::parseFields() {
+    for(uint i = 0; i < 5; i++) {
+        consumeLine();
+        consumeLine(i);
+    }
+    String s;
+    print(s);
+    std::cerr << s;
+}
+
+void Game::selectStone() {
+    lastLocation = 0;
+    directionX = 0;
+    directionY = 0;
+
+    Location from;
+    int rank = -1;
+    for(int bx = 0; bx < 9; bx++) {
+        for(int by = 0; by < 5; by++) {
+            if(fields[bx][by] != EMPTY) {
+                continue;
+            }
+            for(int mx = -1; mx <= 1; mx++) {
+                for(int my = -1; my <= 1; my++) {
+                    int x = bx + mx;
+                    int y = by + my;
+                    if((mx == 0 && my == 0) || !isInRange(x, y) || !areNeighbours(bx, by, x, y) || fields[x][y] != WHITE) {
+                        continue;
+                    }
+                    int rankA = getRank(bx, by, x, y);
+                    int rankB = getRank(x, y, bx, by);
+                    if(rankA > rank) {
+                        rank = rankA;
+                        from = {x, y};
+                    }
+                    if(rankB > rank) {
+                        rank = rankB;
+                        from = {x, y};
+                    }
+                }
+            }
+        }
+    }
+    std::cout << from.x << " " << from.y << "\n";
+}
+
+void Game::selectLocationToMove() {
+    Location to;
+    int rank = -1;
+    for(int mx = -1; mx <= 1; mx++) {
+        for(int my = -1; my <= 1; my++) {
+            int x = activeX + mx;
+            int y = activeY + my;
+            if((mx == 0 && my == 0) || (mx == directionX && my == directionY) || !isInRange(x, y) ||
+                    !areNeighbours(activeX, activeY, x, y) || fields[x][y] != EMPTY || isInQueue(x, y)) {
+                continue;
+            }
+            int rankA = getRank(x, y, activeX, activeY);
+            int rankB = getRank(activeX, activeY, x, y);
+            if(rankA > rank) {
+                rank = rankA;
+                to = {x, y};
+            }
+            if(rankB > rank) {
+                rank = rankB;
+                to = {x, y};
+            }
+        }
+    }
+    addLocation(activeX, activeY);
+    directionX = to.x - activeX;
+    directionY = to.y - activeY;
+    std::cout << to.x << " " << to.y << "\n";
+}
+
+void Game::selectStoneToTake() {
+    int rankA = getRank(activeX, activeY, activeX + directionX, activeY + directionY);
+    int rankB = getRank(activeX + directionX, activeY + directionY, activeX, activeY);
+    if(rankA > rankB) {
+        std::cout << (activeX + directionX * 2) << " " << (activeY + directionY * 2) << "\n";
+    } else {
+        std::cout << (activeX - directionX) << " " << (activeY - directionY) << "\n";
+    }
+}
+
+void Game::move(int fromX, int fromY, int toX, int toY) {
+    addLocation(fromX, fromY);
+    fields[fromX][fromY] = EMPTY;
+    fields[toX][toY] = WHITE;
+}
+
+bool Game::isInRange(int x, int y) const {
+    return x >= 0 && x <= 8 && y >= 0 && y <= 4;
+}
+
+bool Game::areNeighbours(int x, int y, int x2, int y2) const {
+    if(x == x2 && y == y2) {
+        return false;
+    }
+    int diffX = x - x2;
+    int diffY = y - y2;
+    if(diffX < -1 || diffX > 1 || diffY < -1 || diffY > 1) {
+        return false;
+    }
+    if(diffX == 0 || diffY == 0) {
+        return true;
+    }
+    return !((x & 1) ^ (y & 1));
+}
+
+int Game::getRank(int x, int y, int x2, int y2) const {
+    int diffX = x2 - x;
+    int diffY = y2 - y;
+    int rank = 0;
+    while(true) {
+        x2 += diffX;
+        y2 += diffY;
+        if(!isInRange(x2, y2) || fields[x2][y2] != BLACK) {
+            return rank;
+        }
+        rank++;
+    }
+}
+
+void Game::addLocation(int x, int y) {
+    if(lastLocation >= lastLocations.size()) {
+        // safely to ignore, such long turns will never happen
+        return;
+    }
+    lastLocations[lastLocation].x = x;
+    lastLocations[lastLocation].y = y;
+    lastLocation++;
+}
+
+bool Game::isInQueue(int x, int y) const {
+    for(uint i = 0; i < lastLocation; i++) {
+        if(lastLocations[i].x == x && lastLocations[i].y == y) {
+            return true;
+        }
+    }
+    return false;
+}

+ 56 - 0
source/Game.h

@@ -0,0 +1,56 @@
+#ifndef GAME_H
+#define GAME_H
+
+#include <array>
+
+#include "String.h"
+#include "Types.h"
+
+class Game final {
+public:
+
+    enum FieldState {
+        BLACK, WHITE, EMPTY
+    };
+
+    enum State {
+        SELECTION, MOVE, ANOTHER_MOVE, REMOVE_SELECTION
+    };
+
+    Game();
+
+    void parse();
+
+private:
+    void consumeLine() const;
+    void consumeLine(uint index);
+    void parseFields();
+    void selectStone();
+    void selectLocationToMove();
+    void selectStoneToTake();
+
+    void move(int fromX, int fromY, int toX, int toY);
+    bool isInRange(int x, int y) const;
+    void print(String& s) const;
+    void printLine(String& s, int index) const;
+    int getRank(int x, int y, int x2, int y2) const;
+    bool areNeighbours(int x, int y, int x2, int y2) const;
+
+    void addLocation(int x, int y);
+    bool isInQueue(int x, int y) const;
+
+    FieldState fields[9][5];
+    int activeX;
+    int activeY;
+    int directionX;
+    int directionY;
+
+    struct Location {
+        int x;
+        int y;
+    };
+    std::array<Location, 20> lastLocations;
+    uint lastLocation;
+};
+
+#endif

+ 4 - 3
source/Main.cpp

@@ -1,6 +1,7 @@
-#include <iostream>
+#include "Game.h"
 
 int main() {
-    std::cout << "Test2\n";
+    Game game;
+    game.parse();
     return 0;
-}
+}

+ 42 - 0
source/String.cpp

@@ -0,0 +1,42 @@
+#include <iostream>
+#include <cstring>
+
+#include "String.h"
+#include "Types.h"
+
+String::String() : length(0) {
+    data[0] = '\0';
+}
+
+String::operator const char*() const {
+    return data;
+}
+
+String& String::append(char c) {
+    if(length + 1 < LENGTH) {
+        data[length++] = c;
+        data[length] = '\0';
+    }
+    return *this;
+}
+
+String& String::append(const char* str) {
+    for(uint i = 0; length + 1 < LENGTH && str[i] != '\0'; length++, i++) {
+        data[length] = str[i];
+    }
+    data[length] = '\0';
+    return *this;
+}
+
+uint String::getLength() const {
+    return length;
+}
+
+void String::clear() {
+    length = 0;
+    data[0] = '\0';
+}
+
+bool String::operator==(const char* str) const {
+    return strcmp(data, str) == 0;
+}

+ 23 - 0
source/String.h

@@ -0,0 +1,23 @@
+#ifndef STRING_H
+#define STRING_H
+
+#include "Types.h"
+
+class String {
+public:
+    String();
+    operator const char*() const;
+    String& append(char c);
+    String& append(const char* str);
+    void clear();
+    bool operator==(const char* str) const;
+    
+    uint getLength() const;
+    
+private:
+    static const uint LENGTH = 4096;
+    uint length;
+    char data[LENGTH];
+};
+
+#endif

+ 6 - 0
source/Types.h

@@ -0,0 +1,6 @@
+#ifndef TYPES_H
+#define TYPES_H
+
+typedef unsigned int uint;
+
+#endif

+ 10 - 0
source/meson.build

@@ -0,0 +1,10 @@
+project('fanorona server bot', 'cpp')
+
+sources = ['Main.cpp', 'Socket.cpp', 'Client.cpp', 'Utils.cpp', 'Game.cpp', 'String.cpp']
+
+threadDep = dependency('threads')
+
+executable('fanorona_server_bot', 
+    sources: sources,
+    dependencies : [threadDep],
+    cpp_args: ['-Wall', '-Wextra', '-pedantic', '-Werror'])