jja 0.7.1

swiss army knife for chess file formats
Documentation
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
547
548
549
550
551
552
553
554
555
556
557
558
559
560
561
562
563
564
565
566
567
568
569
570
571
572
573
574
575
576
577
578
579
580
581
582
583
584
585
586
587
588
589
590
591
592
593
594
595
596
597
598
599
600
601
602
603
604
605
606
607
608
609
610
611
612
613
614
615
616
617
618
619
620
621
622
623
624
625
626
627
628
629
630
631
632
633
634
635
636
637
638
639
640
641
642
643
644
645
646
647
648
649
650
651
652
653
654
655
656
657
658
659
660
661
662
663
664
665
666
667
668
669
670
671
672
673
674
675
676
677
678
679
680
681
682
683
684
685
686
687
688
689
690
691
692
693
694
695
696
697
698
699
700
701
702
703
704
705
706
707
708
709
710
711
712
713
714
715
716
717
718
719
720
721
722
723
724
725
726
727
728
729
730
731
732
733
734
735
736
737
738
739
740
741
742
743
744
745
746
747
748
749
750
751
752
753
754
755
756
757
758
759
760
761
762
763
764
765
766
767
768
769
770
771
772
773
774
775
776
777
778
779
780
781
782
783
784
785
786
787
788
789
790
791
792
793
794
795
796
797
798
799
800
801
802
803
804
805
806
807
808
809
810
811
812
813
814
815
816
817
818
819
820
821
822
823
824
825
826
827
828
829
830
831
832
833
834
835
836
837
838
839
840
841
842
843
844
845
846
847
848
849
850
851
852
853
854
855
856
857
858
859
860
861
[![](https://git.sr.ht/~alip/jja/blob/main/misc/jja-logo.png)](https://git.sr.ht/~alip/jja)

Swiss army knife for chess file formats
=======================================
[![Jin, Jîyan, Azadî](https://img.shields.io/badge/Jin,%20J%C3%AEyan,%20Azad%C3%AE!-8A2BE2)](https://en.wikipedia.org/wiki/Woman,_Life,_Freedom)
[![crates.io](https://img.shields.io/crates/v/jja.svg)](https://crates.io/crates/jja)
[![msrv](https://img.shields.io/badge/rustc-1.64%2B-green?style=plastic)](https://blog.rust-lang.org/2022/09/22/Rust-1.64.0.html)
[![documentation](https://docs.rs/jja/badge.svg)](https://docs.rs/jja)
[![build status](https://builds.sr.ht/~alip/jja.svg)](https://builds.sr.ht/~alip/jja?)
[![downloads](https://img.shields.io/crates/d/jja.svg)](https://crates.io/crates/jja)
[![Crowdin](https://badges.crowdin.net/jja/localized.svg)](https://crowdin.com/project/jja)
[![stability-beta](https://img.shields.io/badge/stability-beta-33bbff.svg)](https://github.com/mkenney/software-guides/blob/master/STABILITY-BADGES.md#beta)
[![license](https://img.shields.io/crates/l/jja.svg)](https://git.sr.ht/~alip/jja/tree/main/item/COPYING)
[![dependency status](https://deps.rs/repo/sourcehut/~alip/jja/status.svg)](https://deps.rs/repo/sourcehut/~alip/jja)
[![maintenance-status](https://img.shields.io/badge/maintenance-actively--developed-brightgreen.svg)](https://git.sr.ht/~alip/jja)

This is ___jja___, a command line utility to interact with various chess file
formats. It is still in its early stages of development. The initial intention
of the author was to convert their opening books which were saved with ChessBase's
proprietary [CTG](https://www.chessprogramming.org/CTG) format to the free and open
[PolyGlot](http://hgm.nubati.net/book_format.html) format. Overtime they intend to
add support for other chess file formats (
[cbh](https://www.chessprogramming.org/ChessBase_%28Database%29%23Formats),
[epd](https://www.chessprogramming.org/Extended_Position_Description),
[pgn](https://www.chessprogramming.org/Portable_Game_Notation),
[si4](https://www.chessprogramming.org/SCID#Database_Format),
[si5](https://www.chessprogramming.org/SCID#Database_Format)
and so on).

# Demo

<img src="https://chesswob.org/jja/jja-intro-20230326.gif"
alt="jja intro"
title="jja intro"
/>

# Jin, Jîyan, Azadî

I've started hacking this on International Women's Day 2023, a day to honor the
achievements of women and advocate for their rights worldwide. As a person of
Kurdish heritage, I am particularly moved by the slogan "Woman, Life, Freedom",
which has become a symbol of resistance against oppression and a call for
equality. In the spirit of free software and free speech, I strive to
contribute to the creation of a more just and inclusive society, where every
human being is granted the freedom to express themselves and pursue their
dreams. I also honor the memory of Mahsa Amini, whose tragic death reminds us
of the urgent need to fight for women's freedom and safety.

More on Wikipedia,
[WikiPedia::Woman, Life, Freedom](https://en.wikipedia.org/wiki/Woman,_Life,_Freedom)

# Formats

As of version 0.7.0, ___jja___ supports reading/querying:

- [PolyGlot]http://hgm.nubati.net/book_format.html, aka `bin`
- [Arena]https://www.chessprogramming.org/ABK, aka `abk`
- [ChessBase]https://www.chessprogramming.org/CTG, aka `ctg`
- [ChessMaster]https://www.chessprogramming.org/Chessmaster, aka `obk`
  (version 1 and 2, w\o,with text notes)
- [BrainLearn]https://github.com/amchess/BrainLearn, aka `exp`

opening book files, whereas it supports writing/converting to:

- [PolyGlot]http://hgm.nubati.net/book_format.html, aka `bin`
- [Arena]https://www.chessprogramming.org/ABK, aka `abk`
- [BrainLearn]https://github.com/amchess/BrainLearn, aka `exp`

opening book files.

As of version 0.5.0, ___jja___ supports exporting all the supported opening book
formats to [PGN](http://www.saremba.de/chessgml/standards/pgn/pgn-complete.htm). To
use this functionality, specify an output file with `pgn` extension as an argument
to `jja edit`.

During opening book conversion, ___jja___ uses the information provided in various
input opening book formats to come up with a **move weight** which accompanies the
move in the [PolyGlot](http://hgm.nubati.net/book_format.html) opening file.
___jja___ also writes some custom numbers in the **learn** field, such as
[NAGs](https://en.wikipedia.org/wiki/Numeric_Annotation_Glyphs) during `ctg`
conversion or **priority** during `abk` conversion.  You may disable this custom
usage using `--no-learn` as it may confuse other software making use of this field.

Note, [Arena](https://www.chessprogramming.org/ABK), aka `abk`, opening book file
writing support is only supported from
[ChessBase](https://www.chessprogramming.org/CTG), aka `ctg` books. Use the command
line flags `--author`, `--comment`, `--probability-priority`, `--probability-games`,
`--probability-win-percent` to configure [ABK](https://www.chessprogramming.org/ABK)
header metadata. Game statistics (minimum number of games/wins, win percentages for
both sides) are managed automatically by ___jja___.

In-place editing for [Arena](https://www.chessprogramming.org/ABK) opening books is
also possible using `-i, --in-place=SUFFIX` command line option. Conversion from
[PolyGlot](http://hgm.nubati.net/book_format.html), aka `bin`, and
[ChessMaster](https://www.chessprogramming.org/Chessmaster), aka `obk` opening books
to [Arena](https://www.chessprogramming.org/ABK), aka `abk` opening book files is
planned for a future release.

# Usage

- Use `jja info` to get brief information about the chess file.
- Use `jja find` to search for a position in a chess file.
- Use `jja edit` to edit opening book files and convert to
[PolyGlot]http://hgm.nubati.net/book_format.html files or
[Arena]https://www.chessprogramming.org/ABK files.
- Use `jja make` to compile
  [PGN]https://en.wikipedia.org/wiki/Portable_Game_Notation files into
  [PolyGlot]http://hgm.nubati.net/book_format.html opening books.
- Use `jja dump` to dump a [PolyGlot]http://hgm.nubati.net/book_format.html
or [BrainLearn]https://github.com/amchess/BrainLearn file as a stream of
[JSON]https://en.wikipedia.org/wiki/JSON arrays.
- Use `jja restore` to restore a [PolyGlot]http://hgm.nubati.net/book_format.html
or [BrainLearn]https://github.com/amchess/BrainLearn file from a stream of
[JSON]https://en.wikipedia.org/wiki/JSON arrays.
- Use `jja merge` to merge two [PolyGlot]http://hgm.nubati.net/book_format.html
  opening books.
- Use `jja match` to arrange book matches using random playouts.
- Use `jja play` to make random playouts, optionally using books.
- Use `jja hash` to calculate Zobrist hash of a given chess position.
- Use `jja open` to browse [ECO]https://www.chessprogramming.org/ECO
  classification.
- Use `jja quote` to print a chess quote.

___jja___ determines the type of the file using its file extension. Files with the
extension `.bin` are considered [PolyGlot](http://hgm.nubati.net/book_format.html)
books. Files with the extension `.ctg` are considered
[ChessBase](https://www.chessprogramming.org/CTG) books. Files with the extension
`.abk` are considered [Arena](https://www.chessprogramming.org/ABK) books.  Files
with extension `.obk` are considered
[ChessMaster](https://www.chessprogramming.org/Chessmaster) books. Files with
extension `.exp` are considered [BrainLearn](https://github.com/amchess/BrainLearn)
experience files.

By default if the standard output is a
[TTY](https://en.wikipedia.org/wiki/Computer_terminal), ___jja___ will display
information using fancy tables. Use `--porcelain` command line option to get the
output in [CSV](https://en.wikipedia.org/wiki/Comma-separated_values)
(comma-separated values) format instead.

# Install

To compile from source, use `cargo install jja`. This requires the [Rust
Toolchain](https://rustup.rs/) to be installed. In addition you are going to need
[OpenSSL](https://www.openssl.org/) libraries on
[UNIX](https://en.wikipedia.org/wiki/Unix) systems. Moreover you need
[liburing](https://github.com/axboe/liburing) on [Linux](https://kernel.org). If
you're on a [Linux](https://kernel.org) system older than
[5.1](https://en.wikipedia.org/wiki/Io_uring#History) or you are unable to install
[liburing](https://github.com/axboe/liburing) for another reason, you may disable
the feature by building ___jja___ with `cargo install jja --no-default-features`.

As an alternative, release builds of ___jja___ are hosted on
[chesswob.org](https://chesswob.org/jja) for
[64-bit](https://en.wikipedia.org/wiki/64-bit_computing)
[Linux](https://en.wikipedia.org/wiki/Linux) and
[Windows](https://en.wikipedia.org/wiki/Microsoft_Windows). These versions are
signed by [GnuPG](https://gnupg.org/), using key
[D076A377FB27DE70](https://keybase.io/alip/pgp_keys.asc). To install, acquire the
latest version from [chesswob.org](https://chesswob.org/jja), verify the
[checksum](https://en.wikipedia.org/wiki/Checksum) and the
[GnuPG](https://gnupg.org/)
[signature](https://en.wikipedia.org/wiki/Digital_signature):

```
$> export JJA_VERSION=0.6.1
$> export JJA_FLAVOUR=glibc
$> curl https://keybase.io/alip/pgp_keys.asc | gpg --import
  % Total    % Received % Xferd  Average Speed   Time    Time     Time  Current
                                 Dload  Upload   Total   Spent    Left  Speed
100 13292  100 13292    0     0  13535      0 --:--:-- --:--:-- --:--:-- 26584
gpg: key D076A377FB27DE70: public key "Ali Polatel (Caissa AI) <alip@caissa.ai>" imported
gpg: Total number processed: 1
gpg:               imported: 1
$> for f in jja-${JJA_VERSION}-${JJA_FLAVOUR}.bin{,.sha512sum,.sha512sum.asc}; do wget -q https://chesswob.org/jja/${f}; done
$> gpg --verify jja-${JJA_VERSION}.bin.sha512sum.asc jja-${JJA_VERSION}.bin.sha512sum
gpg: Signature made Sun Mar 19 20:52:41 2023 CET
gpg:                using RSA key 5DF763560390A149AC6C14C7D076A377FB27DE70
gpg: Good signature from "Ali Polatel (Caissa AI) ...
$> sha512sum -c jja-${JJA_VERSION}.bin.sha512sum
jja: OK
$> sudo install -m755 jja-${JJA_VERSION}-${JJA_FLAVOUR}.bin /usr/local/bin
```

Finally you may download the builds of the latest git version via the
[SourceHut build server](https://builds.sr.ht/~alip/jja?). There're three flavours,
`windows`, `linux-glibc`, and `linux-musl`. Simply browse to the latest build and
download the artifact listed on the left. Note: these artifacts are kept for 90
days.

# License

__jja__ is free software: you can redistribute it and/or modify it under the
terms of the __GNU General Public License__ as published by the Free Software
Foundation, either __version 3__ of the License, or (at your option) any later
version.

This program is distributed in the hope that it will be useful, but __WITHOUT
ANY WARRANTY__; without even the implied warranty of __MERCHANTABILITY__ or
__FITNESS FOR A PARTICULAR PURPOSE__.  See the GNU General Public License for
more details.

You should have received a copy of the GNU General Public License along with
this program.  If not, see <http://www.gnu.org/licenses/>.

# Bugs

```
Hey you, out there beyond the wall,
Breaking bottles in the hall,
Can you help me?
```

Report bugs to ___jja___'s bug tracker at <https://todo.sr.ht/~alip/jja/>:
1. Always **be polite**, respectful, and kind:
   <https://css-tricks.com/open-source-etiquette-guidebook/>
2. Keep your final change as **small and neat** as possible:
   <https://tirania.org/blog/archive/2010/Dec-31.html>
3. Attaching poems with the bug report encourages consideration tremendously.

# PGN Book Making

Since version 0.4.0, ___jja___ can make
[PolyGlot](http://hgm.nubati.net/book_format.html) books out of
[PGN](http://www.saremba.de/chessgml/standards/pgn/pgn-complete.htm)
files. This feature is similar to `polyglot make-book` with the following
differences:

1. ___jja___ may directly read compressed
[PGN]http://www.saremba.de/chessgml/standards/pgn/pgn-complete.htm files
.pgn.{bz2,gz,lz4,zst}
2. ___jja___ can process
[PGN]http://www.saremba.de/chessgml/standards/pgn/pgn-complete.htm files bigger
than your system's available memory, by persisting statistics in a temporary
[RocksDB]https://rocksdb.org/ database.
3. ___jja___ scales move weights by default to prevent potential overflows with huge
[PGN]http://www.saremba.de/chessgml/standards/pgn/pgn-complete.htm files bigger
files, use `--no-scale` to disable.
4. ___jja___ can filter moves using *Filter Expressions*, allowing the user to
filter-out unwanted games and create specialised opening books.

## Filter Expressions

The filter expression string should contain filter conditions, which consist of a
tag name, a comparison operator, and a value. The following operators are supported:
- `>`  (greater than)
- `>=` (greater than or equal to)
- `<`  (less than)
- `<=` (less than or equal to)
- `=`  (equal to)
- `!=` (not equal to)
- `=~` (regex match, case insensitive)
- `!~` (negated regex match, case insensitive)

Filter conditions can be combined using the following logical operators:
- `AND` (logical AND)
- `OR`  (logical OR)

Example:

`--filter="Event =~ World AND White =~ Carlsen AND ( Result = 1-0 OR ECO = B33 )"`

Supported tags are Event, Site, Date, UTCDate, Round, Black, White, Result,
BlackElo, WhiteElo, BlackRatingDiff, WhiteRatingDiff, BlackTitle, WhiteTitle, ECO,
Opening, TimeControl, Termination, and ScidFlags.

In addition to these are four special variables, namely, Player, Elo, Title, and
RatingDiff. These variables may be used to match the relevant header from either
one of the sides. E.g the filter:

`--filter="Player =~ Carlsen"`

is functionally equivalent to

`--filter="( White =~ Carlsen OR Black =~ Carlsen )"`

Note: The filtering is designed to be simple and fast. The tokens, including
parantheses are split by whitespace. Quoting values is not allowed. For more
sophisticated filtering needs, use pgn-extract.

### Scid Flags
Scid uses one character flags, `DWBMENPTKQ!?U123456`, for each field where:
- `D` - Deleted
- `W` - White opening
- `B` - Black opening
- `M` - Middlegame
- `E` - Endgame
- `N` - Novelty
- `P` - Pawn structure
- `T` - Tactics
- `K` - Kingside play
- `Q` - Queenside play
- `!` - Brilliancy
- `?` - Blunder
- `U` - User-defined
- `1..6` - Custom flags

It is ill-advised to rely on the order of the characters flags.

Use a regex match if/when you can.

## Tips and Tricks about PGN Book Making

0. The defaults run best on my laptop and in my personal benchmarks on the
   [SourceHut build
   server](https://builds.sr.ht/~alip/jja/commits/main/benchmark.yml),
   they're not universal truth.
1. ___jja___ processes input PGN files in parallel. You can use this to your
   advantage by giving many split PGNs as input to increase parallelism and
   performance.
2. Increasing batch size is good as long as you have constant memory usage. When
   threads can't keep up, you'll get increased memory usage so that's the point you
   really know your limit.
3. Try increasing max open files to the point you don't get too many open files
   error from your operating system. You may specify `--max-open-files=-1` to keep
   files always open.
4. Try different compression algorithms for the temporary
   [RocksDB]https://rocksdb.org/ database, or try disabling compression completely
   if you have enough space. ___jja___ writes the temporary
   [RocksDB]https://rocksdb.org/ database in the same directory as the first
   pgn file argument. The default compression algorithm,
   [Lz4]https://lz4.github.io/lz4/, and the default compression level, `4`, are
   aimed at speedy conversion with relatively moderate space usage. If you run out
   of space during conversion, try to use an algorithm like
   [Zstd]https://github.com/facebook/zstd with an "ultra" level, ie. a level
   greater or equal to `20`.
5. Use filters which are processed during PGN traversal when possible. Due to the
   fact that these filters are matched before writing the game data to the temporary
   database, when use wisely, they may have a vast on impact space and memory costs
   and therefore improve overall performance. These filters are `--filter=<expr>`,
   `--max-ply=<ply>`, `--min-pieces=<piece-count>`, `--only-white`, and
   `--only-black`.

# Acknowledgements

Thanks to ___Steinar H. Gunderson___, for publishing the [CTG
Specification](https://web.archive.org/web/20210129162445/https://rybkaforum.net/cgi-bin/rybkaforum/topic_show.pl?tid=2319),
and authoring the [remoteglot](http://git.sesse.net/?p=remoteglot;a=summary) tool:
The [CTG](https://www.chessprogramming.org/CTG) probing code in ___jja___ is very
directly ported from their
[C](https://en.wikipedia.org%2Fwiki%2FC_%28programming_language%29%) probing code,
and the specification has been an enormous help in clearing up various rough edges.
Thanks to ___Fabien Letouzey___, the author of the original
[PolyGlot](http://hgm.nubati.net/book_format.html) software: The
[PolyGlot](http://hgm.nubati.net/book_format.html) probing, book making and merging
code in ___jja___ is mostly ported from their respective
[C](https://en.wikipedia.org%2Fwiki%2FC_%28programming_language%29%) code. Thanks to
___Michel Van den Bergh___, the author of pg\_utils, a collection of tools to
interact with [PolyGlot](http://hgm.nubati.net/book_format.html) opening books: The
[PolyGlot](http://hgm.nubati.net/book_format.html) book editing code of ___jja___
uses many ideas and code excerpts from pg\_utils. Thanks to ___Peter Österlund___,
the author of [DroidFish](https://www.chessprogramming.org/DroidFish): The
[ABK](https://www.chessprogramming.org/ABK) opening book interface code in ___jja___
makes use of ideas and code excerpts from
[DroidFish](https://www.chessprogramming.org/DroidFish). Thanks to ___Jens
Nissen___, the author of [ChessX](https://chessx.sourceforge.io/): The
[CTG](https://www.chessprogramming.org/CTG) and
[ABK](https://www.chessprogramming.org/ABK) probing codes in ___jja___ use ideas and
code excerpts from [ChessX](https://chessx.sourceforge.io/). Thanks to
[LiChess](https://lichess.org), the best chess website on the planet. The `quote`
command of ___jja___ has a selection of quotes imported from the
[LiChess](https://lichess.org) codebase. Thanks to ___Shane Hudson___, the author of
[Scid vs. PC](https://scidvspc.sourceforge.net/): The `jja eco` command uses the
[ECO](https://www.chessprogramming.org/ECO) classification which has been done by
the [Scid](https://scidvspc.sourceforge.net/) project. In addition, the
[PolyGlot](http://hgm.nubati.net/book_format.html) editing code of ___jja___ uses
ideas and code from [Scid](https://scidvspc.sourceforge.net/). Thanks to ___Marcus
Bufett___, the author of
[chess-tactics-cli](https://github.com/marcusbuffett/chess-tactics-cli): The
chessboard displaying code in [PolyGlot](http://hgm.nubati.net/book_format.html) and
[ABK](https://www.chessprogramming.org/ABK) edit screens is borrowed from
[chess-tactics-cli](https://github.com/marcusbuffett/chess-tactics-cli).

# ChangeLog

## 0.7.1

- fix build with `i18n` feature disabled.
- enable the lint `#![deny(clippy::cast_lossless)]` for jja library and fix
  offending code.

## 0.7.0

- enable the lint `#![deny(clippy::cast_precision_loss)]` for jja library and fix
  offending code.
- edit learned to use buffered I/O, rather than direct I/O during PGN conversion to
  improve efficiency, especially with huge opening books.
- edit learned to display a progress bar during PGN conversion.
- CTG interface has seen many **breaking change**s to make the interface leaner,
  with regards to the recent mmap changes. As a result, the function
  `jja::ctg::find_piece` returns an `Option<i32>` rather than a `Result<i32,
  std::io::Error>`, the functions `jja::ctgbook::CtgBook::extract_all{,2}` return a
  `CtgTree` rather than a `Result<CtgTree, std::io::Error>`, and the function
  `jja::ctgbook::CtgBook::lookup_moves` returns an `Option<Vec<CtgEntry>>` rather
  than a `Result<Option<Vec<CtgEntry>>, std:io::Error>`.
- edit now uses a much more efficient PGN converter implementation which does not
  load the whole tree of variations in memory. This makes it possible to convert
  huge opening book files into PGN without running out of system memory. Moreover,
  the PGN converter now respects the `-f, --fen`, and `-p, --pgn` arguments of `jja
  edit` so it is possible to convert a subtree of the opening book into PGN. See the
  respective issues [#14]https://todo.sr.ht/~alip/jja/14, and
  [#16]https://todo.sr.ht/~alip/jja/16 for more details. This change deprecates
  the function `jja::chess::pgn_from_tree`, it is recommended to use the new
  `write_pgn` function of the respective opening book file.
- `jja::polyglotbook::PolyGlotBook` implementation has seen many improvements to be
  a leaner interface. `PolyGlotBook::get{,_key}` functions no longer panic on
  invalid indexes, rather return an `Option<BookEntry>` rather than a `BookEntry`.
  This implementation avoids an ugly hack to map a dummy anonymous memory region for
  PolyGlot books with zero-size to allow creating PolyGlot books from scratch (e.g:
  `touch new-file.bin && jja edit -i new-file.bin`). As these functions are public,
  this is a **breaking change**.
- merge now supports merging BrainLearn experience files together. The option
  `--weight-cutoff` has been renamed to `--cutoff` and now supports filtering out by
  min depth in BrainLearn experience files.
- edit now supports editing BrainLearn experience files. In-place editing of such
  files is also supported.
- reduce minimum supported Rust version (MSRV) from `1.70` to `1.64` for portability.
- bring back the dependency on `is_terminal` crate rather than depending on >=rust-1.70.
- downgrade `pgn-reader` crate from `0.25` to `0.24`.
- downgrade `shakmaty` crate from `0.26` to `0.25`.
- upgrade `regex` crate from `1.8` to `1.9`.
- upgrade `smallvec` crate from `1.10` to `1.11`.
- The `progress_bar` member of `jja::obkbook::ObkBook` has been removed, in return
  the public functions `jja::obkbook::ObkBook::{tree,traverse_tree}` require an
  optional reference to a progress bar now. This avoids a needless clone of the
  progress bar and it is a **breaking change**. Moreover, the function
  `jja::obkbook::ObkBook::read_moves` has been renamed to `load` which is again a
  **breaking change**.
- find now memory maps the OBK opening book files rather than reading the whole file
  into memory at once for efficiency. This caused a change in public function
  signatures of `jja::obkbook::ObkBook::{read_moves,traverse_tree,tree}` which is a
  **breaking change**.
- dump learned `binary` format to dump PGNs in PostgreSQL binary output format. This
  brings in a dependency on crate `pgcopy`.
- find now memory maps the PolyGlot opening book files rather than maintaining a
  `BufReader<File>` handle to them. This removes the `book` public member of
  `jja::polyglotbook::PolyGlotBook`, and changes signatures of public functions
  `jja::polyglotbook::PolyGlotBook::{lookup_moves,tree}` which is a **breaking
  change**. This also changes names of the public functions
  `jja::polyglotbook::PolyGlotBook::{find_book_key,read_book_entry,read_book_key}`
  to `jja::polyglotbook::PolyGlotBook::{find,get,get_key}` respectively which is
  again a **breaking change**.  Moreover `jja::polyglotbook::PolyGlotBook`'s default
  iterator implementation has been changed to iterate over single book entries. New
  function introduced `jja::polyglotbook::PolyGlotBook::into_iter_grouped()` may be
  used to iterate over entries grouped by key.
- check for whether standard error is a TTY, rather than standard output when
  displaying progress bars. This allows commands such as dump to display progress
  bars during execution.
- optimize the stockfish hash function implementation, making it almost double as
  fast.
- hash learned `-B`, `--benchmark`, and `-I`, `--benchmark-iterations` to benchmark
  Stockfish and Zobrist hash functions. This brings in a dependency on
  `benchmarking` crate.
- Avoid translating CSV headers of ABK entries in `jja find` output. Note, this
  change is for CSV output only which is printed when the output is not a TTY or the
  option `--porcelain=csv` is given.
- The `progress_bar` member of `jja::ctgbook::CtgBook` has been removed, in return
  the public functions `jja::ctgbook::CtgBook::extract_all{,2}` require an optional
  reference to a progress bar now. This avoids a needless clone of the progress bar
  and it is a **breaking change**.
- avoid needless conversion to and from EPD in `jja::ctgbook::CtgBook` functions
  improving efficiency. The function `jja::ctgbook::CtgBook::lookup_moves` now
  accepts a `&dyn shakmaty::Position` rather than an EPD string which is a
  **breaking change**.
- drop unused public functions `jja::ctg::find_piece`, `jja::ctg::decode_fen_board`,
  `jja::ctg::invert_board`, `jja::ctg::needs_flipping`, `jja::ctg::flip_board`
  which is a **breaking change**.
- quote now also accepts a search term as a case-insensitive regular expression as
  well as a quote index.
- The `progress_bar` member of `jja::abkbook::AbkBook` has been removed, in return
  many public functions of `jja::abkbook::AbkBook` require an optional reference to
  a progress bar now. This avoids a needless clone of the progress bar and it is a
  **breaking change**.
- find now memory maps the ABK opening book files rather than reading the whole file
  into memory at once for efficiency. `jja::abkbook::AbkBook` no longer implements
  `Clone` which is a **breaking change**.
- edit now uses buffered writing when converting books to the ABK opening book
  format which improves efficiency. The function `jja::AbkBook::write_file` has been
  changed to take an argument a `BufWriter<W: Seek + Write>` rather than a `File`
  which is a **breaking change**.
- the positions table, `p`, in jja-0 databases now have an index on id, `p_idx` so
  as to be able to query for Zobrist hash collisions more efficiently.
- info prints file type in uppercase rather than lowercase now.
- edit learned to print Polyglot book information after a successful CTG conversion,
  like we already do for CTG to ABK conversions.
- CTG gained support for the full range of NAGs, `$0 - $255`, thus edit no longer
  panics when stumbling upon a previously unsupported NAG. Note, only the move
  assessments, `$1 - $9`, are used in Polyglot weight and ABK priority calculation
  during edit. Other NAGs are merely used for display for the find subcommand. Note,
  this changes the `jja::ctg::Nag` public type, and hence is a **breaking change**.
- dump learned `-f=<FORMAT>`, `--format=<FORMAT>` argument to choose the dump format
  of PGN dumps. This option has no effect on non-PGN dumps. The function
  `jja::pgn::pgn2csv` has been renamed to `jja::pgn::pgn_dump` which is a
  **breaking change**.
- prioritize CTG coloured move recommendations over Numeric Annotation Glpyhs (NAGs)
  in ABK priority calculation. edit learned three command-line parameters which are
  `--color-priority-green`, `--color-priority-blue`, and `--color-priority-red`.
  Their default values are `9`, `5`, and `1` respectively.
- make learned `--min-wins` which can be used to filter moves by their win count.
  The default value is `0` which has no effect.
- prioritize CTG coloured recommendations over Numeric Annotation Glpyhs (NAGs) in
  Polyglot weight calculation. edit learned three command-line parameters which are
  `--color-weight-green`, `--color-weight-blue`, and `--color-weight-red`. Their
  default values are `10000`, `1000`, and `1` respectively.
- CTG numerical annotation glyph `7` which is a forced move is now supported.
  Previously, we mistakenly used `8` which was an only move, not a forced move.
  Although the distinction is not really clear, we've implemented `ctg::Nag::Only`
  in addition to `ctg::Nag::Forced`, and the corresponding command-line flags for
  jja edit are `--nag-weight-only=<WEIGHT>`, and `--nag-priority-only=<PRIO>`. Both
  defaults are identical to the default values of `--nag-weight-forced`, and
  `--nag-priority-forced`.
- **important fix** `jja::CtgBook::search_position` function from mistakenly missing
  some positions causing some huge CTG books, larger than ~2,5-3G, to be seen as
  having 0 positions during edit, or causing position lookups to fail during find. A
  multiplication overflow in `jja::CtgBook::read_page` function is also fixed.
- make informs user about the `--{win,draw,loss}-factor` values during filtering as
  they're also important in determining weight, and preserval of entries.
- fix a bug in make where negative values in `--draw-factor`, and `--loss-factor`
  was not counted as deficits.
- dump and restore learned an experimental PGN to an array of chess position Zobrist
  keys and setups in streaming JSON. restore can save these dump into a sqlite3
  database with `.jja-0` extension. The format of this file is **experimental**, and
  is subject to change. Once this format is stable, the extension `.jja-1` is going
  to be used. We're using this format currently only to detect Zobrist hash
  collisions.
- use the [`XorShift`]https://www.jstatsoft.org/v08/i14/paper random number
  generator to randomly pick moves during random playouts using the play command. This
  algorithm is cryptographically insecure but is very fast. See the benchmark
  [here]https://git.sr.ht/~alip/jja/commit/b2659d05c883f33d03d3e5c717759580b528919b
- **breaking change**: `jja::quote::print_quote` now requires a second argument which
  is a boolean which specifies whether the output should be formatted with ANSI
  colour codes or not. By default, when the standard output is not a TTY,
  `print_quote` will now print quote, and author information without styling.
- fix PGN dumps to produce proper JSON arrays, previously the array markers `[]`
  were erroneously not printed out.

## 0.6.1

- fix deserializing of promotions of BrainLearn experience file entries.
- edit learned to convert BrainLearn experience files to PGN. Note a lot of memory
  may be required in converting these files. This is planned to be improved in the
  future.
- fix deserializing of castling moves of BrainLearn experience file entries.
- fix `jja restore` which incorrectly wrote BrainLearn entries in big-endian rather
  than little-endian.
- fix sorting of BrainLearn entries in `jja find` tree output.
- upgrade `rust-embed` crate from `6.7` to `6.8`.
- upgrade `num_cpus` crate from `1.15` to `1.16`.
- find can now correctly query BrainLearn experience files using Stockfish
  compatible Zobrist hashes.
- hash learned `-S, --stockfish` to generate Stockfish compatible Zobrist hashes.
- New public module `jja::stockfish`, and new public function
  `jja::stockfish::stockfish_hash` to generate Stockfish compatible Zobrist hashes.
  Before calling this function `jja::stokfish::zobrist::init` function must be
  called once to compute hashtables used in Zobrist hashing.
- New public functions `jja::chess::{de,}serialize_chess` to serialize/deserialize a
  `shakmaty::Chess` instance to/from an array of 5 unsigned 64-bit numbers. `jja
  dump` uses this functionality in PGN dumps.

## 0.6.0

- `jja::pgnfilt::Operator` and `jja::pgnfilt::LogicalOperator` implements `Eq` as
  well as `PartialEq` now.
- new `restore` command to accompany the `dump` command which restores JSON
  serialized PolyGlot or BrainLearn file entries into the given output file.
- fix `SIGPIPE` handling on UNIX systems so that `jja` does not panic when the
  output is piped to another program such as a pager.
- find learned `-z <HASH>, --hash=<HASH>` to query PolyGlot opening books and
  BrainLearn experience files by Zobrist hash.
- readonly support for BrainLearn experience file format. The subcommands `info`,
  `dump`, and `find` are able to handle files in BrainLearn experience file format
  with the extension `.exp`.
- improve polyglot key lookup by reading only the key rather than the whole entry
  from file. The new public function `polyglotbook::PolyGlotBook::read_book_key` is
  used for that.
- new `dump` command to dump the full contents of a PolyGlot opening book or a PGN
  file. The dump format of the PolyGlot opening book is JSON, whereas for PGN files
  this is CSV.
- **breaking change**: `polyglotbook::PolyglotBook::lookup_moves` has been changed
  to take a zobrist hash of a chess position as an argument rather than the
  `shakmaty::Chess` position itself.
- hash learned `--signed` to print Zobrist hashes as signed decimal numbers.
- new module `jja::file` which exports utilities for binary file i/o.
- **breaking change**: `jja::polyglot::entry_{from,to}_file` have been renamed to
`jja::polyglot::bin_entry_{from,to}_file`.
- use the [`XorShift`]https://www.jstatsoft.org/v08/i14/paper random number
  generator to randomly pick moves during book matches using the match command. This
  algorithm is cryptographically insecure but is very fast.
- fix match command from panicing on certain cases when there is a book lookup miss.
- Print more detailed build information on long version, `--version` output.
- Memory map CTG files to speed up random access in return for increased memory
  usage. This brings a new dependency upon the crate `memmap`.
- use buffered read/write in interacting with PolyGlot books which reduces the
  read/write system calls by a huge margin and thereby improves performance. The
  `book` member of `PolyGlotBook` is now a `BufReader<File>` rather than a `File`
  which is a **breaking change**.
- **important fix** for calculating PolyGlot compatible Zobrist hashes wrt.
en-passant legality. In PolyGlot format, en-passant moves are only pseudo-legal
whereas previously the `jja::hash::zobrist_hash` function mistakenly checked
for full legality.
- **breaking change**: new type `ctgbook::CtgTree` which holds the new return value
  of the functions `CtgBook::extract_all`, and `CtgBook::extract_all2`. The `tree`
  element of `CtgEntry` which were used by these functions has also been dropped and
  the functions have been implemented in a much more performant way using
  considerably less memory. As a result, most ctg to abk/polyglot conversions are
  almost double as fast.
- **breaking change**: `ctgbook::CtgEntry` member uci's type has been changed from
  `String` to `shakmaty::uci::Uci`, and the `nags` member has been renamed to `nag`
  and its type has been changed from `Option<String>` to `Option<Nag>`.
- ctg move comment entries were parsed and silently discarded, they're no longer
  parsed. Moreover, `ctgbook::CtgEntry` no longer has a `comment` member which is a
  **breaking change**.
- **breaking change**: `ctg::colored_uci` function now accepts a
  `shakmaty::uci::Uci` rather than a UCI string. The order of the function arguments
  is also changed.
- ctg: new type `Ctg::Nag` to abstract CTG NAG (Numeric Annotation Glyph) entries
- obk entries with zero weight, are now assigned weight `1` during PolyGlot
  conversion to prevent skipping these entries. We plan to make this
  user-configurable in the future.
- use the standard `IsTerminal` trait, and drop the dependency on `is_terminal`
  crate
- use the standard `to_le_bytes()`, `to_be_bytes()`, and drop the dependency on
  `byteorder` crate
- hash learned `-e=<MODE>`, `--enpassant-mode=<MODE>` to select en-passant mode when
  to include the en-passant square in Zobrist hash calculation.
- **important fix** for encoding castling moves during book conversions to the
  PolyGlot format. See details in the [respective issue]https://todo.sr.ht/~alip/jja/13
- **breaking change**: `polyglot.from_uci` now expects a `bool` argument
  to correctly encode castling positions by determining whether the king to move is
  on their starting square.
- bump minimum supported Rust version (MSRV) from `1.64` to `1.70` due to `shakmaty` bump
- drop dependency on the unmaintained and insecure `chrono` crate
- `PolyGlotBook` has two new public functions: `find_book_key`, and
  `read_book_entry`
- hash learned `-x`, `--hex` to print hash as hexadecimal rather than decimal
- upgrade `pgn-reader` crate from `0.24` to `0.25`
- upgrade `shakmaty` crate from `0.25` to `0.26`
- optimize various ctg functions
- info learned to print the total number of positions for `ctg` opening books
- **breaking change**: `CtgBook::num_entries` has been renamed to `total_positions`.
- **breaking change**: `CtgBook::total_pages` function accepts a reference to
  `self`, rather than consuming `self`. The return type is `usize` now.
- **breaking change**: Drop `close` functions of `CtgBook` and `PolyGlotBook`,
  improve `CtgBook` to close the cto file immediately after open, and keep a `File`,
  rather than an `Option<File>` in `CtgBook`. The now unused `path` element of
  `CtgBook` is also dropped.
- find: simplify & optimize polyglot entry lookup
- display license, version and author information in `--help` output
- set minimum supported Rust version (MSRV) to `1.64` as determined by `cargo-msrv`
- fix a bug which caused key and epd to be displayed incorrectly in editor screen
- edit: implement `--rescale` for PolyGlot books. When specified, edit will rescale
  weights of all entries in the book, rather than a single entry. This is useful to
  quickly correct/optimize PolyGlot books which were generated without weight scaling.
- fix build with `i18n` feature disabled
- upgrade `tempfile` crate from `3.5` to `3.6`
- upgrade `rust-embed` crate from `6.6` to `6.7`
- upgrade `once_cell` crate from `1.17` to `1.18`
- upgrade `ctrlc` crate from `3.3` to `3.4`
- **important fix** CTG to Polyglot weight conversion which caused all entries in
  CTG books with missing performance information to be skipped from the output
  PolyGlot book. Read more about it in the [respective
  issue](https://todo.sr.ht/~alip/jja/12).
- make learned `-H`, `--hashcode` argument to skip duplicate games based on the
  **HashCode** PGN tag. PGN files may be tagged using `pgn-extract --addhashcode`

## 0.5.0

- **important fix, for encoding of castling in polyglot books**. Previously, we
  encoded castling as e1g1, e1c1, e8g8, and e8c8, whereas the correct encoding is
  e1h1, e1a1, e8h8, and e8a8.
- when printing version with `--version`, prefer `git` version over package version
  for git builds
- upgrade `clap` crate from `4.2` to `4.3`
- upgrade `ctrlc` crate from `3.2` to `3.3`
- upgrade `rocksdb` crate to `0.21.0` which bundles RocksDB-8.1.1
- document all the public code and enable the lint `#[deny(missing_docs)]`
- edit learned to export opening books in PGN format, use an output file with `pgn`
  extension to export an opening book to a PGN
- add back the build dependency upon the [built]https://crates.io/crates/built crate
- find learned `-l <max-ply>`, `--line=<max-ply>` to display lines from the opening
  book as a table of opening variations reverse-sorted by cumulative weight
- make `--min-score` now accepts floating point values as argument rather than an
  unsigned 64-bit integer
- make learned `--win-factor`, `--draw-factor`, and `--loss-factor` to specify
  respective factors during score calculation, the defaults, `2`, `1`, and `0`
  respectively, resembles the original `polyglot` tool
- find `--tree=<max-ply>` no longer panics on broken pipe, so it's more convenient
  to use with a pager
- make learned `-p, --min-pieces` to specify the minimum number of pieces on the
  board for a position to be included in the book, defaults to `8`
- merge learned `-c, --weight-cutoff` to specify the minimum weight of entries to be
  included in the book
- merge learned about merge strategies `avg` and `wavg` to merge using average
  weight or weighted average weight respectively; the weighted averages should be
  specified for `wavg` using `-w, --weight1`, and `-W, --weight2`
- in-place editing of polyglot files allows editing empty books which makes it
  practical to create polyglot book from scratch
- merge learned about merge strategies `max`, `min`, `ours`, and `sum`, default is
  `sum` which adds together move weights, `max` picks the one with the maximum
  weight, `min` picks up the one with the minimum weight, and `ours` always picks
  the entries from the first book
- translate the README to German language
- turn i18n support into a feature which defaults to on. For static linking this
  must be disabled as embed makes use of proc macros.
- drop depedency on unused [libc]https://crates.io/crates/libc crate

## 0.4.1

- fix [docs.rs]https://docs.rs/jja build
- update [shakmaty]https://crates.io/crates/shakmaty, and
  [pgn-reader]https://crates.io/crates/pgn-reader creates
- add initial translation to Turkish language
- add initial translation to German language
- use gettext for i18n

## 0.4.0

- replace [atty]https://crates.io/crates/atty with the better maintained
  [is-terminal]https://crates.io/crates/is-terminal crate.
- replace [default-editor]https://crates.io/crates/default-editor with the more
  advanced [dialoguer]https://crates.io/crates/dialoguer crate.
- replace [colored]https://crates.io/crates/colored with the more portable
  [console]https://crates.io/crates/console crate.
- match learned `-i`, `--irreversible` to prefer irreversible moves during random
  playouts. Pawn moves, captures, moves that destroy castling rights, and moves that
  cede en-passant are irreversible.
- replace the [progress_bar]https://crates-io/crates/progress_bar
  crate with the more portable and advanced
  [indicatif]https://crates.io/creates/indicatif crate.
- make learned `-B <games>`, `--batch-size=<games>` to determine write batch size in
  number of games
- Linux builds require [liburing]https://github.com/axboe/liburing to be installed
  by default. This may be disabled using `--no-default-features` on installation.
- make learned about `ScidFlags`, a set of character flags used by the
  [Scid]https://scidvspc.sourceforge.net/ software, which may be used in filter
  expressions
- make learned `--debug` flag to print information on matching filter expressions
- quote learned many more chess quotes which were imported from goodreads.com
- make learned `--max-open-files=<int>` to specify the maximum number of open files
  per thread opened by the temporary [RocksDB]https://rocksdb.org database.
- match command is now multithreaded and uses as many threads as there're cpus on
  the machine by default. Use `-T=<threads>`, `--threads=<threads>` or the
  `JJA_NPROC` environment variable to override
- fix match to alternate colour between books during random playouts
- quote accepts an optional index argument to print the quote at the specified
  index, rather than a random quote
- quote learned many more chess quotes which were imported from archive.org,
  goodreads.com and nitter.net
- make filter learned special variables `Player`, `Elo`, `Title`, and `RatingDiff`
  to match the relevant field from either colour
- make learned `--filter=<expression>` to filter PGN games by headers. The filtering
  is designed to be simple and fast. The tokens, including parantheses are split by
  whitespace. Quoting variables is not allowed. See `jja make --help` for more
  information on Filter Expressions
- find learned `-t <max-ply>`, `--tree=<max-ply>` to display lines from the opening
  book as a tree using the nice [termtree]https://crates.io/crates/termtree crate
- drop the build dependency upon the [built]https://crates.io/crates/built crate
- drop the unused dependency on the
  [ħyphenation]https://crates.io/crates/hyphenation crate
- replace [csv]https://crates.io/crates/csv crate with the more
  lightweight [quick-csv]https://crates.io/crates/quick-csv crate
- replace [cli-table]https://crates.io/crates/cli-table crate with the more
  lightweight [prettytable-rs]https://crates.io/crates/prettytable-rs crate
- use [pgn-reader]https://crates.io/crates/pgn-reader crate instead of
  [pgnparse]https://crates.io/crates/pgnparse for parsing the `--pgn`
  commandline option
- strip off the unneeded
  [cli-table-derive]https://crates.io/crates/cli-table-derive dependency
- fix hash subcommand option parsing causing panic
- ctg find prioritises wins & draws over performance
- make learned to preserve moves with null moves in the book with `-0`, `--null`
- make learned to avoid scaling weights using `--no-scale`
- make learned to configure compression for the rocksdb database using
  `--compression={none,bzip2,lz4,lz4hc,snappy,zlib,zstd}`,`--compression-level=<level>`,
  defaults to `lz4`, level 4
- use a temporary [RocksDB]https://crates.io/crates/rocksdb database during pgn book
  make so as to better make use of memory, this reduces the performance a little,
  however in return makes importing huge PGN files possible.
- new make command to make polyglot books out of pgn files, runs multithreaded
  with as many threads as the cpu number of the system by default, use `-T`, `--threads`
  to override, works transparently with compressed PGN files (zstd, bzip2, gunzip, lz4)

## 0.3.2

- edit writes the name of the user and jja's version as comment to Arena opening
  book metadata on ChessBase to Arena opening book conversion, override with
  --author, --comment
- edit displays a unicode chess board in edit tempfile
- fix book traversal on Arena opening book to PolyGlot conversion
- enable ansi colors when running on windows terminal
- drop unixisms, cross-compiles for windows
- fix yet another bug with castle decoding on polyglot read/query
- edit no longer tries to spawn the default editor if standard output is not a TTY
- fix find and edit for Arena opening book reading, move selection is on par with
  the Arena GUI
- new hash command to calculate the Zobrist hash of the given position
- fix infinite loop while converting some big Arena opening book files
- improve hashing performance by avoiding double hashing using a hasher builder for
  Zobrist hashes
- improve hashing performance using
  [shakmaty]https://docs.rs/shakmaty/latest/shakmaty/zobrist/trait.ZobristHash.html#tymethod.zobrist_hash
  crate's [Zobrist Hashing]https://www.chessprogramming.org/Zobrist_Hashing
  implementation rather than the internal one.

## 0.3.1

- edit learned to calculate & write ABK header game statistics fields
- edit learned to convert CTG book files to ABK book files
- edit learned --author and --comment to specify metadata for Arena opening books
- edit can edit Arena opening book (abk) files in-place with -i, --in-place=SUFFIX
- support for writing Arena (abk) opening books
- open learned to wrap long ECO opening lines into multiple lines
- find no longer panics on some abk books with entries having invalid uci
- edit takes move priority into account for weight on abk to bin conversion
- match learned --move-selection={best\_move,uniform\_random,weighted\_random} to
  pick move selection algoritm for book moves
- fix castle decoding on polyglot read/query
- fix error return when no positions found in abk, obk and ctg find
- fix promotion handling in ctg edit
- many improvements to ctg find (move coloring & sorting, average statistics)
- new merge command to merge two PolyGlot opening books
- new match command to arrange book matches with random playouts

## 0.3.0

- refactor code to unify various opening book reader interfaces
- support obk version 1 as well as 2 (ChessMaster books with and without notes)
- support for reading obk (ChessMaster) books and converting them to polyglot books
- do not display progress bar if standard output is not a TTY
- support for reading abk (Arena) books and converting them to polyglot books
- new play command can be used to make random playouts using opening books

## 0.2.1

- weight conversion in ctg to polyglot edit can be tuned with
  --nag-weight-{good,mistake,hard,blunder,interesting,dubious,forced}=<WEIGHT>
- edit learned --no-scale to avoid scaling weights globally to fit into 16 bits
- the code is now relatively well documented
- edit --in-place now properly deletes the output temp file on interrupt
- edit filters out moves with zero weights, use -0, --null to preserve them

## 0.2.0

- edit window lists position info (key, epd, legal moves) as comment
- edit no longer silently discards illegal moves
- edit can edit PolyGlot files in-place with -i, --in-place=SUFFIX
- edit can convert CTG opening books into PolyGlot opening books
- default to start position when no --fen or --pgn is given for edit and find
- info prints number of total pages in CTG books

## 0.1.1

- new positions can be added to polyglot files
- many bugs fixed with polyglot edit
- quote command added to print a random chess quote
- open command added to query ECO classification

## 0.1.0

- edit polyglot files, only editing present positions work
- read polyglot files
- read ctg files