Routine to identify duplicates in fuzzy data (such as, for example, bibliographical records)
- duplAway.py is the main working version; all the descriptions below are applicable to this script
- diplAway_Mod.py is a working modification that runs in a very similar manner, but calculates ration differently; it compares the results of two fuzzywuzzy algorithms and treats a pair as a match when the results fall within these parameters; essentially, it checks requires that Alg 4 has a very high value (hardcoded into the function; can be changed directly in the script; 95-100%, seem to work best, 95% may be even a bit low—many false positives); then it checks whether Alg 1 gives a rather high value as well (50%, also hardcoded in the function). This combination seems to produce very good results with “false positives” appearing only in the case of very closely connected books (such as continuations and abridgements). This stript does not use certain parameters from the config file, namely
alg
andthr
. - duplAway_Simple.py is another modification.
- all the settings are inside the script, which is particularly convenient if one has to run the script from IDLE (relevant for Windows, where Unicode characters cannot be displayed in the CommandLine).
- The script merges the results (result files now have time stamps, so overwriting should be impossible). When results are merged, old files are removed, so there will not be any clutter.
- The script also checks all four algorithms, so you can play with parameters in the settings. I would start with alg = "[50,0,0,95]" (i.e., alg1=50, alg4=95, alg2 and alg4 do not matter).
- works on Python 3 (will not run on Windows command line if your data is in Unicode; will work via IDLE, but some minor modification to the script are necessary)
- fuzzywuzzy library must be installed
Parameters are as follows:
- file : the name of a data file for processing (must be in the same folder as the script)
- thr[eshold] : the lowest comparison ratio to consider
- alg[orithm] : chooses one of the 4 fuzzywuzzy routines
- [1]
fuzz.ratio
- [2]
fuzz.partial_ratio
- [3]
fuzz.token_sort_ratio
- [4]
fuzz.token_set_ratio
- Detailed descriptions of these algorithms.
- len[gth] : the length (in words) of the first comparison string; works better when comparing longer strings with shorter ones
- sav[e Mode] :
all
, orman
all
- saves all results; advantage: works faster on restart; disadvantage: creates really large result files (duplicID)- a) if you don't mind extra space, restarting is much faster with
all
, since the script does have to calculate anew - b) re-running with
man
will remove all irrelevant results, keeping only manually tagged ones (y
,n
,m
, and other supported choices)
- a) if you don't mind extra space, restarting is much faster with
- id : column with unique identifiers --- required column!
- comp[are] : columns to use for comparison (separated by commas); for example, [author] and [title] for identifying bibliographical records for the same works
- disp[lay] : columns to use for showing during the decision stage (separated by commas); for example, [author], [title], [editor], etc. for bibliographical records
- verb[al]: a word that will be added as an infix into a file with results; makes it easier to understand what kind of data the file contains; for example, the same dataset can be used for different purposes, and while it will be reflected in the infix, where the the column numbers are given, it is not very readable. Adding a simple verbal marker should be helpful (so,
verb=Authors
> analyzing Authors' names;verb=Book
--- book titles, etc.)
NB: arguments (i.e., everything after python3 duplAway.py
) can be given in any order; do not change the name of the script, since it will break the argument analysis logic
$ python3 duplAway.py file=AraCorpus_NewBiblio_TriCollection.tsv thr=90 alg=4 len=5 id=[4] comp=[4] disp=[4,6] verb=Authors sav=all
- the script will analyze file
AraCorpus_NewBiblio_TriCollection.tsv
, - using algorithm 4 and showing only results with 90% likelihood
- comparing longer strings with shorter ones seems to work better, hence the lower limit is set to 5 [words]; you may want to experiment with this parameter (increase it, if you get stuck);
- it will use column 4 for ids
- it will also use values from col 4 for comparison;
- it will show also data from cols 4 and 6, which should help to make the finals decision whether this is a true of false match
- it will add 'Authors' to the name of the file (for readability) this essentially means that we are comparing names of authors
- it will also save ALL results on exit
$ python3 duplAway.py file=AraCorpus_AuthorNames.tsv thr=90 alg=4 len=5 id=[1] comp=[2] disp=[2,3,4] verb=Authors sav=all
$ python3 duplAway.py file=AraCorpus_NewBiblio_TriCollection.tsv thr=90 alg=4 len=5 id=[1] comp=[4,8] disp=[1,3,4,6,8,13] verb=Books sav=all
Using a config file is a more convenient option: all parameters are stored in a file > shorter command
Example for Routine 2 (include the following 11 lines into a file > configFile2.txt); everything after '#' is a comment
============================
~duplAway.py # do not change
~file = AraCorpus_NewBiblio_TriCollection.tsv # this is the file you will analyze
~sav = all # variants: all, man[ually tagged]
~thr = 98 # number between 1 and 100 (100 is usually a very close match [depends of the algorithm])
~alg = 1 # one of the 4 fuzzywuzzy routines
~len = 5 # the shortest length of string (in words) to run comparison on
~id = [1] # the number of the column whenre the IDs are
~comp = [4,8] # the numbers of columns strings from which should be compared (divided by commas)
~disp = [1,3,4,6,8,13] # the numbers of columns strings from which should be printed on the screen (divided by commas)
~verb = Books # this is the infix that will be added to the name of the file with results
============================
The command to run the script then becomes:
$ python3 duplAway.py configFile2.txt
fuzzywuzzy
Library has 4 algorithms that have different level of tolerance, so it make sense to start with algorithm 1 and threshold 100, gradually lowering until too many wrong suggestions start to appear; after that, repeating this for algorithms 2, 3, and 4.
- [1]
fuzz.ratio
- [2]
fuzz.partial_ratio
- [3]
fuzz.token_sort_ratio
- [4]
fuzz.token_set_ratio
- Detailed descriptions of these algorithms.
- Using
sav=all
is likely to result in large files, but this will make it faster to restart if you need to; when you think that you are done, you can rerun the script withsav=man
, which will remove all irrelevant results, saving onlyman
ual decisions - The
len=XX
parameter (number of words): the comparison of longer strings with shorter ones gives better results in the workflow, so you may want to up this parameter if the script gets stuck on an inconclusive string. - If there is an option to run the script remotely, it makes sense (1) to generate statistics for all pairs (large files!) and then (2) review the results; to generate stats, set
sav=all
(to save all results) andthr=101
(or anything higher than 100), which will make the script to run until it is done. Then, lower the threshold and run the script again---it will not be re-calculating anything, just running through the list and showing pairs above the threshold. This will work faster, theoretically...