rdf-ex/lib/rdf/vocabulary/namespace/vocabulary_namespace.ex

285 lines
7.8 KiB
Elixir
Raw Normal View History

defmodule RDF.Vocabulary.Namespace do
@moduledoc """
2021-01-13 15:55:24 +00:00
An RDF vocabulary as a `RDF.Namespace`.
`RDF.Vocabulary.Namespace` modules represent a RDF vocabulary as a `RDF.Namespace`.
They can be defined with the `defvocab/2` macro of this module.
RDF.ex comes with predefined modules for some fundamental vocabularies in
the `RDF.NS` module.
"""
alias RDF.{Description, Graph, Dataset, Vocabulary, Namespace, IRI}
2022-06-04 23:20:27 +00:00
import RDF.Vocabulary.Namespace.{TermMapping, CaseValidation}
import RDF.Vocabulary, only: [term_to_iri: 2, extract_terms: 2]
2020-10-13 13:54:43 +00:00
@type t :: module
defmacro __using__(_opts) do
quote do
import unquote(__MODULE__)
end
end
@doc """
2022-06-04 23:20:27 +00:00
Defines a `RDF.Vocabulary.Namespace` module for a RDF vocabulary.
"""
2022-06-04 23:20:27 +00:00
defmacro defvocab({:__aliases__, _, [module]}, spec) do
env = __CALLER__
module = Namespace.module(env, module)
{base_iri, spec} = Keyword.pop(spec, :base_iri)
{input, opts} = input(module, spec)
no_warn_undefined = if Keyword.get(opts, :strict) == false, do: no_warn_undefined(module)
[
2020-06-29 08:37:42 +00:00
quote do
2022-06-04 23:20:27 +00:00
result =
create!(
unquote(module),
unquote(base_iri),
unquote(input),
unquote(Macro.escape(env)),
unquote(opts)
|> Keyword.put(:moduledoc, Module.delete_attribute(__MODULE__, :vocabdoc))
)
alias unquote(module)
result
2020-06-29 08:37:42 +00:00
end
2022-06-04 23:20:27 +00:00
| List.wrap(no_warn_undefined)
]
end
2022-06-04 23:20:27 +00:00
@required_input_opts ~w[file data terms]a
2022-06-04 23:20:27 +00:00
defp input(module, opts), do: do_input(module, nil, opts, @required_input_opts)
2020-06-29 08:37:42 +00:00
2022-06-04 23:20:27 +00:00
defp do_input(module, nil, _, []) do
raise ArgumentError,
"none of #{Enum.join(@required_input_opts, ", ")} are given on defvocab for #{module}"
end
2022-06-04 23:20:27 +00:00
defp do_input(_, input, opts, []), do: {input, opts}
2020-06-29 08:37:42 +00:00
2022-06-04 23:20:27 +00:00
defp do_input(module, input, opts, [opt | rest]) do
case Keyword.pop(opts, opt) do
{nil, opts} ->
do_input(module, input, opts, rest)
2020-06-29 08:37:42 +00:00
2022-06-04 23:20:27 +00:00
{value, opts} ->
if input do
raise ArgumentError,
"multiple values for #{Enum.join(@required_input_opts, ", ")} are given on defvocab for #{module}"
else
do_input(module, value, opts, rest)
end
end
end
2022-06-04 23:20:27 +00:00
defp no_warn_undefined(module) do
quote do
@compile {:no_warn_undefined, unquote(module)}
end
end
2022-06-04 23:20:27 +00:00
def create(module, base_uri, vocab, location, opts)
2020-06-29 08:37:42 +00:00
2022-06-04 23:20:27 +00:00
def create(module, base_uri, vocab_path, location, opts) when is_binary(vocab_path) do
file = vocab_from_path(vocab_path)
2020-06-29 08:37:42 +00:00
2022-06-04 23:20:27 +00:00
create(
module,
base_uri,
RDF.read_file!(file, base_iri: nil),
location,
Keyword.put(opts, :file, file)
)
end
2022-06-04 23:20:27 +00:00
def create(module, base_uri, %struct{} = data, location, opts)
when struct in [Graph, Dataset] do
do_create(
module,
base_uri,
extract_terms(data, base_uri),
location,
Keyword.put(opts, :data, data)
)
end
2022-06-04 23:20:27 +00:00
def create(module, base_uri, terms, location, opts) do
{terms, opts} = extract_aliases(terms, opts)
do_create(module, base_uri, terms, location, opts)
end
2022-06-04 23:20:27 +00:00
defp do_create(module, base_uri, terms, location, opts) do
base_uri = normalize_base_uri(base_uri)
strict = Keyword.get(opts, :strict, true)
ignored_terms = Keyword.get(opts, :ignore, []) |> normalize_ignored_terms()
{terms, aliases} = normalize_terms(module, terms, ignored_terms, strict, opts)
2022-06-04 23:20:27 +00:00
{terms, ignored_terms} =
{terms, ignored_terms}
|> validate!(opts)
|> validate_case!(Keyword.get(opts, :data), base_uri, aliases, opts)
2022-06-04 23:20:27 +00:00
Namespace.Builder.create(
module,
term_mapping(base_uri, terms, ignored_terms),
location,
if strict do
opts
else
Keyword.put(opts, :add_after, define_undefined_function_handler())
end
|> Keyword.put(
:namespace_functions,
define_namespace_functions(base_uri, terms, ignored_terms, strict, opts)
)
|> Keyword.put(:skip_normalization, true)
)
end
def create!(module, base_uri, vocab, env, opts) do
case create(module, base_uri, vocab, env, opts) do
{:ok, result} -> result
{:error, error} -> raise error
end
end
2022-06-04 23:20:27 +00:00
def define_namespace_functions(base_iri, terms, ignored_terms, strict, opts) do
file = Keyword.get(opts, :file)
2022-06-04 23:20:27 +00:00
quote do
if unquote(file) do
@external_resource unquote(file)
end
2022-06-04 23:20:27 +00:00
@strict unquote(strict)
@spec __strict__ :: boolean
def __strict__, do: @strict
@base_iri unquote(base_iri)
@spec __base_iri__ :: String.t()
def __base_iri__, do: @base_iri
@terms unquote(Macro.escape(terms))
@impl Elixir.RDF.Namespace
def __terms__, do: Map.keys(@terms)
@spec __term_aliases__ :: [atom]
def __term_aliases__, do: RDF.Vocabulary.Namespace.TermMapping.aliases(@terms)
@ignored_terms unquote(Macro.escape(ignored_terms))
@doc """
Returns all known IRIs of the vocabulary.
"""
@impl Elixir.RDF.Namespace
def __iris__ do
@terms
|> Enum.map(fn
{term, true} -> term_to_iri(@base_iri, term)
{_alias, term} -> term_to_iri(@base_iri, term)
end)
|> Enum.uniq()
end
2020-06-29 08:37:42 +00:00
2022-06-04 23:20:27 +00:00
@impl Elixir.RDF.Namespace
@dialyzer {:nowarn_function, __resolve_term__: 1}
def __resolve_term__(term) do
case @terms[term] do
nil ->
if @strict or MapSet.member?(@ignored_terms, term) do
{:error,
%Elixir.RDF.Namespace.UndefinedTermError{
message: "undefined term #{term} in strict vocabulary #{__MODULE__}"
}}
else
{:ok, term_to_iri(@base_iri, term)}
end
2022-06-04 23:20:27 +00:00
true ->
{:ok, term_to_iri(@base_iri, term)}
2020-06-29 08:37:42 +00:00
2022-06-04 23:20:27 +00:00
original_term ->
{:ok, term_to_iri(@base_iri, original_term)}
2020-06-29 08:37:42 +00:00
end
2022-06-04 23:20:27 +00:00
end
end
end
2022-06-04 23:20:27 +00:00
def define_undefined_function_handler do
quote do
def unquote(:"$handle_undefined_function")(term, []) do
if MapSet.member?(@ignored_terms, term) do
raise UndefinedFunctionError
end
2022-06-04 23:20:27 +00:00
term_to_iri(@base_iri, term)
end
2022-06-04 23:20:27 +00:00
def unquote(:"$handle_undefined_function")(term, [subject | objects]) do
if MapSet.member?(@ignored_terms, term) do
raise UndefinedFunctionError
end
2020-06-29 08:37:42 +00:00
2022-06-04 23:20:27 +00:00
objects =
case objects do
[objects] when is_list(objects) -> objects
_ -> objects
end
2020-06-29 08:37:42 +00:00
2022-06-04 23:20:27 +00:00
case subject do
%Description{} -> subject
_ -> Description.new(subject)
end
|> Description.add({term_to_iri(@base_iri, term), objects})
2020-06-29 08:37:42 +00:00
end
end
end
2022-06-04 23:20:27 +00:00
defp vocab_from_path(vocab_path) do
cond do
File.exists?(vocab_path) ->
vocab_path
2022-06-04 23:20:27 +00:00
File.exists?(expanded_filename = Path.expand(vocab_path, Vocabulary.path())) ->
expanded_filename
2022-06-04 23:20:27 +00:00
true ->
raise File.Error, path: vocab_path, action: "find", reason: :enoent
end
end
defp normalize_base_uri(%IRI{} = base_iri), do: IRI.to_string(base_iri)
defp normalize_base_uri(base_uri) when is_binary(base_uri) do
if IRI.valid?(base_uri) do
2022-06-04 23:20:27 +00:00
base_uri
else
raise RDF.Namespace.InvalidVocabBaseIRIError, "invalid base IRI: #{inspect(base_uri)}"
end
end
defp normalize_base_uri(base_uri) do
base_uri |> IRI.new() |> normalize_base_uri()
rescue
[Namespace.UndefinedTermError, IRI.InvalidError, FunctionClauseError] ->
reraise RDF.Namespace.InvalidVocabBaseIRIError,
"invalid base IRI: #{inspect(base_uri)}",
__STACKTRACE__
end
2019-03-30 01:01:30 +00:00
@doc false
2020-03-02 22:42:15 +00:00
@spec vocabulary_namespace?(module) :: boolean
2019-03-30 01:01:30 +00:00
def vocabulary_namespace?(name) do
case Code.ensure_compiled(name) do
{:module, name} -> function_exported?(name, :__base_iri__, 0)
_ -> false
end
2019-03-30 01:01:30 +00:00
end
end