A Jekyll plugin for including RDF data in your static site.
The API Documentation is available at RubyDoc.info.
The easiest and fastest way to install our project is the installation as a gem. The following command automatically installs the project and all required components such as Jekyll and the RDF-library
gem install jekyll-rdf
To install the project with the git-repository you will need git
on your system. The first step is just cloning the repository:
git clone git@github.com:white-gecko/jekyll-rdf.git
A folder named jekyll-rdf
will be automatically generated. You need to switch into this folder and compile the ruby gem to finish the installation:
cd jekyll-rdf
gem build jekyll-rdf.gemspec
gem install jekyll-rdf -*.gem
First, you need a jekyll page. In order to create one, just do:
jekyll new my_page
cd my_page
Further there are some parameters required in your _config.yml
for jekyll-rdf
. I.e. the url
and baseurl
parameters are used for including the resource pages into the root of the site, the plug-in has to be configured, and the path to the RDF file has to be present.
url: "http://www.ifi.uio.no"
baseurl: "/INF3580"
plugins:
- jekyll-rdf
jekyll_rdf:
path: "simpsons.ttl"
Running jekyll build
will render the RDF resources to the _site/…
directory.
RDF resources whose IRIs don't start with the configured jekyll url
and baseurl
are rendered to the _site/rdfsites/…
subdirectory.
Note: Since Jekyll 3.3.0 there is a special behavior of the jekyll serve
command in a development environment.
Jekyll overwrites the site.url
variable with your host and port reference, as described in the Jekyll documentation.
The behavior breaks the jekyll-rdf resource creation.
If you want to use jekyll serve
you have to run jekyll serve --skip-initial-build
or set JEKYLL_ENV=production
.
Now, create one or more files (e.g rdf_index.html
or person.html
) in the _layouts
-directory to edit the temaplate for rdf-pages. For each resource a page will be rendered. See example below:
---
layout: default
---
<div class="home">
<h1 class="page-heading"><b>{{ page.rdf.iri }}</b></h1>
<p>
<h3>Statements in which {{ page.rdf.iri }} occurs as subject:</h3>
{% include statements_table.html collection=page.rdf.statements_as_subject %}
</p>
<p>
<h3>Statements in which {{ page.rdf.iri }} occurs as predicate:</h3>
{% include statements_table.html collection=page.rdf.statements_as_predicate %}
</p>
<p>
<h3>Statements in which {{ page.rdf.iri }} occurs as object:</h3>
{% include statements_table.html collection=page.rdf.statements_as_object %}
</p>
</div>
We included some template examples at
test/source/_layouts/rdf_index.html
test/source/_layouts/person.html
{{ page.rdf }}
Is the currently rendered resource.
{{ page.rdf.iri }}
Returns the IRI of the currently rendered resource.
To access objects which are connected to the current subject via a predicate you can use our custom liquid filters. For single objects or lists of objects use the rdf_property
-filter (see 1 and 2).
To access one object which is connected to the current subject through a given predicate please filter page.rdf
data with the rdf_property
-filter. Example:
Age: {{ page.rdf | rdf_property: '<http://xmlns.com/foaf/0.1/age>' }}
To select a specific language please add a a second parameter to the filter:
Age: {{ page.rdf | rdf_property: '<http://xmlns.com/foaf/0.1/job>','en' }}
To get more than one object connected to the current subject through a given predicate please use the filter rdf_property
in conjunction with a third argument set to true
(the second argument for the language can be omitted by setting it to nil
):
Sisters: <br />
{% assign resultset = page.rdf | rdf_property: '<http://www.ifi.uio.no/INF3580/family#hasSister>', nil, true %}
<ul>
{% for result in resultset %}
<li>{{ result }}</li>
{% endfor %}
</ul>
To select a specific language please add a second parameter to the filter:
Book titles: <br />
{% assign resultset = page.rdf | rdf_property: '<http://xmlns.com/foaf/0.1/currentProject>','de' %}
<ul>
{% for result in resultset %}
<li>{{ result }}</li>
{% endfor %}
</ul>
To support RDF Containers and RDF Collections we provide the rdf_container
and rdf_collection
filters.
In both cases the respective container resource resp. head of the collection needs to be identified and then passed through the respective filter.
For containers we currently support explicit instances of rdf:Bag
, rdf:Seq
and rdf:Alt
with the members identified using the rdfs:ContainerMembershipProperty
s: rdf:_1
, rdf:_2
, rdf:_3
….
Collections are identified using rdf:first
, rdf:rest
and terminated with L rdf:rest rdf:nil
.
Since the head of a collection needs to be identified you cannot use a blank node there, you can identify it indirectly through the predicate which contains the collection.
Example graph:
@prefix ex: <http://example.org/> .
@prefix rdf: <http://www.w3.org/1999/02/22-rdf-syntax-ns#> .
ex:Resource ex:lists ex:List ;
ex:directList ("hello" "from" "turtle") .
ex:hasContainer ex:Container .
ex:List rdf:first "hello" ;
rdf:rest ("rdf" "list") .
ex:Container a rdf:Bag ;
rdf:_1 "hello" ;
rdf:_2 "rdf" ;
rdf:_3 "container" .
The template for ex:Resource
:
{% assign list = page.rdf | rdf_collection: '<http://example.org/directList>' %}
<ol>
{% for item in list %}
<li>{{ item }}</li>
{% endfor %}
</ol>
{% assign container = page.rdf | rdf_property: '<http://example.org/hasContainer>' | rdf_container %}
<ul>
{% for item in container %}
<li>{{ item }}</li>
{% endfor %}
</ul>
We implemented a liquid filter sparql_query
to run custom SPARQL queries. Each occurence of ?resourceUri
gets replaced with the current URI.
Caution: You have to separate query and resultset variables because of Liquids concepts. Example:
{% assign query = 'SELECT ?sub ?pre WHERE { ?sub ?pre ?resourceUri }' %}
{% assign resultset = page.rdf | sparql_query: query %}
<table>
{% for result in resultset %}
<tr>
<td>{{ result.sub }}</td>
<td>{{ result.pre }}</td>
</tr>
{% endfor %}
</table>
It is possible to declare a set of prefixes which can be used in the rdf_property
and sparql_query
liquid-filters.
This allows to shorten the amount of text required for each liquid-filter.
The syntax of the prefix declarations the same as for SPARQL 1.1.
Just put your prefixes in a separate file and include the key rdf_prefix_path
together with a relative path in the YAML Front Matter of a file where your prefixes should be used.
The path gets resolved to <your jekyll-directory>/rdf-data/<rdf_prefix_path>
.
For the prefixes the same rules apply as for other variables defined in the YAML Front Matter. “These variables will then be available to you to access using Liquid tags both further down in the file and also in any layouts or includes that the page or post in question relies on.” (source: YAML Front Matter). This is especially relevant if you are using prefixes in includes.
It is possible to map a specific class (resp. RDF-type) or individual resources to a template.
class_template_mappings:
"http://xmlns.com/foaf/0.1/Person": "person.html"
instance_template_mappings:
"http://aksw.org/Team": "team.html"
A template mapped to a class will be used to render each instance of that class and its subclasses. Each instance is rendered with its most specific class mapped to a template. If the mapping is ambiguous for a resource, a warning will be output to your command window, so watch out!
It is also possible to define a default template, which is used for all resources, which are not covered by the class_template_mappings
or instance_template_mappings
.
default_template: "default.html"
If the URI of a resource contains a fragment identifier (#…
) the resource can be hosted together with other resources with the same base URI up to the fragment identifier on a single page.
The page will by accessible through the base URI, while in the template the individual URIs with a fragment identifier are accessible through the collection page.sub_rdf
.
Example
In the _config.yml
:
'instance_template_mappings' :
'http://www.ifi.uio.no/INF3580/simpsons' : 'family.html'
In _layouts/family.html
:
{% for member in page.sub_rdf%}
{% include simPerson.html person = member%}
{% endfor %}
The example uses the template family.html
to render a single page containing every resource whose URI begins with http://www.ifi.uio.no/INF3580/simpsons#
, was well as the resource http://www.ifi.uio.no/INF3580/simpsons
itself.
Jekyll-rdf collects all resources with a fragment identifier in their URI (from here on called subResources
) and passes them through page.sub_rdf
into the templates of its superResource
(resources whose base URI is the same as of its subResources
except for the fragment identifier).
Additionally, you can restrict the overall resource selection by adding a SPARQL query as restriction
parameter to _config.yml
. Please use ?resourceUri as the placeholder for the resulting literal:
restriction: "SELECT ?resourceUri WHERE { ?resourceUri <http://www.ifi.uio.no/INF3580/family#hasFather> <http://www.ifi.uio.no/INF3580/simpsons#Homer> }"
There are 3 pre-defined keywords for restrictions implemented:
subjects
will load all subject URIspredicates
will load all predicate URIsobjects
will load all object URIs
Furthermore you can decide if you want to render blank nodes or not. You just need to add include_blank
to _config.yml
:
jekyll_rdf:
include_blank: true
Finally it is also possible to set a preferred language for the RDF-literals with the option language
:
jekyll_rdf:
language: "en"
An example configuration could look like this:
jekyll_rdf:
path: "rdf-data/simpsons.ttl"
language: "en"
include_blank: true
restriction: "SELECT ?s WHERE { ?s ?p ?o}"
default_template: "rdf_index.html"
class_template_mappings:
"http://xmlns.com/foaf/0.1/Person": "person.html"
instance_template_mappings:
"http://www.ifi.uio.no/INF3580/simpsons#Abraham": "abraham.html"
Name | Parameter | Optional Parameter | Optional Flag | Description | Example |
---|---|---|---|---|---|
rdf_property | predicate-URI as String | language-tag as String | true to get a list | Returns a single object or an array with objects which are connected to the current subject through a given predicate | {{ page.rdf | rdf_property: '<http://xmlns.com/foaf/0.1/job>','en' }} {% assign resultset = page.rdf | rdf_property: '<http://xmlns.com/foaf/0.1/currentproject>','en', true %}{% for result in resultset %}<li>{{ result }}</li>{% endfor %} |
rdf_inverse_property | predicate-URI as String | language-tag as String | true to get a list | The same as rdf_property, but the predicate is used reversed | {{ page.rdf | rdf_inverse_property: '<http://www.ifi.uio.no/INF3580/family#hasFather>','en' }} <!--Returns a Son instead of a Father--> |
sparql_query | SPARQL-Query as String | - | - | Runs a SPARQL-Query with the current subject as ?resourceURI | {% assign query = 'SELECT ?sub ?pre WHERE { ?sub ?pre ?resourceUri }' %}{% assign resultset = page.rdf | sparql_query: query %}<table>{% for result in resultset %}<tr><td>{{ result.sub }}</td><td>{{ result.pre }}</td></tr>{% endfor %}</table> |
rdf_container | - | - | - | Retrieve an array from an RDF Container | {% assign array = containerResource | rdf_container %}{% for item in array %}{{ item }}{% endfor %} |
rdf_collection | - | predicate-URI as String | - | Retrieve an array from an RDF Collection | {% assign array = Resource | rdf_collection: 'collectionPredicate' %}{% for item in array %}{{ item }}{% endfor %} |
Name | Parameter | Default | Description | Example |
---|---|---|---|---|
path | Relative path to the RDF-File | no default | Specifies the path to the RDF file you want to render the website for | path: "rdf-data/simpsons.ttl" |
language | Language-Tag as String | no default | Specifies the preferred language when you select objects using our Liquid filters | language: "en" |
include_blank | Boolean-Expression | false | Specifies whether blank nodes should also be rendered or not | include_blank: true |
render_orphaned_uris | Boolean-Expression | false | Decide to render resources referenced by URI fragment identifier without a container URI (or not) | render_orphaned_uirs: true |
restriction | SPARQL-Query as String or subjects / objects / predicates | no default | Restricts the resource-selection with a given SPARQL-Query or the three keywords subjects (only subject URIs), objects, predicates | restriction: "SELECT ?resourceUri WHERE { ?resourceUri <http://www.ifi.uio.no/INF3580/family#hasFather> <http://www.ifi.uio.no/INF3580/simpsons#Homer> }" |
default_template | Filename of the default RDF-template in _layouts directory | no default | Specifies the template-file you want Jekyll to use to render all RDF resources | default_template: "rdf_index.html" |
instance_template_mappings | Target URI as String : filename of the template as String | no default | Maps given URIs to template-files for rendering an individual instance | instance_template_mappings: "http://www.ifi.uio.no/INF3580/simpsons#Abraham": "abraham.html" |
class_template_mappings | Target URI as String : filename of the template as String | no default | Maps given URIs to template-files for rendering all instances of that class | class_template_mappings: "http://xmlns.com/foaf/0.1/Person": "person.html" |
bundle exec rake test
Everytime the tests are executed, the Jekyll page inside of test/source
gets processed. Start a slim web server to watch the results in web browser, e.g. Pythons SimpleHTTPServer
(Python 2, for Python 3 it's http.server
):
cd test/source/_site
python -m SimpleHTTPServer 8000
To generate the API Doc please navigate to jekyll-rdf/lib
directory and run
gem install yard
yardoc *
The generated documentation is placed into jekyll-rdf/lib/doc
directory.
jekyll-rdf is licensed under the MIT license.