Dhia-GB commited on
Commit
aa744b6
·
verified ·
1 Parent(s): 83000ad

Chess Challenge submission by Dhia-GB

Browse files
Files changed (7) hide show
  1. README.md +26 -0
  2. config.json +26 -0
  3. model.safetensors +3 -0
  4. special_tokens_map.json +6 -0
  5. tokenizer.py +486 -0
  6. tokenizer_config.json +50 -0
  7. vocab.json +901 -0
README.md ADDED
@@ -0,0 +1,26 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ ---
2
+ library_name: transformers
3
+ tags:
4
+ - chess
5
+ - llm-course
6
+ - chess-challenge
7
+ license: mit
8
+ ---
9
+
10
+ # chess-v1_3
11
+
12
+ Chess model submitted to the LLM Course Chess Challenge.
13
+
14
+ ## Submission Info
15
+
16
+ - **Submitted by**: [Dhia-GB](https://huggingface.co/Dhia-GB)
17
+ - **Parameters**: 935,680
18
+ - **Organization**: LLM-course
19
+
20
+ ## Model Details
21
+
22
+ - **Architecture**: Chess Transformer (GPT-style)
23
+ - **Vocab size**: 899
24
+ - **Embedding dim**: 128
25
+ - **Layers**: 5
26
+ - **Heads**: 4
config.json ADDED
@@ -0,0 +1,26 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ {
2
+ "_name_or_path": "../my_model_v1_3/checkpoint-2784/",
3
+ "architectures": [
4
+ "ChessForCausalLM"
5
+ ],
6
+ "bos_token_id": 1,
7
+ "dropout": 0.1,
8
+ "eos_token_id": 2,
9
+ "layer_norm_epsilon": 1e-06,
10
+ "model_type": "chess_transformer",
11
+ "n_ctx": 256,
12
+ "n_embd": 128,
13
+ "n_head": 4,
14
+ "n_inner": 384,
15
+ "n_layer": 5,
16
+ "pad_token_id": 0,
17
+ "tie_weights": true,
18
+ "torch_dtype": "float32",
19
+ "transformers_version": "4.48.2",
20
+ "use_rms_norm": [
21
+ true
22
+ ],
23
+ "use_rope": true,
24
+ "use_swiglu": true,
25
+ "vocab_size": 899
26
+ }
model.safetensors ADDED
@@ -0,0 +1,3 @@
 
 
 
 
1
+ version https://git-lfs.github.com/spec/v1
2
+ oid sha256:db246035b9081da92277db498cf604bbf25684a7dda9df55da86da55bfaece30
3
+ size 3746840
special_tokens_map.json ADDED
@@ -0,0 +1,6 @@
 
 
 
 
 
 
 
1
+ {
2
+ "bos_token": "[BOS]",
3
+ "eos_token": "[EOS]",
4
+ "pad_token": "[PAD]",
5
+ "unk_token": "[UNK]"
6
+ }
tokenizer.py ADDED
@@ -0,0 +1,486 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ """
2
+ Custom Chess Tokenizer for the Chess Challenge.
3
+
4
+ This tokenizer treats each move as a single token using the extended UCI notation
5
+ from the Lichess dataset (e.g., WPe2e4, BNg8f6).
6
+
7
+ The dataset format uses:
8
+ - W/B prefix for White/Black
9
+ - Piece letter: P=Pawn, N=Knight, B=Bishop, R=Rook, Q=Queen, K=King
10
+ - Source and destination squares (e.g., e2e4)
11
+ - Special suffixes: (x)=capture, (+)=check, (+*)=checkmate, (o)/(O)=castling
12
+ """
13
+
14
+ from __future__ import annotations
15
+
16
+ import json
17
+ import os
18
+ from pathlib import Path
19
+ from typing import Dict, List, Optional
20
+
21
+ from transformers import PreTrainedTokenizer
22
+
23
+
24
+ class ChessTokenizer_v0(PreTrainedTokenizer):
25
+ """
26
+ A custom tokenizer for chess moves using extended UCI notation.
27
+
28
+ This tokenizer maps each possible chess move to a unique token ID.
29
+ The vocabulary is built from the training dataset to ensure all moves
30
+ encountered during training have a corresponding token.
31
+
32
+ Example:
33
+ >>> tokenizer = ChessTokenizer_v0()
34
+ >>> tokenizer.encode("WPe2e4 BPe7e5")
35
+ [1, 42, 87, 2] # [BOS, e2e4, e7e5, EOS]
36
+ """
37
+
38
+ model_input_names = ["input_ids", "attention_mask"]
39
+ vocab_files_names = {"vocab_file": "vocab.json"}
40
+
41
+ # Special tokens
42
+ PAD_TOKEN = "[PAD]"
43
+ BOS_TOKEN = "[BOS]"
44
+ EOS_TOKEN = "[EOS]"
45
+ UNK_TOKEN = "[UNK]"
46
+
47
+ def __init__(
48
+ self,
49
+ vocab_file: Optional[str] = None,
50
+ vocab: Optional[Dict[str, int]] = None,
51
+ **kwargs,
52
+ ):
53
+ """
54
+ Initialize the chess tokenizer.
55
+
56
+ Args:
57
+ vocab_file: Path to a JSON file containing the vocabulary mapping.
58
+ vocab: Dictionary mapping tokens to IDs (alternative to vocab_file).
59
+ **kwargs: Additional arguments passed to PreTrainedTokenizer.
60
+ """
61
+ # Initialize special tokens
62
+ self._pad_token = self.PAD_TOKEN
63
+ self._bos_token = self.BOS_TOKEN
64
+ self._eos_token = self.EOS_TOKEN
65
+ self._unk_token = self.UNK_TOKEN
66
+
67
+ # Remove any duplicate special-token entries passed through kwargs
68
+ # to avoid "multiple values for keyword" errors when loading from disk.
69
+ kwargs.pop("pad_token", None)
70
+ kwargs.pop("bos_token", None)
71
+ kwargs.pop("eos_token", None)
72
+ kwargs.pop("unk_token", None)
73
+
74
+ # Load or create vocabulary
75
+ if vocab is not None:
76
+ self._vocab = vocab
77
+ elif vocab_file is not None and os.path.exists(vocab_file):
78
+ with open(vocab_file, "r", encoding="utf-8") as f:
79
+ self._vocab = json.load(f)
80
+ else:
81
+ # Create a minimal vocabulary with just special tokens
82
+ # The full vocabulary should be built from the dataset
83
+ self._vocab = self._create_default_vocab()
84
+
85
+ # Create reverse mapping
86
+ self._ids_to_tokens = {v: k for k, v in self._vocab.items()}
87
+
88
+ # Call parent init AFTER setting up vocab
89
+ super().__init__(
90
+ pad_token=self._pad_token,
91
+ bos_token=self._bos_token,
92
+ eos_token=self._eos_token,
93
+ unk_token=self._unk_token,
94
+ **kwargs,
95
+ )
96
+
97
+ def _create_default_vocab(self) -> Dict[str, int]:
98
+ """
99
+ Create a minimal default vocabulary with just special tokens.
100
+
101
+ For the full vocabulary, use `build_vocab_from_dataset()`.
102
+ This minimal vocab is just a placeholder - you should build from data.
103
+ """
104
+ special_tokens = [self.PAD_TOKEN, self.BOS_TOKEN, self.EOS_TOKEN, self.UNK_TOKEN]
105
+ vocab = {token: idx for idx, token in enumerate(special_tokens)}
106
+ return vocab
107
+
108
+ @classmethod
109
+ def build_vocab_from_iterator(
110
+ cls,
111
+ iterator,
112
+ min_frequency: int = 1,
113
+ ) -> "ChessTokenizer_v0":
114
+ """
115
+ Build a tokenizer vocabulary from an iterator of game strings.
116
+
117
+ Args:
118
+ iterator: An iterator yielding game strings (space-separated moves).
119
+ min_frequency: Minimum frequency for a token to be included.
120
+
121
+ Returns:
122
+ A ChessTokenizer_v0 with the built vocabulary.
123
+ """
124
+ from collections import Counter
125
+
126
+ token_counts = Counter()
127
+
128
+ for game in iterator:
129
+ moves = game.strip().split()
130
+ token_counts.update(moves)
131
+
132
+ # Filter by frequency
133
+ tokens = [
134
+ token for token, count in token_counts.items()
135
+ if count >= min_frequency
136
+ ]
137
+
138
+ # Sort for reproducibility
139
+ tokens = sorted(tokens)
140
+
141
+ # Build vocabulary
142
+ special_tokens = [cls.PAD_TOKEN, cls.BOS_TOKEN, cls.EOS_TOKEN, cls.UNK_TOKEN]
143
+ vocab = {token: idx for idx, token in enumerate(special_tokens + tokens)}
144
+
145
+ return cls(vocab=vocab)
146
+
147
+ @classmethod
148
+ def build_vocab_from_dataset(
149
+ cls,
150
+ dataset_name: str = "dlouapre/lichess_2025-01_1M",
151
+ split: str = "train",
152
+ column: str = "text",
153
+ min_frequency: int = 500,
154
+ max_samples: Optional[int] = 100000,
155
+ ) -> "ChessTokenizer_v0":
156
+ """
157
+ Build a tokenizer vocabulary from a Hugging Face dataset.
158
+
159
+ Args:
160
+ dataset_name: Name of the dataset on Hugging Face Hub.
161
+ split: Dataset split to use.
162
+ column: Column containing the game strings.
163
+ min_frequency: Minimum frequency for a token to be included (default: 500).
164
+ max_samples: Maximum number of samples to process (default: 100k).
165
+
166
+ Returns:
167
+ A ChessTokenizer_v0 with the built vocabulary.
168
+ """
169
+ from datasets import load_dataset
170
+
171
+ dataset = load_dataset(dataset_name, split=split)
172
+
173
+ if max_samples is not None:
174
+ dataset = dataset.select(range(min(max_samples, len(dataset))))
175
+
176
+ def game_iterator():
177
+ for example in dataset:
178
+ yield example[column]
179
+
180
+ return cls.build_vocab_from_iterator(game_iterator(), min_frequency=min_frequency)
181
+
182
+ @property
183
+ def vocab_size(self) -> int:
184
+ """Return the size of the vocabulary."""
185
+ return len(self._vocab)
186
+
187
+ def get_vocab(self) -> Dict[str, int]:
188
+ """Return the vocabulary as a dictionary."""
189
+ return dict(self._vocab)
190
+
191
+ def _tokenize(self, text: str) -> List[str]:
192
+ """
193
+ Tokenize a string of moves into a list of tokens.
194
+
195
+ Args:
196
+ text: A string of space-separated moves.
197
+
198
+ Returns:
199
+ List of move tokens.
200
+ """
201
+ return text.strip().split()
202
+
203
+ def _convert_token_to_id(self, token: str) -> int:
204
+ """Convert a token to its ID."""
205
+ return self._vocab.get(token, self._vocab.get(self.UNK_TOKEN, 0))
206
+
207
+ def _convert_id_to_token(self, index: int) -> str:
208
+ """Convert an ID to its token."""
209
+ return self._ids_to_tokens.get(index, self.UNK_TOKEN)
210
+
211
+ def convert_tokens_to_string(self, tokens: List[str]) -> str:
212
+ """Convert a list of tokens back to a string."""
213
+ # Filter out special tokens for cleaner output
214
+ special = {self.PAD_TOKEN, self.BOS_TOKEN, self.EOS_TOKEN, self.UNK_TOKEN}
215
+ return " ".join(t for t in tokens if t not in special)
216
+
217
+ def save_vocabulary(
218
+ self,
219
+ save_directory: str,
220
+ filename_prefix: Optional[str] = None,
221
+ ) -> tuple:
222
+ """
223
+ Save the vocabulary to a JSON file.
224
+
225
+ Args:
226
+ save_directory: Directory to save the vocabulary.
227
+ filename_prefix: Optional prefix for the filename.
228
+
229
+ Returns:
230
+ Tuple containing the path to the saved vocabulary file.
231
+ """
232
+ if not os.path.isdir(save_directory):
233
+ os.makedirs(save_directory, exist_ok=True)
234
+
235
+ vocab_file = os.path.join(
236
+ save_directory,
237
+ (filename_prefix + "-" if filename_prefix else "") + "vocab.json",
238
+ )
239
+
240
+ with open(vocab_file, "w", encoding="utf-8") as f:
241
+ json.dump(self._vocab, f, ensure_ascii=False, indent=2)
242
+
243
+ return (vocab_file,)
244
+
245
+
246
+ def count_vocab_from_dataset(
247
+ dataset_name: str = "dlouapre/lichess_2025-01_1M",
248
+ split: str = "train",
249
+ column: str = "text",
250
+ max_samples: Optional[int] = 10000,
251
+ ) -> Dict[str, int]:
252
+ """
253
+ Count token frequencies in a dataset (useful for vocabulary analysis).
254
+
255
+ Args:
256
+ dataset_name: Name of the dataset on Hugging Face Hub.
257
+ split: Dataset split to use.
258
+ column: Column containing the game strings.
259
+ max_samples: Maximum number of samples to process.
260
+
261
+ Returns:
262
+ Dictionary mapping tokens to their frequencies.
263
+ """
264
+ from collections import Counter
265
+ from datasets import load_dataset
266
+
267
+ dataset = load_dataset(dataset_name, split=split)
268
+
269
+ if max_samples is not None:
270
+ dataset = dataset.select(range(min(max_samples, len(dataset))))
271
+
272
+ token_counts = Counter()
273
+
274
+ for example in dataset:
275
+ moves = example[column].strip().split()
276
+ token_counts.update(moves)
277
+
278
+ return dict(token_counts)
279
+
280
+
281
+ # ============================================================================
282
+ # V1 IMPROVEMENTS: Tokenizer with structured vocabulary and piece-type grouping
283
+ # ============================================================================
284
+
285
+ class ChessTokenizer(PreTrainedTokenizer):
286
+ """
287
+ Improved chess tokenizer with structured vocabulary.
288
+
289
+ Improvements over baseline:
290
+ - Vocabulary organized by piece type for better embeddings
291
+ - Lower minimum frequency threshold (100 vs 500) to capture more moves
292
+ - Optimized for the 1M parameter constraint
293
+
294
+ The vocabulary is ordered as:
295
+ 1. Special tokens ([PAD], [BOS], [EOS], [UNK])
296
+ 2. Pawn moves (most frequent)
297
+ 3. Knight moves
298
+ 4. Bishop moves
299
+ 5. Rook moves
300
+ 6. Queen moves
301
+ 7. King moves (including castling)
302
+
303
+ This organization helps the model learn piece-specific patterns.
304
+ """
305
+
306
+ model_input_names = ["input_ids", "attention_mask"]
307
+ vocab_files_names = {"vocab_file": "vocab.json"}
308
+
309
+ # Special tokens
310
+ PAD_TOKEN = "[PAD]"
311
+ BOS_TOKEN = "[BOS]"
312
+ EOS_TOKEN = "[EOS]"
313
+ UNK_TOKEN = "[UNK]"
314
+
315
+ # Piece ordering for structured vocabulary
316
+ PIECE_ORDER = ['P', 'N', 'B', 'R', 'Q', 'K']
317
+
318
+ def __init__(
319
+ self,
320
+ vocab_file: Optional[str] = None,
321
+ vocab: Optional[Dict[str, int]] = None,
322
+ **kwargs,
323
+ ):
324
+ self._pad_token = self.PAD_TOKEN
325
+ self._bos_token = self.BOS_TOKEN
326
+ self._eos_token = self.EOS_TOKEN
327
+ self._unk_token = self.UNK_TOKEN
328
+
329
+ kwargs.pop("pad_token", None)
330
+ kwargs.pop("bos_token", None)
331
+ kwargs.pop("eos_token", None)
332
+ kwargs.pop("unk_token", None)
333
+
334
+ if vocab is not None:
335
+ self._vocab = vocab
336
+ elif vocab_file is not None and os.path.exists(vocab_file):
337
+ with open(vocab_file, "r", encoding="utf-8") as f:
338
+ self._vocab = json.load(f)
339
+ else:
340
+ self._vocab = self._create_default_vocab()
341
+
342
+ self._ids_to_tokens = {v: k for k, v in self._vocab.items()}
343
+
344
+ super().__init__(
345
+ pad_token=self._pad_token,
346
+ bos_token=self._bos_token,
347
+ eos_token=self._eos_token,
348
+ unk_token=self._unk_token,
349
+ **kwargs,
350
+ )
351
+
352
+ def _create_default_vocab(self) -> Dict[str, int]:
353
+ special_tokens = [self.PAD_TOKEN, self.BOS_TOKEN, self.EOS_TOKEN, self.UNK_TOKEN]
354
+ return {token: idx for idx, token in enumerate(special_tokens)}
355
+
356
+ @classmethod
357
+ def _get_piece_type(cls, move: str) -> str:
358
+ """Extract piece type from a move string."""
359
+ # Move format: [W|B][P|N|B|R|Q|K]...
360
+ if len(move) >= 2 and move[0] in 'WB':
361
+ return move[1] if move[1] in cls.PIECE_ORDER else 'P'
362
+ return 'P' # Default to pawn
363
+
364
+ @classmethod
365
+ def build_vocab_from_iterator(
366
+ cls,
367
+ iterator,
368
+ min_frequency: int = 1,
369
+ ) -> "ChessTokenizer":
370
+ from collections import Counter
371
+
372
+ token_counts = Counter()
373
+
374
+ for game in iterator:
375
+ moves = game.strip().split()
376
+ token_counts.update(moves)
377
+
378
+ # Filter by frequency
379
+ tokens = [
380
+ token for token, count in token_counts.items()
381
+ if count >= min_frequency
382
+ ]
383
+
384
+ # Group tokens by piece type for structured vocabulary
385
+ piece_groups = {piece: [] for piece in cls.PIECE_ORDER}
386
+ other_tokens = []
387
+
388
+ for token in tokens:
389
+ piece = cls._get_piece_type(token)
390
+ if piece in piece_groups:
391
+ piece_groups[piece].append(token)
392
+ else:
393
+ other_tokens.append(token)
394
+
395
+ # Sort each group and concatenate in piece order
396
+ ordered_tokens = []
397
+ for piece in cls.PIECE_ORDER:
398
+ ordered_tokens.extend(sorted(piece_groups[piece]))
399
+ ordered_tokens.extend(sorted(other_tokens))
400
+
401
+ # Build vocabulary with special tokens first
402
+ special_tokens = [cls.PAD_TOKEN, cls.BOS_TOKEN, cls.EOS_TOKEN, cls.UNK_TOKEN]
403
+ vocab = {token: idx for idx, token in enumerate(special_tokens + ordered_tokens)}
404
+
405
+ return cls(vocab=vocab)
406
+
407
+ @classmethod
408
+ def build_vocab_from_dataset(
409
+ cls,
410
+ dataset_name: str = "dlouapre/lichess_2025-01_1M",
411
+ split: str = "train",
412
+ column: str = "text",
413
+ min_frequency: int = 1000, # Same as baseline to control vocab size
414
+ max_samples: Optional[int] = 100000,
415
+ ) -> "ChessTokenizer":
416
+ """
417
+ Build vocabulary from dataset with piece-aware organization.
418
+
419
+ Default min_frequency=1000 keeps vocab around ~1500 tokens..
420
+ """
421
+ from datasets import load_dataset
422
+
423
+ dataset = load_dataset(dataset_name, split=split)
424
+
425
+ if max_samples is not None:
426
+ dataset = dataset.select(range(min(max_samples, len(dataset))))
427
+
428
+ def game_iterator():
429
+ for example in dataset:
430
+ yield example[column]
431
+ return cls.build_vocab_from_iterator(game_iterator(), min_frequency=min_frequency)
432
+
433
+ @property
434
+ def vocab_size(self) -> int:
435
+ return len(self._vocab)
436
+
437
+ def get_vocab(self) -> Dict[str, int]:
438
+ return dict(self._vocab)
439
+
440
+ def _tokenize(self, text: str) -> List[str]:
441
+ return text.strip().split()
442
+
443
+ def _convert_token_to_id(self, token: str) -> int:
444
+ return self._vocab.get(token, self._vocab.get(self.UNK_TOKEN, 0))
445
+
446
+ def _convert_id_to_token(self, index: int) -> str:
447
+ return self._ids_to_tokens.get(index, self.UNK_TOKEN)
448
+
449
+ def convert_tokens_to_string(self, tokens: List[str]) -> str:
450
+ special = {self.PAD_TOKEN, self.BOS_TOKEN, self.EOS_TOKEN, self.UNK_TOKEN}
451
+ return " ".join(t for t in tokens if t not in special)
452
+
453
+ def save_vocabulary(
454
+ self,
455
+ save_directory: str,
456
+ filename_prefix: Optional[str] = None,
457
+ ) -> tuple:
458
+ if not os.path.isdir(save_directory):
459
+ os.makedirs(save_directory, exist_ok=True)
460
+
461
+ vocab_file = os.path.join(
462
+ save_directory,
463
+ (filename_prefix + "-" if filename_prefix else "") + "vocab.json",
464
+ )
465
+
466
+ with open(vocab_file, "w", encoding="utf-8") as f:
467
+ json.dump(self._vocab, f, ensure_ascii=False, indent=2)
468
+
469
+ return (vocab_file,)
470
+
471
+ def get_vocab_stats(self) -> Dict[str, int]:
472
+ """Get statistics about vocabulary composition by piece type."""
473
+ stats = {piece: 0 for piece in self.PIECE_ORDER}
474
+ stats['special'] = 4 # PAD, BOS, EOS, UNK
475
+ stats['other'] = 0
476
+
477
+ for token in self._vocab:
478
+ if token.startswith('['):
479
+ continue
480
+ piece = self._get_piece_type(token)
481
+ if piece in stats:
482
+ stats[piece] += 1
483
+ else:
484
+ stats['other'] += 1
485
+
486
+ return stats
tokenizer_config.json ADDED
@@ -0,0 +1,50 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ {
2
+ "added_tokens_decoder": {
3
+ "0": {
4
+ "content": "[PAD]",
5
+ "lstrip": false,
6
+ "normalized": false,
7
+ "rstrip": false,
8
+ "single_word": false,
9
+ "special": true
10
+ },
11
+ "1": {
12
+ "content": "[BOS]",
13
+ "lstrip": false,
14
+ "normalized": false,
15
+ "rstrip": false,
16
+ "single_word": false,
17
+ "special": true
18
+ },
19
+ "2": {
20
+ "content": "[EOS]",
21
+ "lstrip": false,
22
+ "normalized": false,
23
+ "rstrip": false,
24
+ "single_word": false,
25
+ "special": true
26
+ },
27
+ "3": {
28
+ "content": "[UNK]",
29
+ "lstrip": false,
30
+ "normalized": false,
31
+ "rstrip": false,
32
+ "single_word": false,
33
+ "special": true
34
+ }
35
+ },
36
+ "auto_map": {
37
+ "AutoTokenizer": [
38
+ "tokenizer.ChessTokenizer",
39
+ null
40
+ ]
41
+ },
42
+ "bos_token": "[BOS]",
43
+ "clean_up_tokenization_spaces": false,
44
+ "eos_token": "[EOS]",
45
+ "extra_special_tokens": {},
46
+ "model_max_length": 1000000000000000019884624838656,
47
+ "pad_token": "[PAD]",
48
+ "tokenizer_class": "ChessTokenizer",
49
+ "unk_token": "[UNK]"
50
+ }
vocab.json ADDED
@@ -0,0 +1,901 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ {
2
+ "[PAD]": 0,
3
+ "[BOS]": 1,
4
+ "[EOS]": 2,
5
+ "[UNK]": 3,
6
+ "BPa2a1(Q)": 4,
7
+ "BPa3a2": 5,
8
+ "BPa4a3": 6,
9
+ "BPa4b3(x)": 7,
10
+ "BPa5a4": 8,
11
+ "BPa5b4(x)": 9,
12
+ "BPa6a5": 10,
13
+ "BPa6b5(x)": 11,
14
+ "BPa7a5": 12,
15
+ "BPa7a6": 13,
16
+ "BPa7b6(x)": 14,
17
+ "BPb3b2": 15,
18
+ "BPb4b3": 16,
19
+ "BPb4c3(x)": 17,
20
+ "BPb5a4(x)": 18,
21
+ "BPb5b4": 19,
22
+ "BPb5c4(x)": 20,
23
+ "BPb6b5": 21,
24
+ "BPb6c5(x)": 22,
25
+ "BPb7b5": 23,
26
+ "BPb7b6": 24,
27
+ "BPb7c6(x)": 25,
28
+ "BPc3c2": 26,
29
+ "BPc4b3(x)": 27,
30
+ "BPc4c3": 28,
31
+ "BPc5b4(x)": 29,
32
+ "BPc5c4": 30,
33
+ "BPc5d4(x)": 31,
34
+ "BPc6b5(x)": 32,
35
+ "BPc6c5": 33,
36
+ "BPc6d5(x)": 34,
37
+ "BPc7c5": 35,
38
+ "BPc7c6": 36,
39
+ "BPc7d6(x)": 37,
40
+ "BPd3d2": 38,
41
+ "BPd4c3(x)": 39,
42
+ "BPd4d3": 40,
43
+ "BPd4e3(x)": 41,
44
+ "BPd5c4(x)": 42,
45
+ "BPd5d4": 43,
46
+ "BPd5e4(x)": 44,
47
+ "BPd6c5(x)": 45,
48
+ "BPd6d5": 46,
49
+ "BPd6e5(x)": 47,
50
+ "BPd7c6(x)": 48,
51
+ "BPd7d5": 49,
52
+ "BPd7d6": 50,
53
+ "BPe3e2": 51,
54
+ "BPe4d3(x)": 52,
55
+ "BPe4e3": 53,
56
+ "BPe4f3(x)": 54,
57
+ "BPe5d4(x)": 55,
58
+ "BPe5e4": 56,
59
+ "BPe5f4(x)": 57,
60
+ "BPe6d5(x)": 58,
61
+ "BPe6e5": 59,
62
+ "BPe6f5(x)": 60,
63
+ "BPe7e5": 61,
64
+ "BPe7e6": 62,
65
+ "BPe7f6(x)": 63,
66
+ "BPf3f2": 64,
67
+ "BPf4f3": 65,
68
+ "BPf5e4(x)": 66,
69
+ "BPf5f4": 67,
70
+ "BPf5g4(x)": 68,
71
+ "BPf6e5(x)": 69,
72
+ "BPf6f5": 70,
73
+ "BPf6g5(x)": 71,
74
+ "BPf7e6(x)": 72,
75
+ "BPf7f5": 73,
76
+ "BPf7f6": 74,
77
+ "BPf7g6(x)": 75,
78
+ "BPg3g2": 76,
79
+ "BPg4g3": 77,
80
+ "BPg5f4(x)": 78,
81
+ "BPg5g4": 79,
82
+ "BPg5h4(x)": 80,
83
+ "BPg6f5(x)": 81,
84
+ "BPg6g5": 82,
85
+ "BPg6h5(x)": 83,
86
+ "BPg7f6(x)": 84,
87
+ "BPg7g5": 85,
88
+ "BPg7g6": 86,
89
+ "BPg7h6(x)": 87,
90
+ "BPh3h2": 88,
91
+ "BPh4g3(x)": 89,
92
+ "BPh4h3": 90,
93
+ "BPh5g4(x)": 91,
94
+ "BPh5h4": 92,
95
+ "BPh6g5(x)": 93,
96
+ "BPh6h5": 94,
97
+ "BPh7g6(x)": 95,
98
+ "BPh7h5": 96,
99
+ "BPh7h6": 97,
100
+ "WPa2a3": 98,
101
+ "WPa2a4": 99,
102
+ "WPa2b3(x)": 100,
103
+ "WPa3a4": 101,
104
+ "WPa3b4(x)": 102,
105
+ "WPa4a5": 103,
106
+ "WPa4b5(x)": 104,
107
+ "WPa5a6": 105,
108
+ "WPa5b6(x)": 106,
109
+ "WPa6a7": 107,
110
+ "WPa7a8(Q)": 108,
111
+ "WPb2b3": 109,
112
+ "WPb2b4": 110,
113
+ "WPb2c3(x)": 111,
114
+ "WPb3a4(x)": 112,
115
+ "WPb3b4": 113,
116
+ "WPb3c4(x)": 114,
117
+ "WPb4a5(x)": 115,
118
+ "WPb4b5": 116,
119
+ "WPb4c5(x)": 117,
120
+ "WPb5b6": 118,
121
+ "WPb5c6(x)": 119,
122
+ "WPb6b7": 120,
123
+ "WPc2b3(x)": 121,
124
+ "WPc2c3": 122,
125
+ "WPc2c4": 123,
126
+ "WPc2d3(x)": 124,
127
+ "WPc3b4(x)": 125,
128
+ "WPc3c4": 126,
129
+ "WPc3d4(x)": 127,
130
+ "WPc4b5(x)": 128,
131
+ "WPc4c5": 129,
132
+ "WPc4d5(x)": 130,
133
+ "WPc5c6": 131,
134
+ "WPc6c7": 132,
135
+ "WPd2d3": 133,
136
+ "WPd2d4": 134,
137
+ "WPd3c4(x)": 135,
138
+ "WPd3d4": 136,
139
+ "WPd3e4(x)": 137,
140
+ "WPd4c5(x)": 138,
141
+ "WPd4d5": 139,
142
+ "WPd4e5(x)": 140,
143
+ "WPd5c6(x)": 141,
144
+ "WPd5d6": 142,
145
+ "WPd5e6(x)": 143,
146
+ "WPd6d7": 144,
147
+ "WPe2e3": 145,
148
+ "WPe2e4": 146,
149
+ "WPe3d4(x)": 147,
150
+ "WPe3e4": 148,
151
+ "WPe3f4(x)": 149,
152
+ "WPe4d5(x)": 150,
153
+ "WPe4e5": 151,
154
+ "WPe4f5(x)": 152,
155
+ "WPe5d6(x)": 153,
156
+ "WPe5e6": 154,
157
+ "WPe5f6(x)": 155,
158
+ "WPe6e7": 156,
159
+ "WPf2e3(x)": 157,
160
+ "WPf2f3": 158,
161
+ "WPf2f4": 159,
162
+ "WPf2g3(x)": 160,
163
+ "WPf3e4(x)": 161,
164
+ "WPf3f4": 162,
165
+ "WPf3g4(x)": 163,
166
+ "WPf4e5(x)": 164,
167
+ "WPf4f5": 165,
168
+ "WPf4g5(x)": 166,
169
+ "WPf5e6(x)": 167,
170
+ "WPf5f6": 168,
171
+ "WPf5g6(x)": 169,
172
+ "WPf6f7": 170,
173
+ "WPg2f3(x)": 171,
174
+ "WPg2g3": 172,
175
+ "WPg2g4": 173,
176
+ "WPg2h3(x)": 174,
177
+ "WPg3f4(x)": 175,
178
+ "WPg3g4": 176,
179
+ "WPg3h4(x)": 177,
180
+ "WPg4f5(x)": 178,
181
+ "WPg4g5": 179,
182
+ "WPg4h5(x)": 180,
183
+ "WPg5g6": 181,
184
+ "WPg6g7": 182,
185
+ "WPh2g3(x)": 183,
186
+ "WPh2h3": 184,
187
+ "WPh2h4": 185,
188
+ "WPh3g4(x)": 186,
189
+ "WPh3h4": 187,
190
+ "WPh4g5(x)": 188,
191
+ "WPh4h5": 189,
192
+ "WPh5g6(x)": 190,
193
+ "WPh5h6": 191,
194
+ "WPh6h7": 192,
195
+ "BNa5c4": 193,
196
+ "BNa5c6": 194,
197
+ "BNa6c5": 195,
198
+ "BNb4c6": 196,
199
+ "BNb6c4": 197,
200
+ "BNb6d5": 198,
201
+ "BNb6d7": 199,
202
+ "BNb8a6": 200,
203
+ "BNb8c6": 201,
204
+ "BNb8d7": 202,
205
+ "BNc5e4": 203,
206
+ "BNc6a5": 204,
207
+ "BNc6b4": 205,
208
+ "BNc6b4(x)": 206,
209
+ "BNc6b8": 207,
210
+ "BNc6d4": 208,
211
+ "BNc6d4(x)": 209,
212
+ "BNc6e5": 210,
213
+ "BNc6e5(x)": 211,
214
+ "BNc6e7": 212,
215
+ "BNd4f3(x+)": 213,
216
+ "BNd5b4": 214,
217
+ "BNd5b6": 215,
218
+ "BNd5c3(x)": 216,
219
+ "BNd5e3(x)": 217,
220
+ "BNd5f4": 218,
221
+ "BNd5f6": 219,
222
+ "BNd7b6": 220,
223
+ "BNd7c5": 221,
224
+ "BNd7c5(x)": 222,
225
+ "BNd7e5": 223,
226
+ "BNd7e5(x)": 224,
227
+ "BNd7f6": 225,
228
+ "BNd7f6(x)": 226,
229
+ "BNd7f8": 227,
230
+ "BNe4c3(x)": 228,
231
+ "BNe4d2(x)": 229,
232
+ "BNe4d6": 230,
233
+ "BNe4f6": 231,
234
+ "BNe5c4": 232,
235
+ "BNe5c4(x)": 233,
236
+ "BNe5c6": 234,
237
+ "BNe5f3(x+)": 235,
238
+ "BNe5g4": 236,
239
+ "BNe5g6": 237,
240
+ "BNe7c6": 238,
241
+ "BNe7d5": 239,
242
+ "BNe7d5(x)": 240,
243
+ "BNe7f5": 241,
244
+ "BNe7f5(x)": 242,
245
+ "BNe7g6": 243,
246
+ "BNf6d5": 244,
247
+ "BNf6d5(x)": 245,
248
+ "BNf6d7": 246,
249
+ "BNf6e4": 247,
250
+ "BNf6e4(x)": 248,
251
+ "BNf6e8": 249,
252
+ "BNf6g4": 250,
253
+ "BNf6g4(x)": 251,
254
+ "BNf6g8": 252,
255
+ "BNf6h5": 253,
256
+ "BNf6h7": 254,
257
+ "BNg4e3(x)": 255,
258
+ "BNg4e5": 256,
259
+ "BNg4f6": 257,
260
+ "BNg6e5": 258,
261
+ "BNg6f4": 259,
262
+ "BNg8e7": 260,
263
+ "BNg8f6": 261,
264
+ "BNg8f6(x)": 262,
265
+ "BNg8h6": 263,
266
+ "BNh5f4": 264,
267
+ "BNh5f6": 265,
268
+ "BNh6f5": 266,
269
+ "WNa3c4": 267,
270
+ "WNa4c5": 268,
271
+ "WNb1a3": 269,
272
+ "WNb1c3": 270,
273
+ "WNb1c3(x)": 271,
274
+ "WNb1d2": 272,
275
+ "WNb3c5": 273,
276
+ "WNb3d4": 274,
277
+ "WNb5c3": 275,
278
+ "WNc3a4": 276,
279
+ "WNc3b5": 277,
280
+ "WNc3b5(x)": 278,
281
+ "WNc3d5": 279,
282
+ "WNc3d5(x)": 280,
283
+ "WNc3e2": 281,
284
+ "WNc3e4": 282,
285
+ "WNc3e4(x)": 283,
286
+ "WNc4e3": 284,
287
+ "WNc4e5": 285,
288
+ "WNc7a8(x)": 286,
289
+ "WNd2b3": 287,
290
+ "WNd2c4": 288,
291
+ "WNd2c4(x)": 289,
292
+ "WNd2e4": 290,
293
+ "WNd2e4(x)": 291,
294
+ "WNd2f1": 292,
295
+ "WNd2f3": 293,
296
+ "WNd2f3(x)": 294,
297
+ "WNd4b3": 295,
298
+ "WNd4b5": 296,
299
+ "WNd4c6(x)": 297,
300
+ "WNd4e6(x)": 298,
301
+ "WNd4f3": 299,
302
+ "WNd4f5": 300,
303
+ "WNd5f6(x+)": 301,
304
+ "WNe2c3": 302,
305
+ "WNe2d4": 303,
306
+ "WNe2d4(x)": 304,
307
+ "WNe2f4": 305,
308
+ "WNe2g3": 306,
309
+ "WNe4c3": 307,
310
+ "WNe4c5": 308,
311
+ "WNe4d6": 309,
312
+ "WNe4f6(+)": 310,
313
+ "WNe4f6(x+)": 311,
314
+ "WNe4g3": 312,
315
+ "WNe4g5": 313,
316
+ "WNe5c6(x)": 314,
317
+ "WNe5d3": 315,
318
+ "WNe5d7(x)": 316,
319
+ "WNe5f3": 317,
320
+ "WNe5f7(x)": 318,
321
+ "WNe5g4": 319,
322
+ "WNe5g6(x)": 320,
323
+ "WNf1g3": 321,
324
+ "WNf3d2": 322,
325
+ "WNf3d4": 323,
326
+ "WNf3d4(x)": 324,
327
+ "WNf3e1": 325,
328
+ "WNf3e5": 326,
329
+ "WNf3e5(x)": 327,
330
+ "WNf3g5": 328,
331
+ "WNf3g5(x)": 329,
332
+ "WNf3h2": 330,
333
+ "WNf3h4": 331,
334
+ "WNg1e2": 332,
335
+ "WNg1f3": 333,
336
+ "WNg1h3": 334,
337
+ "WNg3e4": 335,
338
+ "WNg3f5": 336,
339
+ "WNg5e4": 337,
340
+ "WNg5e6(x)": 338,
341
+ "WNg5f3": 339,
342
+ "WNg5f7(x)": 340,
343
+ "WNh2g4": 341,
344
+ "WNh4f3": 342,
345
+ "WNh4f5": 343,
346
+ "BBa5b6": 344,
347
+ "BBb4a5": 345,
348
+ "BBb4c3(x)": 346,
349
+ "BBb4c3(x+)": 347,
350
+ "BBb4c5": 348,
351
+ "BBb4d2(x+)": 349,
352
+ "BBb4d6": 350,
353
+ "BBb4e7": 351,
354
+ "BBb7a6": 352,
355
+ "BBb7c6": 353,
356
+ "BBb7c8": 354,
357
+ "BBb7d5(x)": 355,
358
+ "BBb7e4(x)": 356,
359
+ "BBb7f3(x)": 357,
360
+ "BBc5b6": 358,
361
+ "BBc5d4": 359,
362
+ "BBc5d4(x)": 360,
363
+ "BBc5d6": 361,
364
+ "BBc5e3(x)": 362,
365
+ "BBc5e7": 363,
366
+ "BBc5f2(x+)": 364,
367
+ "BBc8a6": 365,
368
+ "BBc8b7": 366,
369
+ "BBc8d7": 367,
370
+ "BBc8e6": 368,
371
+ "BBc8e6(x)": 369,
372
+ "BBc8f5": 370,
373
+ "BBc8f5(x)": 371,
374
+ "BBc8g4": 372,
375
+ "BBc8g4(x)": 373,
376
+ "BBc8h3(x)": 374,
377
+ "BBd6c5": 375,
378
+ "BBd6c7": 376,
379
+ "BBd6e5": 377,
380
+ "BBd6e5(x)": 378,
381
+ "BBd6e7": 379,
382
+ "BBd6f4(x)": 380,
383
+ "BBd6g3(x)": 381,
384
+ "BBd7b5": 382,
385
+ "BBd7c6": 383,
386
+ "BBd7c6(x)": 384,
387
+ "BBd7e6": 385,
388
+ "BBd7e8": 386,
389
+ "BBe6c4(x)": 387,
390
+ "BBe6d5": 388,
391
+ "BBe6d5(x)": 389,
392
+ "BBe6d7": 390,
393
+ "BBe6f5": 391,
394
+ "BBe6g4": 392,
395
+ "BBe7b4": 393,
396
+ "BBe7c5": 394,
397
+ "BBe7c5(x)": 395,
398
+ "BBe7d6": 396,
399
+ "BBe7f6": 397,
400
+ "BBe7f6(x)": 398,
401
+ "BBe7f8": 399,
402
+ "BBe7g5": 400,
403
+ "BBe7g5(x)": 401,
404
+ "BBf5d3(x)": 402,
405
+ "BBf5e4": 403,
406
+ "BBf5e4(x)": 404,
407
+ "BBf5e6": 405,
408
+ "BBf5g4": 406,
409
+ "BBf5g6": 407,
410
+ "BBf6e5(x)": 408,
411
+ "BBf6e7": 409,
412
+ "BBf6g7": 410,
413
+ "BBf8b4": 411,
414
+ "BBf8b4(+)": 412,
415
+ "BBf8c5": 413,
416
+ "BBf8c5(x)": 414,
417
+ "BBf8d6": 415,
418
+ "BBf8d6(x)": 416,
419
+ "BBf8e7": 417,
420
+ "BBf8g7": 418,
421
+ "BBf8h6": 419,
422
+ "BBg4d7": 420,
423
+ "BBg4e2(x)": 421,
424
+ "BBg4e6": 422,
425
+ "BBg4f3(x)": 423,
426
+ "BBg4f5": 424,
427
+ "BBg4h5": 425,
428
+ "BBg7b2(x)": 426,
429
+ "BBg7d4(x)": 427,
430
+ "BBg7e5(x)": 428,
431
+ "BBg7f6": 429,
432
+ "BBg7f6(x)": 430,
433
+ "BBg7f8": 431,
434
+ "BBg7h6": 432,
435
+ "BBh5g6": 433,
436
+ "WBa4b3": 434,
437
+ "WBb2c1": 435,
438
+ "WBb2d4(x)": 436,
439
+ "WBb2e5(x)": 437,
440
+ "WBb2f6(x)": 438,
441
+ "WBb3c2": 439,
442
+ "WBb5a4": 440,
443
+ "WBb5c4": 441,
444
+ "WBb5c6(x)": 442,
445
+ "WBb5c6(x+)": 443,
446
+ "WBb5d3": 444,
447
+ "WBb5d7(x+)": 445,
448
+ "WBc1a3": 446,
449
+ "WBc1b2": 447,
450
+ "WBc1d2": 448,
451
+ "WBc1e3": 449,
452
+ "WBc1f4": 450,
453
+ "WBc1f4(x)": 451,
454
+ "WBc1g5": 452,
455
+ "WBc1g5(x)": 453,
456
+ "WBc1h6": 454,
457
+ "WBc1h6(x)": 455,
458
+ "WBc4a2": 456,
459
+ "WBc4b3": 457,
460
+ "WBc4b5": 458,
461
+ "WBc4b5(+)": 459,
462
+ "WBc4d3": 460,
463
+ "WBc4d5": 461,
464
+ "WBc4d5(x)": 462,
465
+ "WBc4e2": 463,
466
+ "WBc4e6(x)": 464,
467
+ "WBc4f7(x+)": 465,
468
+ "WBd2c3": 466,
469
+ "WBd2c3(x)": 467,
470
+ "WBd2e3": 468,
471
+ "WBd3b5": 469,
472
+ "WBd3c2": 470,
473
+ "WBd3c4": 471,
474
+ "WBd3c4(x)": 472,
475
+ "WBd3e2": 473,
476
+ "WBd3e4": 474,
477
+ "WBd3e4(x)": 475,
478
+ "WBd3f5(x)": 476,
479
+ "WBd3g6(x)": 477,
480
+ "WBe2c4": 478,
481
+ "WBe2c4(x)": 479,
482
+ "WBe2d3": 480,
483
+ "WBe2f3": 481,
484
+ "WBe2f3(x)": 482,
485
+ "WBe2g4": 483,
486
+ "WBe2g4(x)": 484,
487
+ "WBe3c5(x)": 485,
488
+ "WBe3d2": 486,
489
+ "WBe3d4": 487,
490
+ "WBe3d4(x)": 488,
491
+ "WBe3f2": 489,
492
+ "WBe3f4": 490,
493
+ "WBe3g5": 491,
494
+ "WBe3h6": 492,
495
+ "WBf1b5": 493,
496
+ "WBf1b5(+)": 494,
497
+ "WBf1c4": 495,
498
+ "WBf1c4(x)": 496,
499
+ "WBf1d3": 497,
500
+ "WBf1e2": 498,
501
+ "WBf1g2": 499,
502
+ "WBf3e2": 500,
503
+ "WBf4d6(x)": 501,
504
+ "WBf4e3": 502,
505
+ "WBf4e5": 503,
506
+ "WBf4e5(x)": 504,
507
+ "WBf4g3": 505,
508
+ "WBf4g5": 506,
509
+ "WBg2e4(x)": 507,
510
+ "WBg2f1": 508,
511
+ "WBg2f3(x)": 509,
512
+ "WBg2h3": 510,
513
+ "WBg5e3": 511,
514
+ "WBg5e7(x)": 512,
515
+ "WBg5f4": 513,
516
+ "WBg5f6(x)": 514,
517
+ "WBg5h4": 515,
518
+ "WBh4g3": 516,
519
+ "WBh6g7(x)": 517,
520
+ "BRa8a7": 518,
521
+ "BRa8b8": 519,
522
+ "BRa8c8": 520,
523
+ "BRa8d8": 521,
524
+ "BRa8d8(x)": 522,
525
+ "BRa8e8": 523,
526
+ "BRa8f8": 524,
527
+ "BRa8g8": 525,
528
+ "BRb8a8": 526,
529
+ "BRb8b2(x)": 527,
530
+ "BRb8c8": 528,
531
+ "BRb8d8": 529,
532
+ "BRc8b8": 530,
533
+ "BRc8c2": 531,
534
+ "BRc8c3(x)": 532,
535
+ "BRc8c7": 533,
536
+ "BRc8d8": 534,
537
+ "BRc8e8": 535,
538
+ "BRd8b8": 536,
539
+ "BRd8c8": 537,
540
+ "BRd8d1(x)": 538,
541
+ "BRd8d1(x+)": 539,
542
+ "BRd8d2": 540,
543
+ "BRd8d3(x)": 541,
544
+ "BRd8d4(x)": 542,
545
+ "BRd8d5": 543,
546
+ "BRd8d5(x)": 544,
547
+ "BRd8d6": 545,
548
+ "BRd8d6(x)": 546,
549
+ "BRd8d7": 547,
550
+ "BRd8d7(x)": 548,
551
+ "BRd8e8": 549,
552
+ "BRd8f8": 550,
553
+ "BRd8g8": 551,
554
+ "BRe8c8": 552,
555
+ "BRe8d8": 553,
556
+ "BRe8e1(x+)": 554,
557
+ "BRe8e2": 555,
558
+ "BRe8e4(x)": 556,
559
+ "BRe8e5(x)": 557,
560
+ "BRe8e6": 558,
561
+ "BRe8e6(x)": 559,
562
+ "BRe8e7": 560,
563
+ "BRe8e7(x)": 561,
564
+ "BRe8f8": 562,
565
+ "BRf8a8": 563,
566
+ "BRf8b8": 564,
567
+ "BRf8c8": 565,
568
+ "BRf8d8": 566,
569
+ "BRf8d8(x)": 567,
570
+ "BRf8e8": 568,
571
+ "BRf8f1(x+)": 569,
572
+ "BRf8f5(x)": 570,
573
+ "BRf8f6": 571,
574
+ "BRf8f6(x)": 572,
575
+ "BRf8f7": 573,
576
+ "BRf8f7(x)": 574,
577
+ "BRf8g8": 575,
578
+ "BRf8h8": 576,
579
+ "BRg8g7": 577,
580
+ "BRh8d8": 578,
581
+ "BRh8e8": 579,
582
+ "BRh8f8": 580,
583
+ "BRh8g8": 581,
584
+ "WRa1a2": 582,
585
+ "WRa1b1": 583,
586
+ "WRa1c1": 584,
587
+ "WRa1d1": 585,
588
+ "WRa1d1(x)": 586,
589
+ "WRa1e1": 587,
590
+ "WRa1f1": 588,
591
+ "WRa1g1": 589,
592
+ "WRb1c1": 590,
593
+ "WRb1d1": 591,
594
+ "WRc1b1": 592,
595
+ "WRc1c2": 593,
596
+ "WRc1c7": 594,
597
+ "WRc1d1": 595,
598
+ "WRc1e1": 596,
599
+ "WRd1b1": 597,
600
+ "WRd1c1": 598,
601
+ "WRd1d2": 599,
602
+ "WRd1d2(x)": 600,
603
+ "WRd1d3": 601,
604
+ "WRd1d3(x)": 602,
605
+ "WRd1d4": 603,
606
+ "WRd1d4(x)": 604,
607
+ "WRd1d5(x)": 605,
608
+ "WRd1d6(x)": 606,
609
+ "WRd1d7": 607,
610
+ "WRd1d8(x)": 608,
611
+ "WRd1d8(x+)": 609,
612
+ "WRd1e1": 610,
613
+ "WRd1f1": 611,
614
+ "WRd1g1": 612,
615
+ "WRe1c1": 613,
616
+ "WRe1d1": 614,
617
+ "WRe1e2": 615,
618
+ "WRe1e2(x)": 616,
619
+ "WRe1e3": 617,
620
+ "WRe1e3(x)": 618,
621
+ "WRe1e4": 619,
622
+ "WRe1e4(x)": 620,
623
+ "WRe1e5(x)": 621,
624
+ "WRe1e6(x)": 622,
625
+ "WRe1e7": 623,
626
+ "WRe1e8(x+)": 624,
627
+ "WRe1f1": 625,
628
+ "WRf1b1": 626,
629
+ "WRf1c1": 627,
630
+ "WRf1d1": 628,
631
+ "WRf1d1(x)": 629,
632
+ "WRf1e1": 630,
633
+ "WRf1e1(+)": 631,
634
+ "WRf1f2": 632,
635
+ "WRf1f2(x)": 633,
636
+ "WRf1f3": 634,
637
+ "WRf1f3(x)": 635,
638
+ "WRf1f4(x)": 636,
639
+ "WRf1f5(x)": 637,
640
+ "WRf1f6(x)": 638,
641
+ "WRf1f8(x+)": 639,
642
+ "WRf1g1": 640,
643
+ "WRf1h1": 641,
644
+ "WRh1d1": 642,
645
+ "WRh1e1": 643,
646
+ "WRh1f1": 644,
647
+ "WRh1g1": 645,
648
+ "BQa5b6": 646,
649
+ "BQa5c7": 647,
650
+ "BQb6b2(x)": 648,
651
+ "BQb6c7": 649,
652
+ "BQc7b6": 650,
653
+ "BQc7d7": 651,
654
+ "BQd5a5": 652,
655
+ "BQd5d8": 653,
656
+ "BQd7e7": 654,
657
+ "BQd8a5": 655,
658
+ "BQd8a5(+)": 656,
659
+ "BQd8b6": 657,
660
+ "BQd8c7": 658,
661
+ "BQd8c8": 659,
662
+ "BQd8d1(x+)": 660,
663
+ "BQd8d4(x)": 661,
664
+ "BQd8d5": 662,
665
+ "BQd8d5(x)": 663,
666
+ "BQd8d6": 664,
667
+ "BQd8d6(x)": 665,
668
+ "BQd8d7": 666,
669
+ "BQd8d7(x)": 667,
670
+ "BQd8e7": 668,
671
+ "BQd8e7(x)": 669,
672
+ "BQd8e8": 670,
673
+ "BQd8f6": 671,
674
+ "BQd8f6(x)": 672,
675
+ "BQd8g5": 673,
676
+ "BQd8g5(x)": 674,
677
+ "BQd8h4": 675,
678
+ "BQd8h4(+)": 676,
679
+ "BQe7d7": 677,
680
+ "BQe7f6": 678,
681
+ "BQf6e7": 679,
682
+ "BQf6g6": 680,
683
+ "WQb3b7(x)": 681,
684
+ "WQb3c2": 682,
685
+ "WQc2d2": 683,
686
+ "WQd1a4": 684,
687
+ "WQd1a4(+)": 685,
688
+ "WQd1b3": 686,
689
+ "WQd1c1": 687,
690
+ "WQd1c2": 688,
691
+ "WQd1d2": 689,
692
+ "WQd1d2(x)": 690,
693
+ "WQd1d3": 691,
694
+ "WQd1d3(x)": 692,
695
+ "WQd1d4": 693,
696
+ "WQd1d4(x)": 694,
697
+ "WQd1d5(x)": 695,
698
+ "WQd1d8(x+)": 696,
699
+ "WQd1e1": 697,
700
+ "WQd1e2": 698,
701
+ "WQd1e2(x)": 699,
702
+ "WQd1f3": 700,
703
+ "WQd1f3(x)": 701,
704
+ "WQd1g4": 702,
705
+ "WQd1g4(x)": 703,
706
+ "WQd1h5": 704,
707
+ "WQd1h5(+)": 705,
708
+ "WQd2e2": 706,
709
+ "WQd2e3": 707,
710
+ "WQd4d1": 708,
711
+ "WQe2d2": 709,
712
+ "WQe2d3": 710,
713
+ "WQe2e3": 711,
714
+ "WQe2f2": 712,
715
+ "WQe2f3": 713,
716
+ "WQf3e2": 714,
717
+ "WQf3e3": 715,
718
+ "WQf3g3": 716,
719
+ "WQh5f3": 717,
720
+ "BKb8a8": 718,
721
+ "BKc7b6": 719,
722
+ "BKc8b7": 720,
723
+ "BKc8b8": 721,
724
+ "BKc8c7": 722,
725
+ "BKc8d7": 723,
726
+ "BKd6c5": 724,
727
+ "BKd6c6": 725,
728
+ "BKd7c6": 726,
729
+ "BKd7c7": 727,
730
+ "BKd7c8": 728,
731
+ "BKd7d6": 729,
732
+ "BKd7e6": 730,
733
+ "BKd7e7": 731,
734
+ "BKd7e8": 732,
735
+ "BKd8c7": 733,
736
+ "BKd8c8": 734,
737
+ "BKd8d7": 735,
738
+ "BKd8e7": 736,
739
+ "BKd8e8": 737,
740
+ "BKe6d5": 738,
741
+ "BKe6d6": 739,
742
+ "BKe6d7": 740,
743
+ "BKe6e5": 741,
744
+ "BKe6f5": 742,
745
+ "BKe6f6": 743,
746
+ "BKe6f7": 744,
747
+ "BKe7d6": 745,
748
+ "BKe7d7": 746,
749
+ "BKe7d8": 747,
750
+ "BKe7e6": 748,
751
+ "BKe7e8": 749,
752
+ "BKe7f6": 750,
753
+ "BKe7f7": 751,
754
+ "BKe7f8": 752,
755
+ "BKe8c8(O)": 753,
756
+ "BKe8d7": 754,
757
+ "BKe8d8": 755,
758
+ "BKe8d8(x)": 756,
759
+ "BKe8e7": 757,
760
+ "BKe8f7": 758,
761
+ "BKe8f7(x)": 759,
762
+ "BKe8f8": 760,
763
+ "BKe8g8(o)": 761,
764
+ "BKf6e5": 762,
765
+ "BKf6e6": 763,
766
+ "BKf6e7": 764,
767
+ "BKf6f5": 765,
768
+ "BKf6g5": 766,
769
+ "BKf6g6": 767,
770
+ "BKf6g7": 768,
771
+ "BKf7e6": 769,
772
+ "BKf7e7": 770,
773
+ "BKf7e8": 771,
774
+ "BKf7f6": 772,
775
+ "BKf7f8": 773,
776
+ "BKf7g6": 774,
777
+ "BKf7g7": 775,
778
+ "BKf7g8": 776,
779
+ "BKf8e7": 777,
780
+ "BKf8e8": 778,
781
+ "BKf8f7": 779,
782
+ "BKf8g7": 780,
783
+ "BKf8g8": 781,
784
+ "BKg6f5": 782,
785
+ "BKg6f6": 783,
786
+ "BKg6f7": 784,
787
+ "BKg6g5": 785,
788
+ "BKg6h5": 786,
789
+ "BKg7f6": 787,
790
+ "BKg7f7": 788,
791
+ "BKg7f8": 789,
792
+ "BKg7g6": 790,
793
+ "BKg7g8": 791,
794
+ "BKg7h6": 792,
795
+ "BKg7h7": 793,
796
+ "BKg8f7": 794,
797
+ "BKg8f7(x)": 795,
798
+ "BKg8f8": 796,
799
+ "BKg8f8(x)": 797,
800
+ "BKg8g7": 798,
801
+ "BKg8g7(x)": 799,
802
+ "BKg8h7": 800,
803
+ "BKg8h8": 801,
804
+ "BKh6g7": 802,
805
+ "BKh7g6": 803,
806
+ "BKh7g7": 804,
807
+ "BKh7g8": 805,
808
+ "BKh7h6": 806,
809
+ "BKh7h8": 807,
810
+ "BKh8g7": 808,
811
+ "BKh8g8": 809,
812
+ "BKh8h7": 810,
813
+ "WKb1a1": 811,
814
+ "WKc1b1": 812,
815
+ "WKc1b2": 813,
816
+ "WKc1c2": 814,
817
+ "WKc1d2": 815,
818
+ "WKc2b3": 816,
819
+ "WKd1c1": 817,
820
+ "WKd1c2": 818,
821
+ "WKd1e2": 819,
822
+ "WKd2c1": 820,
823
+ "WKd2c2": 821,
824
+ "WKd2c3": 822,
825
+ "WKd2d3": 823,
826
+ "WKd2e2": 824,
827
+ "WKd2e3": 825,
828
+ "WKd3c3": 826,
829
+ "WKd3c4": 827,
830
+ "WKd3e3": 828,
831
+ "WKe1c1(O)": 829,
832
+ "WKe1d1": 830,
833
+ "WKe1d1(x)": 831,
834
+ "WKe1d2": 832,
835
+ "WKe1e2": 833,
836
+ "WKe1f1": 834,
837
+ "WKe1f2": 835,
838
+ "WKe1g1(o)": 836,
839
+ "WKe2d1": 837,
840
+ "WKe2d2": 838,
841
+ "WKe2d3": 839,
842
+ "WKe2e3": 840,
843
+ "WKe2f1": 841,
844
+ "WKe2f2": 842,
845
+ "WKe2f3": 843,
846
+ "WKe3d2": 844,
847
+ "WKe3d3": 845,
848
+ "WKe3d4": 846,
849
+ "WKe3e4": 847,
850
+ "WKe3f3": 848,
851
+ "WKe3f4": 849,
852
+ "WKf1e1": 850,
853
+ "WKf1e2": 851,
854
+ "WKf1f2": 852,
855
+ "WKf1g1": 853,
856
+ "WKf1g2": 854,
857
+ "WKf2e1": 855,
858
+ "WKf2e2": 856,
859
+ "WKf2e3": 857,
860
+ "WKf2f3": 858,
861
+ "WKf2g1": 859,
862
+ "WKf2g2": 860,
863
+ "WKf2g3": 861,
864
+ "WKf3e2": 862,
865
+ "WKf3e3": 863,
866
+ "WKf3e4": 864,
867
+ "WKf3f4": 865,
868
+ "WKf3g2": 866,
869
+ "WKf3g3": 867,
870
+ "WKf3g4": 868,
871
+ "WKg1f1": 869,
872
+ "WKg1f1(x)": 870,
873
+ "WKg1f2": 871,
874
+ "WKg1f2(x)": 872,
875
+ "WKg1g2": 873,
876
+ "WKg1g2(x)": 874,
877
+ "WKg1h1": 875,
878
+ "WKg1h2": 876,
879
+ "WKg2f1": 877,
880
+ "WKg2f2": 878,
881
+ "WKg2f3": 879,
882
+ "WKg2g1": 880,
883
+ "WKg2g3": 881,
884
+ "WKg2h2": 882,
885
+ "WKg2h3": 883,
886
+ "WKg3f2": 884,
887
+ "WKg3f3": 885,
888
+ "WKg3f4": 886,
889
+ "WKg3g4": 887,
890
+ "WKg3h4": 888,
891
+ "WKh1g1": 889,
892
+ "WKh1g2": 890,
893
+ "WKh1h2": 891,
894
+ "WKh2g1": 892,
895
+ "WKh2g2": 893,
896
+ "WKh2g3": 894,
897
+ "WKh2h1": 895,
898
+ "WKh2h3": 896,
899
+ "WKh3g2": 897,
900
+ "WKh3g4": 898
901
+ }