Skip to content

Hex AI distributed API. Sends AI moves computations and game analyzes to peers.

Notifications You must be signed in to change notification settings

playhex/hex-ai-distributed

Repository files navigation

Hex AI distributed

API that process Hex move by sending move calculation to peers.

Install

Clone this repo, then:

yarn install

This library requires a Redis instance. Configure your redis url in an empty .env file:

REDIS_URL=0.0.0.0:6379

With Docker, you can quickly start a local instance of redis with:

docker run -p 6379:6379 redis

Then run server and/or a worker in separate processes:

yarn peer-server
yarn server
yarn worker

Build for production

Build typescript files with:

yarn build

Then run server and/or worker in separate processes:

node dist/src/peer-server
node dist/src/server
node dist/src/worker

Or run these with pm2.

With docker

cp .env.docker .env

docker compose up
# only server
docker compose up redis server

# only worker
docker compose up worker
  • To test with multiple workers:
docker compose up --scale worker=2
  • To empty all redis queues
docker compose run server sh -c "cd /app && yarn obliterate"

API

Server can accept this request in order to process an AI move calculation through workers:

POST http://0.0.0.0:8088/calculate-move
Content-Type: application/json

{
    "game": {
        "size": 11,
        "currentPlayer": "black",
        "movesHistory": "e4 d6",
        "swapRule": false
    },
    "ai": {
        "engine": "mohex",
        "maxGames": 20
    }
}

Result:

f7

See all the possible HTTP requests in this postman collection:

https://www.postman.com/alcalyn/workspace/hex/collection/32226212-905e39fc-27b0-4792-9634-88a2dba2de5c

Katahex

To use Katahex, you first need to install a pre-trained model, currently downloadable here:

https://drive.google.com/file/d/1xMvP_75xgo0271nQbmlAJ40rvpKiFTgP/view

See https://github.com/selinger/katahex#running.

Then place the .bin.gz model file in this repo, in katahex/ folder to make it available in Docker.

Other usages

If you just want to use preinstalled Hex AI engines in docker using command line, you can run and enter in the worker container with:

docker compose run worker bash

# Use Mohex
mohex

# Use Katahex
katahex gtp -config /app/katahex/config.cfg -model /app/katahex/katahex_model_20220618.bin.gz

Example:

$> docker compose run worker bash
$> mohex
MoHex 2.0.CMake Dec 14 2023
Copyright (C) 2007-2012 by the authors of the Benzene project.
...

boardsize 5
=

showboard
=

  e1275ef128b3b312
  a  b  c  d  e
 1\.  .  .  .  .\1
  2\.  .  .  .  .\2
   3\.  .  .  .  .\3
    4\.  .  .  .  .\4
     5\.  .  .  .  .\5
        a  b  c  d  e

Architecture

  • Server (express)

Can request an API move, or game analysis.

Will create a job and queue it (redis).

Once job is done, send result back to client.

client <-> API <-> bullmq/redis
  • Peer server (NodeJs.net)

An open socket, workers will connect to this socket. Keep all workers online, knows which one is available or working.

Will dequeue jobs to send to next available worker.

bullmq/redis <-> Peer server <-> workers
  • Worker (NodeJs.net)

Spawn Hex AI engines, connect to peer server and wait for a job.

Will compute it, and send result back to peer server.

Peer server <-> worker <-> mohex/katahex

General workflow

ApiInput        ApiOutput
    v               ^
        API
    v               ^
JobInput        JobOutput
    v               ^
        redis

        ...

        redis
    v               ^
JobInput        JobOutput
    v               ^
        Peer server
    v               ^
WorkerInput     WorkerOutput
    v               ^
        Worker

Output types are ResultType

API then returns either 200 with data, or 400 with error message.

  • Calculate move

Compute a move from AI. Can choose AI and level.

CalculateMoveInput      CalculateMoveOutput
    v                           ^
        API
    v                           ^
CalculateMoveInput      CalculateMoveOutput
    v                           ^
        redis

        ...

        redis
    v                           ^
CalculateMoveInput      CalculateMoveOutput
    v                           ^
        Peer server
    v                           ^
CalculateMoveInput      CalculateMoveOutput
    v                           ^
        Worker
  • Analyze game

Uses an AI (katahex) to analyze game moves (best moves, blunders...).

AnalyzeGameInput        AnalyzeGameOutput
    v                           ^
        API
    v                           ^
AnalyzeGameInput        AnalyzeGameOutput
> AnalyzeMoveInput[]    > AnalyzeMoveOutput[]
    v                           ^
        redis

        ...

        redis
    v                           ^
AnalyzeMoveInput        AnalyzeMoveOutput
    v                           ^
        Peer server
    v                           ^
AnalyzeMoveInput        AnalyzeMoveOutput
    v                           ^
        Worker

Uses bullmq job children feature.

AnalyzeGameInput is kept as parent job, creates multiple AnalyzeMoveInput jobs to send to workers.

Once every AnalyzeMoveInput is done, will be "reconsolided" back to AnalyzeGameOutput which is sent to client.

Message queues

  • Queue worker_tasks

Contains WorkerTask jobs. Can be calculate move, or analyze a position.

Analyze a position (AnalyzeMove) have lower priority.

  • Queue analyzes

Contains AnalyzeGame jobs. Every AnalyzeGame job is the parent of multiple AnalyzeMove jobs in the worker_tasks queue.

Build worker standalone docker image

To run a worker from an image with all inside (source code and katahex model),

  • check there is a model in ./katahex/ folder, and is the one configured in .env

  • then run:

# build
docker build -f src/worker/Dockerfile.standalone -t alcalyn/hex-distributed-ai-worker:standalone .

# publish
docker push alcalyn/hex-distributed-ai-worker:standalone

Start a worker

Pull or update worker:

docker pull alcalyn/hex-distributed-ai-worker:standalone
  • start a worker with:
docker run -it alcalyn/hex-distributed-ai-worker:standalone

# (ctrl + C to stop it)
  • start it in background:
docker run -d --name hex-worker alcalyn/hex-distributed-ai-worker:standalone

# and stop it with:
docker rm -f hex-worker

# or show logs with:
docker logs -ft hex-worker
  • run and enter in the container without running a worker:
docker run -it alcalyn/hex-distributed-ai-worker:standalone bash

Simulate stale connection between peer and server

Peer keep a long running tcp connection to peer-server.

There is chance that peer lose connection without having time to gracefully closing the connection (wifi down, unpluged, laptop closed without exiting worker).

This makes the connection stale, peer-server think the worker is available but is not, and when sending him a job, he lose time before re-sending job to another peer.

This should be handled, peer-server must keep connections alive, and disconnect stale peers.

For developments, to simulate losing connection, with docker:

# get network name (the name is this project folder name by default)
docker network ls

# get worker container id (the name is something like alcalyn/hex-distributed-ai-worker:latest)
docker ps

# disconnect
docker network disconnect hex-ai-distributed_default ead82e1f53e8

Then check how long the server take to remove this peer:

GET /status

Or send him a task, and check how long it takes to re-attribute the job to another available peer.

For this, you can run 2 workers with docker compose up --scale worker=2.

To restart the test, you need to reconnect worker to the network:

docker network connect hex-ai-distributed_default ead82e1f53e8

About

Hex AI distributed API. Sends AI moves computations and game analyzes to peers.

Resources

Stars

Watchers

Forks