All notable changes to this project will be documented in this file. This project adheres to Semantic Versioning and Keep a CHANGELOG.
Elixir versions < 1.14 and OTP version < 24 are no longer supported
RDF.JSON
datatype
- Fixed compilation error when defining
RDF.Vocabulary.Namespace
s for large vocabularies (like Schema.org) caused by exceeding implementation limits. - The case violation check during
RDF.Vocabulary.Namespace
creation didn't always recognize properties correctly. - Namespace delegator modules defined with
RDF.Namespace.act_as_namespace/1
were not properly recompiled on changes of the underlying vocabulary file.
- The Turtle/TriG encoder didn't escape strings properly when using the long literal form, i.e. when the encoded string contains newlines, which could result in invalid output in edge-cases.
Elixir versions < 1.13 and OTP version < 23 are no longer supported
An update to the recent more extensive Turtle test suite revealed that a bug in
Elixir's URI.merge/2
function affects the relative URI handling in the Turtle decoder.
It is therefore strongly recommended to use the Turtle decoder with
Elixir v1.15 or later, where this issue has been resolved.
RDF.TriG
with an implementation of the TriG serialization language.- Several new options on the Turtle/TriG encoders for more customizations and
performance optimizations:
- Capability to add custom content on the Turtle/TriG encoders with the
:content
option. :line_prefix
for a function defining custom line prefixes:indent_width
to customize the indentation width:pn_local_validation
for controlling IRI validation when encoding prefixed names:rdf_star
allowing to skip an RDF-star related preprocessing step
- Capability to add custom content on the Turtle/TriG encoders with the
RDF.Dataset.named_graphs/1
to get a list of all named graphs of a dataset.RDF.Dataset.graph_names/1
to get a list of all graph names of a dataset.RDF.Dataset.update_all_graphs/2
to apply a function on all graphs of a dataset.RDF.Graph.update_all_descriptions/2
to apply a function on all descriptions of a graph.RDF.Description.update_all_predicates/2
andRDF.Description.update_all_objects/2
to apply a function on all predications resp. objects of a description.RDF.Graph.rename_resource/2
andRDF.Description.rename_resource/2
to replace all occurrences of an id.RDF.turtle_prefixes/1
andRDF.sparql_prefixes/1
for creating respective prefix headersRDF.Graph.prefixes/2
which allows to specify a custom return value when the prefixes are empty.RDF.PrefixMap.empty?/1
to check of aRDF.PrefixMap
is empty.RDF.PrefixMap.limit/2
to limit aRDF.PrefixMap
to a subset of some given prefixes.RDF.BlankNode.Generator.UUID
andRDF.BlankNode.Generator.Random
implementations ofRDF.BlankNode.Generator.Algorithm
:bnode_gen
option on the Turtle/TriG decoders, allowing customization of blank node generation and aturtle_trig_decoder_bnode_gen
application config for setting the default blank node generator globally.- Performance improvements of N-Triples and N-Quads encoders.
- Default blank node generation in Turtle decoder now generates UUID blank node
identifiers instead of the previous deterministic incremented identifiers.
This change ensures unique blank nodes across multiple parsing operations.
You can opt back to the previous behaviour with the new
turtle_trig_decoder_bnode_gen
application config using the:increment
value. - Aliases defined within
RDF.Graph.build
blocks are no longer supported due to changes
in Elixir 1.17. Aliases from the caller context are still available and automatically
re-aliased in the build block. However, instead of using aliases for vocabulary namespaces, use@prefix
declarations inside the build block, as it provides additional benefits. Please refer to the user guide for more information. - The
prefixes
of anRDF.Graph
are now always aRDF.PrefixMap
and no longernil
initially, since this had the confusing consequence that anRDF.Graph
where all prefixes were deleted was not equal to same graph where the deleted were never set, e.g.RDF.graph() |> Graph.add_prefixes(ex: EX) |> Graph.delete_prefixes(:ex) == RDF.graph()
did not hold previously. This behaviour was used before to differentiate graphs which should use theRDF.default_prefixes/0
(in caseprefixes
wasnil
) from those which should not use any prefixes (emptyPrefixMap
) when serialized to Turtle. You'll now have to add theprefixes: []
on Turtle serialization explicitly. The old behaviour of gettingnil
on empty prefixes can be achieved with the newRDF.Graph.prefixes/2
function. - Update to change in N-Triples and N-Quads specs disallowing colons in bnode labels.
- Rename
:only
option ofRDF.Turtle.Encoder
to:content
to reflect the enhanced capabilities. - The
RDF.BlankNode.Generator
andRDF.BlankNode.Generator.Algorithm
behaviour used internally at various places was redesigned. - Deprecated
RDF.Diff.merge/2
was removed. UseRDF.Diff.union/2
instead. - Replacement of
elixir_uuid
withuniq
dependency for UUID generation and make it no longer optional
- The
RDF.Turtle.Encoder
was not falling back to usingRDF.default_prefixes/0
when the encoded graph had prefixes which were removed afterwards. - Fixed the
RDF.Turtle.Encoder
validation to ensure IRIs with permissible characters, such as hyphens, can be correctly encoded as prefixed names. Previously, the validation was overly strict, preventing some valid IRIs from being encoded as prefixed names. RDF.NTriples.Encoder
andRDF.NQuads.Encoder
could not stream quoted RDF-star triples could as iodata.
Elixir versions < 1.12 are no longer supported
RDF.Namespace.act_as_namespace/1
macro which can be used to let a module act as a specifiedRDF.Namespace
orRDF.Vocabulary.Namespace
.canonical_hash/2
function onRDF.Dataset
,RDF.Graph
andRDF.Description
which computes a hash value for the data based on theRDF.Canonicalization
algorithmintersection/2
function onRDF.Dataset
,RDF.Graph
andRDF.Description
which create respective graph data intersections. Since this feature relies onMap.intersect/3
that was added in Elixir v1.15, it is only available with a respective Elixir version.RDF.Dataset.put_graph/3
adds new graphs overwriting any existing graphsRDF.Dataset.update/4
to update graphs in aRDF.Dataset
RDF.Graph.delete_predications/3
to delete all statements in aRDF.Graph
with the given subjects and predicatesRDF.PrefixMap.to_header/3
,RDF.PrefixMap.to_turtle/1
andRDF.PrefixMap.to_sparql/1
to get header string representations of aRDF.PrefixMap
in the respective styleRDF.PrefixMap.to_sorted_list/1
which returns the prefix map as keyword list sorted by prefix (this should become useful with OTP 26)RDF.PropertyMap.to_sorted_list/1
which returns the property map as keyword list sorted by property- The Turtle encoder now sorts the prefixes (based on
RDF.PrefixMap.to_sorted_list/1
), which has become necessary, since OTP 26 maps are now unordered even in smaller cases (previously only larger maps were unordered). - The N-Triples and N-Quads encoders now support a flag option
:sort
which, when
activated, encodes the statements sorted in Unicode code point order. - The hash algorithm to be used for RDF canonicalization can be configured either
with the
:hash_algorithm
keyword option or the:canon_hash_algorithm
application runtime configuration. - Add Hash N-Degree Quads algorithm call limit to canonicalization as a countermeasure against poison dataset attacks
- Compile-time application configuration
:optimize_regexes
that allows to switch internal usage to the faster Erlang:re.run/2
function for regex pattern matching (@jkrueger) - Some optimizations on
RDF.IRI
(@jkrueger) RDF.EarlFormatter
as anExUnit.Formatter
implementation that generates EARL reports
RDF.Canonicalization.canonicalize/2
now returns the canonicalized dataset in a tuple along with final state containing the input blank node identifier map and the issued identifiers map as required by the RDF dataset canonicalization specificationRDF.Diff.merge/2
was deprecated and will be replaced in future versions with a different merge algorithm. UseRDF.Diff.union/2
now for the current algorithm.- Statements as lists (instead of tuples) in the
Collectable
implementations ofRDF.Description
,RDF.Graph
andRDF.Dataset
were deprecated. Support of those will be removed in RDF.ex v2.0. - The following deprecated types were removed:
RDF.Statement.coercible_t
(new type:RDF.Statement.coercible
)RDF.Star.Statement.coercible_t
(new type:RDF.Star.Statement.coercible
)RDF.Triple.t_values
(new type:RDF.Triple.mapping_value
)RDF.Quad.t_values
(new type:RDF.Quad.mapping_value
)
RDF.Dataset.put/3
with aRDF.Dataset
input didn't respect the:graph
option to aggregate everything into single target graph
- a custom option
:content_only
on theInspect
implementation ofRDF.Graph
which returns only the (possibly abbreviated) Turtle representation of the graph; this can be used in otherInspect
implementations that want to include thisRDF.Graph
representation RDF.prefixes/1
as another alias for the creation of aRDF.PrefixMap
RDF.Graph.new/2
didn't respect the:init
opt when the first argument was aRDF.Graph
- implementation of the Standard RDF Dataset Canonicalization Algorithm
which can be used with
RDF.Graph.canonicalize/1
andRDF.Dataset.canonicalize/1
functions RDF.Graph.isomorphic?/2
andRDF.Dataset.isomorphic?/2
to compare if two graphs or datasets are the same, regardless of the concrete names of the blank nodes they containRDF.Statement.bnodes/1
,RDF.Triple.bnodes/1
,RDF.Quad.bnodes/1
to get a list of all blank nodes within a statementRDF.Statement.include_value?/2
,RDF.Triple.include_value?/2
,RDF.Quad.include_value?/2
to check whether a given value is a component of a statement- performance improvements of the
RDF.Turtle.Encoder
RDF.XSD.Double
andRDF.XSD.Float
literals created from Elixir floats and integers are now interpreted to be in the canonical lexical formRDF.BlankNode.new/1
ignores the prefix"_:"
in a given blank node name
- the N-Triples, N-Quads and Turtle encoder were creating too many backslashes,
when escaping a backslash in a string;
BEWARE: You'll have to fix the generated Turtle files you've produced with earlier versions! - the N-Triples, N-Quads and Turtle encoder didn't apply proper escaping in typed literals of unknown type in general
- the Turtle encoder didn't encode the descriptions of blank nodes which occurred in
a blank node cycle, e.g. in
_:b1 :p1 _:b2 . _:b2 :p2 _:b1 .
neither the description of_:b1
nor of_:b2
were rendered - the Turtle encoder now preserves the lexical form of a literal instead of always
encoding the canonical form - a regression in
defvocab
prevented its use with fully qualified vocabulary namespace module names (i.e. which include a dot) - the
term_to_iri/1
macro didn't work properly in all types of pattern matches - the
Inspect
protocol implementation for decimal literals wasn't using the lexical in case of an uncanonical lexical form of a decimal literal
In this version RDF.Namespace
and RDF.Vocabulary.Namespace
were completely rewritten.
The generated namespaces are much more flexible now and compile faster.
For more details on how to migrate from an earlier version read this wiki page.
Elixir versions < 1.11 are no longer supported
RDF.Vocabulary.Namespace.create/5
for dynamic creation ofRDF.Vocabulary.Namespace
sRDF.Namespace
buildersdefnamespace/3
andcreate/4
inside of pattern matchesRDF.Vocabulary.Namespace
modules now have a__file__/0
function which returns the path to the vocabulary file they were generated fromRDF.Vocabulary.path/1
returning the path to the vocabulary directory of an application andRDF.Vocabulary.path/2
returning the path to the files within it- The property functions on the
RDF.Namespace
andRDF.Vocabulary.Namespace
modules now also have a single argument variant, which allows to query the objects for the respective property from aRDF.Description
- Aliases on a
RDF.Vocabulary.Namespace
can now be specified directly in the:terms
list. - The
:terms
option can now also be used in conjunction with the:file
and:data
options to restrict the terms loaded from the vocabulary data with a list of the terms or a restriction function. - The
case_violations
option ofdefvocab
now supports an:auto_fix
option which adapts the first letter of violating term accordingly. It also supports custom handler functions, either as an inline function or as a function on a separate module. - New option
allow_lowercase_resource_terms
option which can be set totrue
to no longer complain about lowercased terms for non-property resources. RDF.Graph.build/2
now supports the creation of ad-hoc vocabulary namespaces with a@prefix
declaration providing the URI of the namespace as a stringRDF.Namespace.IRI.term_to_iri/1
macro which allows to resolveRDF.Namespace
terms- a lot of new
RDF.Guards
:is_rdf_iri/1
,is_rdf_bnode/1
,is_rdf_literal/1
,is_rdf_literal/2
,is_plain_rdf_literal/1
,is_typed_rdf_literal/1
,is_rdf_resource/1
,is_rdf_term/1
,is_rdf_triple/1
,is_rdf_quad/1
andis_rdf_statement/1
- an implementation of
__using__
on the top-levelRDF
module, which allows to add basic imports and aliases with a simpleuse RDF
RDF.IRI.starts_with?/2
andRDF.IRI.ends_with?/2
RDF.Graph.quads/2
andRDF.Dataset.quads/2
to get all statements of aRDF.Graph
andRDF.Dataset
as quadsRDF.Dataset.triples/2
to get all statements of aRDF.Dataset
as triplesRDF.PrefixMap.to_list/1
RDF.PropertyMap.to_list/1
- support for Elixir 1.14
- support for Decimal v2
- Support for passing multiple objects as separate arguments to the property functions of the description DSL on the vocabulary namespaces was removed to create space for further arguments for other purposes in the future. Multiple objects must be given now in a list instead.
- All errors found during the compilation of
RDF.Vocabulary.Namespace
are now collectively reported under a singleRDF.Vocabulary.Namespace.CompileError
. - An
ignore
term in adefvocab
definition which actually is not a term of the vocabulary namespace is now considered an error. - When defining an alias for a term of vocabulary which would be invalid as an
Elixir term, the original term is now implicitly ignored and won't any longer
be returned by the
__terms__/0
function of aRDF.Vocabulary.Namespace
. RDF.Graph.build/2
blocks are now wrapped in a function, so the aliases and import no longer affect the caller context.alias
es in the caller context are still available in the build block, butimport
s not and must be reimported in the build block. Variables in the caller context are also no longer available in abuild
block but must be passed explicitly as bindings in a keyword list on the new optional first argument ofRDF.Graph.build/3
.RDF.BlankNode.Increment
was renamed toRDF.BlankNode.Generator.Increment
RDF.XSD.Datatype.Mismatch
exception was renamed toRDF.XSD.Datatype.MismatchError
for consistency reasons
- The
defvocab
macro can now be safely used in any module and guarantees cleanliness of the base module. So, a surrounding namespace (likeNS
) is no longer necessary. Although still useful for foreign vocabularies, this can be useful eg. to define aMyApplication.Vocab
module directly under the root module of the application. - The
:base_iri
specified indefvocab
can now be given in any form supported byRDF.IRI.new/1
. There are also no longer restrictions on the expression of this value. While previously the value had to be provided as a literal value, now any expression returning a value accepted byRDF.IRI.new/1
can be given (e.g. function calls, module attributes etc.). The:base_iri
also no longer has to end with a/
or#
. RDF.Data.merge/2
andRDF.Data.equal?/2
are now commutative, i.e. structs which implement theRDF.Data
protocol can be given also as the second argument (previously custom structs withRDF.Data
protocol implementations always had to be given as the first argument).- the
Inspect
implementation for theRDF.Literal
,RDF.PrefixMap
and
RDF.PropertyMap
structs now return a string with a valid Elixir expression that recreates the struct when evaluated - several performance improvements
- The RDF vocabulary namespaces used in
@prefix
and@base
declarations in aRDF.Graph.build
block no longer have to be written out, which had to be done previously even when parts of the module were available as an alias. - No warning on lowercased non-property resources in vocabularies
This version introduces a new graph builder DSL. See the new guide for an introduction.
- a
RDF.Graph
builder DSL available under theRDF.Graph.build/2
function - new
RDF.Sigils
~i
,~b
and~l
as variants of the~I
,~B
and~L
sigils, which support string interpolation RDF.Graph.new/2
andRDF.Graph.add/2
support the addition ofRDF.Dataset
sRDF.Description.empty?/1
,RDF.Graph.empty?/1
,RDF.Dataset.empty?/1
andRDF.Data.empty?/1
which are significantly faster thanEnum.empty?/1
- By replacing all
Enum.empty?/1
uses over the RDF data structures with these newempty?/1
functions throughout the code base, several functions benefit from this performance improvement.
- By replacing all
RDF.Description.first/2
now has aRDF.Description.first/3
variant which supports a default value- new guards in
RDF.Guards
:is_statement/1
andis_quad/1
RDF.PropertyMap.terms/1
andRDF.PropertyMap.iris/1
RDF.Graph.description/2
is no longer an alias forRDF.Graph.get/2
, but has a different behaviour now: it will return an empty description when no description for the requested subject exists in the graph- The inspect string of
RDF.Description
now includes the subject separately, so it can be seen also when the description is empty.
- When triples with an empty object list where added to an
RDF.Graph
, it included empty descriptions, which lead to inconsistent behaviour (for example it would be counted inRDF.Graph.subject_count/1
). - When an
RDF.Graph
contained empty descriptions these were rendered by theRDF.Turtle.Encoder
to a subject without predicates and objects, i.e. invalid Turtle. This actually shouldn't happen and is either caused by misuse or a bug. So instead, aRDF.Graph.EmptyDescriptionError
with a detailed message will be raised now when this case is detected.
The main feature of this version are the RDF.Resource.Generator
s.
For an introduction see this guide.
RDF.Resource.Generator
s which can be used to generate configurable ids:implicit_base
option on theRDF.Turtle.Encoder
:base_description
option on theRDF.Turtle.Encoder
- several new types:
RDF.Resource.t
for all node identifiers, i.e.RDF.IRI
s andRDF.BlankNode
sRDF.Triple.coercible
,RDF.Quad.coercible
,RDF.Star.Triple.coercible
andRDF.Star.Quad.coercible
for tuples which can be coerced to the respective statements
- some types were renamed for consistency reasons; the old types were deprecated
and will be removed
RDF.Statement.coercible_t
->RDF.Statement.coercible
RDF.Star.Statement.coercible_t
->RDF.Star.Statement.coercible
RDF.Triple.t_values
->RDF.Triple.mapping_value
RDF.Quad.t_values
->RDF.Quad.mapping_value
- the interface of
RDF.BlankNode.Generator.start_link/1
was fixed, so that generators can be started supervised
This release adds RDF-star support on the RDF data structures, the N-Triples, N-Quads, Turtle encoders and decoders and the BGP query engine. For an introduction read the new page on the RDF.ex guide. For more details on how to migrate from an earlier version read this wiki page.
Elixir versions < 1.10 are no longer supported
- Support for
RDF.PropertyMap
onRDF.Statement.new/2
andRDF.Statement.coerce/2
. RDF.Dataset.graph_count/1
- The
RDF.NQuads.Encoder
now supports a:default_graph_name
option, which allows to specify the graph name to be used as the default for triples from aRDF.Graph
orRDF.Description
.
- The
RDF.Turtle.Encoder
no longer supports the encoding ofRDF.Dataset
s.
You'll have to aggregate aRDF.Dataset
to aRDF.Graph
on your own now. - The
RDF.NQuads.Encoder
now uses theRDF.Graph.name/1
as the graph name for
the triples of aRDF.Graph
. Previously the triples of anRDF.Graph
were always encoded as part of default graph. You can use the new:default_graph_name
option and set it tonil
to get the old behaviour.
RDF.statement/1
,RDF.statement/3
andRDF.statement/4
constructor functions- the
:default_prefixes
configuration option now allows to set a{mod, fun}
tuple, with a function which should be called to determine the default prefixes - the
:default_base_iri
configuration option now allows to set a{mod, fun}
tuple, with a function which should be called to determine the default base IRI - support for Elixir 1.12 and OTP 24
- the Turtle encoder was encoding IRIs as prefixed names even when they were resulting in non-conform prefixed names
- the Turtle encoder didn't properly escape special characters in language-tagged literals
- the N-Triples and N-Quads encoders didn't properly escape special characters in both language-tagged and plain literals
- the
Inspect
protocol implementation forRDF.Diff
was causing an error when both graphs had prefixes defined
Note: In the canonical form of -0.0 in XSD doubles and floats the sign is removed in OTP versions < 24 although this is not standard conform. This has the consequence that the sign of -0.0 is also removed when casting these doubles and floats to decimals. These bugs won't be fixed. If you rely on the correct behavior in these cases, you'll have to upgrade to OTP 24 and a respective Elixir version.
:indent
option onRDF.Turtle.Encoder
, which allows to specify the number of spaces the output should be indented
- the performance of the
Enumerable
protocol implementations of the RDF data structures was significantly improved (for graphs almost 10x), which in turn increases the performance of all functions built on top of that, eg. the N-Triples and N-Quads encoders - improvement of the Inspect forms of the RDF data structures: the content is now enclosed in angle brackets and indented
- strings of the form
".0"
and"0."
weren't recognized as valid XSD float
and double literals - the Turtle encoder handles base URIs without a trailing slash or hash properly
(no longer raising a warning and ignoring them)
RDF.XSD.Base64Binary
datatype (@pukkamustard)
- a new option
:as_value
to enforce interpretation of an input string as a value instead of a lexical, which is needed on datatypes where the lexical space and the value space both consist of strings RDF.XSD.Date
andRDF.XSD.Time
both can now be initialized with tuples of an ElixirDate
resp.Time
value and a timezone string (previously XSD date and time values with time zones could only be created from strings)
Elixir versions < 1.9 are no longer supported
- general serialization functions for reading from and writing to streams and implementations for N-Triples and N-Quads (Turtle still to come)
- a
:gzip
option flag on allread_file/3
andwrite_file/3
functions allows to read and write all supported serialization formats from and to gzipped files (works also with the new possibility to read and write files via streams) RDF.Dataset.prefixes/1
for getting an aggregatedRDF.PrefixMap
over all graphsRDF.PrefixMap.put/3
for adding a prefix mapping and overwrite an existing oneRDF.BlankNode.value/1
for getting the internal string representation of a blank nodeRDF.IRI.in_namespace?/2
for determining whether an IRI lies in a namespace
- all
read_file/3
andwrite_file/3
functions onRDF.Serialization
and the modules of RDF serialization formats can use streaming via the:stream
flag option; forread_file/3
andwrite_file/3
it defaults tofalse
, while forread_file!/3
andwrite_file!/3
it defaults totrue
when the respective format supports streams - the Inspect form of the RDF data structures are now Turtle-based and respect
the usual
:limit
behaviour - more compact Inspect form for
RDF.PrefixMap
- the
RDF.Turtle.Encoder
acceptsRDF.Vocabulary.Namespace
modules asbase
- the performance of the
RDF.Turtle.Encoder
was improved (by using a for most use cases more efficient method for resolving IRIs to prefixed names) RDF.BlankNode.new/0
creates integer-based blank nodes, which is much more efficient in terms of performance and memory consumption than the previous ref-based blank nodes
RDF.BlankNode
s based on refs weren't serializable to TurtleRDF.Vocabulary.Namespace
s couldn't contain terms conflicting with functions from Elixirs Kernel module; most of them are supported now, while for the
remaining unsupported ones a proper error message is produced during compilation
The API of the all three RDF datastructures RDF.Dataset
, RDF.Graph
and
RDF.Description
were changed, so that the functions taking input data consist only
of one field in order to open the possibility of introducing options on these
functions. The supported ways with which RDF statements can be passed to the
RDF data structures were extended and unified to be supported across all functions
accepting input data. This includes also the way in which patterns for BGP queries
are specified. Also, the performance for adding data has been improved.
For an introduction on the new data structure API and the commonly supported input formats read the updated page on the RDF data structures in the guide. For more details on how to migrate from an earlier version read this wiki page.
RDF.PropertyMap
which allow definition of atoms for RDF properties. Such property maps can be provided to all RDF data structure functions accepting input data and BGP query patterns with the:context
opt, allowing the use of the atoms from the property map in the input data.- on
RDF.Description
RDF.Description.subject/1
RDF.Description.change_subject/2
- on
RDF.Graph
RDF.Graph.name/1
RDF.Graph.change_name/2
RDF.Graph.base_iri/1
RDF.Graph.prefixes/1
RDF.Graph.put_properties/3
- on
RDF.Dataset
RDF.Dataset.name/1
RDF.Dataset.change_name/2
RDF.Dataset.put_properties/3
RDF.IRI.append/2
- the format for the specification of BGP queries with
RDF.Graph.query/2
,RDF.Graph.query_stream/2
andRDF.Query.bgp/1
has been changed to be consistent
with the supported formats for input data in the rest of the library RDF.Description.new
now requires thesubject
to be passed always as first argument; if you want to add some initial data this must be done with the:init
option- The
put/3
functions onRDF.Graph
andRDF.Dataset
now overwrite all statements with same subject. Previously only statements with the same subject AND predicate were overwritten, which was probably not the expected behaviour, since it's not inline with the commonput
semantics in Elixir. A function with the previous behaviour was added onRDF.Graph
andRDF.Dataset
with theput_properties/3
function.- CAUTION: This means the
RDF.Graph.put/2
andRDF.Dataset.put/2
function have become more destructive now when not specified otherwise. - Note: Although one could argue, that following this route
RDF.Dataset.put/3
would consequently have to overwrite whole graphs, this was not implemented for practical reasons. It's probably not what's wanted in most cases.
- CAUTION: This means the
- The
Access
protocol implementation ofget_and_update/3
onRDF.Graph
andRDF.Dataset
previously relied on theput/2
functions with the old behaviour of overwriting only statements with the same subject and predicate, which was almost never the expected behaviour. This is fixed now by relying on the newput/2
behaviour. - the
values/2
functions ofRDF.Statement
,RDF.Triple
,RDF.Quad
,RDF.Description
,RDF.Graph
andRDF.Dataset
now accept on their second argument an optionalRDF.PropertyMap
which will be used to map predicates accordingly; the variant of thesevalues/2
functions to provide a custom mapping function was extracted into a new functionmap/2
on all of these modules - for consistency reasons the internal
:id
struct field ofRDF.BlankNode
was renamed to:value
- allow the
base_iri
ofRDF.Vocabulary.Namespace
s to end with a.
to support vocabularies which use dots in the IRIs for further structuring (e.g. CIM-based formats like CGMES) RDF.Triple.new/1
now also accepts four-element tuples and simple ignores fourth elementRDF.Quad.new/1
now also accepts three-element tuples and simple assumes the fourth element to benil
- the
put
functions onRDF.Description
,RDF.Graph
andRDF.Dataset
didn't add all statements properly under certain circumstances RDF.Graph.put/2
ignores empty descriptions; this should be the final piece to ensure thatRDF.Graph
s never contain empty descriptions, which would distort results of functions likeRDF.Graph.subjects/1
,RDF.Graph.subject_count/1
,RDF.Graph.descriptions/1
- the Turtle encoder can now produce partial Turtle documents with the
:only
option and any combination of the following values::triples
,:directives
,:base
,:prefixes
- the style of the Turtle directives produced by the Turtle encoder can be
switched to SPARQL style with the option
:directive_style
and the value:sparql
- the most common conflict resolution strategies on
RDF.PrefixMap.merge/3
can now be chosen directly with the atoms:ignore
and:overwrite
RDF.PrefixMap.prefixed_name/2
to convert an IRI to a prefixed nameRDF.PrefixMap.prefixed_name_to_iri/2
to convert a prefixed name to an IRI
- when serializing a
RDF.Dataset
with the Turtle encoder the prefixes of all of its graphs are used now
- adding an empty
RDF.Description
with a subject to an emptyRDF.Graph
resulted in an invalid non-empty graph (@pukkamustard)
- query functions for basic graph pattern matching (incl. streaming-support)
RDF literals and their datatypes were completely redesigned to support derived XSD datatypes and allow for defining custom datatypes. For an introduction on how literals work now read the updated page on literals in the guide. For more details on how to migrate from an earlier version read this wiki page.
Elixir versions < 1.8 are no longer supported
- a lot of new datatypes like
xsd:float
,xsd:byte
orxsd:anyURI
-- all numeric XSD datatypes are now available; see this page of the API documentation for an up-to-date list of all supported and missing XSD datatypes - an implementation of XSD facet system now makes it easy to define own custom datatypes via restriction of the existing XSD datatypes
RDF.Literal.update/2
updates the value of aRDF.Literal
without changing anything else, eg. the language or datatype
- the
RDF.Literal
struct now consists entirely of a datatype-specific structs in theliteral
field, which besides being more memory-efficient (since literals no longer consist of all possible fields a literal might have), allows pattern matching now on the datatype of literals. - RDF XSD datatypes are now defined in the
RDF.XSD
namespace - alias constructor functions for the XSD datatypes are now defined on
RDF.XSD
matches?
,less_than?
,greater_than
as higher level functions were removed from theRDF.Literal.Datatype
modulesless_than?
,greater_than?
now always return a boolean and no longernil
when incomparable; you can still determine if two terms are comparable by checking ifcompare/2
returnsnil
- the
language
option is not supported on theRDF.XSD.String.new/2
constructor - the
language
option onRDF.Literal.new/2
is no longer ignored if it's empty (nil
or""
), so this either produces an invalidRDF.LangString
now or, if anotherdatatype
is provided will fail with anArgumentError
canonical
now performs implicit coercions when passed plain Elixir values- the inspect format for literals was changed and is now much more informative and uniform, since you now always see the value, the lexical form and if the literal is valid
RDF.Namespace.resolve_term/1
now returns ok or error tuples, but a new functionRDF.Namespace.resolve_term!/1
with the old behaviour was added
- numeric operations on invalid numeric literals no longer fail, but return
nil
instead - Datetimes preserve the original lexical form of the timezone when casting from a date
- BEAM error warnings when trying to use top-level modules as vocabulary terms
- proper typespecs so that Dialyzer passes without warnings (@rustra)
RDF.XSD.Time
didn't handle 24h overflows with an offset correctly
RDF.Diff
data structure for diffs between RDF graphs and descriptionsRDF.Description.update/4
updates the objects of a predicate in a description with a custom update functionRDF.Graph.update/4
updates the descriptions of a subject in a graph with a custom update functionRDF.Description.take/2
creates a description from another one by limiting its statements to a set of predicatesRDF.Graph.take/3
creates a graph from another one by limiting its statements to a set of subjects and optionally also a set of predicatesRDF.Graph.clear/1
removes the triples from a graph- Mix formatter configuration for using
defvocab
without parens
RDF.Serialization.Writer.write_file/4
which is the basis used by all thewrite_file/3
andwrite_file!/3
functions of all serialization format modules likeRDF.NTriples
,RDF.Turtle
,JSON.LD
etc. now opens file in a different mode: it no longer opens them with the:utf8
option. First, this by default slowed down the writing, but more importantly could lead to unexpected encoding issues. This is a breaking change: If your code relied on this file mode, you can get the old behaviour, by specifying thefile_mode
on these functions accordingly as[:utf8, :write, :exclusive]
. For example, to write a Turtle file with the old behaviour, you can do it like this:
RDF.Turtle.write_file!(some_data, some_path, file_mode: ~w[utf8 write exclusive]a)
- field
base_iri
onRDF.Graph
structure which can be set via newbase_iri
option onRDF.Graph.new
or the new functionsRDF.Graph.set_base_iri/2
andRDF.Graph.clear_base_iri/1
RDF.Graph.clear_metadata/1
which clears the base IRI and the prefixesRDF.IRI.coerce_base/1
which coerces base IRIs; as opposed toRDF.IRI.new/1
it also accepts bareRDF.Vocabulary.Namespace
modules
RDF.Turtle.Decoder
saves the base IRI in theRDF.Graph
nowRDF.Turtle.Encoder
now takes the base IRI to be used during serialization in
the following order of precedence:- from the
base
option or its new aliasbase_iri
- from the
base_iri
field of the given graph - from the
RDF.default_base_iri
returning the one from the application configuration
- from the
RDF.PrefixMap.new
andRDF.PrefixMap.add
now also accepts terms fromRDF.Vocabulary.Namespace
s as namespaces
- Vocabulary namespace modules weren't always detected properly
RDF.IRI.to_string/1
returns the string representation of anRDF.IRI
(implicitly resolving vocabulary namespace terms)RDF.Literal.matches?/3
for XQuery regex pattern matchingRDF.Decimal.digit_count/1
andRDF.Decimal.fraction_digit_count/1
for
determining the number of digits of decimal literals
- language literals were not properly unescaped during Turtle parsing
RDF.Literal.new/1
can take decimals and infers the datatypexsd:decimal
correctlytrue
andfalse
with capital letters are no longer validRDF.Boolean
s following the XSD specification; the same applies for booleans in Turtle+INF
is no longer a validRDF.Double
(positive infinity doesn't expect a sign)- slightly improve output of errors during parsing of Turtle, N-Triples and N-Quads
see here for upgrading notes to RDF.ex 0.6
RDF.PrefixMap
- prefix management of
RDF.Graph
s:- the structure now has a
prefixes
field with an optionalRDF.PrefixMap
- new functions
add_prefixes/2
,delete_prefixes/2
andclear_prefixes/1
- the structure now has a
- configurable
RDF.default_prefixes
RDF.Description.equal?/2
,RDF.Graph.equal?/2
,RDF.Dataset.equal?/2
andRDF.Data.equal?/2
- the constructor functions for
RDF.Graph
s andRDF.Dataset
s now take the graph name resp. dataset name through aname
option, instead of the first argument RDF.Graph.new
supports an additionalprefixes
argument to initialize theprefixes
field- when
RDF.Graph.add
andRDF.Graph.put
are called with another graph, its prefixes are merged RDF.Turtle.Decoder
saves the prefixes nowRDF.Turtle.Encoder
now takes the prefixes to be serialized in the following order of precedence:- from the
prefixes
option (as before) - from the
prefixes
field of the given graph - from the
RDF.default_prefixes
- from the
- drop support for OTP < 20, since prefixes can consist of UTF characters which are not supported in atoms on these versions
- issue with Elixir 1.8
RDF.write_file
andRDF.write_file!
delegators had wrong signatures
RDF.Triple.valid?/1
,RDF.Quad.valid?/1
andRDF.Statement.valid?/1
, which validate if a tuple is a valid RDF triple or RDF quad
RDF.Term.value/1
returning the native Elixir value of a RDF termRDF.Statement.values/1
,RDF.Triple.values/1
andRDF.Quad.values/1
returning a tuple ofRDF.Term.value/1
converted native Elixir values from a tuple of RDF termsRDF.Description.values/1
,RDF.Graph.values/1
,RDF.Dataset.values/1
andRDF.Data.values/1
returning a map ofRDF.Term.value/1
converted native Elixir values from the respective structure of RDF terms- for all of aforementioned
values/1
functions a variantvalues/2
which allows to specify custom mapping function to be applied when creating the resp. structure RDF.Literal.compare/2
,RDF.Literal.less_than?/2
andRDF.Literal.greater_than?/2
forRDF.Datatype
aware comparisons ofRDF.Literal
s
RDF.DateTime.equal_value?/2
andRDF.Date.equal_value?/2
did not handle timezones correctly-00:00
is a valid timezone offset onRDF.DateTime
- generated Erlang output files of Leex and Yecc are excluded from Hex package
Elixir versions < 1.6 are no longer supported
- Possibility to execute simple SPARQL queries against
RDF.Graph
s with SPARQL 0.2 - New
RDF.Term
protocol implemented for all structs representing RDF nodes and
all native Elixir datatypes which are coercible to those modules. For now, it
mainly offers, besides the coercion, just the functionRDF.Term.equal?/2
andRDF.Term.equal_value?/2
for term- and value comparisons. - New
RDF.Decimal
datatype forxsd:decimal
literals and support for decimal literals in Turtle encoder RDF.Numeric
module with a list of all numeric datatypes and shared functions for all numeric literals, e.g. arithmetic functions- Various new
RDF.Datatype
functionRDF.Datatype.cast/1
for casting betweenRDF.Literal
s as specified in the XSD spec on allRDF.Datatype
s- logical operators and the Effective Boolean Value (EBV) coercion algorithm
from the XPath and SPARQL specs on
RDF.Boolean
- various functions on the
RDF.DateTime
andRDF.Time
datatypes RDF.LangString.match_language?/2
- Many new convenience functions on the top-level
RDF
module- constructors for all the supported
RDF.Datatype
s - constant functions
RDF.true
andRDF.false
for the two booleanRDF.Literal
values
- constructors for all the supported
RDF.Literal.Guards
which allow pattern matching of common literal datatypesRDF.BlankNode.Generator
- Possibility to configure an application-specific default base IRI; for now it
is used only on reading of RDF serializations (when no
base
specified)
RDF.String.new/2
andRDF.String.new!/2
produce ardf:langString
when given a language tag- Some of the defined structs now enforce keys at compile-time (via Elixirs
@enforce_keys
feature) when not setting the corresponding fields would lead to invalid structs, namely the following fields:RDF.IRI.value
RDF.BlankNode.id
RDF.Description.subject
RDF.List.head
RDF.resource?/1
does not fail anymore when called with unresolvable atoms but returnsfalse
insteadRDF.IRI.absolute/2
does not fail with aFunctionClauseError
when the given base is not absolute, but returnsnil
insteadRDF.DateTime
andRDF.Time
store microsecondsRDF.DateTime
: '24:00:00' is a valid time in a xsd:dateTime; the dateTime value so represented is the first instant of the following dayRDF.LangString
: non-strings or the empty string as language produce invalid literals
RDF.Literal.new!/2
which fails when creating an invalid literal
RDF.Literal.new/2
can createrdf:langString
literals without failing, they
are simply invalid; if you want to fail without a language tag use the newRDF.Literal.new!/2
function
- renamed
RDF.Serialization
behaviour toRDF.Serialization.Format
; the newRDF.Serialization
module contains just simple RDF serialization related functions - renamed
RDF.Serialization.Format
functioncontent_type/0
tomedia_type/0
- moved
RDF.Reader
andRDF.Writer
intoRDF.Serialization
module - removed the limitation to serialization formats defined in the core RDF.ex package
for use as a source of
RDF.Vocabulary.Namespace
s; so you can now also define vocabulary namespaces from JSON-LD files for example, provided that the corresponding Hex package is defined as a dependency
RDF.Serialization.Format
s define aname
atom- all
RDF.Serialization.Reader
andRDF.Serialization.Writer
functions are now available on theRDF.Serialization
module (or aliased on the top-levelRDF
module) and the format can be specified instead of aRDF.Serialization.Format
argument, via theformat
ormedia_type
option or in case of*_file
functions, without explicit specification of the format, but inferred from file name extension instead; see the updated README section about RDF serializations - the following functions to access available
RDF.Serialization.Format
s:RDF.Serialization.formats/0
RDF.Serialization.available_formats/0
RDF.Serialization.format/1
RDF.Serialization.format_by_media_type/1
RDF.Serialization.format_by_extension/1
Collectable
implementations for allRDF.Data
structures, so they can be used as destinations ofEnum.into
andfor
comprehensionsRDF.Quad
can be created from triple andRDF.Triple
can be created from quadRDF.Statement.map/2
which creates a statement with mapped nodes from another statementRDF.Statement
functions to get the coerced components of a statement
- Fix
unescape_map
inparse_helper
for Elixir 1.6 (@ajkeys)
RDF.IRI
as a more suitable URI/IRI representation for RDF, bringing enormous performance and memory consumption benefits (see here for the details about the improvements)
- use
RDF.IRI
instead of ElixirsURI
everywhere - use the term iri instead of uri consistently, leading to the following
function renamings:
base_iri
instead ofbase_uri
for the definition ofRDF.Vocabulary.Namespace
s__base_iri__
instead of__base_uri__
in allRDF.Vocabulary.Namespace
s__iris__
instead of__uris__
in allRDF.Vocabulary.Namespace
sRDF.IRI.InvalidError
instead ofRDF.InvalidURIError
RDF.Literal.InvalidError
instead ofRDF.InvalidLiteralError
RDF.Namespace.InvalidVocabBaseIRIError
instead ofRDF.Namespace.InvalidVocabBaseURIError
- show compilation message of vocabulary namespaces always to be able to relate resp. errors and warnings
- when trying to resolve a term from an undefined module a
RDF.Namespace.UndefinedTermError
exception
Elixir versions < 1.4 are no longer supported
- full Turtle support
RDF.List
structure for the representation of RDF listsdescribes?/1
onRDF.Data
protocol and all RDF data structures which checks
if statements about a given resource existRDF.Data.descriptions/1
which returns all descriptions within an RDF data structureRDF.Description.first/2
which returns a single object to a predicate of aRDF.Description
RDF.Description.objects/2
now supports a custom filter functionRDF.bnode?/1
which checks if the given value is a blank node
- Rename
RDF.Statement.convert*
functions toRDF.Statement.coerce*
RDF.uri/1
and URI parsing of N-Triples and N-Quads decoders preserve empty fragments- booleans weren't recognized as coercible literals on object positions
- N-Triples and N-Quads decoder didn't handle escaping properly
- Add
src
directory to package files.
Initial release
Note: This version is not usable, since the src
directory is not part of the
package, which has been immediately fixed on version 0.1.1.