gaspardbd commited on
Commit
916a5ba
·
verified ·
1 Parent(s): 1ecbd8d

Chess Challenge submission by gaspardbd

Browse files
Files changed (8) hide show
  1. README.md +26 -0
  2. config.json +24 -0
  3. model.py +438 -0
  4. model.safetensors +3 -0
  5. special_tokens_map.json +6 -0
  6. tokenizer.py +368 -0
  7. tokenizer_config.json +50 -0
  8. vocab.json +2042 -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_gasp
11
+
12
+ Chess model submitted to the LLM Course Chess Challenge.
13
+
14
+ ## Submission Info
15
+
16
+ - **Submitted by**: [gaspardbd](https://huggingface.co/gaspardbd)
17
+ - **Parameters**: 955,648
18
+ - **Organization**: LLM-course
19
+
20
+ ## Model Details
21
+
22
+ - **Architecture**: Chess Transformer (GPT-style)
23
+ - **Vocab size**: 2040
24
+ - **Embedding dim**: 128
25
+ - **Layers**: 4
26
+ - **Heads**: 4
config.json ADDED
@@ -0,0 +1,24 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ {
2
+ "architectures": [
3
+ "ChessForCausalLM"
4
+ ],
5
+ "auto_map": {
6
+ "AutoConfig": "model.ChessConfig",
7
+ "AutoModelForCausalLM": "model.ChessForCausalLM"
8
+ },
9
+ "bos_token_id": 1,
10
+ "dropout": 0.1,
11
+ "dtype": "float32",
12
+ "eos_token_id": 2,
13
+ "layer_norm_epsilon": 1e-05,
14
+ "model_type": "chess_transformer",
15
+ "n_ctx": 256,
16
+ "n_embd": 128,
17
+ "n_head": 4,
18
+ "n_inner": 384,
19
+ "n_layer": 4,
20
+ "pad_token_id": 0,
21
+ "tie_weights": true,
22
+ "transformers_version": "4.57.6",
23
+ "vocab_size": 2040
24
+ }
model.py ADDED
@@ -0,0 +1,438 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ """
2
+ Chess Transformer Model for the Chess Challenge.
3
+
4
+ This module provides a simple GPT-style transformer architecture
5
+ designed to fit within the 1M parameter constraint.
6
+
7
+ Key components:
8
+ - ChessConfig: Configuration class for model hyperparameters
9
+ - ChessForCausalLM: The main model class for next-move prediction
10
+ """
11
+
12
+ from __future__ import annotations
13
+
14
+ import math
15
+ from dataclasses import dataclass
16
+ from typing import Optional, Tuple, Union
17
+
18
+ import torch
19
+ import torch.nn as nn
20
+ import torch.nn.functional as F
21
+ from transformers import PretrainedConfig, PreTrainedModel
22
+ from transformers.modeling_outputs import CausalLMOutputWithPast
23
+
24
+
25
+ class ChessConfig(PretrainedConfig):
26
+ """
27
+ Configuration class for the Chess Transformer model.
28
+
29
+ This configuration is designed for a ~1M parameter model.
30
+ Students can adjust these values to explore different architectures.
31
+
32
+ Parameter budget breakdown (with default values):
33
+ - Embeddings (vocab): 1200 x 128 = 153,600
34
+ - Position Embeddings: 256 x 128 = 32,768
35
+ - Transformer Layers: 6 x ~120,000 = ~720,000
36
+ - LM Head (with weight tying): 0 (shared with embeddings)
37
+ - Total: ~906,000 parameters
38
+
39
+ Attributes:
40
+ vocab_size: Size of the vocabulary (number of unique moves).
41
+ n_embd: Embedding dimension (d_model).
42
+ n_layer: Number of transformer layers.
43
+ n_head: Number of attention heads.
44
+ n_ctx: Maximum sequence length (context window).
45
+ n_inner: Feed-forward inner dimension (default: 3 * n_embd).
46
+ dropout: Dropout probability.
47
+ layer_norm_epsilon: Epsilon for layer normalization.
48
+ tie_weights: Whether to tie embedding and output weights.
49
+ """
50
+
51
+ model_type = "chess_transformer"
52
+
53
+ def __init__(
54
+ self,
55
+ vocab_size: int = 1200,
56
+ n_embd: int = 128,
57
+ n_layer: int = 6,
58
+ n_head: int = 4,
59
+ n_ctx: int = 256,
60
+ n_inner: Optional[int] = None,
61
+ dropout: float = 0.1,
62
+ layer_norm_epsilon: float = 1e-5,
63
+ tie_weights: bool = True,
64
+ pad_token_id: int = 0,
65
+ bos_token_id: int = 1,
66
+ eos_token_id: int = 2,
67
+ **kwargs,
68
+ ):
69
+ super().__init__(
70
+ pad_token_id=pad_token_id,
71
+ bos_token_id=bos_token_id,
72
+ eos_token_id=eos_token_id,
73
+ **kwargs,
74
+ )
75
+
76
+ self.vocab_size = vocab_size
77
+ self.n_embd = n_embd
78
+ self.n_layer = n_layer
79
+ self.n_head = n_head
80
+ self.n_ctx = n_ctx
81
+ self.n_inner = n_inner if n_inner is not None else 3 * n_embd # Reduced from 4x to 3x
82
+ self.dropout = dropout
83
+ self.layer_norm_epsilon = layer_norm_epsilon
84
+ self.tie_weights = tie_weights
85
+ # Inform HF base class about tying behavior
86
+ self.tie_word_embeddings = bool(tie_weights)
87
+
88
+
89
+ class MultiHeadAttention(nn.Module):
90
+ """
91
+ Multi-head self-attention module.
92
+
93
+ This is a standard scaled dot-product attention implementation
94
+ with causal masking for autoregressive generation.
95
+ """
96
+
97
+ def __init__(self, config: ChessConfig):
98
+ super().__init__()
99
+
100
+ assert config.n_embd % config.n_head == 0, \
101
+ f"n_embd ({config.n_embd}) must be divisible by n_head ({config.n_head})"
102
+
103
+ self.n_head = config.n_head
104
+ self.n_embd = config.n_embd
105
+ self.head_dim = config.n_embd // config.n_head
106
+
107
+ # Combined QKV projection for efficiency
108
+ self.c_attn = nn.Linear(config.n_embd, 3 * config.n_embd)
109
+ self.c_proj = nn.Linear(config.n_embd, config.n_embd)
110
+
111
+ self.dropout = nn.Dropout(config.dropout)
112
+
113
+ # Causal mask (will be created on first forward pass)
114
+ self.register_buffer(
115
+ "bias",
116
+ torch.tril(torch.ones(config.n_ctx, config.n_ctx)).view(
117
+ 1, 1, config.n_ctx, config.n_ctx
118
+ ),
119
+ persistent=False,
120
+ )
121
+
122
+ def forward(
123
+ self,
124
+ x: torch.Tensor,
125
+ attention_mask: Optional[torch.Tensor] = None,
126
+ ) -> torch.Tensor:
127
+ batch_size, seq_len, _ = x.size()
128
+
129
+ # Compute Q, K, V
130
+ qkv = self.c_attn(x)
131
+ q, k, v = qkv.split(self.n_embd, dim=2)
132
+
133
+ # Reshape for multi-head attention
134
+ q = q.view(batch_size, seq_len, self.n_head, self.head_dim).transpose(1, 2)
135
+ k = k.view(batch_size, seq_len, self.n_head, self.head_dim).transpose(1, 2)
136
+ v = v.view(batch_size, seq_len, self.n_head, self.head_dim).transpose(1, 2)
137
+
138
+ # Scaled dot-product attention
139
+ attn_weights = torch.matmul(q, k.transpose(-2, -1)) / math.sqrt(self.head_dim)
140
+
141
+ # Apply causal mask
142
+ causal_mask = self.bias[:, :, :seq_len, :seq_len]
143
+ attn_weights = attn_weights.masked_fill(causal_mask == 0, float("-inf"))
144
+
145
+ # Apply attention mask (for padding)
146
+ if attention_mask is not None:
147
+ # attention_mask shape: (batch_size, seq_len) -> (batch_size, 1, 1, seq_len)
148
+ attention_mask = attention_mask.unsqueeze(1).unsqueeze(2)
149
+ attn_weights = attn_weights.masked_fill(attention_mask == 0, float("-inf"))
150
+
151
+ attn_weights = F.softmax(attn_weights, dim=-1)
152
+ attn_weights = self.dropout(attn_weights)
153
+
154
+ # Apply attention to values
155
+ attn_output = torch.matmul(attn_weights, v)
156
+
157
+ # Reshape back
158
+ attn_output = attn_output.transpose(1, 2).contiguous().view(
159
+ batch_size, seq_len, self.n_embd
160
+ )
161
+
162
+ # Output projection
163
+ attn_output = self.c_proj(attn_output)
164
+
165
+ return attn_output
166
+
167
+
168
+ class FeedForward(nn.Module):
169
+ """
170
+ Feed-forward network (MLP) module.
171
+
172
+ Standard two-layer MLP with GELU activation.
173
+ """
174
+
175
+ def __init__(self, config: ChessConfig):
176
+ super().__init__()
177
+
178
+ self.c_fc = nn.Linear(config.n_embd, config.n_inner)
179
+ self.c_proj = nn.Linear(config.n_inner, config.n_embd)
180
+ self.dropout = nn.Dropout(config.dropout)
181
+
182
+ def forward(self, x: torch.Tensor) -> torch.Tensor:
183
+ x = self.c_fc(x)
184
+ x = F.gelu(x)
185
+ x = self.c_proj(x)
186
+ x = self.dropout(x)
187
+ return x
188
+
189
+
190
+ class TransformerBlock(nn.Module):
191
+ """
192
+ A single transformer block with attention and feed-forward layers.
193
+
194
+ Uses pre-normalization (LayerNorm before attention/FFN) for better
195
+ training stability.
196
+ """
197
+
198
+ def __init__(self, config: ChessConfig):
199
+ super().__init__()
200
+
201
+ self.ln_1 = nn.LayerNorm(config.n_embd, eps=config.layer_norm_epsilon)
202
+ self.attn = MultiHeadAttention(config)
203
+ self.ln_2 = nn.LayerNorm(config.n_embd, eps=config.layer_norm_epsilon)
204
+ self.mlp = FeedForward(config)
205
+
206
+ def forward(
207
+ self,
208
+ x: torch.Tensor,
209
+ attention_mask: Optional[torch.Tensor] = None,
210
+ ) -> torch.Tensor:
211
+ # Pre-norm attention
212
+ x = x + self.attn(self.ln_1(x), attention_mask=attention_mask)
213
+ # Pre-norm FFN
214
+ x = x + self.mlp(self.ln_2(x))
215
+ return x
216
+
217
+
218
+ class ChessForCausalLM(PreTrainedModel):
219
+ """
220
+ Chess Transformer for Causal Language Modeling (next-move prediction).
221
+
222
+ This model is designed to predict the next chess move given a sequence
223
+ of previous moves. It uses a GPT-style architecture with:
224
+ - Token embeddings for chess moves
225
+ - Learned positional embeddings
226
+ - Stacked transformer blocks
227
+ - Linear head for next-token prediction
228
+
229
+ The model supports weight tying between the embedding layer and the
230
+ output projection to save parameters.
231
+
232
+ Example:
233
+ >>> config = ChessConfig(vocab_size=1200, n_embd=128, n_layer=6)
234
+ >>> model = ChessForCausalLM(config)
235
+ >>> inputs = {"input_ids": torch.tensor([[1, 42, 87]])}
236
+ >>> outputs = model(**inputs)
237
+ >>> next_move_logits = outputs.logits[:, -1, :]
238
+ """
239
+
240
+ config_class = ChessConfig
241
+ base_model_prefix = "transformer"
242
+ supports_gradient_checkpointing = True
243
+ # Suppress missing-key warning for tied lm_head when loading
244
+ keys_to_ignore_on_load_missing = ["lm_head.weight"]
245
+
246
+ def __init__(self, config: ChessConfig):
247
+ super().__init__(config)
248
+
249
+ # Token and position embeddings
250
+ self.wte = nn.Embedding(config.vocab_size, config.n_embd)
251
+ self.wpe = nn.Embedding(config.n_ctx, config.n_embd)
252
+
253
+ self.drop = nn.Dropout(config.dropout)
254
+
255
+ # Transformer blocks
256
+ self.h = nn.ModuleList([
257
+ TransformerBlock(config) for _ in range(config.n_layer)
258
+ ])
259
+
260
+ # Final layer norm
261
+ self.ln_f = nn.LayerNorm(config.n_embd, eps=config.layer_norm_epsilon)
262
+
263
+ # Output head
264
+ self.lm_head = nn.Linear(config.n_embd, config.vocab_size, bias=False)
265
+
266
+ # Declare tied weights for proper serialization
267
+ if config.tie_weights:
268
+ self._tied_weights_keys = ["lm_head.weight"]
269
+
270
+ # Initialize weights
271
+ self.post_init()
272
+
273
+ # Tie weights if configured
274
+ if config.tie_weights:
275
+ self.tie_weights()
276
+
277
+ def get_input_embeddings(self) -> nn.Module:
278
+ return self.wte
279
+
280
+ def set_input_embeddings(self, new_embeddings: nn.Module):
281
+ self.wte = new_embeddings
282
+ if getattr(self.config, "tie_weights", False):
283
+ self.tie_weights()
284
+
285
+ def get_output_embeddings(self) -> nn.Module:
286
+ return self.lm_head
287
+
288
+ def set_output_embeddings(self, new_embeddings: nn.Module):
289
+ self.lm_head = new_embeddings
290
+
291
+ def tie_weights(self):
292
+ # Use HF helper to tie or clone depending on config
293
+ if getattr(self.config, "tie_weights", False) or getattr(self.config, "tie_word_embeddings", False):
294
+ self._tie_or_clone_weights(self.lm_head, self.wte)
295
+
296
+ def _init_weights(self, module: nn.Module):
297
+ """Initialize weights following GPT-2 style."""
298
+ if isinstance(module, nn.Linear):
299
+ torch.nn.init.normal_(module.weight, mean=0.0, std=0.02)
300
+ if module.bias is not None:
301
+ torch.nn.init.zeros_(module.bias)
302
+ elif isinstance(module, nn.Embedding):
303
+ torch.nn.init.normal_(module.weight, mean=0.0, std=0.02)
304
+ elif isinstance(module, nn.LayerNorm):
305
+ torch.nn.init.ones_(module.weight)
306
+ torch.nn.init.zeros_(module.bias)
307
+
308
+ def forward(
309
+ self,
310
+ input_ids: torch.LongTensor,
311
+ attention_mask: Optional[torch.Tensor] = None,
312
+ position_ids: Optional[torch.LongTensor] = None,
313
+ labels: Optional[torch.LongTensor] = None,
314
+ return_dict: Optional[bool] = None,
315
+ **kwargs,
316
+ ) -> Union[Tuple, CausalLMOutputWithPast]:
317
+ """
318
+ Forward pass of the model.
319
+
320
+ Args:
321
+ input_ids: Token IDs of shape (batch_size, seq_len).
322
+ attention_mask: Attention mask of shape (batch_size, seq_len).
323
+ position_ids: Position IDs of shape (batch_size, seq_len).
324
+ labels: Labels for language modeling loss.
325
+ return_dict: Whether to return a ModelOutput object.
326
+
327
+ Returns:
328
+ CausalLMOutputWithPast containing loss (if labels provided) and logits.
329
+ """
330
+ return_dict = return_dict if return_dict is not None else self.config.use_return_dict
331
+
332
+ batch_size, seq_len = input_ids.size()
333
+ device = input_ids.device
334
+
335
+ # Create position IDs if not provided
336
+ if position_ids is None:
337
+ position_ids = torch.arange(seq_len, device=device).unsqueeze(0).expand(batch_size, -1)
338
+
339
+ # Get embeddings
340
+ token_embeds = self.wte(input_ids)
341
+ position_embeds = self.wpe(position_ids)
342
+ hidden_states = self.drop(token_embeds + position_embeds)
343
+
344
+ # Pass through transformer blocks
345
+ for block in self.h:
346
+ hidden_states = block(hidden_states, attention_mask=attention_mask)
347
+
348
+ # Final layer norm
349
+ hidden_states = self.ln_f(hidden_states)
350
+
351
+ # Get logits
352
+ logits = self.lm_head(hidden_states)
353
+
354
+ # Compute loss if labels are provided
355
+ loss = None
356
+ if labels is not None:
357
+ # Shift logits and labels for next-token prediction
358
+ shift_logits = logits[..., :-1, :].contiguous()
359
+ shift_labels = labels[..., 1:].contiguous()
360
+
361
+ # Flatten for cross-entropy
362
+ loss_fct = nn.CrossEntropyLoss(ignore_index=-100)
363
+ # loss_fct = nn.CrossEntropyLoss(ignore_index=self.config.pad_token_id)
364
+ loss = loss_fct(
365
+ shift_logits.view(-1, shift_logits.size(-1)),
366
+ shift_labels.view(-1),
367
+ )
368
+
369
+ if not return_dict:
370
+ output = (logits,)
371
+ return ((loss,) + output) if loss is not None else output
372
+
373
+ return CausalLMOutputWithPast(
374
+ loss=loss,
375
+ logits=logits,
376
+ past_key_values=None,
377
+ hidden_states=None,
378
+ attentions=None,
379
+ )
380
+
381
+ @torch.no_grad()
382
+ def generate_move(
383
+ self,
384
+ input_ids: torch.LongTensor,
385
+ temperature: float = 1.0,
386
+ top_k: Optional[int] = None,
387
+ top_p: Optional[float] = None,
388
+ ) -> int:
389
+ """
390
+ Generate the next move given a sequence of moves.
391
+
392
+ Args:
393
+ input_ids: Token IDs of shape (1, seq_len).
394
+ temperature: Sampling temperature (1.0 = no change).
395
+ top_k: If set, only sample from top k tokens.
396
+ top_p: If set, use nucleus sampling with this threshold.
397
+
398
+ Returns:
399
+ The token ID of the predicted next move.
400
+ """
401
+ self.eval()
402
+
403
+ # Get logits for the last position
404
+ outputs = self(input_ids)
405
+ logits = outputs.logits[:, -1, :] / temperature
406
+
407
+ # Apply top-k filtering
408
+ if top_k is not None:
409
+ indices_to_remove = logits < torch.topk(logits, top_k)[0][..., -1, None]
410
+ logits[indices_to_remove] = float("-inf")
411
+
412
+ # Apply top-p (nucleus) filtering
413
+ if top_p is not None:
414
+ sorted_logits, sorted_indices = torch.sort(logits, descending=True)
415
+ cumulative_probs = torch.cumsum(F.softmax(sorted_logits, dim=-1), dim=-1)
416
+
417
+ # Remove tokens with cumulative probability above the threshold
418
+ sorted_indices_to_remove = cumulative_probs > top_p
419
+ sorted_indices_to_remove[..., 1:] = sorted_indices_to_remove[..., :-1].clone()
420
+ sorted_indices_to_remove[..., 0] = 0
421
+
422
+ indices_to_remove = sorted_indices_to_remove.scatter(
423
+ dim=-1, index=sorted_indices, src=sorted_indices_to_remove
424
+ )
425
+ logits[indices_to_remove] = float("-inf")
426
+
427
+ # Sample from the distribution
428
+ probs = F.softmax(logits, dim=-1)
429
+ next_token = torch.multinomial(probs, num_samples=1)
430
+
431
+ return next_token.item()
432
+
433
+
434
+ # Register the model with Auto classes for easy loading
435
+ from transformers import AutoConfig, AutoModelForCausalLM
436
+
437
+ AutoConfig.register("chess_transformer", ChessConfig)
438
+ AutoModelForCausalLM.register(ChessConfig, ChessForCausalLM)
model.safetensors ADDED
@@ -0,0 +1,3 @@
 
 
 
 
1
+ version https://git-lfs.github.com/spec/v1
2
+ oid sha256:53cd3a04a99576f0f30153478e7a8da37e1f5a3d3efecede3eb5dbab73587ea1
3
+ size 3826992
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,368 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ """
2
+ Custom Chess Tokenizer for the Chess Challenge.
3
+
4
+ This tokenizer uses a compact hybrid scheme optimized for small models:
5
+ - Frequent moves are single tokens (e.g., WPe2e4).
6
+ - Rare moves fall back to two tokens: piece+from (e.g., WPe2) and to-square (e.g., e4).
7
+ - Promotions add a third token (q/r/b/n).
8
+
9
+ The dataset format uses:
10
+ - W/B prefix for White/Black
11
+ - Piece letter: P=Pawn, N=Knight, B=Bishop, R=Rook, Q=Queen, K=King
12
+ - Source and destination squares (e.g., e2e4)
13
+ - Special suffixes: (x)=capture, (+)=check, (+*)=checkmate, (o)/(O)=castling
14
+ """
15
+
16
+ from __future__ import annotations
17
+
18
+ import json
19
+ import os
20
+ import re
21
+ from pathlib import Path
22
+ from typing import Dict, List, Optional, Tuple
23
+
24
+ from transformers import PreTrainedTokenizer
25
+
26
+
27
+ class ChessTokenizer(PreTrainedTokenizer):
28
+ """
29
+ A custom tokenizer for chess moves using extended UCI notation.
30
+
31
+ This tokenizer uses a compact base vocabulary (piece+from, to-square,
32
+ promotion tokens) and optionally adds frequent full-move tokens for
33
+ shorter sequences and better sample efficiency.
34
+
35
+ Example:
36
+ >>> tokenizer = ChessTokenizer()
37
+ >>> tokenizer.encode("WPe2e4 BPe7e5")
38
+ [1, 42, 87, 2] # [BOS, e2e4, e7e5, EOS]
39
+ """
40
+
41
+ model_input_names = ["input_ids", "attention_mask"]
42
+ vocab_files_names = {"vocab_file": "vocab.json"}
43
+
44
+ # Special tokens
45
+ PAD_TOKEN = "[PAD]"
46
+ BOS_TOKEN = "[BOS]"
47
+ EOS_TOKEN = "[EOS]"
48
+ UNK_TOKEN = "[UNK]"
49
+
50
+ _MOVE_RE = re.compile(
51
+ r"^(?P<color>[WB])(?P<piece>[PNBRQK])(?P<from>[a-h][1-8])(?P<to>[a-h][1-8])(?P<rest>.*)$"
52
+ )
53
+ _PROMO_RE = re.compile(r"=([NBRQnrbq])")
54
+
55
+ def __init__(
56
+ self,
57
+ vocab_file: Optional[str] = None,
58
+ vocab: Optional[Dict[str, int]] = None,
59
+ **kwargs,
60
+ ):
61
+ """
62
+ Initialize the chess tokenizer.
63
+
64
+ Args:
65
+ vocab_file: Path to a JSON file containing the vocabulary mapping.
66
+ vocab: Dictionary mapping tokens to IDs (alternative to vocab_file).
67
+ **kwargs: Additional arguments passed to PreTrainedTokenizer.
68
+ """
69
+ # Initialize special tokens
70
+ self._pad_token = self.PAD_TOKEN
71
+ self._bos_token = self.BOS_TOKEN
72
+ self._eos_token = self.EOS_TOKEN
73
+ self._unk_token = self.UNK_TOKEN
74
+
75
+ # Remove any duplicate special-token entries passed through kwargs
76
+ # to avoid "multiple values for keyword" errors when loading from disk.
77
+ kwargs.pop("pad_token", None)
78
+ kwargs.pop("bos_token", None)
79
+ kwargs.pop("eos_token", None)
80
+ kwargs.pop("unk_token", None)
81
+
82
+ # Load or create vocabulary
83
+ if vocab is not None:
84
+ self._vocab = vocab
85
+ elif vocab_file is not None and os.path.exists(vocab_file):
86
+ with open(vocab_file, "r", encoding="utf-8") as f:
87
+ self._vocab = json.load(f)
88
+ else:
89
+ # Create a compact default vocabulary that can tokenize any move
90
+ self._vocab = self._create_default_vocab()
91
+
92
+ # Create reverse mapping
93
+ self._ids_to_tokens = {v: k for k, v in self._vocab.items()}
94
+
95
+ # Call parent init AFTER setting up vocab
96
+ super().__init__(
97
+ pad_token=self._pad_token,
98
+ bos_token=self._bos_token,
99
+ eos_token=self._eos_token,
100
+ unk_token=self._unk_token,
101
+ **kwargs,
102
+ )
103
+
104
+ def _create_default_vocab(self) -> Dict[str, int]:
105
+ """
106
+ Create a compact default vocabulary with full move coverage.
107
+
108
+ For better compression, use `build_vocab_from_dataset()` to add
109
+ frequent full-move tokens.
110
+ """
111
+ tokens = self._create_base_vocab_tokens()
112
+ return {token: idx for idx, token in enumerate(tokens)}
113
+
114
+ @classmethod
115
+ def _create_base_vocab_tokens(cls) -> List[str]:
116
+ special_tokens = [cls.PAD_TOKEN, cls.BOS_TOKEN, cls.EOS_TOKEN, cls.UNK_TOKEN]
117
+ pieces = ["P", "N", "B", "R", "Q", "K"]
118
+ colors = ["W", "B"]
119
+ files = "abcdefgh"
120
+ ranks = "12345678"
121
+ squares = [f"{f}{r}" for f in files for r in ranks]
122
+ piece_from_tokens = [f"{c}{p}{sq}" for c in colors for p in pieces for sq in squares]
123
+ to_tokens = squares
124
+ promo_tokens = ["q", "r", "b", "n"]
125
+ return special_tokens + piece_from_tokens + to_tokens + promo_tokens
126
+
127
+ @classmethod
128
+ def _parse_move(cls, token: str) -> Optional[Tuple[str, str, str, str, Optional[str]]]:
129
+ match = cls._MOVE_RE.match(token)
130
+ if not match:
131
+ return None
132
+ color = match.group("color")
133
+ piece = match.group("piece")
134
+ from_sq = match.group("from")
135
+ to_sq = match.group("to")
136
+ rest = match.group("rest")
137
+ promo_match = cls._PROMO_RE.search(rest)
138
+ promo = promo_match.group(1).upper() if promo_match else None
139
+ return color, piece, from_sq, to_sq, promo
140
+
141
+ @classmethod
142
+ def build_vocab_from_iterator(
143
+ cls,
144
+ iterator,
145
+ min_frequency: int = 1,
146
+ max_full_move_tokens: Optional[int] = 1200,
147
+ ) -> "ChessTokenizer":
148
+ """
149
+ Build a tokenizer vocabulary from an iterator of game strings.
150
+
151
+ Args:
152
+ iterator: An iterator yielding game strings (space-separated moves).
153
+ min_frequency: Minimum frequency for a token to be included.
154
+ max_full_move_tokens: Maximum number of full-move tokens to keep.
155
+
156
+ Returns:
157
+ A ChessTokenizer with the built vocabulary.
158
+ """
159
+ from collections import Counter
160
+
161
+ token_counts = Counter()
162
+
163
+ for game in iterator:
164
+ moves = game.strip().split()
165
+ for move in moves:
166
+ parsed = cls._parse_move(move)
167
+ if not parsed:
168
+ continue
169
+ color, piece, from_sq, to_sq, promo = parsed
170
+ if promo:
171
+ continue
172
+ token_counts[f"{color}{piece}{from_sq}{to_sq}"] += 1
173
+
174
+ # Filter by frequency
175
+ tokens = [
176
+ token for token, count in token_counts.items()
177
+ if count >= min_frequency
178
+ ]
179
+
180
+ # Sort by frequency, then lexicographically for reproducibility
181
+ tokens.sort(key=lambda t: (-token_counts[t], t))
182
+
183
+ if max_full_move_tokens is not None:
184
+ tokens = tokens[:max_full_move_tokens]
185
+
186
+ base_tokens = cls._create_base_vocab_tokens()
187
+ vocab = {token: idx for idx, token in enumerate(base_tokens + tokens)}
188
+
189
+ return cls(vocab=vocab)
190
+
191
+ @classmethod
192
+ def build_vocab_from_dataset(
193
+ cls,
194
+ dataset_name: str = "dlouapre/lichess_2025-01_1M",
195
+ split: str = "train",
196
+ column: str = "text",
197
+ min_frequency: int = 500,
198
+ max_samples: Optional[int] = 100000,
199
+ max_full_move_tokens: Optional[int] = 1200,
200
+ ) -> "ChessTokenizer":
201
+ """
202
+ Build a tokenizer vocabulary from a Hugging Face dataset.
203
+
204
+ Args:
205
+ dataset_name: Name of the dataset on Hugging Face Hub.
206
+ split: Dataset split to use.
207
+ column: Column containing the game strings.
208
+ min_frequency: Minimum frequency for a token to be included (default: 500).
209
+ max_samples: Maximum number of samples to process (default: 100k).
210
+ max_full_move_tokens: Maximum number of full-move tokens to keep.
211
+
212
+ Returns:
213
+ A ChessTokenizer with the built vocabulary.
214
+ """
215
+ from datasets import load_dataset
216
+
217
+ dataset = load_dataset(dataset_name, split=split)
218
+
219
+ if max_samples is not None:
220
+ dataset = dataset.select(range(min(max_samples, len(dataset))))
221
+
222
+ def game_iterator():
223
+ for example in dataset:
224
+ yield example[column]
225
+
226
+ return cls.build_vocab_from_iterator(
227
+ game_iterator(),
228
+ min_frequency=min_frequency,
229
+ max_full_move_tokens=max_full_move_tokens,
230
+ )
231
+
232
+ @property
233
+ def vocab_size(self) -> int:
234
+ """Return the size of the vocabulary."""
235
+ return len(self._vocab)
236
+
237
+ def get_vocab(self) -> Dict[str, int]:
238
+ """Return the vocabulary as a dictionary."""
239
+ return dict(self._vocab)
240
+
241
+ def _tokenize(self, text: str) -> List[str]:
242
+ """
243
+ Tokenize a string of moves into a list of tokens.
244
+
245
+ Args:
246
+ text: A string of space-separated moves.
247
+
248
+ Returns:
249
+ List of move tokens.
250
+ """
251
+ raw = text.strip()
252
+ if not raw:
253
+ return []
254
+
255
+ parts = raw.split()
256
+ out: List[str] = []
257
+ special = {self.PAD_TOKEN, self.BOS_TOKEN, self.EOS_TOKEN, self.UNK_TOKEN}
258
+
259
+ for part in parts:
260
+ if part in special:
261
+ out.append(part)
262
+ continue
263
+
264
+ parsed = self._parse_move(part)
265
+ if not parsed:
266
+ out.append(self.UNK_TOKEN)
267
+ continue
268
+
269
+ color, piece, from_sq, to_sq, promo = parsed
270
+ full_move = f"{color}{piece}{from_sq}{to_sq}"
271
+ if promo is None and full_move in self._vocab:
272
+ out.append(full_move)
273
+ continue
274
+
275
+ piece_from = f"{color}{piece}{from_sq}"
276
+ to_token = f"{to_sq}"
277
+ out.append(piece_from if piece_from in self._vocab else self.UNK_TOKEN)
278
+ out.append(to_token if to_token in self._vocab else self.UNK_TOKEN)
279
+
280
+ if promo:
281
+ promo_token = promo.lower()
282
+ out.append(promo_token if promo_token in self._vocab else self.UNK_TOKEN)
283
+
284
+ return out
285
+
286
+ def _convert_token_to_id(self, token: str) -> int:
287
+ """Convert a token to its ID."""
288
+ return self._vocab.get(token, self._vocab.get(self.UNK_TOKEN, 0))
289
+
290
+ def _convert_id_to_token(self, index: int) -> str:
291
+ """Convert an ID to its token."""
292
+ return self._ids_to_tokens.get(index, self.UNK_TOKEN)
293
+
294
+ def convert_tokens_to_string(self, tokens: List[str]) -> str:
295
+ """Convert a list of tokens back to a string."""
296
+ # Filter out special tokens for cleaner output
297
+ special = {self.PAD_TOKEN, self.BOS_TOKEN, self.EOS_TOKEN, self.UNK_TOKEN}
298
+ return " ".join(t for t in tokens if t not in special)
299
+
300
+ def save_vocabulary(
301
+ self,
302
+ save_directory: str,
303
+ filename_prefix: Optional[str] = None,
304
+ ) -> tuple:
305
+ """
306
+ Save the vocabulary to a JSON file.
307
+
308
+ Args:
309
+ save_directory: Directory to save the vocabulary.
310
+ filename_prefix: Optional prefix for the filename.
311
+
312
+ Returns:
313
+ Tuple containing the path to the saved vocabulary file.
314
+ """
315
+ if not os.path.isdir(save_directory):
316
+ os.makedirs(save_directory, exist_ok=True)
317
+
318
+ vocab_file = os.path.join(
319
+ save_directory,
320
+ (filename_prefix + "-" if filename_prefix else "") + "vocab.json",
321
+ )
322
+
323
+ with open(vocab_file, "w", encoding="utf-8") as f:
324
+ json.dump(self._vocab, f, ensure_ascii=False, indent=2)
325
+
326
+ return (vocab_file,)
327
+
328
+
329
+ def count_vocab_from_dataset(
330
+ dataset_name: str = "dlouapre/lichess_2025-01_1M",
331
+ split: str = "train",
332
+ column: str = "text",
333
+ max_samples: Optional[int] = 10000,
334
+ ) -> Dict[str, int]:
335
+ """
336
+ Count normalized move frequencies in a dataset (useful for vocabulary analysis).
337
+
338
+ Args:
339
+ dataset_name: Name of the dataset on Hugging Face Hub.
340
+ split: Dataset split to use.
341
+ column: Column containing the game strings.
342
+ max_samples: Maximum number of samples to process.
343
+
344
+ Returns:
345
+ Dictionary mapping normalized full-move tokens to their frequencies.
346
+ """
347
+ from collections import Counter
348
+ from datasets import load_dataset
349
+
350
+ dataset = load_dataset(dataset_name, split=split)
351
+
352
+ if max_samples is not None:
353
+ dataset = dataset.select(range(min(max_samples, len(dataset))))
354
+
355
+ token_counts = Counter()
356
+
357
+ for example in dataset:
358
+ moves = example[column].strip().split()
359
+ for move in moves:
360
+ parsed = ChessTokenizer._parse_move(move)
361
+ if not parsed:
362
+ continue
363
+ color, piece, from_sq, to_sq, promo = parsed
364
+ if promo:
365
+ continue
366
+ token_counts[f"{color}{piece}{from_sq}{to_sq}"] += 1
367
+
368
+ return dict(token_counts)
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,2042 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ {
2
+ "[PAD]": 0,
3
+ "[BOS]": 1,
4
+ "[EOS]": 2,
5
+ "[UNK]": 3,
6
+ "WPa1": 4,
7
+ "WPa2": 5,
8
+ "WPa3": 6,
9
+ "WPa4": 7,
10
+ "WPa5": 8,
11
+ "WPa6": 9,
12
+ "WPa7": 10,
13
+ "WPa8": 11,
14
+ "WPb1": 12,
15
+ "WPb2": 13,
16
+ "WPb3": 14,
17
+ "WPb4": 15,
18
+ "WPb5": 16,
19
+ "WPb6": 17,
20
+ "WPb7": 18,
21
+ "WPb8": 19,
22
+ "WPc1": 20,
23
+ "WPc2": 21,
24
+ "WPc3": 22,
25
+ "WPc4": 23,
26
+ "WPc5": 24,
27
+ "WPc6": 25,
28
+ "WPc7": 26,
29
+ "WPc8": 27,
30
+ "WPd1": 28,
31
+ "WPd2": 29,
32
+ "WPd3": 30,
33
+ "WPd4": 31,
34
+ "WPd5": 32,
35
+ "WPd6": 33,
36
+ "WPd7": 34,
37
+ "WPd8": 35,
38
+ "WPe1": 36,
39
+ "WPe2": 37,
40
+ "WPe3": 38,
41
+ "WPe4": 39,
42
+ "WPe5": 40,
43
+ "WPe6": 41,
44
+ "WPe7": 42,
45
+ "WPe8": 43,
46
+ "WPf1": 44,
47
+ "WPf2": 45,
48
+ "WPf3": 46,
49
+ "WPf4": 47,
50
+ "WPf5": 48,
51
+ "WPf6": 49,
52
+ "WPf7": 50,
53
+ "WPf8": 51,
54
+ "WPg1": 52,
55
+ "WPg2": 53,
56
+ "WPg3": 54,
57
+ "WPg4": 55,
58
+ "WPg5": 56,
59
+ "WPg6": 57,
60
+ "WPg7": 58,
61
+ "WPg8": 59,
62
+ "WPh1": 60,
63
+ "WPh2": 61,
64
+ "WPh3": 62,
65
+ "WPh4": 63,
66
+ "WPh5": 64,
67
+ "WPh6": 65,
68
+ "WPh7": 66,
69
+ "WPh8": 67,
70
+ "WNa1": 68,
71
+ "WNa2": 69,
72
+ "WNa3": 70,
73
+ "WNa4": 71,
74
+ "WNa5": 72,
75
+ "WNa6": 73,
76
+ "WNa7": 74,
77
+ "WNa8": 75,
78
+ "WNb1": 76,
79
+ "WNb2": 77,
80
+ "WNb3": 78,
81
+ "WNb4": 79,
82
+ "WNb5": 80,
83
+ "WNb6": 81,
84
+ "WNb7": 82,
85
+ "WNb8": 83,
86
+ "WNc1": 84,
87
+ "WNc2": 85,
88
+ "WNc3": 86,
89
+ "WNc4": 87,
90
+ "WNc5": 88,
91
+ "WNc6": 89,
92
+ "WNc7": 90,
93
+ "WNc8": 91,
94
+ "WNd1": 92,
95
+ "WNd2": 93,
96
+ "WNd3": 94,
97
+ "WNd4": 95,
98
+ "WNd5": 96,
99
+ "WNd6": 97,
100
+ "WNd7": 98,
101
+ "WNd8": 99,
102
+ "WNe1": 100,
103
+ "WNe2": 101,
104
+ "WNe3": 102,
105
+ "WNe4": 103,
106
+ "WNe5": 104,
107
+ "WNe6": 105,
108
+ "WNe7": 106,
109
+ "WNe8": 107,
110
+ "WNf1": 108,
111
+ "WNf2": 109,
112
+ "WNf3": 110,
113
+ "WNf4": 111,
114
+ "WNf5": 112,
115
+ "WNf6": 113,
116
+ "WNf7": 114,
117
+ "WNf8": 115,
118
+ "WNg1": 116,
119
+ "WNg2": 117,
120
+ "WNg3": 118,
121
+ "WNg4": 119,
122
+ "WNg5": 120,
123
+ "WNg6": 121,
124
+ "WNg7": 122,
125
+ "WNg8": 123,
126
+ "WNh1": 124,
127
+ "WNh2": 125,
128
+ "WNh3": 126,
129
+ "WNh4": 127,
130
+ "WNh5": 128,
131
+ "WNh6": 129,
132
+ "WNh7": 130,
133
+ "WNh8": 131,
134
+ "WBa1": 132,
135
+ "WBa2": 133,
136
+ "WBa3": 134,
137
+ "WBa4": 135,
138
+ "WBa5": 136,
139
+ "WBa6": 137,
140
+ "WBa7": 138,
141
+ "WBa8": 139,
142
+ "WBb1": 140,
143
+ "WBb2": 141,
144
+ "WBb3": 142,
145
+ "WBb4": 143,
146
+ "WBb5": 144,
147
+ "WBb6": 145,
148
+ "WBb7": 146,
149
+ "WBb8": 147,
150
+ "WBc1": 148,
151
+ "WBc2": 149,
152
+ "WBc3": 150,
153
+ "WBc4": 151,
154
+ "WBc5": 152,
155
+ "WBc6": 153,
156
+ "WBc7": 154,
157
+ "WBc8": 155,
158
+ "WBd1": 156,
159
+ "WBd2": 157,
160
+ "WBd3": 158,
161
+ "WBd4": 159,
162
+ "WBd5": 160,
163
+ "WBd6": 161,
164
+ "WBd7": 162,
165
+ "WBd8": 163,
166
+ "WBe1": 164,
167
+ "WBe2": 165,
168
+ "WBe3": 166,
169
+ "WBe4": 167,
170
+ "WBe5": 168,
171
+ "WBe6": 169,
172
+ "WBe7": 170,
173
+ "WBe8": 171,
174
+ "WBf1": 172,
175
+ "WBf2": 173,
176
+ "WBf3": 174,
177
+ "WBf4": 175,
178
+ "WBf5": 176,
179
+ "WBf6": 177,
180
+ "WBf7": 178,
181
+ "WBf8": 179,
182
+ "WBg1": 180,
183
+ "WBg2": 181,
184
+ "WBg3": 182,
185
+ "WBg4": 183,
186
+ "WBg5": 184,
187
+ "WBg6": 185,
188
+ "WBg7": 186,
189
+ "WBg8": 187,
190
+ "WBh1": 188,
191
+ "WBh2": 189,
192
+ "WBh3": 190,
193
+ "WBh4": 191,
194
+ "WBh5": 192,
195
+ "WBh6": 193,
196
+ "WBh7": 194,
197
+ "WBh8": 195,
198
+ "WRa1": 196,
199
+ "WRa2": 197,
200
+ "WRa3": 198,
201
+ "WRa4": 199,
202
+ "WRa5": 200,
203
+ "WRa6": 201,
204
+ "WRa7": 202,
205
+ "WRa8": 203,
206
+ "WRb1": 204,
207
+ "WRb2": 205,
208
+ "WRb3": 206,
209
+ "WRb4": 207,
210
+ "WRb5": 208,
211
+ "WRb6": 209,
212
+ "WRb7": 210,
213
+ "WRb8": 211,
214
+ "WRc1": 212,
215
+ "WRc2": 213,
216
+ "WRc3": 214,
217
+ "WRc4": 215,
218
+ "WRc5": 216,
219
+ "WRc6": 217,
220
+ "WRc7": 218,
221
+ "WRc8": 219,
222
+ "WRd1": 220,
223
+ "WRd2": 221,
224
+ "WRd3": 222,
225
+ "WRd4": 223,
226
+ "WRd5": 224,
227
+ "WRd6": 225,
228
+ "WRd7": 226,
229
+ "WRd8": 227,
230
+ "WRe1": 228,
231
+ "WRe2": 229,
232
+ "WRe3": 230,
233
+ "WRe4": 231,
234
+ "WRe5": 232,
235
+ "WRe6": 233,
236
+ "WRe7": 234,
237
+ "WRe8": 235,
238
+ "WRf1": 236,
239
+ "WRf2": 237,
240
+ "WRf3": 238,
241
+ "WRf4": 239,
242
+ "WRf5": 240,
243
+ "WRf6": 241,
244
+ "WRf7": 242,
245
+ "WRf8": 243,
246
+ "WRg1": 244,
247
+ "WRg2": 245,
248
+ "WRg3": 246,
249
+ "WRg4": 247,
250
+ "WRg5": 248,
251
+ "WRg6": 249,
252
+ "WRg7": 250,
253
+ "WRg8": 251,
254
+ "WRh1": 252,
255
+ "WRh2": 253,
256
+ "WRh3": 254,
257
+ "WRh4": 255,
258
+ "WRh5": 256,
259
+ "WRh6": 257,
260
+ "WRh7": 258,
261
+ "WRh8": 259,
262
+ "WQa1": 260,
263
+ "WQa2": 261,
264
+ "WQa3": 262,
265
+ "WQa4": 263,
266
+ "WQa5": 264,
267
+ "WQa6": 265,
268
+ "WQa7": 266,
269
+ "WQa8": 267,
270
+ "WQb1": 268,
271
+ "WQb2": 269,
272
+ "WQb3": 270,
273
+ "WQb4": 271,
274
+ "WQb5": 272,
275
+ "WQb6": 273,
276
+ "WQb7": 274,
277
+ "WQb8": 275,
278
+ "WQc1": 276,
279
+ "WQc2": 277,
280
+ "WQc3": 278,
281
+ "WQc4": 279,
282
+ "WQc5": 280,
283
+ "WQc6": 281,
284
+ "WQc7": 282,
285
+ "WQc8": 283,
286
+ "WQd1": 284,
287
+ "WQd2": 285,
288
+ "WQd3": 286,
289
+ "WQd4": 287,
290
+ "WQd5": 288,
291
+ "WQd6": 289,
292
+ "WQd7": 290,
293
+ "WQd8": 291,
294
+ "WQe1": 292,
295
+ "WQe2": 293,
296
+ "WQe3": 294,
297
+ "WQe4": 295,
298
+ "WQe5": 296,
299
+ "WQe6": 297,
300
+ "WQe7": 298,
301
+ "WQe8": 299,
302
+ "WQf1": 300,
303
+ "WQf2": 301,
304
+ "WQf3": 302,
305
+ "WQf4": 303,
306
+ "WQf5": 304,
307
+ "WQf6": 305,
308
+ "WQf7": 306,
309
+ "WQf8": 307,
310
+ "WQg1": 308,
311
+ "WQg2": 309,
312
+ "WQg3": 310,
313
+ "WQg4": 311,
314
+ "WQg5": 312,
315
+ "WQg6": 313,
316
+ "WQg7": 314,
317
+ "WQg8": 315,
318
+ "WQh1": 316,
319
+ "WQh2": 317,
320
+ "WQh3": 318,
321
+ "WQh4": 319,
322
+ "WQh5": 320,
323
+ "WQh6": 321,
324
+ "WQh7": 322,
325
+ "WQh8": 323,
326
+ "WKa1": 324,
327
+ "WKa2": 325,
328
+ "WKa3": 326,
329
+ "WKa4": 327,
330
+ "WKa5": 328,
331
+ "WKa6": 329,
332
+ "WKa7": 330,
333
+ "WKa8": 331,
334
+ "WKb1": 332,
335
+ "WKb2": 333,
336
+ "WKb3": 334,
337
+ "WKb4": 335,
338
+ "WKb5": 336,
339
+ "WKb6": 337,
340
+ "WKb7": 338,
341
+ "WKb8": 339,
342
+ "WKc1": 340,
343
+ "WKc2": 341,
344
+ "WKc3": 342,
345
+ "WKc4": 343,
346
+ "WKc5": 344,
347
+ "WKc6": 345,
348
+ "WKc7": 346,
349
+ "WKc8": 347,
350
+ "WKd1": 348,
351
+ "WKd2": 349,
352
+ "WKd3": 350,
353
+ "WKd4": 351,
354
+ "WKd5": 352,
355
+ "WKd6": 353,
356
+ "WKd7": 354,
357
+ "WKd8": 355,
358
+ "WKe1": 356,
359
+ "WKe2": 357,
360
+ "WKe3": 358,
361
+ "WKe4": 359,
362
+ "WKe5": 360,
363
+ "WKe6": 361,
364
+ "WKe7": 362,
365
+ "WKe8": 363,
366
+ "WKf1": 364,
367
+ "WKf2": 365,
368
+ "WKf3": 366,
369
+ "WKf4": 367,
370
+ "WKf5": 368,
371
+ "WKf6": 369,
372
+ "WKf7": 370,
373
+ "WKf8": 371,
374
+ "WKg1": 372,
375
+ "WKg2": 373,
376
+ "WKg3": 374,
377
+ "WKg4": 375,
378
+ "WKg5": 376,
379
+ "WKg6": 377,
380
+ "WKg7": 378,
381
+ "WKg8": 379,
382
+ "WKh1": 380,
383
+ "WKh2": 381,
384
+ "WKh3": 382,
385
+ "WKh4": 383,
386
+ "WKh5": 384,
387
+ "WKh6": 385,
388
+ "WKh7": 386,
389
+ "WKh8": 387,
390
+ "BPa1": 388,
391
+ "BPa2": 389,
392
+ "BPa3": 390,
393
+ "BPa4": 391,
394
+ "BPa5": 392,
395
+ "BPa6": 393,
396
+ "BPa7": 394,
397
+ "BPa8": 395,
398
+ "BPb1": 396,
399
+ "BPb2": 397,
400
+ "BPb3": 398,
401
+ "BPb4": 399,
402
+ "BPb5": 400,
403
+ "BPb6": 401,
404
+ "BPb7": 402,
405
+ "BPb8": 403,
406
+ "BPc1": 404,
407
+ "BPc2": 405,
408
+ "BPc3": 406,
409
+ "BPc4": 407,
410
+ "BPc5": 408,
411
+ "BPc6": 409,
412
+ "BPc7": 410,
413
+ "BPc8": 411,
414
+ "BPd1": 412,
415
+ "BPd2": 413,
416
+ "BPd3": 414,
417
+ "BPd4": 415,
418
+ "BPd5": 416,
419
+ "BPd6": 417,
420
+ "BPd7": 418,
421
+ "BPd8": 419,
422
+ "BPe1": 420,
423
+ "BPe2": 421,
424
+ "BPe3": 422,
425
+ "BPe4": 423,
426
+ "BPe5": 424,
427
+ "BPe6": 425,
428
+ "BPe7": 426,
429
+ "BPe8": 427,
430
+ "BPf1": 428,
431
+ "BPf2": 429,
432
+ "BPf3": 430,
433
+ "BPf4": 431,
434
+ "BPf5": 432,
435
+ "BPf6": 433,
436
+ "BPf7": 434,
437
+ "BPf8": 435,
438
+ "BPg1": 436,
439
+ "BPg2": 437,
440
+ "BPg3": 438,
441
+ "BPg4": 439,
442
+ "BPg5": 440,
443
+ "BPg6": 441,
444
+ "BPg7": 442,
445
+ "BPg8": 443,
446
+ "BPh1": 444,
447
+ "BPh2": 445,
448
+ "BPh3": 446,
449
+ "BPh4": 447,
450
+ "BPh5": 448,
451
+ "BPh6": 449,
452
+ "BPh7": 450,
453
+ "BPh8": 451,
454
+ "BNa1": 452,
455
+ "BNa2": 453,
456
+ "BNa3": 454,
457
+ "BNa4": 455,
458
+ "BNa5": 456,
459
+ "BNa6": 457,
460
+ "BNa7": 458,
461
+ "BNa8": 459,
462
+ "BNb1": 460,
463
+ "BNb2": 461,
464
+ "BNb3": 462,
465
+ "BNb4": 463,
466
+ "BNb5": 464,
467
+ "BNb6": 465,
468
+ "BNb7": 466,
469
+ "BNb8": 467,
470
+ "BNc1": 468,
471
+ "BNc2": 469,
472
+ "BNc3": 470,
473
+ "BNc4": 471,
474
+ "BNc5": 472,
475
+ "BNc6": 473,
476
+ "BNc7": 474,
477
+ "BNc8": 475,
478
+ "BNd1": 476,
479
+ "BNd2": 477,
480
+ "BNd3": 478,
481
+ "BNd4": 479,
482
+ "BNd5": 480,
483
+ "BNd6": 481,
484
+ "BNd7": 482,
485
+ "BNd8": 483,
486
+ "BNe1": 484,
487
+ "BNe2": 485,
488
+ "BNe3": 486,
489
+ "BNe4": 487,
490
+ "BNe5": 488,
491
+ "BNe6": 489,
492
+ "BNe7": 490,
493
+ "BNe8": 491,
494
+ "BNf1": 492,
495
+ "BNf2": 493,
496
+ "BNf3": 494,
497
+ "BNf4": 495,
498
+ "BNf5": 496,
499
+ "BNf6": 497,
500
+ "BNf7": 498,
501
+ "BNf8": 499,
502
+ "BNg1": 500,
503
+ "BNg2": 501,
504
+ "BNg3": 502,
505
+ "BNg4": 503,
506
+ "BNg5": 504,
507
+ "BNg6": 505,
508
+ "BNg7": 506,
509
+ "BNg8": 507,
510
+ "BNh1": 508,
511
+ "BNh2": 509,
512
+ "BNh3": 510,
513
+ "BNh4": 511,
514
+ "BNh5": 512,
515
+ "BNh6": 513,
516
+ "BNh7": 514,
517
+ "BNh8": 515,
518
+ "BBa1": 516,
519
+ "BBa2": 517,
520
+ "BBa3": 518,
521
+ "BBa4": 519,
522
+ "BBa5": 520,
523
+ "BBa6": 521,
524
+ "BBa7": 522,
525
+ "BBa8": 523,
526
+ "BBb1": 524,
527
+ "BBb2": 525,
528
+ "BBb3": 526,
529
+ "BBb4": 527,
530
+ "BBb5": 528,
531
+ "BBb6": 529,
532
+ "BBb7": 530,
533
+ "BBb8": 531,
534
+ "BBc1": 532,
535
+ "BBc2": 533,
536
+ "BBc3": 534,
537
+ "BBc4": 535,
538
+ "BBc5": 536,
539
+ "BBc6": 537,
540
+ "BBc7": 538,
541
+ "BBc8": 539,
542
+ "BBd1": 540,
543
+ "BBd2": 541,
544
+ "BBd3": 542,
545
+ "BBd4": 543,
546
+ "BBd5": 544,
547
+ "BBd6": 545,
548
+ "BBd7": 546,
549
+ "BBd8": 547,
550
+ "BBe1": 548,
551
+ "BBe2": 549,
552
+ "BBe3": 550,
553
+ "BBe4": 551,
554
+ "BBe5": 552,
555
+ "BBe6": 553,
556
+ "BBe7": 554,
557
+ "BBe8": 555,
558
+ "BBf1": 556,
559
+ "BBf2": 557,
560
+ "BBf3": 558,
561
+ "BBf4": 559,
562
+ "BBf5": 560,
563
+ "BBf6": 561,
564
+ "BBf7": 562,
565
+ "BBf8": 563,
566
+ "BBg1": 564,
567
+ "BBg2": 565,
568
+ "BBg3": 566,
569
+ "BBg4": 567,
570
+ "BBg5": 568,
571
+ "BBg6": 569,
572
+ "BBg7": 570,
573
+ "BBg8": 571,
574
+ "BBh1": 572,
575
+ "BBh2": 573,
576
+ "BBh3": 574,
577
+ "BBh4": 575,
578
+ "BBh5": 576,
579
+ "BBh6": 577,
580
+ "BBh7": 578,
581
+ "BBh8": 579,
582
+ "BRa1": 580,
583
+ "BRa2": 581,
584
+ "BRa3": 582,
585
+ "BRa4": 583,
586
+ "BRa5": 584,
587
+ "BRa6": 585,
588
+ "BRa7": 586,
589
+ "BRa8": 587,
590
+ "BRb1": 588,
591
+ "BRb2": 589,
592
+ "BRb3": 590,
593
+ "BRb4": 591,
594
+ "BRb5": 592,
595
+ "BRb6": 593,
596
+ "BRb7": 594,
597
+ "BRb8": 595,
598
+ "BRc1": 596,
599
+ "BRc2": 597,
600
+ "BRc3": 598,
601
+ "BRc4": 599,
602
+ "BRc5": 600,
603
+ "BRc6": 601,
604
+ "BRc7": 602,
605
+ "BRc8": 603,
606
+ "BRd1": 604,
607
+ "BRd2": 605,
608
+ "BRd3": 606,
609
+ "BRd4": 607,
610
+ "BRd5": 608,
611
+ "BRd6": 609,
612
+ "BRd7": 610,
613
+ "BRd8": 611,
614
+ "BRe1": 612,
615
+ "BRe2": 613,
616
+ "BRe3": 614,
617
+ "BRe4": 615,
618
+ "BRe5": 616,
619
+ "BRe6": 617,
620
+ "BRe7": 618,
621
+ "BRe8": 619,
622
+ "BRf1": 620,
623
+ "BRf2": 621,
624
+ "BRf3": 622,
625
+ "BRf4": 623,
626
+ "BRf5": 624,
627
+ "BRf6": 625,
628
+ "BRf7": 626,
629
+ "BRf8": 627,
630
+ "BRg1": 628,
631
+ "BRg2": 629,
632
+ "BRg3": 630,
633
+ "BRg4": 631,
634
+ "BRg5": 632,
635
+ "BRg6": 633,
636
+ "BRg7": 634,
637
+ "BRg8": 635,
638
+ "BRh1": 636,
639
+ "BRh2": 637,
640
+ "BRh3": 638,
641
+ "BRh4": 639,
642
+ "BRh5": 640,
643
+ "BRh6": 641,
644
+ "BRh7": 642,
645
+ "BRh8": 643,
646
+ "BQa1": 644,
647
+ "BQa2": 645,
648
+ "BQa3": 646,
649
+ "BQa4": 647,
650
+ "BQa5": 648,
651
+ "BQa6": 649,
652
+ "BQa7": 650,
653
+ "BQa8": 651,
654
+ "BQb1": 652,
655
+ "BQb2": 653,
656
+ "BQb3": 654,
657
+ "BQb4": 655,
658
+ "BQb5": 656,
659
+ "BQb6": 657,
660
+ "BQb7": 658,
661
+ "BQb8": 659,
662
+ "BQc1": 660,
663
+ "BQc2": 661,
664
+ "BQc3": 662,
665
+ "BQc4": 663,
666
+ "BQc5": 664,
667
+ "BQc6": 665,
668
+ "BQc7": 666,
669
+ "BQc8": 667,
670
+ "BQd1": 668,
671
+ "BQd2": 669,
672
+ "BQd3": 670,
673
+ "BQd4": 671,
674
+ "BQd5": 672,
675
+ "BQd6": 673,
676
+ "BQd7": 674,
677
+ "BQd8": 675,
678
+ "BQe1": 676,
679
+ "BQe2": 677,
680
+ "BQe3": 678,
681
+ "BQe4": 679,
682
+ "BQe5": 680,
683
+ "BQe6": 681,
684
+ "BQe7": 682,
685
+ "BQe8": 683,
686
+ "BQf1": 684,
687
+ "BQf2": 685,
688
+ "BQf3": 686,
689
+ "BQf4": 687,
690
+ "BQf5": 688,
691
+ "BQf6": 689,
692
+ "BQf7": 690,
693
+ "BQf8": 691,
694
+ "BQg1": 692,
695
+ "BQg2": 693,
696
+ "BQg3": 694,
697
+ "BQg4": 695,
698
+ "BQg5": 696,
699
+ "BQg6": 697,
700
+ "BQg7": 698,
701
+ "BQg8": 699,
702
+ "BQh1": 700,
703
+ "BQh2": 701,
704
+ "BQh3": 702,
705
+ "BQh4": 703,
706
+ "BQh5": 704,
707
+ "BQh6": 705,
708
+ "BQh7": 706,
709
+ "BQh8": 707,
710
+ "BKa1": 708,
711
+ "BKa2": 709,
712
+ "BKa3": 710,
713
+ "BKa4": 711,
714
+ "BKa5": 712,
715
+ "BKa6": 713,
716
+ "BKa7": 714,
717
+ "BKa8": 715,
718
+ "BKb1": 716,
719
+ "BKb2": 717,
720
+ "BKb3": 718,
721
+ "BKb4": 719,
722
+ "BKb5": 720,
723
+ "BKb6": 721,
724
+ "BKb7": 722,
725
+ "BKb8": 723,
726
+ "BKc1": 724,
727
+ "BKc2": 725,
728
+ "BKc3": 726,
729
+ "BKc4": 727,
730
+ "BKc5": 728,
731
+ "BKc6": 729,
732
+ "BKc7": 730,
733
+ "BKc8": 731,
734
+ "BKd1": 732,
735
+ "BKd2": 733,
736
+ "BKd3": 734,
737
+ "BKd4": 735,
738
+ "BKd5": 736,
739
+ "BKd6": 737,
740
+ "BKd7": 738,
741
+ "BKd8": 739,
742
+ "BKe1": 740,
743
+ "BKe2": 741,
744
+ "BKe3": 742,
745
+ "BKe4": 743,
746
+ "BKe5": 744,
747
+ "BKe6": 745,
748
+ "BKe7": 746,
749
+ "BKe8": 747,
750
+ "BKf1": 748,
751
+ "BKf2": 749,
752
+ "BKf3": 750,
753
+ "BKf4": 751,
754
+ "BKf5": 752,
755
+ "BKf6": 753,
756
+ "BKf7": 754,
757
+ "BKf8": 755,
758
+ "BKg1": 756,
759
+ "BKg2": 757,
760
+ "BKg3": 758,
761
+ "BKg4": 759,
762
+ "BKg5": 760,
763
+ "BKg6": 761,
764
+ "BKg7": 762,
765
+ "BKg8": 763,
766
+ "BKh1": 764,
767
+ "BKh2": 765,
768
+ "BKh3": 766,
769
+ "BKh4": 767,
770
+ "BKh5": 768,
771
+ "BKh6": 769,
772
+ "BKh7": 770,
773
+ "BKh8": 771,
774
+ "a1": 772,
775
+ "a2": 773,
776
+ "a3": 774,
777
+ "a4": 775,
778
+ "a5": 776,
779
+ "a6": 777,
780
+ "a7": 778,
781
+ "a8": 779,
782
+ "b1": 780,
783
+ "b2": 781,
784
+ "b3": 782,
785
+ "b4": 783,
786
+ "b5": 784,
787
+ "b6": 785,
788
+ "b7": 786,
789
+ "b8": 787,
790
+ "c1": 788,
791
+ "c2": 789,
792
+ "c3": 790,
793
+ "c4": 791,
794
+ "c5": 792,
795
+ "c6": 793,
796
+ "c7": 794,
797
+ "c8": 795,
798
+ "d1": 796,
799
+ "d2": 797,
800
+ "d3": 798,
801
+ "d4": 799,
802
+ "d5": 800,
803
+ "d6": 801,
804
+ "d7": 802,
805
+ "d8": 803,
806
+ "e1": 804,
807
+ "e2": 805,
808
+ "e3": 806,
809
+ "e4": 807,
810
+ "e5": 808,
811
+ "e6": 809,
812
+ "e7": 810,
813
+ "e8": 811,
814
+ "f1": 812,
815
+ "f2": 813,
816
+ "f3": 814,
817
+ "f4": 815,
818
+ "f5": 816,
819
+ "f6": 817,
820
+ "f7": 818,
821
+ "f8": 819,
822
+ "g1": 820,
823
+ "g2": 821,
824
+ "g3": 822,
825
+ "g4": 823,
826
+ "g5": 824,
827
+ "g6": 825,
828
+ "g7": 826,
829
+ "g8": 827,
830
+ "h1": 828,
831
+ "h2": 829,
832
+ "h3": 830,
833
+ "h4": 831,
834
+ "h5": 832,
835
+ "h6": 833,
836
+ "h7": 834,
837
+ "h8": 835,
838
+ "q": 836,
839
+ "r": 837,
840
+ "b": 838,
841
+ "n": 839,
842
+ "WNg1f3": 840,
843
+ "BNg8f6": 841,
844
+ "WPe2e4": 842,
845
+ "WPd2d4": 843,
846
+ "WNb1c3": 844,
847
+ "WKe1g1": 845,
848
+ "BNb8c6": 846,
849
+ "BKe8g8": 847,
850
+ "BPd7d5": 848,
851
+ "BPe7e6": 849,
852
+ "BPe7e5": 850,
853
+ "BPd7d6": 851,
854
+ "WPc2c3": 852,
855
+ "WPh2h3": 853,
856
+ "BPg7g6": 854,
857
+ "BPc7c6": 855,
858
+ "BPh7h6": 856,
859
+ "BPc7c5": 857,
860
+ "BPa7a6": 858,
861
+ "WPc2c4": 859,
862
+ "WNf3e5": 860,
863
+ "BBf8e7": 861,
864
+ "WPa2a3": 862,
865
+ "WPg2g3": 863,
866
+ "WPe2e3": 864,
867
+ "BNb8d7": 865,
868
+ "WPf2f4": 866,
869
+ "WBf1c4": 867,
870
+ "WRf1e1": 868,
871
+ "WPd2d3": 869,
872
+ "WNb1d2": 870,
873
+ "WPe4e5": 871,
874
+ "BPb7b6": 872,
875
+ "WBf1d3": 873,
876
+ "BNf6e4": 874,
877
+ "BPf7f6": 875,
878
+ "WPb2b3": 876,
879
+ "BPb7b5": 877,
880
+ "WBc1g5": 878,
881
+ "WBc1e3": 879,
882
+ "WPf2f3": 880,
883
+ "BBc8g4": 881,
884
+ "WPe4d5": 882,
885
+ "BRf8e8": 883,
886
+ "BBf8g7": 884,
887
+ "WBf1e2": 885,
888
+ "BPf7f5": 886,
889
+ "WNf3d4": 887,
890
+ "WPb2b4": 888,
891
+ "WBc1f4": 889,
892
+ "BPc5d4": 890,
893
+ "BBc8b7": 891,
894
+ "BNf6d5": 892,
895
+ "BNc6d4": 893,
896
+ "WNc3d5": 894,
897
+ "WPg2g4": 895,
898
+ "WPa2a4": 896,
899
+ "BBf8d6": 897,
900
+ "BBc8d7": 898,
901
+ "WPh2h4": 899,
902
+ "BPa7a5": 900,
903
+ "WPd4d5": 901,
904
+ "BBc8e6": 902,
905
+ "WPd4e5": 903,
906
+ "WRa1d1": 904,
907
+ "BRa8c8": 905,
908
+ "BBf8c5": 906,
909
+ "BNc6e5": 907,
910
+ "WQd1e2": 908,
911
+ "BNg8e7": 909,
912
+ "WNc3e4": 910,
913
+ "BPh7h5": 911,
914
+ "BBc8f5": 912,
915
+ "BRa8d8": 913,
916
+ "BPe6e5": 914,
917
+ "BQd8e7": 915,
918
+ "WRa1c1": 916,
919
+ "WNf3g5": 917,
920
+ "BPe5d4": 918,
921
+ "WQd1f3": 919,
922
+ "BPd5e4": 920,
923
+ "BPc6c5": 921,
924
+ "WBf1b5": 922,
925
+ "WKe1c1": 923,
926
+ "WKg1h1": 924,
927
+ "BPg7g5": 925,
928
+ "WQd1d2": 926,
929
+ "WBc1d2": 927,
930
+ "BBf8b4": 928,
931
+ "WBf1g2": 929,
932
+ "BKg8h8": 930,
933
+ "WPc4d5": 931,
934
+ "BKg8g7": 932,
935
+ "WBc1b2": 933,
936
+ "BPe5e4": 934,
937
+ "BRa8b8": 935,
938
+ "WPh4h5": 936,
939
+ "BKe8c8": 937,
940
+ "BPe6d5": 938,
941
+ "BPd5d4": 939,
942
+ "BPd6d5": 940,
943
+ "BQd8c7": 941,
944
+ "WPc3c4": 942,
945
+ "WRa1b1": 943,
946
+ "BQd8f6": 944,
947
+ "BPb5b4": 945,
948
+ "BPc6d5": 946,
949
+ "WKg1g2": 947,
950
+ "BNf6g4": 948,
951
+ "BQd8d7": 949,
952
+ "BPd6e5": 950,
953
+ "BPb7c6": 951,
954
+ "WPf4f5": 952,
955
+ "WPc3d4": 953,
956
+ "WPe3e4": 954,
957
+ "BPa5a4": 955,
958
+ "WPg4g5": 956,
959
+ "WRf1d1": 957,
960
+ "BRf8d8": 958,
961
+ "WRa1e1": 959,
962
+ "WNg1e2": 960,
963
+ "WPb2c3": 961,
964
+ "BBg4f3": 962,
965
+ "WPh3h4": 963,
966
+ "WKg1h2": 964,
967
+ "BQd8b6": 965,
968
+ "BNd7e5": 966,
969
+ "BPc5c4": 967,
970
+ "WQd1c2": 968,
971
+ "WBg5f6": 969,
972
+ "WPd3d4": 970,
973
+ "BRa8e8": 971,
974
+ "BPg6g5": 972,
975
+ "BKg8h7": 973,
976
+ "WPc4c5": 974,
977
+ "BNf6d7": 975,
978
+ "BBe7f6": 976,
979
+ "WPb4b5": 977,
980
+ "WPa4a5": 978,
981
+ "BPh5h4": 979,
982
+ "WNd2f3": 980,
983
+ "BPa6a5": 981,
984
+ "BNd7f6": 982,
985
+ "BQd8d5": 983,
986
+ "WPf3f4": 984,
987
+ "BPg5g4": 985,
988
+ "WKg1f2": 986,
989
+ "BPf6f5": 987,
990
+ "WNc3e2": 988,
991
+ "WPg3g4": 989,
992
+ "BPh6h5": 990,
993
+ "BPf5f4": 991,
994
+ "BKg8f7": 992,
995
+ "WKg1f1": 993,
996
+ "WNd2e4": 994,
997
+ "BKg8f8": 995,
998
+ "BPd5c4": 996,
999
+ "WNc3b5": 997,
1000
+ "BNf6h5": 998,
1001
+ "WBb5c6": 999,
1002
+ "WPd4c5": 1000,
1003
+ "WPf4e5": 1001,
1004
+ "WBe2f3": 1002,
1005
+ "BBb4c3": 1003,
1006
+ "WBc4b3": 1004,
1007
+ "WQd1d3": 1005,
1008
+ "WQd1b3": 1006,
1009
+ "BNc6e7": 1007,
1010
+ "BPb6b5": 1008,
1011
+ "WNf3d2": 1009,
1012
+ "BNc6b4": 1010,
1013
+ "WPa3a4": 1011,
1014
+ "WNf3h4": 1012,
1015
+ "WQd1d4": 1013,
1016
+ "BNd7c5": 1014,
1017
+ "BNe7f5": 1015,
1018
+ "WPg2f3": 1016,
1019
+ "WPb3b4": 1017,
1020
+ "BPg7f6": 1018,
1021
+ "BNc6a5": 1019,
1022
+ "BPf7e6": 1020,
1023
+ "BNe7g6": 1021,
1024
+ "BNd7b6": 1022,
1025
+ "WKc1b1": 1023,
1026
+ "BRf8f7": 1024,
1027
+ "WNe2g3": 1025,
1028
+ "WNd2c4": 1026,
1029
+ "BPe5f4": 1027,
1030
+ "WBg5h4": 1028,
1031
+ "BQd8d6": 1029,
1032
+ "BBg4h5": 1030,
1033
+ "WQd1h5": 1031,
1034
+ "BQd8a5": 1032,
1035
+ "WPe3d4": 1033,
1036
+ "WRf1f2": 1034,
1037
+ "WPe5f6": 1035,
1038
+ "BRh8g8": 1036,
1039
+ "WBd3e4": 1037,
1040
+ "BBd7c6": 1038,
1041
+ "WNc3a4": 1039,
1042
+ "BKc8b8": 1040,
1043
+ "BPf6e5": 1041,
1044
+ "BRf8c8": 1042,
1045
+ "WRd1e1": 1043,
1046
+ "WPe4f5": 1044,
1047
+ "WBe3d4": 1045,
1048
+ "BKe8d7": 1046,
1049
+ "BPh6g5": 1047,
1050
+ "WPe5e6": 1048,
1051
+ "WRa1f1": 1049,
1052
+ "BRa8f8": 1050,
1053
+ "WRd1d2": 1051,
1054
+ "BKe8f7": 1052,
1055
+ "BBe7d6": 1053,
1056
+ "WNd4c6": 1054,
1057
+ "BKe8d8": 1055,
1058
+ "BKe8e7": 1056,
1059
+ "BPf5e4": 1057,
1060
+ "WPf2e3": 1058,
1061
+ "WRh1g1": 1059,
1062
+ "WNd2b3": 1060,
1063
+ "WRe1e2": 1061,
1064
+ "WNe2f4": 1062,
1065
+ "BRd8d7": 1063,
1066
+ "WPd5d6": 1064,
1067
+ "WPd3e4": 1065,
1068
+ "WPh3g4": 1066,
1069
+ "BPa6b5": 1067,
1070
+ "BPa4a3": 1068,
1071
+ "WPh5h6": 1069,
1072
+ "WBf4g3": 1070,
1073
+ "BRd8e8": 1071,
1074
+ "WRf1c1": 1072,
1075
+ "WRe1d1": 1073,
1076
+ "WNe4f6": 1074,
1077
+ "BBd6e5": 1075,
1078
+ "BKf8e7": 1076,
1079
+ "WBf4e5": 1077,
1080
+ "WKf1e2": 1078,
1081
+ "BPd4d3": 1079,
1082
+ "BPh7g6": 1080,
1083
+ "BRe8e7": 1081,
1084
+ "WPa3b4": 1082,
1085
+ "WBc4d5": 1083,
1086
+ "WRh1e1": 1084,
1087
+ "WBd2c3": 1085,
1088
+ "WRd1d8": 1086,
1089
+ "BRh8f8": 1087,
1090
+ "BNe7d5": 1088,
1091
+ "WRe1e3": 1089,
1092
+ "WPa5a6": 1090,
1093
+ "BRd8d1": 1091,
1094
+ "BKg7f6": 1092,
1095
+ "BBc8a6": 1093,
1096
+ "WPe5d6": 1094,
1097
+ "WBe2d3": 1095,
1098
+ "BRh8e8": 1096,
1099
+ "WNe5c6": 1097,
1100
+ "WPa4b5": 1098,
1101
+ "WRf1f3": 1099,
1102
+ "WPf3e4": 1100,
1103
+ "BKe8f8": 1101,
1104
+ "BQd8h4": 1102,
1105
+ "BBc5b6": 1103,
1106
+ "WQd1g4": 1104,
1107
+ "BPe6f5": 1105,
1108
+ "BBe7g5": 1106,
1109
+ "BPe4e3": 1107,
1110
+ "WKe1d2": 1108,
1111
+ "BQd8g5": 1109,
1112
+ "WKe1e2": 1110,
1113
+ "WQd1a4": 1111,
1114
+ "WKg2f3": 1112,
1115
+ "WKe1d1": 1113,
1116
+ "BNb8a6": 1114,
1117
+ "BPa5b4": 1115,
1118
+ "BPf7g6": 1116,
1119
+ "BNg8h6": 1117,
1120
+ "WNe2d4": 1118,
1121
+ "WBc4d3": 1119,
1122
+ "WKf2e3": 1120,
1123
+ "WRd1d7": 1121,
1124
+ "WRe1e8": 1122,
1125
+ "BNe5f3": 1123,
1126
+ "BPb5c4": 1124,
1127
+ "BRf8f6": 1125,
1128
+ "BRe8d8": 1126,
1129
+ "BPh4h3": 1127,
1130
+ "BBe6d5": 1128,
1131
+ "BBg7e5": 1129,
1132
+ "WRh1f1": 1130,
1133
+ "WPh4g5": 1131,
1134
+ "BPh5g4": 1132,
1135
+ "BNd5c3": 1133,
1136
+ "BBh5g6": 1134,
1137
+ "WPd5c6": 1135,
1138
+ "WNb1a3": 1136,
1139
+ "WBh4g3": 1137,
1140
+ "BRe8e1": 1138,
1141
+ "BKf7e6": 1139,
1142
+ "WRe1e4": 1140,
1143
+ "WBg5e7": 1141,
1144
+ "BPb4b3": 1142,
1145
+ "BPd4c3": 1143,
1146
+ "BNe7c6": 1144,
1147
+ "BBf5g6": 1145,
1148
+ "BBg7f6": 1146,
1149
+ "BBe7c5": 1147,
1150
+ "WKh1g1": 1148,
1151
+ "WBd3e2": 1149,
1152
+ "BPg6f5": 1150,
1153
+ "WNe5d7": 1151,
1154
+ "BKh8g8": 1152,
1155
+ "WPa2b3": 1153,
1156
+ "WPf5f6": 1154,
1157
+ "WNg5e6": 1155,
1158
+ "BRd8d2": 1156,
1159
+ "WBd3c4": 1157,
1160
+ "WRe1e5": 1158,
1161
+ "WRe1e7": 1159,
1162
+ "BKg7h6": 1160,
1163
+ "WRd1d3": 1161,
1164
+ "WPh2g3": 1162,
1165
+ "BNe4c3": 1163,
1166
+ "WPc5c6": 1164,
1167
+ "WBe3c5": 1165,
1168
+ "WBc4b5": 1166,
1169
+ "WPb5b6": 1167,
1170
+ "BRe8e6": 1168,
1171
+ "BKe7d6": 1169,
1172
+ "WKe1f1": 1170,
1173
+ "WBc4e6": 1171,
1174
+ "WBd3f5": 1172,
1175
+ "BPc4c3": 1173,
1176
+ "WRe1f1": 1174,
1177
+ "WPg5g6": 1175,
1178
+ "WBf4d6": 1176,
1179
+ "BPb6c5": 1177,
1180
+ "BPc7d6": 1178,
1181
+ "WBe3f4": 1179,
1182
+ "WKh1h2": 1180,
1183
+ "WKe1f2": 1181,
1184
+ "WBb5a4": 1182,
1185
+ "WPb4c5": 1183,
1186
+ "WPh5g6": 1184,
1187
+ "WPb3c4": 1185,
1188
+ "BNe5c4": 1186,
1189
+ "BBc5d4": 1187,
1190
+ "WNe4d6": 1188,
1191
+ "BBf5e4": 1189,
1192
+ "BPf4f3": 1190,
1193
+ "BBf5d3": 1191,
1194
+ "WRh1d1": 1192,
1195
+ "WQd1e1": 1193,
1196
+ "WKg2g3": 1194,
1197
+ "BQd8e8": 1195,
1198
+ "WBe3g5": 1196,
1199
+ "BRh8d8": 1197,
1200
+ "WPg4f5": 1198,
1201
+ "WBd3c2": 1199,
1202
+ "WKh2g3": 1200,
1203
+ "WBc4f7": 1201,
1204
+ "WRd1c1": 1202,
1205
+ "BRd8d6": 1203,
1206
+ "WNg5f3": 1204,
1207
+ "WNd5f6": 1205,
1208
+ "WQf3g3": 1206,
1209
+ "WKe2d3": 1207,
1210
+ "WKg2h3": 1208,
1211
+ "BRc8d8": 1209,
1212
+ "BKf7g6": 1210,
1213
+ "BNd4f3": 1211,
1214
+ "WKf2g3": 1212,
1215
+ "BBb7d5": 1213,
1216
+ "BRe8e5": 1214,
1217
+ "BKf7e7": 1215,
1218
+ "WRd1d4": 1216,
1219
+ "BRe8f8": 1217,
1220
+ "WNf3h2": 1218,
1221
+ "BNd5f4": 1219,
1222
+ "BPe4f3": 1220,
1223
+ "BRe8e2": 1221,
1224
+ "BNe5d3": 1222,
1225
+ "BRd8c8": 1223,
1226
+ "WPd5e6": 1224,
1227
+ "WPc2d3": 1225,
1228
+ "WBb5d7": 1226,
1229
+ "BBd6e7": 1227,
1230
+ "WNd4f5": 1228,
1231
+ "BNh5f4": 1229,
1232
+ "WNg5f7": 1230,
1233
+ "BRd8d5": 1231,
1234
+ "BBd6f4": 1232,
1235
+ "WBg5e3": 1233,
1236
+ "WPf2g3": 1234,
1237
+ "BRc8c7": 1235,
1238
+ "WNe4c5": 1236,
1239
+ "BPg4g3": 1237,
1240
+ "WNb5d6": 1238,
1241
+ "WRc1d1": 1239,
1242
+ "BKh8h7": 1240,
1243
+ "BKf7f6": 1241,
1244
+ "WBe2g4": 1242,
1245
+ "WRd1d5": 1243,
1246
+ "BPa7b6": 1244,
1247
+ "WRe1e6": 1245,
1248
+ "WKf2e2": 1246,
1249
+ "WBc1h6": 1247,
1250
+ "WRd1d6": 1248,
1251
+ "BKg7g6": 1249,
1252
+ "BNg4e3": 1250,
1253
+ "WBc1a3": 1251,
1254
+ "WPg3f4": 1252,
1255
+ "WNe5f3": 1253,
1256
+ "WBe2c4": 1254,
1257
+ "WRd1f1": 1255,
1258
+ "BRc8c2": 1256,
1259
+ "BNa5c4": 1257,
1260
+ "BKh7g8": 1258,
1261
+ "BPa3a2": 1259,
1262
+ "WNe2c3": 1260,
1263
+ "WBe3d2": 1261,
1264
+ "BKh7g6": 1262,
1265
+ "WKc1d2": 1263,
1266
+ "WKf2f3": 1264,
1267
+ "BQd8c8": 1265,
1268
+ "BRd8d4": 1266,
1269
+ "BKe7d7": 1267,
1270
+ "BNe4d2": 1268,
1271
+ "BRf8g8": 1269,
1272
+ "BBc5e3": 1270,
1273
+ "BBd7e6": 1271,
1274
+ "BBb7e4": 1272,
1275
+ "WKh2g1": 1273,
1276
+ "WNd4f3": 1274,
1277
+ "WBd2e3": 1275,
1278
+ "WNg1h3": 1276,
1279
+ "WBf4g5": 1277,
1280
+ "BRd8f8": 1278,
1281
+ "WRc1c2": 1279,
1282
+ "BNb6c4": 1280,
1283
+ "WNe4g5": 1281,
1284
+ "WBd3g6": 1282,
1285
+ "BBd7b5": 1283,
1286
+ "BRe8e4": 1284,
1287
+ "WPa6a7": 1285,
1288
+ "WNg3f5": 1286,
1289
+ "BKc8d7": 1287,
1290
+ "BRf8b8": 1288,
1291
+ "BNe4f6": 1289,
1292
+ "WNd2f1": 1290,
1293
+ "BKe7f6": 1291,
1294
+ "WRf1g1": 1292,
1295
+ "WBg5f4": 1293,
1296
+ "BBe6c4": 1294,
1297
+ "WKh1g2": 1295,
1298
+ "BNf6e8": 1296,
1299
+ "WBf4e3": 1297,
1300
+ "WQe2f3": 1298,
1301
+ "BRd8d3": 1299,
1302
+ "WRc1c7": 1300,
1303
+ "WKe2d2": 1301,
1304
+ "BBd6c5": 1302,
1305
+ "WBd3b5": 1303,
1306
+ "BNg6f4": 1304,
1307
+ "BKh8g7": 1305,
1308
+ "BPg5f4": 1306,
1309
+ "BKf8g7": 1307,
1310
+ "BBg7h6": 1308,
1311
+ "BKf8e8": 1309,
1312
+ "BQf6g6": 1310,
1313
+ "WPc4b5": 1311,
1314
+ "WKe2f3": 1312,
1315
+ "BKf8g8": 1313,
1316
+ "BBb7c6": 1314,
1317
+ "BBg4e2": 1315,
1318
+ "WNg5e4": 1316,
1319
+ "BKg7f7": 1317,
1320
+ "BKd7c6": 1318,
1321
+ "BNf6h7": 1319,
1322
+ "BNb6d5": 1320,
1323
+ "WRf1b1": 1321,
1324
+ "BBg7d4": 1322,
1325
+ "WQd1d8": 1323,
1326
+ "BNd7f8": 1324,
1327
+ "WPc3b4": 1325,
1328
+ "BNg4f6": 1326,
1329
+ "BKf7g8": 1327,
1330
+ "WNd5e7": 1328,
1331
+ "BNc5e4": 1329,
1332
+ "WNh4f5": 1330,
1333
+ "BNd5e3": 1331,
1334
+ "BQe7f6": 1332,
1335
+ "BNb4d3": 1333,
1336
+ "WNd4e6": 1334,
1337
+ "BBb4a5": 1335,
1338
+ "WBe3h6": 1336,
1339
+ "WKf1g1": 1337,
1340
+ "WQd1d5": 1338,
1341
+ "WPf4g5": 1339,
1342
+ "WKe2e3": 1340,
1343
+ "BRf8f5": 1341,
1344
+ "BBb4d2": 1342,
1345
+ "BQd8d4": 1343,
1346
+ "BKe7e6": 1344,
1347
+ "BPd6c5": 1345,
1348
+ "BNd5f6": 1346,
1349
+ "BQd8d1": 1347,
1350
+ "WKf1g2": 1348,
1351
+ "WKf1e1": 1349,
1352
+ "BQb6c7": 1350,
1353
+ "BPc6b5": 1351,
1354
+ "WNa4c5": 1352,
1355
+ "WRf1f4": 1353,
1356
+ "WPe3f4": 1354,
1357
+ "WPh6h7": 1355,
1358
+ "BRb8b2": 1356,
1359
+ "BPg7h6": 1357,
1360
+ "BBe6f5": 1358,
1361
+ "BBc5d6": 1359,
1362
+ "BPf5g4": 1360,
1363
+ "WNb5c3": 1361,
1364
+ "BNg4e5": 1362,
1365
+ "WPd6d7": 1363,
1366
+ "WNe5g6": 1364,
1367
+ "BBg4f5": 1365,
1368
+ "BNg6e5": 1366,
1369
+ "WKg2f2": 1367,
1370
+ "BQd5d8": 1368,
1371
+ "WBg2e4": 1369,
1372
+ "BKf7e8": 1370,
1373
+ "BNd4e2": 1371,
1374
+ "WQd1c1": 1372,
1375
+ "BKf6e5": 1373,
1376
+ "BKg7g8": 1374,
1377
+ "WBa4b3": 1375,
1378
+ "BNe5g4": 1376,
1379
+ "BRb8c8": 1377,
1380
+ "WBh6g7": 1378,
1381
+ "WBb3c2": 1379,
1382
+ "WNc4e5": 1380,
1383
+ "WNd4b5": 1381,
1384
+ "WKf3e4": 1382,
1385
+ "WNg3e4": 1383,
1386
+ "WPd3c4": 1384,
1387
+ "BPc5b4": 1385,
1388
+ "BKd7c7": 1386,
1389
+ "BBc5b4": 1387,
1390
+ "WRe1c1": 1388,
1391
+ "WBg2f3": 1389,
1392
+ "BBg4e6": 1390,
1393
+ "BBe7h4": 1391,
1394
+ "WNb3c5": 1392,
1395
+ "BBb7f3": 1393,
1396
+ "WNe5f7": 1394,
1397
+ "WNb5c7": 1395,
1398
+ "BNb4c6": 1396,
1399
+ "WNe4g3": 1397,
1400
+ "BRe8e3": 1398,
1401
+ "BRc8c1": 1399,
1402
+ "BNf5d4": 1400,
1403
+ "BKc8b7": 1401,
1404
+ "BPd3d2": 1402,
1405
+ "WQe2e3": 1403,
1406
+ "WPc6c7": 1404,
1407
+ "BKg7f8": 1405,
1408
+ "WKg3f4": 1406,
1409
+ "BPf6g5": 1407,
1410
+ "BBc8h3": 1408,
1411
+ "BRc8c3": 1409,
1412
+ "BBf6e5": 1410,
1413
+ "BRa8a7": 1411,
1414
+ "WQd2e3": 1412,
1415
+ "BPg6h5": 1413,
1416
+ "BBe7b4": 1414,
1417
+ "WRb1b7": 1415,
1418
+ "BBe6g4": 1416,
1419
+ "BBg7f8": 1417,
1420
+ "WRf1f7": 1418,
1421
+ "BPb4c3": 1419,
1422
+ "WKd2c3": 1420,
1423
+ "BBb7a6": 1421,
1424
+ "BBc5f2": 1422,
1425
+ "BBd6g3": 1423,
1426
+ "BRc8c4": 1424,
1427
+ "WNf3e1": 1425,
1428
+ "BNc5d3": 1426,
1429
+ "WKd2e3": 1427,
1430
+ "BKf7g7": 1428,
1431
+ "WKg2f1": 1429,
1432
+ "WKh2h3": 1430,
1433
+ "BKd7e6": 1431,
1434
+ "BRe8c8": 1432,
1435
+ "BRb8d8": 1433,
1436
+ "BPb3b2": 1434,
1437
+ "WKc1b2": 1435,
1438
+ "WPg2h3": 1436,
1439
+ "BPc3c2": 1437,
1440
+ "WBb5d3": 1438,
1441
+ "WPb6b7": 1439,
1442
+ "WRb1c1": 1440,
1443
+ "WBd2b4": 1441,
1444
+ "WRc1c8": 1442,
1445
+ "BKe7f7": 1443,
1446
+ "WNe5g4": 1444,
1447
+ "WKh2h1": 1445,
1448
+ "BBb7c8": 1446,
1449
+ "WKf3e3": 1447,
1450
+ "WNd5c7": 1448,
1451
+ "BKd8c7": 1449,
1452
+ "BBe6d7": 1450,
1453
+ "WQe2d3": 1451,
1454
+ "BNd4c2": 1452,
1455
+ "WBc4e2": 1453,
1456
+ "BKg6f5": 1454,
1457
+ "WKh2g2": 1455,
1458
+ "BNe5g6": 1456,
1459
+ "WKg3g4": 1457,
1460
+ "BKf6f5": 1458,
1461
+ "WKe3d3": 1459,
1462
+ "BPh3h2": 1460,
1463
+ "BKd7d6": 1461,
1464
+ "BRc8c6": 1462,
1465
+ "WPf5g6": 1463,
1466
+ "WRc1e1": 1464,
1467
+ "WRf1f8": 1465,
1468
+ "WPf3g4": 1466,
1469
+ "BKh7g7": 1467,
1470
+ "WKf3f4": 1468,
1471
+ "WRd1g1": 1469,
1472
+ "BNd5b4": 1470,
1473
+ "BKd8e7": 1471,
1474
+ "BPh4g3": 1472,
1475
+ "WPe6e7": 1473,
1476
+ "WKe3d4": 1474,
1477
+ "WBb5c4": 1475,
1478
+ "WKd2c2": 1476,
1479
+ "WKg3f3": 1477,
1480
+ "BQb6b2": 1478,
1481
+ "WNc4d6": 1479,
1482
+ "BRc8b8": 1480,
1483
+ "BNf5e3": 1481,
1484
+ "BNd5b6": 1482,
1485
+ "BRc8e8": 1483,
1486
+ "WRf1f6": 1484,
1487
+ "BKd7e7": 1485,
1488
+ "BKf6e6": 1486,
1489
+ "BQc7b6": 1487,
1490
+ "BKc7b6": 1488,
1491
+ "WBe2b5": 1489,
1492
+ "WBb2d4": 1490,
1493
+ "BKc8c7": 1491,
1494
+ "BNb4c2": 1492,
1495
+ "WKf3g4": 1493,
1496
+ "BBf5g4": 1494,
1497
+ "WQe2e4": 1495,
1498
+ "BKf8f7": 1496,
1499
+ "BKe6d5": 1497,
1500
+ "WRb1d1": 1498,
1501
+ "WQd2e2": 1499,
1502
+ "WBe2h5": 1500,
1503
+ "WRf1f5": 1501,
1504
+ "BKg6f6": 1502,
1505
+ "WNb3d4": 1503,
1506
+ "WKg2g1": 1504,
1507
+ "BKf6g5": 1505,
1508
+ "BRf8f4": 1506,
1509
+ "WKf2g1": 1507,
1510
+ "WKc1c2": 1508,
1511
+ "WBg3e5": 1509,
1512
+ "BKe6d6": 1510,
1513
+ "WKd2d3": 1511,
1514
+ "BBf5e6": 1512,
1515
+ "BKh7h8": 1513,
1516
+ "WRc1c3": 1514,
1517
+ "BPe4d3": 1515,
1518
+ "WRa1a2": 1516,
1519
+ "BKh7h6": 1517,
1520
+ "WBb3d5": 1518,
1521
+ "BNh6f5": 1519,
1522
+ "BRf8f2": 1520,
1523
+ "WBd2f4": 1521,
1524
+ "WKf1f2": 1522,
1525
+ "BBb4c5": 1523,
1526
+ "BRf8f1": 1524,
1527
+ "BBd7f5": 1525,
1528
+ "WQd2d3": 1526,
1529
+ "WQe2d2": 1527,
1530
+ "WRc1c6": 1528,
1531
+ "BRf8a8": 1529,
1532
+ "WBb2e5": 1530,
1533
+ "BNg4f2": 1531,
1534
+ "BBb4d6": 1532,
1535
+ "BPe3e2": 1533,
1536
+ "BKd6c5": 1534,
1537
+ "BBd6c7": 1535,
1538
+ "WNc3d1": 1536,
1539
+ "BNh5f6": 1537,
1540
+ "WBg2d5": 1538,
1541
+ "BNe5c6": 1539,
1542
+ "BQe7e6": 1540,
1543
+ "BKg6g5": 1541,
1544
+ "BQe7d7": 1542,
1545
+ "WQd2f4": 1543,
1546
+ "BBe7f8": 1544,
1547
+ "BNe4g3": 1545,
1548
+ "BNc6d8": 1546,
1549
+ "WKd2e2": 1547,
1550
+ "BQc7d7": 1548,
1551
+ "BKf6e7": 1549,
1552
+ "BBf6g5": 1550,
1553
+ "WKe2f2": 1551,
1554
+ "BNe4g5": 1552,
1555
+ "BRf8f3": 1553,
1556
+ "WBb2c3": 1554,
1557
+ "WKd1e2": 1555,
1558
+ "WQb3c2": 1556,
1559
+ "BBd6b4": 1557,
1560
+ "WKe3e4": 1558,
1561
+ "BKe7f8": 1559,
1562
+ "WKe3f4": 1560,
1563
+ "WKc2b3": 1561,
1564
+ "WNe5c4": 1562,
1565
+ "BPa2a1": 1563,
1566
+ "BNe4f2": 1564,
1567
+ "BBc5e7": 1565,
1568
+ "WPg4h5": 1566,
1569
+ "BRb8b7": 1567,
1570
+ "WKg3h4": 1568,
1571
+ "WKf2g2": 1569,
1572
+ "BPa4b3": 1570,
1573
+ "WRc1b1": 1571,
1574
+ "BKe6f5": 1572,
1575
+ "WBe3f2": 1573,
1576
+ "BBf6d4": 1574,
1577
+ "BRg8g7": 1575,
1578
+ "BBf8h6": 1576,
1579
+ "WRf1a1": 1577,
1580
+ "BQd7e7": 1578,
1581
+ "BRa8g8": 1579,
1582
+ "WKf3e2": 1580,
1583
+ "BRf8h8": 1581,
1584
+ "WBd3h7": 1582,
1585
+ "WNh4g6": 1583,
1586
+ "WQf3e2": 1584,
1587
+ "BKe7d8": 1585,
1588
+ "WKf3g3": 1586,
1589
+ "BBg7c3": 1587,
1590
+ "BQe7d6": 1588,
1591
+ "BNe4c5": 1589,
1592
+ "WBf3e4": 1590,
1593
+ "WQe2c4": 1591,
1594
+ "WPg6g7": 1592,
1595
+ "WKg3f2": 1593,
1596
+ "WBc4a2": 1594,
1597
+ "BKf6g6": 1595,
1598
+ "WKe3f3": 1596,
1599
+ "BPe7f6": 1597,
1600
+ "BNh5g3": 1598,
1601
+ "WKd1c2": 1599,
1602
+ "WBd2g5": 1600,
1603
+ "WNd4b3": 1601,
1604
+ "BPb5a4": 1602,
1605
+ "WNf4d5": 1603,
1606
+ "BRd8g8": 1604,
1607
+ "BBa5b6": 1605,
1608
+ "BBf6e7": 1606,
1609
+ "WBb2f6": 1607,
1610
+ "WKe2f1": 1608,
1611
+ "BQc7e5": 1609,
1612
+ "WKd3c4": 1610,
1613
+ "WNg3h5": 1611,
1614
+ "WNh2g4": 1612,
1615
+ "WPa7a8": 1613,
1616
+ "BQa5b6": 1614,
1617
+ "WBg5h6": 1615,
1618
+ "WBg2h3": 1616,
1619
+ "WKf2e1": 1617,
1620
+ "BPd7c6": 1618,
1621
+ "WQc2d2": 1619,
1622
+ "BNe4d6": 1620,
1623
+ "BQd7e6": 1621,
1624
+ "WRd1b1": 1622,
1625
+ "WNe5d3": 1623,
1626
+ "WRc1c5": 1624,
1627
+ "WBe3b6": 1625,
1628
+ "BQe7e5": 1626,
1629
+ "WQe2g4": 1627,
1630
+ "WRa1g1": 1628,
1631
+ "BKe6e5": 1629,
1632
+ "BKd6c6": 1630,
1633
+ "BBd7g4": 1631,
1634
+ "BRc8c5": 1632,
1635
+ "WPf6f7": 1633,
1636
+ "BNa5c6": 1634,
1637
+ "BPd4e3": 1635,
1638
+ "BNa6c5": 1636,
1639
+ "WPb4a5": 1637,
1640
+ "WQf3e3": 1638,
1641
+ "WQd2c3": 1639,
1642
+ "BKd8c8": 1640,
1643
+ "BBe7d8": 1641,
1644
+ "BKe6f6": 1642,
1645
+ "BKg6f7": 1643,
1646
+ "BNb4d5": 1644,
1647
+ "BBb4e7": 1645,
1648
+ "WRg1g2": 1646,
1649
+ "BKg6h5": 1647,
1650
+ "BQc7c6": 1648,
1651
+ "WNb5d4": 1649,
1652
+ "WNh4f3": 1650,
1653
+ "BKd7c8": 1651,
1654
+ "BQa5c7": 1652,
1655
+ "WNa3c4": 1653,
1656
+ "WQc2d3": 1654,
1657
+ "BKg7h7": 1655,
1658
+ "BQd5a5": 1656,
1659
+ "WRb1b2": 1657,
1660
+ "WPg5f6": 1658,
1661
+ "BRd8b8": 1659,
1662
+ "BQc7d6": 1660,
1663
+ "BQe7c5": 1661,
1664
+ "BNb6d7": 1662,
1665
+ "BQd7c6": 1663,
1666
+ "WQe2f2": 1664,
1667
+ "BQe7f7": 1665,
1668
+ "WQc2b3": 1666,
1669
+ "WNf4e6": 1667,
1670
+ "WPb5c6": 1668,
1671
+ "WRa1a7": 1669,
1672
+ "BQe7g5": 1670,
1673
+ "BBd7e8": 1671,
1674
+ "WKe2d1": 1672,
1675
+ "WNe4c3": 1673,
1676
+ "WQb3b7": 1674,
1677
+ "WBf3e2": 1675,
1678
+ "BKc8d8": 1676,
1679
+ "WNc4e3": 1677,
1680
+ "BPg5h4": 1678,
1681
+ "BPg3g2": 1679,
1682
+ "BRa8a6": 1680,
1683
+ "BRb2a2": 1681,
1684
+ "WKd3c3": 1682,
1685
+ "WBb2a3": 1683,
1686
+ "BBf6g7": 1684,
1687
+ "BNc6b8": 1685,
1688
+ "WRb1e1": 1686,
1689
+ "BPf3f2": 1687,
1690
+ "WPf5e6": 1688,
1691
+ "BNf5h4": 1689,
1692
+ "BRa8a2": 1690,
1693
+ "WNc7a8": 1691,
1694
+ "WQf3f4": 1692,
1695
+ "BKh6g7": 1693,
1696
+ "BQd7d6": 1694,
1697
+ "BKf6g7": 1695,
1698
+ "BBg6e4": 1696,
1699
+ "WKg2h2": 1697,
1700
+ "BRb8a8": 1698,
1701
+ "BPf4g3": 1699,
1702
+ "WQf3e4": 1700,
1703
+ "BKe6d7": 1701,
1704
+ "WKb1a1": 1702,
1705
+ "WQd2g5": 1703,
1706
+ "WBf3d5": 1704,
1707
+ "WBg2f1": 1705,
1708
+ "WKh3g2": 1706,
1709
+ "WKf3g2": 1707,
1710
+ "BRh8c8": 1708,
1711
+ "WBf3g4": 1709,
1712
+ "WPa5b6": 1710,
1713
+ "BNc5e6": 1711,
1714
+ "BKd8d7": 1712,
1715
+ "WNf1g3": 1713,
1716
+ "WKe3d2": 1714,
1717
+ "BKd7e8": 1715,
1718
+ "BKd8e8": 1716,
1719
+ "WQh5f3": 1717,
1720
+ "WNf5e7": 1718,
1721
+ "WBf4c7": 1719,
1722
+ "WKh3g4": 1720,
1723
+ "WKc1d1": 1721,
1724
+ "WPg3h4": 1722,
1725
+ "WRc1c4": 1723,
1726
+ "BBg4d7": 1724,
1727
+ "BPc4b3": 1725,
1728
+ "BQc7e7": 1726,
1729
+ "BRb8e8": 1727,
1730
+ "WQd3e3": 1728,
1731
+ "BKh6g5": 1729,
1732
+ "BNd4f5": 1730,
1733
+ "WQc2e4": 1731,
1734
+ "BNf4d3": 1732,
1735
+ "BNf6g8": 1733,
1736
+ "WQd2h6": 1734,
1737
+ "BNf4e2": 1735,
1738
+ "WKd2c1": 1736,
1739
+ "BBg7b2": 1737,
1740
+ "WQf3f6": 1738,
1741
+ "WRf1h1": 1739,
1742
+ "BKf7f8": 1740,
1743
+ "BQd7f5": 1741,
1744
+ "WKd3d4": 1742,
1745
+ "BNa5b3": 1743,
1746
+ "BRa8a1": 1744,
1747
+ "WQe2h5": 1745,
1748
+ "BKc7d6": 1746,
1749
+ "BRh8h7": 1747,
1750
+ "WKd3e3": 1748,
1751
+ "WBf4h6": 1749,
1752
+ "BKe7e8": 1750,
1753
+ "BKd6d5": 1751,
1754
+ "WKh3h4": 1752,
1755
+ "BKb8a8": 1753,
1756
+ "BQf6e7": 1754,
1757
+ "BKh6h5": 1755,
1758
+ "WPh7h8": 1756,
1759
+ "BRc8a8": 1757,
1760
+ "WKc2d3": 1758,
1761
+ "WRd1h1": 1759,
1762
+ "WBb2c1": 1760,
1763
+ "WQc2e2": 1761,
1764
+ "BNg6h4": 1762,
1765
+ "BKg7h8": 1763,
1766
+ "WQd3d2": 1764,
1767
+ "WRa1a8": 1765,
1768
+ "WRf3g3": 1766,
1769
+ "WQf3g4": 1767,
1770
+ "WBf1h3": 1768,
1771
+ "BNc4e3": 1769,
1772
+ "BQb6d4": 1770,
1773
+ "BQe7b4": 1771,
1774
+ "BBf5c2": 1772,
1775
+ "WKd3e4": 1773,
1776
+ "WKd1c1": 1774,
1777
+ "WRb7a7": 1775,
1778
+ "BQf6e5": 1776,
1779
+ "WQf3h5": 1777,
1780
+ "BKd6e6": 1778,
1781
+ "BKc7c6": 1779,
1782
+ "WBf4d2": 1780,
1783
+ "BNd4c6": 1781,
1784
+ "WQd3e2": 1782,
1785
+ "BRa2a3": 1783,
1786
+ "WBg5d2": 1784,
1787
+ "WNc5e6": 1785,
1788
+ "BBc6d5": 1786,
1789
+ "WRb1a1": 1787,
1790
+ "BKe6f7": 1788,
1791
+ "WBe5f6": 1789,
1792
+ "WNd5f4": 1790,
1793
+ "WPb3a4": 1791,
1794
+ "BPb2b1": 1792,
1795
+ "WPc5b6": 1793,
1796
+ "BRb8b6": 1794,
1797
+ "WNf4h5": 1795,
1798
+ "BNa6c7": 1796,
1799
+ "WKg4g5": 1797,
1800
+ "WPc2b3": 1798,
1801
+ "BNc2a1": 1799,
1802
+ "BKb8a7": 1800,
1803
+ "BKd6e5": 1801,
1804
+ "WRb1b3": 1802,
1805
+ "WBc2b3": 1803,
1806
+ "BNa6b4": 1804,
1807
+ "BNg6e7": 1805,
1808
+ "WQd3e4": 1806,
1809
+ "BRc2b2": 1807,
1810
+ "WNd5e3": 1808,
1811
+ "BRc8f8": 1809,
1812
+ "WNh3f4": 1810,
1813
+ "WPb7b8": 1811,
1814
+ "WKe3f2": 1812,
1815
+ "BKd6e7": 1813,
1816
+ "WNg3e2": 1814,
1817
+ "WRa1a3": 1815,
1818
+ "WQe2b5": 1816,
1819
+ "BKe5d4": 1817,
1820
+ "WKd1d2": 1818,
1821
+ "WKf4g5": 1819,
1822
+ "BKc6b5": 1820,
1823
+ "BQd8f8": 1821,
1824
+ "WQd4d1": 1822,
1825
+ "BBb6d4": 1823,
1826
+ "BKf5e4": 1824,
1827
+ "BNd4e6": 1825,
1828
+ "WNf5d6": 1826,
1829
+ "BKd6c7": 1827,
1830
+ "WNa4c3": 1828,
1831
+ "BNd5e7": 1829,
1832
+ "WKb1a2": 1830,
1833
+ "BKf5f4": 1831,
1834
+ "WPc5d6": 1832,
1835
+ "WRh1h2": 1833,
1836
+ "WBd3b1": 1834,
1837
+ "BNh7g5": 1835,
1838
+ "BPc4d3": 1836,
1839
+ "BPh2h1": 1837,
1840
+ "WBe2d1": 1838,
1841
+ "BQc7d8": 1839,
1842
+ "BNh6g4": 1840,
1843
+ "WQc2c3": 1841,
1844
+ "WBg2c6": 1842,
1845
+ "WNe3f5": 1843,
1846
+ "WNe3d5": 1844,
1847
+ "WPc7c8": 1845,
1848
+ "BQf6f5": 1846,
1849
+ "BQe7h4": 1847,
1850
+ "WRe1g1": 1848,
1851
+ "WNb3d2": 1849,
1852
+ "WKf4f5": 1850,
1853
+ "WQd2c2": 1851,
1854
+ "WRe1b1": 1852,
1855
+ "WKb1c1": 1853,
1856
+ "WBb5e2": 1854,
1857
+ "WNg5h3": 1855,
1858
+ "BBd6h2": 1856,
1859
+ "WKd3c2": 1857,
1860
+ "WKf2f1": 1858,
1861
+ "BKc7d7": 1859,
1862
+ "BQf6f3": 1860,
1863
+ "BQf6g5": 1861,
1864
+ "WKd2e1": 1862,
1865
+ "BNe5d7": 1863,
1866
+ "WKb1c2": 1864,
1867
+ "WBh6g5": 1865,
1868
+ "WRh1c1": 1866,
1869
+ "BBc6b5": 1867,
1870
+ "WBe2f1": 1868,
1871
+ "BQb6a5": 1869,
1872
+ "BQd7c7": 1870,
1873
+ "WRh1h3": 1871,
1874
+ "WBc2e4": 1872,
1875
+ "WQd3f3": 1873,
1876
+ "BPg4f3": 1874,
1877
+ "BBe6b3": 1875,
1878
+ "WQd2d4": 1876,
1879
+ "WRc7b7": 1877,
1880
+ "WPb2a3": 1878,
1881
+ "BKg5g4": 1879,
1882
+ "WRa7a6": 1880,
1883
+ "WNa3b5": 1881,
1884
+ "BNf4h3": 1882,
1885
+ "BRa2a1": 1883,
1886
+ "BQd7g4": 1884,
1887
+ "WNd4e2": 1885,
1888
+ "BQb6c6": 1886,
1889
+ "BNg4h6": 1887,
1890
+ "WNh3g5": 1888,
1891
+ "WNa4b6": 1889,
1892
+ "BBe6h3": 1890,
1893
+ "WKf4e5": 1891,
1894
+ "BKc6b6": 1892,
1895
+ "BRh8h6": 1893,
1896
+ "BRb2b3": 1894,
1897
+ "BKd5c4": 1895,
1898
+ "WKd3e2": 1896,
1899
+ "WKe4d5": 1897,
1900
+ "BRd8a8": 1898,
1901
+ "WNc6e7": 1899,
1902
+ "BQb2c3": 1900,
1903
+ "BBe6f7": 1901,
1904
+ "BRd8h8": 1902,
1905
+ "BQd6e7": 1903,
1906
+ "WKg2h1": 1904,
1907
+ "BQf6e6": 1905,
1908
+ "WBe3a7": 1906,
1909
+ "WRc1f1": 1907,
1910
+ "WQf3f7": 1908,
1911
+ "WQe2e5": 1909,
1912
+ "BBg4h3": 1910,
1913
+ "WKg3g2": 1911,
1914
+ "BNf5d6": 1912,
1915
+ "BBf6c3": 1913,
1916
+ "BNf8g6": 1914,
1917
+ "BQd5e6": 1915,
1918
+ "WKc2d2": 1916,
1919
+ "BRg8g6": 1917,
1920
+ "WNa3c2": 1918,
1921
+ "WNf5h6": 1919,
1922
+ "WRc1a1": 1920,
1923
+ "WRd1a1": 1921,
1924
+ "WKd1e1": 1922,
1925
+ "BKf6f7": 1923,
1926
+ "WKd4c5": 1924,
1927
+ "WQf3d5": 1925,
1928
+ "BRe8b8": 1926,
1929
+ "BRe8g8": 1927,
1930
+ "WQh5h6": 1928,
1931
+ "BBe5d4": 1929,
1932
+ "BKf5g4": 1930,
1933
+ "BNd4b3": 1931,
1934
+ "BRg8f8": 1932,
1935
+ "BBc6e4": 1933,
1936
+ "WQd3d4": 1934,
1937
+ "BPc2c1": 1935,
1938
+ "BQd6e5": 1936,
1939
+ "BNe6f4": 1937,
1940
+ "BKc7b7": 1938,
1941
+ "WQf3d3": 1939,
1942
+ "BPb7a6": 1940,
1943
+ "WNd5c3": 1941,
1944
+ "BRa8a5": 1942,
1945
+ "WRa1a6": 1943,
1946
+ "BNf5g3": 1944,
1947
+ "WBe4d5": 1945,
1948
+ "WKc2c3": 1946,
1949
+ "WQd3c4": 1947,
1950
+ "BKc6c5": 1948,
1951
+ "WNc3b1": 1949,
1952
+ "BRa3a2": 1950,
1953
+ "BBa6b7": 1951,
1954
+ "BNc4e5": 1952,
1955
+ "WRg1g3": 1953,
1956
+ "BKg5f4": 1954,
1957
+ "WNe3g4": 1955,
1958
+ "WRa1a4": 1956,
1959
+ "BKb8c7": 1957,
1960
+ "WRa7a8": 1958,
1961
+ "WNc5d7": 1959,
1962
+ "WBd4e3": 1960,
1963
+ "BPc7b6": 1961,
1964
+ "WQd4e3": 1962,
1965
+ "WKc3b4": 1963,
1966
+ "WKe4e5": 1964,
1967
+ "WBh4f6": 1965,
1968
+ "BRc2a2": 1966,
1969
+ "BQd6d7": 1967,
1970
+ "WNd6b7": 1968,
1971
+ "WBb3c4": 1969,
1972
+ "WBg3d6": 1970,
1973
+ "BBd7a4": 1971,
1974
+ "BRb8b4": 1972,
1975
+ "WBd2e1": 1973,
1976
+ "BQb6d8": 1974,
1977
+ "WRa6a7": 1975,
1978
+ "BKb8c8": 1976,
1979
+ "BNe6d4": 1977,
1980
+ "WBb3e6": 1978,
1981
+ "WKe2e1": 1979,
1982
+ "WPg7g8": 1980,
1983
+ "WQf3d1": 1981,
1984
+ "WKf3f2": 1982,
1985
+ "WPg5h6": 1983,
1986
+ "WRb7b6": 1984,
1987
+ "WNg4f6": 1985,
1988
+ "BKe6e7": 1986,
1989
+ "BBa6c4": 1987,
1990
+ "WBd5c6": 1988,
1991
+ "WRh1h7": 1989,
1992
+ "BRf6g6": 1990,
1993
+ "BNc5d7": 1991,
1994
+ "WKf4e3": 1992,
1995
+ "BNc4b2": 1993,
1996
+ "BBb7g2": 1994,
1997
+ "BNc4d2": 1995,
1998
+ "WKg4f3": 1996,
1999
+ "WRg1f1": 1997,
2000
+ "WKe3e2": 1998,
2001
+ "WKe4f5": 1999,
2002
+ "WQd3b5": 2000,
2003
+ "BRa8a3": 2001,
2004
+ "WNd3e5": 2002,
2005
+ "BBe4f3": 2003,
2006
+ "BQc7b7": 2004,
2007
+ "BNd6e4": 2005,
2008
+ "WKg3h3": 2006,
2009
+ "WNd5b6": 2007,
2010
+ "BKe5e4": 2008,
2011
+ "BQf6d4": 2009,
2012
+ "WBg5d8": 2010,
2013
+ "WRh1h5": 2011,
2014
+ "BNc5b3": 2012,
2015
+ "BQd8a8": 2013,
2016
+ "BKg6g7": 2014,
2017
+ "BKd7d8": 2015,
2018
+ "BPb6a5": 2016,
2019
+ "WNh2f3": 2017,
2020
+ "BNd6f5": 2018,
2021
+ "BBh5f3": 2019,
2022
+ "BKf5e5": 2020,
2023
+ "WBb2g7": 2021,
2024
+ "WBf4h2": 2022,
2025
+ "BQd6e6": 2023,
2026
+ "WRe2d2": 2024,
2027
+ "BQe7d8": 2025,
2028
+ "WQd3c3": 2026,
2029
+ "WBe4d3": 2027,
2030
+ "BKc5b4": 2028,
2031
+ "WNe6f8": 2029,
2032
+ "WNg4e5": 2030,
2033
+ "WQf3f5": 2031,
2034
+ "WRe3f3": 2032,
2035
+ "WKg4f5": 2033,
2036
+ "BRc3c2": 2034,
2037
+ "WKf4e4": 2035,
2038
+ "WRc7a7": 2036,
2039
+ "BBb6c7": 2037,
2040
+ "BKg6h6": 2038,
2041
+ "BBc5a7": 2039
2042
+ }