NOTE: This document is a work-in-progress as of 2019-07-12.
This repository documents the 4-gram study conducted in 2013 by the Wordtree Foundation, specifically Duane & Chris Johnson. The study connected the Book of Mormon (1830) and The Late War Between the United States and Great Britain, by Gilbert Hunt (1812). Other books of interest included The First Book of Napoleon and the Apocrypha.
The intent of this document is to make it easy to reproduce and/or extend the results of the study.
The 4-gram study was, at its core, a search algorithm. Using a computer algorithm, the study compared approximately 100,000 books published prior to 1830 with the Book of Mormon (1830 edition).
The authors sought to find a particular kind of similarity between the Book of Mormon & other books. The "similarity function" was the use of contextually rare 4-grams (i.e. "any four words in sequence") to discover possible influence. The central idea was that if there were a book (or books) that showed an overwhelming number of unusual similarities, then that book may have influenced the author of the Book of Mormon (i.e. probably Joseph Smith).
The study used data published by archive.org--English books scanned and OCRed to produce text files suitable to language analytics software.
The following directions should replicate the results of the study, with as much fidelity as possible to the original study. However, given that some data may have changed in the meantime (e.g. archive.org updates its library with new scans of old books, and may improve quality of OCR over time), it's possible to get slightly divergent results.
We assume a Unix-type environment. All of the commands should work on Mac OS X. In Windows, you can install the Ubuntu subsystem and it will work as well.
You should have the following compilers, tools, and utilities:
gcc
(GNU C Compiler)ruby
(Ruby interpreter)git
(Distributed Version Control System)- text tools like
awk
,sed
,sort
,uniq
,xargs
parallel
(GNU Parallel)
Hardware & Network
This documentation assumes hardware specifications as follows:
- 3.5GHz 4-core CPU
- 64 GB RAM Memory
- 1 TB SSD Hard Drive
In addition, a broadband connection to the internet was used with 150 MBit download speed.
Throughout this guide, we will assume a folder structure as follows:
+- study
+- library <-- ~140,000 books from archive.org
+- bomdb <-- various editions of the Book of Mormon
+- ngram-tools <-- fast C-based tools for processing text as ngrams
To algorithmically compare the Book of Mormon & other books, we need a text-only version of the Book of Mormon (scanned images of pages, even in PDF format, won't do). It's surprisingly difficult to find this data on the larger web, so we've built some data repositories and tools to make it easier.
(~1 minute on a 150Mbit network)
We have a repository of pseudo-biblical books (i.e. books that mimic the "Biblical Style" of the King James Version of the bible--if you're interested in this American phenomenon, look up Eran Shalev and American Zion). This repository includes the Book of Mormon.
Download the Book of Mormon - 1830 edition from our Books repository
(~20min on a 150Mbit network)
The bomdb command-line tool (developed by the Wordtree Foundation) has the 1830 original edition, among several other editions. It is very flexible, and allows you to do things like only compare ranges of chapters or verses, or choose which edition of the Book of Mormon you'd like to compare, or exclude verses marked as very similar to the Bible.
Download & install bomdb, then write the entire Book of Mormon, 1830 edition, without verse or chapter numbers to "bom.txt":
$ cd bomdb
$ bundle install
$ bundle exec bin/bomdb show --edition=1830 --no-color --no-verses >bom.txt
(~5min on a 150Mbit network)
- Download the Book of Mormon - unknown edition (TXT) from Project Gutenberg
- Download the Book of Mormon - 1830 edition (PDF) from Library of Congress
- Download the Book of Mormon - 1830 edition (PDF) from Xristian.org
The PDF versions will need some work by you before they can be used as text in the next step.
(~1hr on a 150Mbit network)
We've prepared an archive of books from 1650 to 1829, which you can download here (about 30GB). This archive was downloaded in June of 2019.
These books are text-only, with a small (human-readable) YAML header at the top to identify its title, year, and author(s). They've been stored in sub-folders and sub-sub-folders so that a normal filesystem will handle the large quantity of books.
(~3 days on a 150Mbit network)
The archdown command-line tool (developed by the Wordtree Foundation) is a helpful tool for querying & downloading directly from archive.org. It also arranges downloads in a tree structure so the filesystem can hold lots of books.
Prerequisites: Ruby 2.x
Download and unzip https://github.com/wordtreefoundation/archdown/archive/master.zip (or use git clone [email protected]:wordtreefoundation/archdown.git
). Then install & run archdown
:
$ cd archdown
$ bundle install
$ mkdir ../library
$ bundle exec bin/archdown -l ../library -y 1650-1830
Note that archive.org has since changed its policies to limit the tool to downloading 10,000 files at a time, so it may require some manual restarting, updating the start year each time to avoid redundant downloading.
(~10min on a 150Mbit network)
We've prepared a baseline using the archive of books from 1650 to 1829, which you can download here (about 6GB). This baseline was calculated based on the archive of books downloaded in June of 2019, using the following procedure ("Calculate Baseline").
(~2 days on a 4-core 3.5Ghz machine with 1TB SSD & 64GB of RAM)
Let's count the 4-grams in the Book of Mormon to get started, and then use that as a starting-off point to count 4-grams in other books.
Use ngram-tools to count each 4-gram in the Book of Mormon and store it in a .4grams
file. Build instructions and usage examples are also contained in the ngram-tools readme.
If we just wanted a list of all of the 4-grams in the Book of Mormon, we could execute text-to-ngrams
with the -n4
argument:
$ ./text-to-ngrams -n4 ../bomdb/bom.txt
But what we'd really like is a tally of the 4-grams, so that, for example, all of the occurrences of "it came to pass" are summed one one line, next to the "it came to pass" 4-gram. The following pipe from text-to-ngrams
to tally-lines
will get us what we need:
$ ./text-to-ngrams -n4 ../bomdb/bom.txt | ./tally-lines -c >bom.4grams.tallied
We can check that the bom.4grams.tallied
file contains a list of tallied ngrams, sorted alphabetically:
$ less bom.4grams.tallied
1 a and he spake
1 a bad one for
1 a ball or director
1 a band of christians
1 a band which had
1 a banner upon the
1 a battle commenced between
...
Once you have the tallied 4-grams list, you can also sort by tally rather than alphabetically (i.e. most frequently used phrases in descending order):
$ sort -bgr bom.4grams.tallied | less
1395 it came to pass
1285 came to pass that
1163 and it came to
266 i say unto you
233 to pass that the
168 to pass that when
160 now it came to
...
Next: Counting 4-grams in the entire library
Now that we know how to count 4-grams in one book, let's apply this to counting the 4-grams in all of our downloaded archive.org books:
$ find ../library/ -name "*.md" | shuf >library.toc
$ cat library.toc \
| xargs -n 1 -I {} \
sh -c "./text-to-ngrams -n4 {} | ./tally-lines -c >{}.4grams.tallied"
Breaking it down:
- The
find
command recursively looks for files in a folder that matches a pattern. We need this because filesystems typically can't contain 100,000 files in a single directory so we've created little sub-folders that each contain just a subset of the files. We're looking for ".md" files because thearchdown
command stores the plain-text archive.org books with a yaml header file that is compatible with markdown file format. - The
shuf
command randomly shuffles the order of the text files, and we store it in a "table of contents" file calledlibrary.toc
. - We use the
library.toc
file to pipe the filenames that need processing toxargs
. - The
xargs
command takes the list of files as input and sends each one as an argument to the next command (text-to-ngrams
). We need-n 1
becausetext-to-ngrams
doesn't take multiple files as input (just one file). We also use-I {}
because we need to be able to specify where to "insert" each book filename in the next command. - The
sh
command creates a new shell. This is necessary because we need to pipe several commands together. - Next, we emit each 4-gram in each file via
text-to-ngrams
, then sort and count (viauniq -c
) the ngrams. - Finally, we redirect the output of the whole pipe chain to the original file location (in its original
library
sub-folder) but with the.4grams
suffix appended so that we don't clobber the original text ('.md') file.
For faster processing on a mult-core CPU, use GNU Parallel:
$ find ../library/ -name "*.md" | shuf >library.toc
$ parallel -j4 --progress \
-a library.toc \
./text-to-ngrams -n4 {} \
'|' ./tally-lines -c \
'>' {}.4grams.tallied
The *.md.4grams.tallied
files will contain two columns, the tally column (numbers), followed by the ngrams column (words). The two columns will be separated by a tab character ('\t'):
5 a added providing for
1 a added providing that
1 a added provisions relative
6 a added regulating the
1 a added regulating workmens
18 a added relative to
7 a added requiring the
1 a added restricting the
1 a added to prevent
1 a added under heading
1 a adjacent to erithmo
2 a adopt for the
... etc ...
Next: Process pre-1830s books to calculate a rough Language Model
In order to properly weight the significance of matching ngrams, we need a rough "language model". What this means is that rather than using intuition to tell us whether a sequence of words is rare, we should use math. For instance, "it came to pass" might be in both the Book of Mormon and a sermon preached in 1820, but that doesn't mean they are connected because "it came to pass" is a very common 4-gram.
By "rough" language model, we mean that this isn't a sophisticated language model, it's just a tally of occurrences. In a future study, it would be better to use a proper language model. But in 2013, a tally of 4-grams is what we used, so we'll stick with that here.
Using our ngram-tools, you can fairly easily clean up & tally ngrams (specifically, 4-grams) from any book in the archive.org library that you downloaded.
We'll use a shell-script based "map-reduce" algorithm to vastly speed up calculating a baseline. Specifically, our map-reduce is a "merge + sum". First, let's get a randomly shuffled list of the entire library's books:
$ find ../library/ -name '*.md.4grams.tallied' | shuf >library.toc
Then, let's merge and sum up the ngrams:
$ mkdir output
$ parallel -a library.toc \
--progress -j4 --xargs \
./scripts/merge.sh {} \
'|' ./scripts/sum-consecutive.sh \
'>' ./output/{#}.t
Even after this merge + sum step, there will be over 100 files in the output
folder. We need 1 more reduction step (merge + sum) to get to a single file that represents all of the 4-grams in all of our pre-1830s books:
$ cd output
$ ../scripts/merge.sh *.t | ../scripts/sum-consecutive.sh > baseline.4grams.tallied
The reason we needed to use parallel
to do this in the step prior is because we can't fit all of the ~140,000 book filenames on a single line as arguments to merge.sh
. Plus, it parallelizes the processing and makes things go a bit faster.
Lastly, in order to reduce the size of this file & speed things up in future steps, we remove single-occurrence 4-grams from the file, and gzip it:
$ cat baseline.4grams.tallied \
| awk -F $'\t' '{if ($1 > 1) print $0}' baseline.4grams.tallied \
| gzip -c \
> baseline.4grams.culled.tallied.gz
Note that calculating the baseline may take a long time (i.e. let it run over night).
TODO: steps to reproduce scoring
TODO: steps to reproduce ranking