mirror of
https://github.com/syuilo/ai.git
synced 2025-03-25 21:12:56 +00:00
back.ts
This commit is contained in:
parent
a8926f927f
commit
c8e31bf25b
1 changed files with 62 additions and 62 deletions
|
@ -1,9 +1,9 @@
|
||||||
/**
|
/**
|
||||||
* -AI-
|
* -AI-
|
||||||
* Botのバックエンド(思考を担当)
|
* Bot backend (in charge of thinking)
|
||||||
*
|
*
|
||||||
* 対話と思考を同じプロセスで行うと、思考時間が長引いたときにストリームから
|
* If you do the interaction and thinking in the same process, they will be * disconnected from the stream when the thinking time is too long, so they should be done in separate processes.
|
||||||
* 切断されてしまうので、別々のプロセスで行うようにします
|
* * Keep them in separate processes to avoid disconnection from the stream when the thinking time is too long
|
||||||
*/
|
*/
|
||||||
|
|
||||||
import got from 'got';
|
import got from 'got';
|
||||||
|
@ -33,27 +33,27 @@ class Session {
|
||||||
private appliedOps: string[] = [];
|
private appliedOps: string[] = [];
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* 隅周辺のインデックスリスト(静的評価に利用)
|
* Index list around corners (used for static evaluation)
|
||||||
*/
|
*/
|
||||||
private sumiNearIndexes: number[] = [];
|
private sumiNearIndexes: number[] = [];
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* 隅のインデックスリスト(静的評価に利用)
|
* Corner index list (used for static evaluation)
|
||||||
*/
|
*/
|
||||||
private sumiIndexes: number[] = [];
|
private sumiIndexes: number[] = [];
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* 最大のターン数
|
* Maximum number of turns
|
||||||
*/
|
*/
|
||||||
private maxTurn;
|
private maxTurn;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* 現在のターン数
|
* Number of current turns
|
||||||
*/
|
*/
|
||||||
private currentTurn = 0;
|
private currentTurn = 0;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* 対局が開始したことを知らせた投稿
|
* A post announcing the start of a game
|
||||||
*/
|
*/
|
||||||
private startedNote: any = null;
|
private startedNote: any = null;
|
||||||
|
|
||||||
|
@ -96,7 +96,7 @@ class Session {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// 親プロセスからデータをもらう
|
// Get data from the parent process
|
||||||
private onInit = (msg: any) => {
|
private onInit = (msg: any) => {
|
||||||
this.game = msg.game;
|
this.game = msg.game;
|
||||||
this.form = msg.form;
|
this.form = msg.form;
|
||||||
|
@ -104,23 +104,23 @@ class Session {
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* 対局が始まったとき
|
* When the match started
|
||||||
*/
|
*/
|
||||||
private onStarted = (msg: any) => {
|
private onStarted = (msg: any) => {
|
||||||
this.game = msg.game;
|
this.game = msg.game;
|
||||||
if (this.game.canPutEverywhere) { // 対応してない
|
if (this.game.canPutEverywhere) { // Not supported
|
||||||
process.send!({
|
process.send!({
|
||||||
type: 'ended'
|
type: 'ended'
|
||||||
});
|
});
|
||||||
process.exit();
|
process.exit();
|
||||||
}
|
}
|
||||||
|
|
||||||
// TLに投稿する
|
// TPost to TL
|
||||||
this.postGameStarted().then(note => {
|
this.postGameStarted().then(note => {
|
||||||
this.startedNote = note;
|
this.startedNote = note;
|
||||||
});
|
});
|
||||||
|
|
||||||
// リバーシエンジン初期化
|
// Reversi Engine Initialization
|
||||||
this.engine = new Reversi.Game(this.game.map, {
|
this.engine = new Reversi.Game(this.game.map, {
|
||||||
isLlotheo: this.game.isLlotheo,
|
isLlotheo: this.game.isLlotheo,
|
||||||
canPutEverywhere: this.game.canPutEverywhere,
|
canPutEverywhere: this.game.canPutEverywhere,
|
||||||
|
@ -129,9 +129,9 @@ class Session {
|
||||||
|
|
||||||
this.maxTurn = this.engine.map.filter(p => p === 'empty').length - this.engine.board.filter(x => x != null).length;
|
this.maxTurn = this.engine.map.filter(p => p === 'empty').length - this.engine.board.filter(x => x != null).length;
|
||||||
|
|
||||||
//#region 隅の位置計算など
|
//#region Calculating corner positions, etc.
|
||||||
|
|
||||||
//#region 隅
|
//#region corner
|
||||||
this.engine.map.forEach((pix, i) => {
|
this.engine.map.forEach((pix, i) => {
|
||||||
if (pix == 'null') return;
|
if (pix == 'null') return;
|
||||||
|
|
||||||
|
@ -169,7 +169,7 @@ class Session {
|
||||||
});
|
});
|
||||||
//#endregion
|
//#endregion
|
||||||
|
|
||||||
//#region 隅の隣
|
//#region Next to the corner
|
||||||
this.engine.map.forEach((pix, i) => {
|
this.engine.map.forEach((pix, i) => {
|
||||||
if (pix == 'null') return;
|
if (pix == 'null') return;
|
||||||
if (this.sumiIndexes.includes(i)) return;
|
if (this.sumiIndexes.includes(i)) return;
|
||||||
|
@ -182,14 +182,14 @@ class Session {
|
||||||
};
|
};
|
||||||
|
|
||||||
const isSumiNear = (
|
const isSumiNear = (
|
||||||
check(x - 1, y - 1) || // 左上
|
check(x - 1, y - 1) || // Top left
|
||||||
check(x , y - 1) || // 上
|
check(x , y - 1) || // top
|
||||||
check(x + 1, y - 1) || // 右上
|
check(x + 1, y - 1) || // Top right
|
||||||
check(x + 1, y ) || // 右
|
check(x + 1, y ) || // right
|
||||||
check(x + 1, y + 1) || // 右下
|
check(x + 1, y + 1) || // Bottom right
|
||||||
check(x , y + 1) || // 下
|
check(x , y + 1) || // bottom
|
||||||
check(x - 1, y + 1) || // 左下
|
check(x - 1, y + 1) || // lower left
|
||||||
check(x - 1, y ) // 左
|
check(x - 1, y ) // left
|
||||||
)
|
)
|
||||||
|
|
||||||
if (isSumiNear) this.sumiNearIndexes.push(i);
|
if (isSumiNear) this.sumiNearIndexes.push(i);
|
||||||
|
@ -206,10 +206,10 @@ class Session {
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* 対局が終わったとき
|
* When the game is over
|
||||||
*/
|
*/
|
||||||
private onEnded = async (msg: any) => {
|
private onEnded = async (msg: any) => {
|
||||||
// ストリームから切断
|
// Disconnect from a stream
|
||||||
process.send!({
|
process.send!({
|
||||||
type: 'ended'
|
type: 'ended'
|
||||||
});
|
});
|
||||||
|
@ -250,7 +250,7 @@ class Session {
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* 打たれたとき
|
* When you get hit
|
||||||
*/
|
*/
|
||||||
private onLog = (log: any) => {
|
private onLog = (log: any) => {
|
||||||
if (log.id == null || !this.appliedOps.includes(log.id)) {
|
if (log.id == null || !this.appliedOps.includes(log.id)) {
|
||||||
|
@ -272,9 +272,9 @@ class Session {
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Botにとってある局面がどれだけ有利か静的に評価する
|
* Statically evaluate how advantageous a certain situation is for the bot
|
||||||
* static(静的)というのは、先読みはせずに盤面の状態のみで評価するということ。
|
* "Static" means that it evaluates only based on the state of the board without looking ahead.
|
||||||
* TODO: 接待時はまるっと処理の中身を変え、とにかく相手が隅を取っていること優先な評価にする
|
* TODO: When entertaining, completely change the content of the process and prioritize the evaluation of whether the opponent has taken the corner.
|
||||||
*/
|
*/
|
||||||
private staticEval = () => {
|
private staticEval = () => {
|
||||||
let score = this.engine.getPuttablePlaces(this.botColor).length;
|
let score = this.engine.getPuttablePlaces(this.botColor).length;
|
||||||
|
@ -283,28 +283,28 @@ class Session {
|
||||||
const stone = this.engine.board[index];
|
const stone = this.engine.board[index];
|
||||||
|
|
||||||
if (stone === this.botColor) {
|
if (stone === this.botColor) {
|
||||||
score += 1000; // 自分が隅を取っていたらスコアプラス
|
score += 1000; // If you take a corner, you get a plus point.
|
||||||
} else if (stone !== null) {
|
} else if (stone !== null) {
|
||||||
score -= 1000; // 相手が隅を取っていたらスコアマイナス
|
score -= 1000; // If the opponent takes the corner, the score is subtracted.
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// TODO: ここに (隅以外の確定石の数 * 100) をスコアに加算する処理を入れる
|
// TODO: Here we put a process to add (number of confirmed stones other than corners * 100) to the score.
|
||||||
|
|
||||||
for (const index of this.sumiNearIndexes) {
|
for (const index of this.sumiNearIndexes) {
|
||||||
const stone = this.engine.board[index];
|
const stone = this.engine.board[index];
|
||||||
|
|
||||||
if (stone === this.botColor) {
|
if (stone === this.botColor) {
|
||||||
score -= 10; // 自分が隅の周辺を取っていたらスコアマイナス(危険なので)
|
score -= 10; // If you take the corner area, you will get a minus point (because it is dangerous).
|
||||||
} else if (stone !== null) {
|
} else if (stone !== null) {
|
||||||
score += 10; // 相手が隅の周辺を取っていたらスコアプラス
|
score += 10; // If the opponent takes the corner area, you get a plus point.
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// ロセオならスコアを反転
|
// If you use Roseo, you can reverse the score.
|
||||||
if (this.game.isLlotheo) score = -score;
|
if (this.game.isLlotheo) score = -score;
|
||||||
|
|
||||||
// 接待ならスコアを反転
|
// Reverse the score for entertainment
|
||||||
if (this.isSettai) score = -score;
|
if (this.isSettai) score = -score;
|
||||||
|
|
||||||
return score;
|
return score;
|
||||||
|
@ -314,55 +314,55 @@ class Session {
|
||||||
console.log(`(${this.currentTurn}/${this.maxTurn}) Thinking...`);
|
console.log(`(${this.currentTurn}/${this.maxTurn}) Thinking...`);
|
||||||
console.time('think');
|
console.time('think');
|
||||||
|
|
||||||
// 接待モードのときは、全力(5手先読みくらい)で負けるようにする
|
// When in entertainment mode, try to lose with all your might (about 5 moves ahead)
|
||||||
// TODO: 接待のときは、どちらかというと「自分が不利になる手を選ぶ」というよりは、「相手に角を取らせられる手を選ぶ」ように思考する
|
// TODO: When entertaining, think more like "select a move that will allow the opponent to take the bishop" rather than "select a move that will put you at a disadvantage"
|
||||||
// 自分が不利になる手を選ぶというのは、換言すれば自分が打てる箇所を減らすことになるので、
|
// Choosing a move that will put you at a disadvantage means, in other words, reducing the number of places you can play, so
|
||||||
// 自分が打てる箇所が少ないと結果的に思考の選択肢が狭まり、対局をコントロールするのが難しくなるジレンマのようなものがある。
|
// There is a dilemma in that if you have fewer places you can play, your options for thinking are narrowed and it becomes difficult to control the game.
|
||||||
// つまり「相手を勝たせる」という意味での正しい接待は、「ゲーム序盤・中盤までは(通常通り)自分の有利になる手を打ち、終盤になってから相手が勝つように打つ」こと。
|
// In other words, the correct entertainment in the sense of "making the opponent win" is to "play a move that is advantageous to you (as usual) until the early and middle stages of the game, and then play in a way that will allow the opponent to win in the late stages."
|
||||||
// とはいえ藍に求められているのは、そういった「本物の」接待ではなく、単に「角を取らせてくれる」接待だと思われるので、
|
// However, what is required of Ai is not that kind of "real" entertainment, but simply entertainment that "lets you take the bishop," so
|
||||||
// 静的評価で「角に相手の石があるかどうか(と、ゲームが終わったときは相手が勝っているかどうか)」を考慮するようにすれば良いかもしれない。
|
// It may be a good idea to consider "whether the opponent has a stone in the bishop (and whether the opponent wins when the game ends)" in the static evaluation.
|
||||||
const maxDepth = this.isSettai ? 5 : this.strength;
|
const maxDepth = this.isSettai ? 5 : this.strength;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* αβ法での探索
|
* Search using the αβ method
|
||||||
*/
|
*/
|
||||||
const dive = (pos: number, alpha = -Infinity, beta = Infinity, depth = 0): number => {
|
const dive = (pos: number, alpha = -Infinity, beta = Infinity, depth = 0): number => {
|
||||||
// 試し打ち
|
// Test shot
|
||||||
this.engine.putStone(pos);
|
this.engine.putStone(pos);
|
||||||
|
|
||||||
const isBotTurn = this.engine.turn === this.botColor;
|
const isBotTurn = this.engine.turn === this.botColor;
|
||||||
|
|
||||||
// 勝った
|
// Won
|
||||||
if (this.engine.turn === null) {
|
if (this.engine.turn === null) {
|
||||||
const winner = this.engine.winner;
|
const winner = this.engine.winner;
|
||||||
|
|
||||||
// 勝つことによる基本スコア
|
// Base score by winning
|
||||||
const base = 10000;
|
const base = 10000;
|
||||||
|
|
||||||
let score;
|
let score;
|
||||||
|
|
||||||
if (this.game.isLlotheo) {
|
if (this.game.isLlotheo) {
|
||||||
// 勝ちは勝ちでも、より自分の石を少なくした方が美しい勝ちだと判定する
|
// A win is a win, but the more stones your opponent has, the more beautiful the win will be.
|
||||||
score = this.engine.winner ? base - (this.engine.blackCount * 100) : base - (this.engine.whiteCount * 100);
|
score = this.engine.winner ? base - (this.engine.blackCount * 100) : base - (this.engine.whiteCount * 100);
|
||||||
} else {
|
} else {
|
||||||
// 勝ちは勝ちでも、より相手の石を少なくした方が美しい勝ちだと判定する
|
// A win is a win, but the more stones your opponent has, the more beautiful the win will be.
|
||||||
score = this.engine.winner ? base + (this.engine.blackCount * 100) : base + (this.engine.whiteCount * 100);
|
score = this.engine.winner ? base + (this.engine.blackCount * 100) : base + (this.engine.whiteCount * 100);
|
||||||
}
|
}
|
||||||
|
|
||||||
// 巻き戻し
|
// Rewind
|
||||||
this.engine.undo();
|
this.engine.undo();
|
||||||
|
|
||||||
// 接待なら自分が負けた方が高スコア
|
// If you're entertaining someone, you get a higher score if you lose.
|
||||||
return this.isSettai
|
return this.isSettai
|
||||||
? winner !== this.botColor ? score : -score
|
? winner !== this.botColor ? score : -score
|
||||||
: winner === this.botColor ? score : -score;
|
: winner === this.botColor ? score : -score;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (depth === maxDepth) {
|
if (depth === maxDepth) {
|
||||||
// 静的に評価
|
// Statically Evaluated
|
||||||
const score = this.staticEval();
|
const score = this.staticEval();
|
||||||
|
|
||||||
// 巻き戻し
|
// Rewind
|
||||||
this.engine.undo();
|
this.engine.undo();
|
||||||
|
|
||||||
return score;
|
return score;
|
||||||
|
@ -373,11 +373,11 @@ class Session {
|
||||||
let a = alpha;
|
let a = alpha;
|
||||||
let b = beta;
|
let b = beta;
|
||||||
|
|
||||||
// TODO: 残りターン数というよりも「空いているマスが12以下」の場合に完全読みさせる
|
// TODO: Rather than the number of turns remaining, it is a perfect reading when there are 12 or less vacant spaces.
|
||||||
const nextDepth = (this.strength >= 4) && ((this.maxTurn - this.currentTurn) <= 12) ? Infinity : depth + 1;
|
const nextDepth = (this.strength >= 4) && ((this.maxTurn - this.currentTurn) <= 12) ? Infinity : depth + 1;
|
||||||
|
|
||||||
// 次のターンのプレイヤーにとって最も良い手を取得
|
// Get the best hand for the next player
|
||||||
// TODO: cansをまず浅く読んで(または価値マップを利用して)から有益そうな手から順に並べ替え、効率よく枝刈りできるようにする
|
// TODO: First, read through the list shallowly (or use a value map), then sort the list by the most profitable moves, allowing for efficient pruning.
|
||||||
for (const p of cans) {
|
for (const p of cans) {
|
||||||
if (isBotTurn) {
|
if (isBotTurn) {
|
||||||
const score = dive(p, a, beta, nextDepth);
|
const score = dive(p, a, beta, nextDepth);
|
||||||
|
@ -392,7 +392,7 @@ class Session {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// 巻き戻し
|
// Rewind
|
||||||
this.engine.undo();
|
this.engine.undo();
|
||||||
|
|
||||||
return value;
|
return value;
|
||||||
|
@ -425,19 +425,19 @@ class Session {
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* 対局が始まったことをMisskeyに投稿します
|
* Post to Misskey that the game has started
|
||||||
*/
|
*/
|
||||||
private postGameStarted = async () => {
|
private postGameStarted = async () => {
|
||||||
const text = this.isSettai
|
const text = this.isSettai
|
||||||
? serifs.reversi.startedSettai(this.userName)
|
? serifs.reversi.startedSettai(this.userName)
|
||||||
: serifs.reversi.started(this.userName, this.strength.toString());
|
: serifs.reversi.started(this.userName, this.strength.toString());
|
||||||
|
|
||||||
return await this.post(`${text}\n→[観戦する](${this.url})`);
|
return await this.post(`${text}\n→[Spectate](${this.url})`);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Misskeyに投稿します
|
* Post to Misskey
|
||||||
* @param text 投稿内容
|
* @param text Post Content
|
||||||
*/
|
*/
|
||||||
private post = async (text: string, renote?: any) => {
|
private post = async (text: string, renote?: any) => {
|
||||||
if (this.allowPost) {
|
if (this.allowPost) {
|
||||||
|
|
Loading…
Reference in a new issue