mjai


Namemjai JSON
Version 0.2.1 PyPI version JSON
download
home_pageNone
SummaryMahjong game simulator for evaluating submission files on mjai.app
upload_time2024-07-20 13:50:22
maintainerNone
docs_urlNone
authorNone
requires_python>=3.10
licenseNone
keywords mahjong mjai game simulator
VCS
bugtrack_url
requirements No requirements were recorded.
Travis-CI No Travis.
coveralls test coverage No coveralls.
            # mjai-simulator

[mjai.app](https://mjai.app) is a platform for mahjong AI competition.
This repository contains the implementation of mahjong game simulator for evaluating submission files on mjai.app.

## Installation

`mjai` package is available at the Python Package Index (PyPI).
You can install it with `pip` command.

```sh
$ pip install mjai
```

## Usage

You can simulate a mahjong game by specifying submission files as shown in the code below.
The Simulator runs [Docker](https://www.docker.com/) internally. The `docker` command must be installed and be available to run with user privileges.

```py
import mjai

submissions = [
    "examples/shanten.zip",
    "examples/tsumogiri.zip",
    "examples/tsumogiri.zip",
    "examples/invalidbot2.zip",
]
mjai.Simulator(submissions, logs_dir="./logs").run()
```

This package also provides a base class for developing bots that communicate via the mjai protocol.

```py
from mjai import Bot


class RulebaseBot(Bot):
    def think(self) -> str:
        if self.can_tsumo_agari:
            return self.action_tsumo_agari()
        elif self.can_ron_agari:
            return self.action_ron_agari()
        elif self.can_riichi:
            return self.action_riichi()

        ...

if __name__ == "__main__":
    RulebaseBot(player_id=int(sys.argv[1])).start()
```

### Docker Image

Submission files are deployed in a Docker container and run as a Docker container.

This repository contains Dockerfile and other resources that will be used to create the Docker container.
The docker image is pushed to Docker Hub (`docker.io/smly/mjai-client:v3`).

### Submission format

Please prepare a program that outputs the appropriate [mjai protocol message](https://mjai.app/docs/mjai-protocol) to the standard output when given input in the mjai protocol format from standard input. Name this program "bot.py" and pack it into a zip file. The zip file should contain bot.py directly under the root directory.

bot.py must be a Python script, but it is also possible to include precompiled libraries if they are executable. The program will be executed in a `linux/amd64` environment. The submission file must be 1000 MB or less.
bot.py takes a player ID as its first argument. Player ID must be 0, 1, 2, or 3. Player ID 0 represents the chicha 起家, and subsequent numbers represent the shimocha 下家, toimen 対面 or kamicha 上家 of the chicha 起家. See [example code](https://github.com/smly/mjai.app/blob/main/examples/tsumogiri/bot.py) for details.

### Timeout

When the `mjai.Simulator` instance creates an environment to run the submission file in docker, it specifies the `--platform linux/amd64` option. If you want to run on a different architecture, you will have to emulate and run the container, which will be much slower. If you are debugging on an architecture other than `linux/amd64`, you can avoid timeout errors by relaxing the timeout limit. Specify the `timeout` argument as follows. The `timeout` is set to 2.0 by default.

```py
Simulator(submissions, logs_dir="./logs", timeout=10.0).run()
```

If the simulation is interrupted, the docker container may not terminate successfully. If past docker containers remain, the new docker container may fail to launch due to duplicate HTTP ports. You can remove all docker containers that have the `smly/mjai-client` image as an ancestor in a batch as follows:

```sh
% for cid in `docker ps -a --filter ancestor=smly/mjai-client:v3 --format "{{.ID}}"`; do docker rm -f $cid; done
```

## Mjai Protocol

[Our Mjai protocol](https://mjai.app/docs/mjai-protocol) is basically based on the [Gimite's original Mjai Protocol](https://gimite.net/pukiwiki/index.php?Mjai%20%E9%BA%BB%E9%9B%80AI%E5%AF%BE%E6%88%A6%E3%82%B5%E3%83%BC%E3%83%90), but customized based on [Mortal's Mjai Engine implementation](https://github.com/Equim-chan/Mortal/blob/main/libriichi/src/mjai/event.rs). The following points are customized:

- Messages are sent and received by standard input and standard output.
- The game server sends messages collectively up to the event that the player can act on.
- When the player does not send the appropriate event message, it is treated as a chombo and a mangan-sized penalty is paid.
- If the player does not send a message within 2 seconds, it is treated as a chombo and a mangan-sized penalty is paid.
- First round (kyoku) is not necessarily East 1.

### First round is not necessarily East 1

If an error occurs in any one of the 4 players, the game is treated as a abortive draw (ryukyoku; 流局). At this time, all 4 players' containers are terminated. The players' containers are restarted and the game is resumed from the next round in which an error occurs. Therefore, the first round received by the player may not be East 1.

The following are messages sent and received as seen from player 0.
`<-` denotes events for the player.
`->` denotes events from the player.

```
# Game resumed from S1-1
0 <- [{"type":"start_game","names":["0","1","2","3"],"id":0}]
0 -> {"type":"none"}

# NOTE: No ryukyoku events are sent from the game server.
0 <- [{"type":"end_kyoku"}]
0 -> {"type":"none"}

# S2-2 first tsumo tile
0 <- [{"type":"start_kyoku","bakaze":"S","dora_marker":"1p","kyoku":2,"honba":2,"kyotaku":0,"oya":1,"scores":[800,61100,11300,26800],"tehais":[["4p","4s","P","3p","1p","5s","2m","F","1m","7s","9m","6m","9s"],["?","?","?","?","?","?","?","?","?","?","?","?","?"],["?","?","?","?","?","?","?","?","?","?","?","?","?"],["?","?","?","?","?","?","?","?","?","?","?","?","?"]]},{"type":"tsumo","actor":1,"pai":"?"},{"type":"dahai","actor":1,"pai":"F","tsumogiri":false},{"type":"tsumo","actor":2,"pai":"?"},{"type":"dahai","actor":2,"pai":"3m","tsumogiri":true},{"type":"tsumo","actor":3,"pai":"?"},{"type":"dahai","actor":3,"pai":"1m","tsumogiri":true},{"type":"tsumo","actor":0,"pai":"3s"}]
0 -> {"type":"dahai","pai":"3s","actor":0,"tsumogiri":true}
```

### Case Study: Furiten (振聴)

When the player has already made a tenpai but the hand is furiten. Since the player cannot Ron, even if the waiting tile is discarded by the opponent, no action is possible.
For example, let's say you have `2333678m 678s 678p`. The waiting tile is `14m` and `1m` has already been discarded.
Since the hand is a Furiten, even if the other player discards `1m`, the player cannot Ron.

```
3 <- [{"type":"dahai","actor":3,"pai":"P","tsumogiri":true},{"type":"tsumo","actor":0,"pai":"?"},{"type":"dahai","actor":0,"pai":"2p","tsumogiri":true},{"type":"tsumo","actor":1,"pai":"?"},{"type":"dahai","actor":1,"pai":"4m","tsumogiri":true},{"type":"tsumo","actor":2,"pai":"?"},{"type":"dahai","actor":2,"pai":"6m","tsumogiri":true}]
```

In this case, immediately after actor 2 discards 6m, input is given to actor 3. actor 3 needs to decide whether to call `chi` on 6m.

### Case Study: Ankan (暗槓)

In the case of an ankan, the dora event comes first, followed by the tsumo event.

```
3 -> {"type": "ankan", "actor": 3, "consumed": ["6s", "6s", "6s", "6s"]}
3 <- [{"type":"ankan","actor":3,"consumed":["6s","6s","6s","6s"]},{"type":"dora","dora_marker":"6p"},{"type":"tsumo","actor":3,"pai":"7p"}]
3 -> {"type":dahai","actor":3,"pai":"7p","tsumogiri":true}
```

## For Developers

### Debug with interactive shell

The procedures executed by Simulator can be checked and debugged one by one as follows:

```bash
# pull latest docker image
% docker pull docker.io/smly/mjai-client:v3

# launch
% CONTAINER_ID=`docker run -d --rm -p 28080:3000 --mount "type=bind,src=/Users/smly/gitws/mjai.app/examples/rulebase.zip,dst=/bot.zip,readonly" smly/mjai-client:v3 sleep infinity`

# install bot program
% docker exec ${CONTAINER_ID} unzip -q /bot.zip

# debug
% docker exec -it ${CONTAINER_ID} /workspace/.pyenv/shims/python -u bot.py 0
[{"type":"start_game","id":0}]  <-- Input
{"type":"none"}  <-- Output
[{"type":"start_kyoku","bakaze":"E","dora_marker":"2s","kyoku":1,"honba":0,"kyotaku":0,"oya":0,"scores":[25000,25000,25000,25000],"tehais":[["E","6p","9m","8m","C","2s","7m","S","6m","1m","S","3s","8m"],["?","?","?","?","?","?","?","?","?","?","?","?","?"],["?","?","?","?","?","?","?","?","?","?","?","?","?"],["?","?","?","?","?","?","?","?","?","?","?","?","?"]]},{"type":"tsumo","actor":0,"pai":"1m"}]  <-- Input
{"type": "dahai", "actor": 0, "pai": "C", "tsumogiri": false}  <-- Output
[{"type":"dahai","actor":0,"pai":"C","tsumogiri":false},{"type":"tsumo","actor":1,"pai":"?"},{"type":"dahai","actor":1,"pai":"3m","tsumogiri":false},{"type":"tsumo","actor":2,"pai":"?"},{"type":"dahai","actor":2,"pai":"1m","tsumogiri":false}]  <-- Input
{"type": "none"}  <-- Output
[{"type":"tsumo","actor":3,"pai":"?"},{"type":"dahai","actor":3,"pai":"1m","tsumogiri":false}]  <-- Input
{"type": "none"}  <-- Output
[{"type":"tsumo","actor":0,"pai":"C"}]  <-- Input
{"type": "dahai", "actor": 0, "pai": "C", "tsumogiri": true}  <-- Output
```

### Debug with http sever

```bash
# install http server interface of mjai protocol
% docker cp python/mjai/http_server/server.py ${CONTAINER_ID}:/workspace/00__server__.py

# http server mode. `0` is the player index.
% docker exec -it ${CONTAINER_ID} /workspace/.pyenv/shims/python 00__server__.py 0
```

```bash
% curl http://localhost:28080/
OK

% curl -X POST -d '[{"type":"start_game","id":0}]' http://localhost:28080/
{"type":"none"}

% cat > request.json
[{"type":"start_kyoku","bakaze":"E","dora_marker":"2s","kyoku":1,"honba":0,"kyotaku":0,"oya":0,"scores":[25000,25000,25000,25000],"tehais":[["E","6p","9m","8m","C","2s","7m","S","6m","1m","S","3s","8m"],["?","?","?","?","?","?","?","?","?","?","?","?","?"],["?","?","?","?","?","?","?","?","?","?","?","?","?"],["?","?","?","?","?","?","?","?","?","?","?","?","?"]]},{"type":"tsumo","actor":0,"pai":"1m"}]

% curl -X POST -d '@request.json' http://localhost:28080/
{"type":"dahai","actor":0,"pai":"6p","tsumogiri":false}
```

## Development

Confirmed working with rustc 1.75.0 (82e1608df 2023-12-21).

```sh
$ pip install maturin  # install build tool
$ maturin build --release --locked --target aarch64-apple-darwin --out dist
```

## Special Thanks

The code in `./src` directory is Mortal's libriichi with minor updates. Mortal is distributed under the AGPL-3.0 and is copyrighted by Equim.


            

Raw data

            {
    "_id": null,
    "home_page": null,
    "name": "mjai",
    "maintainer": null,
    "docs_url": null,
    "requires_python": ">=3.10",
    "maintainer_email": null,
    "keywords": "mahjong, mjai, game, simulator",
    "author": null,
    "author_email": "smly <eown.er+mjai@gmail.com>",
    "download_url": "https://files.pythonhosted.org/packages/db/77/1298b7b1307ee150fb5f341248de517efcc48124ff8e5370c968648e1db4/mjai-0.2.1.tar.gz",
    "platform": null,
    "description": "# mjai-simulator\n\n[mjai.app](https://mjai.app) is a platform for mahjong AI competition.\nThis repository contains the implementation of mahjong game simulator for evaluating submission files on mjai.app.\n\n## Installation\n\n`mjai` package is available at the Python Package Index (PyPI).\nYou can install it with `pip` command.\n\n```sh\n$ pip install mjai\n```\n\n## Usage\n\nYou can simulate a mahjong game by specifying submission files as shown in the code below.\nThe Simulator runs [Docker](https://www.docker.com/) internally. The `docker` command must be installed and be available to run with user privileges.\n\n```py\nimport mjai\n\nsubmissions = [\n    \"examples/shanten.zip\",\n    \"examples/tsumogiri.zip\",\n    \"examples/tsumogiri.zip\",\n    \"examples/invalidbot2.zip\",\n]\nmjai.Simulator(submissions, logs_dir=\"./logs\").run()\n```\n\nThis package also provides a base class for developing bots that communicate via the mjai protocol.\n\n```py\nfrom mjai import Bot\n\n\nclass RulebaseBot(Bot):\n    def think(self) -> str:\n        if self.can_tsumo_agari:\n            return self.action_tsumo_agari()\n        elif self.can_ron_agari:\n            return self.action_ron_agari()\n        elif self.can_riichi:\n            return self.action_riichi()\n\n        ...\n\nif __name__ == \"__main__\":\n    RulebaseBot(player_id=int(sys.argv[1])).start()\n```\n\n### Docker Image\n\nSubmission files are deployed in a Docker container and run as a Docker container.\n\nThis repository contains Dockerfile and other resources that will be used to create the Docker container.\nThe docker image is pushed to Docker Hub (`docker.io/smly/mjai-client:v3`).\n\n### Submission format\n\nPlease prepare a program that outputs the appropriate [mjai protocol message](https://mjai.app/docs/mjai-protocol) to the standard output when given input in the mjai protocol format from standard input. Name this program \"bot.py\" and pack it into a zip file. The zip file should contain bot.py directly under the root directory.\n\nbot.py must be a Python script, but it is also possible to include precompiled libraries if they are executable. The program will be executed in a `linux/amd64` environment. The submission file must be 1000 MB or less.\nbot.py takes a player ID as its first argument. Player ID must be 0, 1, 2, or 3. Player ID 0 represents the chicha \u8d77\u5bb6, and subsequent numbers represent the shimocha \u4e0b\u5bb6, toimen \u5bfe\u9762 or kamicha \u4e0a\u5bb6 of the chicha \u8d77\u5bb6. See [example code](https://github.com/smly/mjai.app/blob/main/examples/tsumogiri/bot.py) for details.\n\n### Timeout\n\nWhen the `mjai.Simulator` instance creates an environment to run the submission file in docker, it specifies the `--platform linux/amd64` option. If you want to run on a different architecture, you will have to emulate and run the container, which will be much slower. If you are debugging on an architecture other than `linux/amd64`, you can avoid timeout errors by relaxing the timeout limit. Specify the `timeout` argument as follows. The `timeout` is set to 2.0 by default.\n\n```py\nSimulator(submissions, logs_dir=\"./logs\", timeout=10.0).run()\n```\n\nIf the simulation is interrupted, the docker container may not terminate successfully. If past docker containers remain, the new docker container may fail to launch due to duplicate HTTP ports. You can remove all docker containers that have the `smly/mjai-client` image as an ancestor in a batch as follows:\n\n```sh\n% for cid in `docker ps -a --filter ancestor=smly/mjai-client:v3 --format \"{{.ID}}\"`; do docker rm -f $cid; done\n```\n\n## Mjai Protocol\n\n[Our Mjai protocol](https://mjai.app/docs/mjai-protocol) is basically based on the [Gimite's original Mjai Protocol](https://gimite.net/pukiwiki/index.php?Mjai%20%E9%BA%BB%E9%9B%80AI%E5%AF%BE%E6%88%A6%E3%82%B5%E3%83%BC%E3%83%90), but customized based on [Mortal's Mjai Engine implementation](https://github.com/Equim-chan/Mortal/blob/main/libriichi/src/mjai/event.rs). The following points are customized:\n\n- Messages are sent and received by standard input and standard output.\n- The game server sends messages collectively up to the event that the player can act on.\n- When the player does not send the appropriate event message, it is treated as a chombo and a mangan-sized penalty is paid.\n- If the player does not send a message within 2 seconds, it is treated as a chombo and a mangan-sized penalty is paid.\n- First round (kyoku) is not necessarily East 1.\n\n### First round is not necessarily East 1\n\nIf an error occurs in any one of the 4 players, the game is treated as a abortive draw (ryukyoku; \u6d41\u5c40). At this time, all 4 players' containers are terminated. The players' containers are restarted and the game is resumed from the next round in which an error occurs. Therefore, the first round received by the player may not be East 1.\n\nThe following are messages sent and received as seen from player 0.\n`<-` denotes events for the player.\n`->` denotes events from the player.\n\n```\n# Game resumed from S1-1\n0 <- [{\"type\":\"start_game\",\"names\":[\"0\",\"1\",\"2\",\"3\"],\"id\":0}]\n0 -> {\"type\":\"none\"}\n\n# NOTE: No ryukyoku events are sent from the game server.\n0 <- [{\"type\":\"end_kyoku\"}]\n0 -> {\"type\":\"none\"}\n\n# S2-2 first tsumo tile\n0 <- [{\"type\":\"start_kyoku\",\"bakaze\":\"S\",\"dora_marker\":\"1p\",\"kyoku\":2,\"honba\":2,\"kyotaku\":0,\"oya\":1,\"scores\":[800,61100,11300,26800],\"tehais\":[[\"4p\",\"4s\",\"P\",\"3p\",\"1p\",\"5s\",\"2m\",\"F\",\"1m\",\"7s\",\"9m\",\"6m\",\"9s\"],[\"?\",\"?\",\"?\",\"?\",\"?\",\"?\",\"?\",\"?\",\"?\",\"?\",\"?\",\"?\",\"?\"],[\"?\",\"?\",\"?\",\"?\",\"?\",\"?\",\"?\",\"?\",\"?\",\"?\",\"?\",\"?\",\"?\"],[\"?\",\"?\",\"?\",\"?\",\"?\",\"?\",\"?\",\"?\",\"?\",\"?\",\"?\",\"?\",\"?\"]]},{\"type\":\"tsumo\",\"actor\":1,\"pai\":\"?\"},{\"type\":\"dahai\",\"actor\":1,\"pai\":\"F\",\"tsumogiri\":false},{\"type\":\"tsumo\",\"actor\":2,\"pai\":\"?\"},{\"type\":\"dahai\",\"actor\":2,\"pai\":\"3m\",\"tsumogiri\":true},{\"type\":\"tsumo\",\"actor\":3,\"pai\":\"?\"},{\"type\":\"dahai\",\"actor\":3,\"pai\":\"1m\",\"tsumogiri\":true},{\"type\":\"tsumo\",\"actor\":0,\"pai\":\"3s\"}]\n0 -> {\"type\":\"dahai\",\"pai\":\"3s\",\"actor\":0,\"tsumogiri\":true}\n```\n\n### Case Study: Furiten (\u632f\u8074)\n\nWhen the player has already made a tenpai but the hand is furiten. Since the player cannot Ron, even if the waiting tile is discarded by the opponent, no action is possible.\nFor example, let's say you have `2333678m 678s 678p`. The waiting tile is `14m` and `1m` has already been discarded.\nSince the hand is a Furiten, even if the other player discards `1m`, the player cannot Ron.\n\n```\n3 <- [{\"type\":\"dahai\",\"actor\":3,\"pai\":\"P\",\"tsumogiri\":true},{\"type\":\"tsumo\",\"actor\":0,\"pai\":\"?\"},{\"type\":\"dahai\",\"actor\":0,\"pai\":\"2p\",\"tsumogiri\":true},{\"type\":\"tsumo\",\"actor\":1,\"pai\":\"?\"},{\"type\":\"dahai\",\"actor\":1,\"pai\":\"4m\",\"tsumogiri\":true},{\"type\":\"tsumo\",\"actor\":2,\"pai\":\"?\"},{\"type\":\"dahai\",\"actor\":2,\"pai\":\"6m\",\"tsumogiri\":true}]\n```\n\nIn this case, immediately after actor 2 discards 6m, input is given to actor 3. actor 3 needs to decide whether to call `chi` on 6m.\n\n### Case Study: Ankan (\u6697\u69d3)\n\nIn the case of an ankan, the dora event comes first, followed by the tsumo event.\n\n```\n3 -> {\"type\": \"ankan\", \"actor\": 3, \"consumed\": [\"6s\", \"6s\", \"6s\", \"6s\"]}\n3 <- [{\"type\":\"ankan\",\"actor\":3,\"consumed\":[\"6s\",\"6s\",\"6s\",\"6s\"]},{\"type\":\"dora\",\"dora_marker\":\"6p\"},{\"type\":\"tsumo\",\"actor\":3,\"pai\":\"7p\"}]\n3 -> {\"type\":dahai\",\"actor\":3,\"pai\":\"7p\",\"tsumogiri\":true}\n```\n\n## For Developers\n\n### Debug with interactive shell\n\nThe procedures executed by Simulator can be checked and debugged one by one as follows:\n\n```bash\n# pull latest docker image\n% docker pull docker.io/smly/mjai-client:v3\n\n# launch\n% CONTAINER_ID=`docker run -d --rm -p 28080:3000 --mount \"type=bind,src=/Users/smly/gitws/mjai.app/examples/rulebase.zip,dst=/bot.zip,readonly\" smly/mjai-client:v3 sleep infinity`\n\n# install bot program\n% docker exec ${CONTAINER_ID} unzip -q /bot.zip\n\n# debug\n% docker exec -it ${CONTAINER_ID} /workspace/.pyenv/shims/python -u bot.py 0\n[{\"type\":\"start_game\",\"id\":0}]  <-- Input\n{\"type\":\"none\"}  <-- Output\n[{\"type\":\"start_kyoku\",\"bakaze\":\"E\",\"dora_marker\":\"2s\",\"kyoku\":1,\"honba\":0,\"kyotaku\":0,\"oya\":0,\"scores\":[25000,25000,25000,25000],\"tehais\":[[\"E\",\"6p\",\"9m\",\"8m\",\"C\",\"2s\",\"7m\",\"S\",\"6m\",\"1m\",\"S\",\"3s\",\"8m\"],[\"?\",\"?\",\"?\",\"?\",\"?\",\"?\",\"?\",\"?\",\"?\",\"?\",\"?\",\"?\",\"?\"],[\"?\",\"?\",\"?\",\"?\",\"?\",\"?\",\"?\",\"?\",\"?\",\"?\",\"?\",\"?\",\"?\"],[\"?\",\"?\",\"?\",\"?\",\"?\",\"?\",\"?\",\"?\",\"?\",\"?\",\"?\",\"?\",\"?\"]]},{\"type\":\"tsumo\",\"actor\":0,\"pai\":\"1m\"}]  <-- Input\n{\"type\": \"dahai\", \"actor\": 0, \"pai\": \"C\", \"tsumogiri\": false}  <-- Output\n[{\"type\":\"dahai\",\"actor\":0,\"pai\":\"C\",\"tsumogiri\":false},{\"type\":\"tsumo\",\"actor\":1,\"pai\":\"?\"},{\"type\":\"dahai\",\"actor\":1,\"pai\":\"3m\",\"tsumogiri\":false},{\"type\":\"tsumo\",\"actor\":2,\"pai\":\"?\"},{\"type\":\"dahai\",\"actor\":2,\"pai\":\"1m\",\"tsumogiri\":false}]  <-- Input\n{\"type\": \"none\"}  <-- Output\n[{\"type\":\"tsumo\",\"actor\":3,\"pai\":\"?\"},{\"type\":\"dahai\",\"actor\":3,\"pai\":\"1m\",\"tsumogiri\":false}]  <-- Input\n{\"type\": \"none\"}  <-- Output\n[{\"type\":\"tsumo\",\"actor\":0,\"pai\":\"C\"}]  <-- Input\n{\"type\": \"dahai\", \"actor\": 0, \"pai\": \"C\", \"tsumogiri\": true}  <-- Output\n```\n\n### Debug with http sever\n\n```bash\n# install http server interface of mjai protocol\n% docker cp python/mjai/http_server/server.py ${CONTAINER_ID}:/workspace/00__server__.py\n\n# http server mode. `0` is the player index.\n% docker exec -it ${CONTAINER_ID} /workspace/.pyenv/shims/python 00__server__.py 0\n```\n\n```bash\n% curl http://localhost:28080/\nOK\n\n% curl -X POST -d '[{\"type\":\"start_game\",\"id\":0}]' http://localhost:28080/\n{\"type\":\"none\"}\n\n% cat > request.json\n[{\"type\":\"start_kyoku\",\"bakaze\":\"E\",\"dora_marker\":\"2s\",\"kyoku\":1,\"honba\":0,\"kyotaku\":0,\"oya\":0,\"scores\":[25000,25000,25000,25000],\"tehais\":[[\"E\",\"6p\",\"9m\",\"8m\",\"C\",\"2s\",\"7m\",\"S\",\"6m\",\"1m\",\"S\",\"3s\",\"8m\"],[\"?\",\"?\",\"?\",\"?\",\"?\",\"?\",\"?\",\"?\",\"?\",\"?\",\"?\",\"?\",\"?\"],[\"?\",\"?\",\"?\",\"?\",\"?\",\"?\",\"?\",\"?\",\"?\",\"?\",\"?\",\"?\",\"?\"],[\"?\",\"?\",\"?\",\"?\",\"?\",\"?\",\"?\",\"?\",\"?\",\"?\",\"?\",\"?\",\"?\"]]},{\"type\":\"tsumo\",\"actor\":0,\"pai\":\"1m\"}]\n\n% curl -X POST -d '@request.json' http://localhost:28080/\n{\"type\":\"dahai\",\"actor\":0,\"pai\":\"6p\",\"tsumogiri\":false}\n```\n\n## Development\n\nConfirmed working with rustc 1.75.0 (82e1608df 2023-12-21).\n\n```sh\n$ pip install maturin  # install build tool\n$ maturin build --release --locked --target aarch64-apple-darwin --out dist\n```\n\n## Special Thanks\n\nThe code in `./src` directory is Mortal's libriichi with minor updates. Mortal is distributed under the AGPL-3.0 and is copyrighted by Equim.\n\n",
    "bugtrack_url": null,
    "license": null,
    "summary": "Mahjong game simulator for evaluating submission files on mjai.app",
    "version": "0.2.1",
    "project_urls": {
        "Docuemntation": "https://github.com/smly/mjai.app/blob/main/README.md",
        "Repository": "https://github.com/smly/mjai.app"
    },
    "split_keywords": [
        "mahjong",
        " mjai",
        " game",
        " simulator"
    ],
    "urls": [
        {
            "comment_text": "",
            "digests": {
                "blake2b_256": "612b563ecdb4e9ce39de5f5933dd9e1b58e01670381102b389205d89e501b447",
                "md5": "fbebe7a68cab234d66b5d9aa9972cef7",
                "sha256": "f5ff6a389b76fd4a3bf4aff760c19c29475074cc483c95953bcf3df001d989c7"
            },
            "downloads": -1,
            "filename": "mjai-0.2.1-cp310-cp310-macosx_11_0_arm64.whl",
            "has_sig": false,
            "md5_digest": "fbebe7a68cab234d66b5d9aa9972cef7",
            "packagetype": "bdist_wheel",
            "python_version": "cp310",
            "requires_python": ">=3.10",
            "size": 1476986,
            "upload_time": "2024-07-20T13:50:07",
            "upload_time_iso_8601": "2024-07-20T13:50:07.549294Z",
            "url": "https://files.pythonhosted.org/packages/61/2b/563ecdb4e9ce39de5f5933dd9e1b58e01670381102b389205d89e501b447/mjai-0.2.1-cp310-cp310-macosx_11_0_arm64.whl",
            "yanked": false,
            "yanked_reason": null
        },
        {
            "comment_text": "",
            "digests": {
                "blake2b_256": "30aa704ba4c2e4e4351834ffabde39a6d046f02e3f3a37cbb8936ad9cb7f4fbd",
                "md5": "199a0a0b03d585a1f2dd45911702612e",
                "sha256": "0eecddb86746eb48827f2d81f7dccd2fff28648ea9c388f8dd55ab4774e9e7f9"
            },
            "downloads": -1,
            "filename": "mjai-0.2.1-cp310-cp310-manylinux_2_17_x86_64.manylinux2014_x86_64.whl",
            "has_sig": false,
            "md5_digest": "199a0a0b03d585a1f2dd45911702612e",
            "packagetype": "bdist_wheel",
            "python_version": "cp310",
            "requires_python": ">=3.10",
            "size": 1692377,
            "upload_time": "2024-07-20T13:50:09",
            "upload_time_iso_8601": "2024-07-20T13:50:09.454116Z",
            "url": "https://files.pythonhosted.org/packages/30/aa/704ba4c2e4e4351834ffabde39a6d046f02e3f3a37cbb8936ad9cb7f4fbd/mjai-0.2.1-cp310-cp310-manylinux_2_17_x86_64.manylinux2014_x86_64.whl",
            "yanked": false,
            "yanked_reason": null
        },
        {
            "comment_text": "",
            "digests": {
                "blake2b_256": "66f17880c30bf840c1b610f599d955d8c2f908445d20b5a3be738065aa4acc28",
                "md5": "03c4f73c49c88a3fde399a229adca10e",
                "sha256": "33853f451a7c5858c808f965d7180c20816a7302779f9f7331bc2f9d9e97dcd4"
            },
            "downloads": -1,
            "filename": "mjai-0.2.1-cp310-none-win_amd64.whl",
            "has_sig": false,
            "md5_digest": "03c4f73c49c88a3fde399a229adca10e",
            "packagetype": "bdist_wheel",
            "python_version": "cp310",
            "requires_python": ">=3.10",
            "size": 1353518,
            "upload_time": "2024-07-20T13:50:11",
            "upload_time_iso_8601": "2024-07-20T13:50:11.244960Z",
            "url": "https://files.pythonhosted.org/packages/66/f1/7880c30bf840c1b610f599d955d8c2f908445d20b5a3be738065aa4acc28/mjai-0.2.1-cp310-none-win_amd64.whl",
            "yanked": false,
            "yanked_reason": null
        },
        {
            "comment_text": "",
            "digests": {
                "blake2b_256": "b2997b248dba856f85aaef3301536bacd6c5046fb370ab9012530cbf0f001fb3",
                "md5": "16c7889d0c0830e2e5c1839644f4914f",
                "sha256": "31a066f250df8538b8d7f4f833e7bc0f91baef23b58899f565e4dbc5ba8ed564"
            },
            "downloads": -1,
            "filename": "mjai-0.2.1-cp311-cp311-macosx_11_0_arm64.whl",
            "has_sig": false,
            "md5_digest": "16c7889d0c0830e2e5c1839644f4914f",
            "packagetype": "bdist_wheel",
            "python_version": "cp311",
            "requires_python": ">=3.10",
            "size": 1476611,
            "upload_time": "2024-07-20T13:50:13",
            "upload_time_iso_8601": "2024-07-20T13:50:13.043974Z",
            "url": "https://files.pythonhosted.org/packages/b2/99/7b248dba856f85aaef3301536bacd6c5046fb370ab9012530cbf0f001fb3/mjai-0.2.1-cp311-cp311-macosx_11_0_arm64.whl",
            "yanked": false,
            "yanked_reason": null
        },
        {
            "comment_text": "",
            "digests": {
                "blake2b_256": "05d53c95799b5f802a9706bb57d36fef8df1fbd67dd3ca9735136ca6e3c26aea",
                "md5": "2261d06595b956833c0d2c757e3337e4",
                "sha256": "7e9bd89d0dfe8c00ecb1e157f889d4c2aa1098d21f3025dff5e434448e46667d"
            },
            "downloads": -1,
            "filename": "mjai-0.2.1-cp311-cp311-manylinux_2_17_x86_64.manylinux2014_x86_64.whl",
            "has_sig": false,
            "md5_digest": "2261d06595b956833c0d2c757e3337e4",
            "packagetype": "bdist_wheel",
            "python_version": "cp311",
            "requires_python": ">=3.10",
            "size": 1692141,
            "upload_time": "2024-07-20T13:50:14",
            "upload_time_iso_8601": "2024-07-20T13:50:14.765726Z",
            "url": "https://files.pythonhosted.org/packages/05/d5/3c95799b5f802a9706bb57d36fef8df1fbd67dd3ca9735136ca6e3c26aea/mjai-0.2.1-cp311-cp311-manylinux_2_17_x86_64.manylinux2014_x86_64.whl",
            "yanked": false,
            "yanked_reason": null
        },
        {
            "comment_text": "",
            "digests": {
                "blake2b_256": "73921dee7027fc3c44473769c888b24c9b3341e1cc4f5483f7c2e3322d513f9a",
                "md5": "65e60f849fcde2c05d3b2e27447b1e8a",
                "sha256": "80e5f908e0a3ea86ae278d4a7d320babc7a608421f9442719f2a01e5d484ee3a"
            },
            "downloads": -1,
            "filename": "mjai-0.2.1-cp311-none-win_amd64.whl",
            "has_sig": false,
            "md5_digest": "65e60f849fcde2c05d3b2e27447b1e8a",
            "packagetype": "bdist_wheel",
            "python_version": "cp311",
            "requires_python": ">=3.10",
            "size": 1353337,
            "upload_time": "2024-07-20T13:50:16",
            "upload_time_iso_8601": "2024-07-20T13:50:16.585111Z",
            "url": "https://files.pythonhosted.org/packages/73/92/1dee7027fc3c44473769c888b24c9b3341e1cc4f5483f7c2e3322d513f9a/mjai-0.2.1-cp311-none-win_amd64.whl",
            "yanked": false,
            "yanked_reason": null
        },
        {
            "comment_text": "",
            "digests": {
                "blake2b_256": "e508a19b1cc890390ec45dee2e308efec753d9beffe007def3c00b730d586d6f",
                "md5": "66577de64c98e4cc6f084f60020b99c4",
                "sha256": "14e8126a0370d74444a807ea46cb75f7b2f5e7f24124755e711c298285d82464"
            },
            "downloads": -1,
            "filename": "mjai-0.2.1-cp312-cp312-macosx_11_0_arm64.whl",
            "has_sig": false,
            "md5_digest": "66577de64c98e4cc6f084f60020b99c4",
            "packagetype": "bdist_wheel",
            "python_version": "cp312",
            "requires_python": ">=3.10",
            "size": 1475339,
            "upload_time": "2024-07-20T13:50:17",
            "upload_time_iso_8601": "2024-07-20T13:50:17.875471Z",
            "url": "https://files.pythonhosted.org/packages/e5/08/a19b1cc890390ec45dee2e308efec753d9beffe007def3c00b730d586d6f/mjai-0.2.1-cp312-cp312-macosx_11_0_arm64.whl",
            "yanked": false,
            "yanked_reason": null
        },
        {
            "comment_text": "",
            "digests": {
                "blake2b_256": "41932a0b0c1a1f02aff5ed7156c28ea21607a6c85c53ac250c2ec22cc7646706",
                "md5": "de73e14ecdcb5a323d82d33cdeebe2fb",
                "sha256": "9e261bef81938fa122b79811412b770e361d00775335b00e3d55bd90a9a4fc88"
            },
            "downloads": -1,
            "filename": "mjai-0.2.1-cp312-cp312-manylinux_2_17_x86_64.manylinux2014_x86_64.whl",
            "has_sig": false,
            "md5_digest": "de73e14ecdcb5a323d82d33cdeebe2fb",
            "packagetype": "bdist_wheel",
            "python_version": "cp312",
            "requires_python": ">=3.10",
            "size": 1690719,
            "upload_time": "2024-07-20T13:50:19",
            "upload_time_iso_8601": "2024-07-20T13:50:19.552299Z",
            "url": "https://files.pythonhosted.org/packages/41/93/2a0b0c1a1f02aff5ed7156c28ea21607a6c85c53ac250c2ec22cc7646706/mjai-0.2.1-cp312-cp312-manylinux_2_17_x86_64.manylinux2014_x86_64.whl",
            "yanked": false,
            "yanked_reason": null
        },
        {
            "comment_text": "",
            "digests": {
                "blake2b_256": "7ede77e59de167c851588d9d6c7521331b0d498ebd4db9d344e3b15456a626ea",
                "md5": "f81cffe38f4738689e2f683980b9c0e3",
                "sha256": "327cdf462f20261b85be2f61bea1f1f713c10f20b037bde2e2b2e8d6e1597f12"
            },
            "downloads": -1,
            "filename": "mjai-0.2.1-cp312-none-win_amd64.whl",
            "has_sig": false,
            "md5_digest": "f81cffe38f4738689e2f683980b9c0e3",
            "packagetype": "bdist_wheel",
            "python_version": "cp312",
            "requires_python": ">=3.10",
            "size": 1353124,
            "upload_time": "2024-07-20T13:50:21",
            "upload_time_iso_8601": "2024-07-20T13:50:21.170021Z",
            "url": "https://files.pythonhosted.org/packages/7e/de/77e59de167c851588d9d6c7521331b0d498ebd4db9d344e3b15456a626ea/mjai-0.2.1-cp312-none-win_amd64.whl",
            "yanked": false,
            "yanked_reason": null
        },
        {
            "comment_text": "",
            "digests": {
                "blake2b_256": "db771298b7b1307ee150fb5f341248de517efcc48124ff8e5370c968648e1db4",
                "md5": "77b196add969ebdbd057f83ebfc38ffe",
                "sha256": "7678ef75265192e0bf475ffbea27129de1a4c4c496de4ce95783d9178d33fbbe"
            },
            "downloads": -1,
            "filename": "mjai-0.2.1.tar.gz",
            "has_sig": false,
            "md5_digest": "77b196add969ebdbd057f83ebfc38ffe",
            "packagetype": "sdist",
            "python_version": "source",
            "requires_python": ">=3.10",
            "size": 373406,
            "upload_time": "2024-07-20T13:50:22",
            "upload_time_iso_8601": "2024-07-20T13:50:22.747470Z",
            "url": "https://files.pythonhosted.org/packages/db/77/1298b7b1307ee150fb5f341248de517efcc48124ff8e5370c968648e1db4/mjai-0.2.1.tar.gz",
            "yanked": false,
            "yanked_reason": null
        }
    ],
    "upload_time": "2024-07-20 13:50:22",
    "github": true,
    "gitlab": false,
    "bitbucket": false,
    "codeberg": false,
    "github_user": "smly",
    "github_project": "mjai.app",
    "travis_ci": false,
    "coveralls": false,
    "github_actions": true,
    "lcname": "mjai"
}
        
Elapsed time: 0.78203s