A pretty sweet RML engine
(Disclaimer: The current state of CARML is early beta. The next release will offer improved code quality, more test coverage, more documentation and several features currently on the product backlog.)
CARML is being developed by Taxonic in cooperation with Kadaster.
- Releases
- Introduction
- Getting started
- Validating your RML mapping
- Input stream extension
- Function extension
- MultiTermMap extension
- XML namespace extension
- Supported data source types
- Roadmap
20 Sep 2017 - CARML 0.0.1
21 Oct 2017 - CARML 0.1.0
05 Dec 2017 - CARML 0.1.1
12 Feb 2018 - CARML 0.1.2
20 May 2018 - CARML 0.2.0
03 Aug 2018 - CARML 0.2.1
13 Nov 2018 - CARML 0.2.2
17 Nov 2018 - CARML 0.2.3
CARML is a java library that transforms structured sources to RDF based as declared in and RML mapping, in accordance with the RML spec. It is considered by many as the optimal choice for mapping structured sources to RDF.
The best place to start learning about RML is at the source, but basically RML is defined as a superset of R2RML which is a W3C recommendation that describes a language for expressing mappings from relational databases to RDF datasets. RML allows not only the expression of mappings for relational databases, but generalizes this to any structured source. All you need is a way to iterate over and query the source.
CARML is available from the Central Maven Repository.
<dependency>
<groupId>com.taxonic.carml</groupId>
<artifactId>carml-engine</artifactId>
<version>0.2.3</version>
</dependency>
<!-- Choose the resolvers to suit your need -->
<dependency>
<groupId>com.taxonic.carml</groupId>
<artifactId>carml-logical-source-resolver-jsonpath</artifactId>
<version>0.2.3</version>
</dependency>
<dependency>
<groupId>com.taxonic.carml</groupId>
<artifactId>carml-logical-source-resolver-xpath</artifactId>
<version>0.2.3</version>
</dependency>
<dependency>
<groupId>com.taxonic.carml</groupId>
<artifactId>carml-logical-source-resolver-csv</artifactId>
<version>0.2.3</version>
</dependency>
CARML is built on RDF4J, and currently the Mapper directly outputs an RDF4J Model.
Set<TriplesMap> mapping =
RmlMappingLoader
.build()
.load(RDFFormat.TURTLE, Paths.get("path-to-mapping-file"));
RmlMapper mapper =
RmlMapper
.newBuilder()
// Add the resolvers to suit your need
.setLogicalSourceResolver(Rdf.Ql.JsonPath, new JsonPathResolver())
.setLogicalSourceResolver(Rdf.Ql.XPath, new XPathResolver())
.setLogicalSourceResolver(Rdf.Ql.Csv, new CsvResolver())
// optional:
// specify IRI unicode normalization form (default = NFC)
// see http://www.unicode.org/unicode/reports/tr15/tr15-23.html
.iriUnicodeNormalization(Form.NFKC)
// set file directory for sources in mapping
.fileResolver("/some/dir/")
// set classpath basepath for sources in mapping
.classPathResolver("/some/path")
.build();
Model result = mapper.map(mapping);
We're not set up for full mapping validation yet. But, to help you get those first nasty mapping errors out, we've created a SHACL shapes graph (here) that validates RML mappings. You can use the SHACL playground to easily test your mapping.
When it comes to non-database sources, the current RML spec only supports the specification of file based sources in an rml:LogicalSource
. However, it is often very useful to be able to transform stream sources.
To this end CARML introduces the notion of 'Named Streams'. Which follows the ontology defined here.
So now, you can define streams in your mapping like so:
:SomeLogicalSource
rml:source [
a carml:Stream ;
# NOTE: name is not mandatory and can be left unspecified, when working with a single stream
carml:streamName "stream-A" ;
];
rml:referenceFormulation ql:JSONPath;
rml:iterator "$"
.
In order to provide access to the input stream, it needs to be registered on the mapper.
RmlMapper mapper =
RmlMapper
.newBuilder()
.setLogicalSourceResolver(Rdf.Ql.JsonPath, new JsonPathResolver())
.build();
mapper.bindInputStream("stream-A", inputStream);
Note that it is possible to register several streams, allowing you to combine several streams to create your desired RDF output.
A recent development related to RML is the possibility of adding functions to the mix. This is a powerful extension that increases the expressivity of mappings immensely. Do note that the function ontology is still under development at UGhent.
Because we believe that this extension can already be of great value, we've already adopted it in CARML.
The way it works is, you describe the execution of a function in terms of the Function Ontology (FnO).
Take for example the SumFunction example of the FnO spec. This defines an instance ex:sumFunction
of class fno:Function
that is able to compute the sum of two values provided as parameters of the function at execution time.
It also describes an instance ex:sumExecution
of fno:execution
, which fno:executes
ex:sumFunction
which descibes an instance of an execution of the defined sum function. In this case with parameters 2 and 4.
To be able to use this in RML mappings we use executions of instances of fno:Function
to determine the value of a term map. The execution of a function can be seen as a post-processing step in the evaluation of a term map.
@prefix rr: <http://www.w3.org/ns/r2rml#> .
@prefix rml: <http://semweb.mmlab.be/ns/rml#> .
@prefix fnml: <http://semweb.mmlab.be/ns/fnml#> .
@prefix xsd: <http://www.w3.org/2001/XMLSchema#> .
@prefix fno: <http://semweb.datasciencelab.be/ns/function#> .
@prefix ex: <http://example.org/> .
ex:sumValuePredicateObjectMap
rr:predicate ex:total ;
rr:objectMap [
a fnml:FunctionMap ;
fnml:functionValue [
rml:logicalSource ex:LogicalSource ;
rr:subjectMap [
rr:template "functionExec";
rr:termType rr:BlankNode ;
rr:class fno:Execution
] ;
rr:predicateObjectMap
[
rr:predicate fno:executes ;
rr:objectMap [
rr:constant ex:sumFunction ;
]
] ,
[
rr:predicate ex:intParameterA ;
rr:objectMap [ rml:reference "foo" ]
] ,
[
rr:predicate ex:intParameterB ;
rr:objectMap [ rml:reference "bar" ]
]
] ;
rr:datatype xsd:integer ;
]
.
A function can be created in any .java
class. The function should be annotated with @FnoFunction
, providing the string value of the function IRI, and the function parameters with @FnoParam
, providing the string value of the function parameter IRIs.
public class RmlFunctions {
@FnoFunction("http://example.org/sumFunction")
public int sumFunction(
@FnoParam("http://example.org/intParameterA") int intA,
@FnoParam("http://example.org/intParameterB") int intB
) {
return intA + intB;
}
}
The class or classes containing the annotated functions can then be registered on the mapper via the RmlMapper#addFunctions
method.
RmlMapper mapper =
RmlMapper
.newBuilder()
.setLogicalSourceResolver(Rdf.Ql.JsonPath, new JsonPathResolver())
.addFunctions(new YourRmlFunctions())
.build();
Model result = mapper.map(mapping);
It is recommended to describe and publish new functions in terms of FnO for interpretability of mappings, and, possibly, reuse of functions, but it's not mandatory for use in CARML.
Note that it is currently possible to specify and use function executions as parameters of other function executions in CARML, although this is not (yet?) expressible in FnO.
As RML is a superset of R2RML that was developed after R2RML, and R2RML only supports term maps that return a single RDF term, it is not possible to have a term map return multiple values according to the current RML spec. However, in structured file sources like XML and JSON, it is often the case that a collection of similar nodes is nested in another.
For example:
{
"name":"John Doe",
"cars":[ "BMW", "Seat", "Porsche" ]
}
The only way to get a ex:John%20Doe ex:ownsCar X
triple per value in cars[]
with RML would be something like the following mapping:
@prefix rr: <http://www.w3.org/ns/r2rml#>.
@prefix rml: <http://semweb.mmlab.be/ns/rml#>.
@prefix ql: <http://semweb.mmlab.be/ns/ql#> .
@prefix carml: <http://carml.taxonic.com/carml/> .
@prefix ex: <http://www.example.com/> .
<#SubjectMapping> a rr:TriplesMap ;
rml:logicalSource [
rml:source [
a carml:Stream ;
] ;
rml:referenceFormulation ql:JSONPath ;
rml:iterator "$" ;
] ;
rr:subjectMap [
rr:template "http://www.example.com/{name}" ;
] ;
rr:predicateObjectMap [
rr:predicate ex:ownsCar ;
rr:objectMap [
rml:reference "cars[0]" ;
] ;
] ;
rr:predicateObjectMap [
rr:predicate ex:ownsCar ;
rr:objectMap [
rml:reference "cars[1]" ;
] ;
] ;
rr:predicateObjectMap [
rr:predicate ex:ownsCar ;
rr:objectMap [
rml:reference "cars[2]" ;
] ;
] ;
.
This is not very flexible.
To solve this issue, CARML introduces the notion of a MultiTermMap and the following properties:
carml:multiTemplate
carml:multiReference
carml:multiFunctionValue
carml:multiJoinCondition
This allows the earlier example to be mapped as follows:
@prefix rr: <http://www.w3.org/ns/r2rml#>.
@prefix rml: <http://semweb.mmlab.be/ns/rml#>.
@prefix ql: <http://semweb.mmlab.be/ns/ql#> .
@prefix carml: <http://carml.taxonic.com/carml/> .
@prefix ex: <http://www.example.com/> .
<#SubjectMapping> a rr:TriplesMap ;
rml:logicalSource [
rml:source [
a carml:Stream ;
] ;
rml:referenceFormulation ql:JSONPath ;
rml:iterator "$" ;
] ;
rr:subjectMap [
rr:template "http://www.example.com/{name}" ;
] ;
rr:predicateObjectMap [
rr:predicate ex:ownsCar ;
rr:objectMap [
carml:multiReference "cars" ;
rr:datatype xsd:string ;
] ;
] ;
.
returning:
<http://www.example.com/John%20Doe>
<http://www.example.com/ownsCar>
"BMW" ,
"Seat" ,
"Porsche" .
Adding a multiTemplate to the mapping:
rr:predicateObjectMap [
rr:predicate ex:ownsCar2 ;
rr:objectMap [
carml:multiTemplate "Car: {cars}" ;
rr:datatype xsd:string ;
] ;
] ;
one gets:
<http://www.example.com/John%20Doe>
<http://www.example.com/ownsCar>
"BMW" ,
"Seat" ,
"Porsche" ;
<http://www.example.com/ownsCar2>
"Car: BMW" ,
"Car: Seat" ,
"Car: Porsche" .
Or using a multiJoinCondition:
rr:predicateObjectMap [
rr:predicate ex:ownsCar3 ;
rr:objectMap [
rr:parentTriplesMap <#CarMapping> ;
carml:multiJoinCondition [
rr:child "cars" ;
rr:parent "$" ;
] ;
] ;
] ;
.
<#CarMapping> a rr:TriplesMap ;
rml:logicalSource [
rml:source [
a carml:Stream ;
] ;
rml:referenceFormulation ql:JSONPath ;
rml:iterator "$.cars" ;
] ;
rr:subjectMap [
rr:template "http://www.example.com/{$}" ;
] ;
rr:predicateObjectMap [
rr:predicate rdfs:label ;
rr:objectMap [
rml:reference "$" ;
rr:datatype xsd:string ;
] ;
] ;
.
yields:
<http://www.example.com/John%20Doe>
<http://www.example.com/ownsCar>
"BMW" ,
"Seat" ,
"Porsche" ;
<http://www.example.com/ownsCar2>
"Car: BMW" ,
"Car: Seat" ,
"Car: Porsche" ;
<http://www.example.com/ownsCar3>
<http://www.example.com/BMW> ,
<http://www.example.com/Seat> ,
<http://www.example.com/Porsche> .
<http://www.example.com/BMW>
<http://www.w3.org/2000/01/rdf-schema#label> "BMW" .
<http://www.example.com/Seat>
<http://www.w3.org/2000/01/rdf-schema#label> "Seat" .
<http://www.example.com/Porsche>
<http://www.w3.org/2000/01/rdf-schema#label> "Porsche" .
you get the drift.
Note that currently CARML only supports MultiObjectMaps. Future versions may support this for graphs, subjects and predicates as well.
When working with XML documents, it is often necessary specify namespaces to identify a node's qualified name.
Most XPath implementations allow you to register these namespaces, in order to be able to use them in executing XPath expressions.
In order to convey these expressions to the CARML engine, CARML introduces the class carml:XmlDocument
that can be used as a value of rml:source
. An instance of carml:XmlDocument
can, if it is a file source, specify a location via the carml:url
property, and specify namespace declarations via the carml:declaresNamespace
property.
For example, given the following XML document:
<?xml version="1.0" encoding="UTF-8"?>
<ex:bookstore xmlns:ex="http://www.example.com/books/1.0/">
<ex:book category="children">
<ex:title lang="en">Harry Potter</ex:title>
<ex:author>J K. Rowling</ex:author>
<ex:year>2005</ex:year>
<ex:price>29.99</ex:price>
</ex:book>
</ex:bookstore>
one can now use the following mapping, declaring namespaces, to use them in XPath expressions:
@prefix rr: <http://www.w3.org/ns/r2rml#>.
@prefix rml: <http://semweb.mmlab.be/ns/rml#>.
@prefix ql: <http://semweb.mmlab.be/ns/ql#> .
@prefix carml: <http://carml.taxonic.com/carml/> .
@prefix ex: <http://www.example.com/> .
<#SubjectMapping> a rr:TriplesMap ;
rml:logicalSource [
rml:source [
a carml:Stream ;
# or in case of a file source use:
# carml:url "path-to-souce" ;
carml:declaresNamespace [
carml:namespacePrefix "ex" ;
carml:namespaceName "http://www.example.com/books/1.0/" ;
] ;
] ;
rml:referenceFormulation ql:XPath ;
rml:iterator "/ex:bookstore/*" ;
] ;
rr:subjectMap [
rr:template "http://www.example.com/{./ex:title}" ;
rr:class ex:Book ;
rr:termType rr:IRI ;
] ;
.
which yields:
<http://www.example.com/Harry%20Potter> a <http://www.example.com/Book> .
Data source type | Reference query language | Implementation |
---|---|---|
JSON | JsonPath | Jayway JsonPath 2.4.0 |
XML | XPath | Saxon-HE 9.8.0-12 |
CSV | n/a | Univocity 2.6.3 |
- CARML Command line interface
- Better support for large sources
- Improved join / parent triples map performance
- Support RDF store connections
- Split off provisional RDF Mapper as separate library