Ecosyste.ms: Awesome

An open API service indexing awesome lists of open source software.

Awesome Lists | Featured Topics | Projects

https://github.com/yetanalytics/flint

SPARQL DSL library for Clojure(Script)
https://github.com/yetanalytics/flint

clojure clojurescript dsl rdf sparql

Last synced: 6 days ago
JSON representation

SPARQL DSL library for Clojure(Script)

Awesome Lists containing this project

README

        

# flint

Flint Logo

[![CI](https://github.com/yetanalytics/flint/actions/workflows/test.yml/badge.svg)](https://github.com/yetanalytics/flint/actions/workflows/test.yml)
[![Clojars Project](https://img.shields.io/clojars/v/com.yetanalytics/flint.svg)](https://clojars.org/com.yetanalytics/flint)
[![Contributor Covenant](https://img.shields.io/badge/Contributor%20Covenant-2.1-5e0b73.svg)](CODE_OF_CONDUCT.md)

_The fire i' the flint shows not till it be struck_
\- William Shakespeare, _Timon of Athens_, Act I, Scene 1

A Clojure(Script) DSL for creating SPARQL query and update strings.

If you are using Apache Jena, check out the [flint-jena](https://github.com/yetanalytics/flint-jena) companion library.

## Installation

Add the following to your `deps.edn` map.

```clojure
com.yetanalytics/flint {:mvn/version "0.3.0"
:exclusions [org.clojure/clojure
org.clojure/clojurescript]}
```

See [Clojars](https://clojars.org/com.yetanalytics/flint) for installation using Leiningen, Boot, etc; do not forget to adapt `:exclusions` to your method.

## Outline

Documentation is also available on [cljdoc](https://cljdoc.org/d/com.yetanalytics/flint).

- Queries and Updates
- [Queries](doc/query.md)
- [Updates](doc/update.md)
- Clauses and Subforms
- [Graph IRIs](doc/graph.md)
- [Graph Patterns](doc/where.md)
- [Modifiers](doc/modifier.md)
- [Prologue](doc/prologue.md)
- [Expressions](doc/expr.md)
- [Triples](doc/triple.md)
- [RDF Terms](doc/axiom.md)

## API

Three functions exist in the Flint API:
- `format-query`
- `format-update`
- `format-updates`

The first two functions format a single SPARQL query or update, respectively, while the third formats a collection of SPARQL updates into a single update string.

Each function takes in the following keyword arguments:

| Argument | Description |
| --- | --- |
| `:pretty?` | If `true`, adds line breaks and indentation to the resulting SPARQL string. Default `false`.
| `:validate?` | If `true`, validates that prefixed IRIs are expandable and that certain restrictions on variables and blank nodes are met. Default `true`.
| `:spec-ed?` | If `true`, let the exception data map be the spec error data map (i.e. with `::s/problems`) upon conformance failure, instead of Flint's default error map. Spec error data maps can get quite large, hence this is default `false`.
| `:force-iris?` | If `true`, let all literals be formatted with their datatype IRIs (e.g. `` for string literals); if `false` (the default), then string, numeric, and boolean literals will not have such IRIs appended. Language-tagged literals will never have an appended datatype IRI.

## Examples

The following is a simple query that queries the name of the author who wrote the popular manga series [Attack on Titan](https://en.wikipedia.org/wiki/Attack_on_Titan):
```clojure
(def query
'{:prefixes {:dc ""}
:select [?author]
:where [[?aot :dc/title "Attack on Titan"]
[?aot :dc/creator ?author]]})
```
Note that the map needs to be quoted due to the presence of symbols in the map. We can then pass the query to the function `format-query`:
```clojure
(require '[com.yetanalytics.flint :as f])

(f/format-query query :pretty? true)
```
and it will return a SPARQL string:
```sparql
PREFIX dc:
SELECT ?author
WHERE {
?aot dc:title "Attack on Titan" .
?aot dc:creator ?author .
}
```
One can then pass this query string to a Resource Description Framework (RDF) database and, depending on the data in the system, should return that `?author` is [Hajime Isayama](https://en.wikipedia.org/wiki/Hajime_Isayama).

The following is a more comprehensive example - a query that looks for the publisher of Attack on Titan, then returns the titles of all the works it published in 2010 or after:
```clojure
(def query-2
'{:prefixes {:dc ""
:xsd ""}
:select [?title]
:from [""]
:where [[:union [{_b1 {:dc/title #{{:en "Attack on Titan"}}
:dc/publisher #{?publisher}}}]
[{_b2 {:dc/title #{{:jp "進撃の巨人"}}
:dc/publisher #{?publisher}}}]]
{?work {:dc/publisher #{?publisher}
:dc/title #{?title}
:dc/date #{?date}}}
[:filter (<= #inst "2010-01-01T00:00:00Z" ?date)]]})
```
which demonstrates several additional features, such as an alternate triple syntax using maps instead of vectors, blank nodes, language tags, and the `:union`, `:filter` and `:from` clauses. When passed to `format-query`, it is translated to:
```sparql
PREFIX dc:
PREFIX xsd:
SELECT ?title
FROM
WHERE {
{
_:b1 dc:title "Attack on Titan"@en ;
dc:publisher ?publisher .
}
UNION
{
_:b2 dc:title "進撃の巨人"@jp ;
dc:publisher ?publisher .
}
?work dc:publisher ?publisher ;
dc:title ?title ;
dc:date ?date .
FILTER ("2010-01-01T00:00:00Z"^^xsd:dateTime <= ?date)
}
```

## Prior Art
- Flint is based off of the grammar of [SPARQL 1.1](https://www.w3.org/TR/sparql11-query/).
- The idea of a SPARQL DSL was inspired by [HoneySQL](https://github.com/seancorfield/honeysql), a DSL for creating SQL queries.
- [Matsu](https://github.com/boutros/matsu) is a previous SPARQL DSL implementation that uses an expression-based approach to query construction.
- Flint borrows certain syntactic conventions from the [Datomic](https://docs.datomic.com/on-prem/query/query.html) and [Asami](https://github.com/threatgrid/asami) query and update languages.
- The map-based triples syntax is based on the normal form used in the [IGraph protocol](https://github.com/ont-app/igraph).

## License

Copyright © 2022-2024 Yet Analytics, Inc.

Distributed under the Apache License version 2.0.