In the Metadata Quality Assurance Framework as a first step we create some CSV files. The structure of the CSV (in case of Europeana) looks like this:
[record ID],[dataset ID],[data provider ID],score1,score2,...scoreN
[record ID],[dataset ID],[data provider ID],score1,score2,...scoreN
...
where each score belong to a field- or record-level quality metric. There are several measurements (such as frequency and cardinatity of fields, multilinguality, language distribution, metadata anti-pattern detection, uniqueness), each produce such CSV files.
Once we have them we run statistical analyses on these files to get a collection level overview and we present the result in a web UI. From the web UI it is not easy to go back to the records, or at least to figure it out which records have a give value (or range of values). We get feedbacks from the Data Quality Committee members – most recently from Tom Miles (British Library) –, that in some cases it would be necessary to check the records itself e.g. ‘which record contains values in Dutch?’, ‘which records have lots of dc:subject values?’ etc.
What we need is a Solr index which contains all metrics of all records. Since we don’t have one CSV file which contain all information, we have to build the index incrementally. The increment in this case is not vertical (adding new records), but horizontal (adding new fields). How to do that? A quick recipie follows.
I will use R’s famous mtcars
dataset to show the process instead of real Europeana files.
Solr supports reading from CSV as original data, but doesn’t support it as the source or updating
existing records (or document in Solr-speak), so we create two files: one CSV and a JSON:
library(tidyverse)
library(jsonlite)
df <- as.tibble(mtcars)
df %>%
select(1:6) %>%
write_csv("cars1.csv", col_names=T)
json <- df %>%
select(1,7:12) %>%
toJSON()
write(json, file = 'cars2.json')
The first file contains the first 6 features (columns), the second contains the rest. Since it doesn’t have ID, I manually added IDs to the files. The final version of the CSV looks like this:
1,21,6,160,110,3.9
2,21,6,160,110,3.9
3,22.8,4,108,93,3.85
4,21.4,6,258,110,3.08
5,18.7,8,360,175,3.15
6,18.1,6,225,105,2.76
7,14.3,8,360,245,3.21
8,24.4,4,146.7,62,3.69
9,22.8,4,140.8,95,3.92
The final version of JSON looks like this:
[{"id":1,"wt_f":{"set":2.62},"qsec_f":{"set":16.46},"vs_f":{"set":0},"am_f":{"set":1},"gear_f":{"set":4},"carb_f":{"set":4}},
{"id":2,"wt_f":{"set":2.875},"qsec_f":{"set":17.02},"vs_f":{"set":0},"am_f":{"set":1},"gear_f":{"set":4},"carb_f":{"set":4}},
{"id":3,"wt_f":{"set":2.32},"qsec_f":{"set":18.61},"vs_f":{"set":1},"am_f":{"set":1},"gear_f":{"set":4},"carb_f":{"set":1}},
{"id":4,"wt_f":{"set":3.215},"qsec_f":{"set":19.44},"vs_f":{"set":1},"am_f":{"set":0},"gear_f":{"set":3},"carb_f":{"set":1}},
{"id":5,"wt_f":{"set":3.44},"qsec_f":{"set":17.02},"vs_f":{"set":0},"am_f":{"set":0},"gear_f":{"set":3},"carb_f":{"set":2}},
{"id":6,"wt_f":{"set":3.46},"qsec_f":{"set":20.22},"vs_f":{"set":1},"am_f":{"set":0},"gear_f":{"set":3},"carb_f":{"set":1}},
{"id":7,"wt_f":{"set":3.57},"qsec_f":{"set":15.84},"vs_f":{"set":0},"am_f":{"set":0},"gear_f":{"set":3},"carb_f":{"set":4}},
{"id":8,"wt_f":{"set":3.19},"qsec_f":{"set":20},"vs_f":{"set":1},"am_f":{"set":0},"gear_f":{"set":4},"carb_f":{"set":2}},
{"id":9,"wt_f":{"set":3.15},"qsec_f":{"set":22.9},"vs_f":{"set":1},"am_f":{"set":0},"gear_f":{"set":4},"carb_f":{"set":2}}]
As you can see the JSON contains an array of document, where id is the record identifier. The rest of the fields have the following syntax
"[name]":{"[command]":[value]}
In our case the command is always set
but there are number of other command of your use case is not to add a single value (consult the Solr manual here). We use _f
suffix everywhere, which denotes a single float value dynamic field, this way we don’t have to touch Solr’s schema.xml
file (or any other configuration).
We have the files prepared. its time to index:
curl 'http://localhost:8984/solr/qa-2018-03/update?commit=true&header=false&fieldnames=id,mpg_f,cyl_f,disp_f,hp_f,drat_f' \
--data-binary @/path/to/cars1.csv \
-H 'Content-type:application/csv'
We set header
false to not take the first line of CSV as header, and put the field names explicitly in the fieldnames
parameter.
curl http://localhost:8984/solr/qa-2018-03/update?commit=true \
--data-binary @/path/to/cars2.json \
-H 'Content-type:application/json'
So the task is (which is missing from this piece) is to transform the CSV files (except the one we are starting with) to Solr compatible JSON update files. Right now we have 5 such files.
Caveats:
(a) cURL first read the files into memory then sends it to the server. Our CSV files are typicaly around
30-60 GB large, so we have to split them before sending. The JSON files are minimum 3 times larger, since it should
contain the fields names, the command and the extra characters required by the JSON syntax, so the process involves
some preliminary playing with the sizes. Undex linux you can split the files to smaller chunks with the split
command, e.g.:
split -l 1000000 \
-d \
--additional-suffix .csv \
result29-multilingual-saturation.csv multilingual-part
creates multilingual-part00.csv
, multilingual-part01.csv
… from result29-multilingual-saturation.csv
each
containing 100 000 lines.
(b) in this example we supposed that each value is a float, which in Solr is a stored
type. In your case there might
be other data types. Be sure that you chose types which are stored in the Solr index, otherwise this technique will
not work. Check the schema.xml
(or managed-schema
) file that your choosed suffix has the following
attribute: stored="true"
.
(c) Solr’s commit
command is time consuming. It works while you are playing with the sizes, but slows down
the bulk process. You can either sends commits only in every Nth request (say after indexing 10 000 or 100 000
records), or even better you can set Solr’s autoCommit
feature which does it for you.
Details here.
ps. I would like to thank to Mark Philips, who showed me the University of North Texas digital library’s metadata management tool, in which the metrics are stored in Solr alogside the original metadata values, so they can easily find examples for given metadata problems.