All Downloads are FREE. Search and download functionalities are using the official Maven repository.

no.hasmac.jsonld.context.TermDefinitionBuilder Maven / Gradle / Ivy

The newest version!
/*
 * Copyright 2020 APICATALOG and HASMAC.
 *
 * Licensed under the Apache License, Version 2.0 (the "License");
 * you may not use this file except in compliance with the License.
 * You may obtain a copy of the License at
 *
 * http://www.apache.org/licenses/LICENSE-2.0
 *
 * Unless required by applicable law or agreed to in writing, software
 * distributed under the License is distributed on an "AS IS" BASIS,
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 * See the License for the specific language governing permissions and
 * limitations under the License.
 * 
 * SPDX-License-Identifier: Apache-2.0
 */

package no.hasmac.jsonld.context;

import jakarta.json.JsonArray;
import jakarta.json.JsonObject;
import jakarta.json.JsonString;
import jakarta.json.JsonValue;
import no.hasmac.jsonld.JsonLdError;
import no.hasmac.jsonld.JsonLdErrorCode;
import no.hasmac.jsonld.JsonLdVersion;
import no.hasmac.jsonld.StringUtils;
import no.hasmac.jsonld.json.JsonUtils;
import no.hasmac.jsonld.lang.BlankNode;
import no.hasmac.jsonld.lang.CompactUri;
import no.hasmac.jsonld.lang.DirectionType;
import no.hasmac.jsonld.lang.Keywords;
import no.hasmac.jsonld.lang.LanguageTag;
import no.hasmac.jsonld.uri.UriUtils;

import java.net.URI;
import java.util.ArrayList;
import java.util.Collection;
import java.util.Collections;
import java.util.Map;
import java.util.Set;
import java.util.logging.Level;
import java.util.logging.Logger;

/**
 * @see Create
 * Term Definition
 */
public final class TermDefinitionBuilder {

    private static final Logger LOGGER = Logger.getLogger(TermDefinitionBuilder.class.getName());
    public static final Set KEYWORDS = Set.of(Keywords.ID, Keywords.REVERSE, Keywords.CONTAINER,
            Keywords.CONTEXT, Keywords.DIRECTION, Keywords.INDEX, Keywords.LANGUAGE, Keywords.NEST, Keywords.PREFIX,
            Keywords.PROTECTED, Keywords.TYPE);

    // mandatory
    private final ActiveContext activeContext;

    private final JsonObject localContext;

    private final Map defined;

    // optional
    private URI baseUrl;

    private boolean protectedFlag;

    private boolean overrideProtectedFlag;

    private Collection remoteContexts;

    private TermDefinitionBuilder(ActiveContext activeContext, JsonObject localContext, Map defined) {
        this.activeContext = activeContext;
        this.localContext = localContext;
        this.defined = defined;

        // default values
        this.baseUrl = null;
        this.protectedFlag = false;
        this.overrideProtectedFlag = false;
        this.remoteContexts = new ArrayList<>();
    }

    public static TermDefinitionBuilder with(ActiveContext activeContext, JsonObject localContext, Map defined) {
        return new TermDefinitionBuilder(activeContext, localContext, defined);
    }

    public TermDefinitionBuilder baseUrl(URI baseUrl) {
        this.baseUrl = baseUrl;
        return this;
    }

    public TermDefinitionBuilder protectedFlag(boolean protectedFlag) {
        this.protectedFlag = protectedFlag;
        return this;
    }

    public TermDefinitionBuilder overrideProtectedFlag(boolean overrideProtectedFlag) {
        this.overrideProtectedFlag = overrideProtectedFlag;
        return this;
    }

    public TermDefinitionBuilder remoteContexts(Collection remoteContexts) {
        this.remoteContexts = remoteContexts;
        return this;
    }

    public void create(final String term) throws JsonLdError {

        if (StringUtils.isBlank(term)) {
            throw new JsonLdError(JsonLdErrorCode.INVALID_TERM_DEFINITION);
        }

        // 1.
        Boolean isDefined = defined.get(term);
        if (isDefined != null) {
            if (isDefined) {
                return;
            }

            throw new JsonLdError(JsonLdErrorCode.CYCLIC_IRI_MAPPING);
        }

        // 2.
        defined.put(term, false);

        // 3.
        final JsonValue value = localContext.get(term);

        // 4.
        if (Keywords.TYPE.equals(term)) {

            if (activeContext.inMode(JsonLdVersion.V1_0)) {
                throw new JsonLdError(JsonLdErrorCode.KEYWORD_REDEFINITION);
            }

            if (JsonUtils.isObject(value)) {

                final JsonObject map = value.asJsonObject();

                if (map.size() == 1 && map.containsKey(Keywords.CONTAINER)) {

                    JsonValue container = map.get(Keywords.CONTAINER);
                    if (JsonUtils.isNotString(container)
                            || !Keywords.SET.equals(((JsonString) container).getString())) {
                        throw new JsonLdError(JsonLdErrorCode.KEYWORD_REDEFINITION);
                    }

                } else if (map.size() == 2 && map.containsKey(Keywords.CONTAINER)
                        && map.containsKey(Keywords.PROTECTED)) {

                    final JsonValue containerValue = map.get(Keywords.CONTAINER);

                    if (!JsonUtils.contains(Keywords.SET, containerValue)) {
                        throw new JsonLdError(JsonLdErrorCode.KEYWORD_REDEFINITION);
                    }

                } else if (map.size() != 1 || !map.containsKey(Keywords.PROTECTED)) {
                    throw new JsonLdError(JsonLdErrorCode.KEYWORD_REDEFINITION);
                }

            } else {
                throw new JsonLdError(JsonLdErrorCode.KEYWORD_REDEFINITION);
            }

            // 5.
        } else if (Keywords.contains(term)) {
            throw new JsonLdError(JsonLdErrorCode.KEYWORD_REDEFINITION, "A keyword [" + term + "] redefinition has been detected.");

        } else if (Keywords.matchForm(term)) {
            LOGGER.log(Level.WARNING, "Term [{0}] has form of a keyword. Keywords cannot be overridden.", term);
            if (activeContext.getOptions().isExceptionOnWarning()) {
                throw new JsonLdError(JsonLdErrorCode.UNSPECIFIED, "Term [" + term + "] has form of a keyword. Keywords cannot be overridden.");
            }
            return;
        }

        // 6.
        final TermDefinition previousDefinition = activeContext.removeTerm(term).orElse(null);

        final Map valueObject;

        final boolean simpleTerm;
        final JsonValue idValue;

        // 7.
        if (JsonUtils.isNull(value)) {

            valueObject = Collections.emptyMap();
            idValue = JsonValue.NULL;
            simpleTerm = false;

            // 8.
        } else if (JsonUtils.isString(value)) {

            valueObject = Collections.emptyMap();
            idValue = value;
            simpleTerm = true;

            // 9.
        } else if (JsonUtils.isObject(value)) {

            valueObject = value.asJsonObject();
            idValue = valueObject.get(Keywords.ID);
            simpleTerm = false;

        } else {
            throw new JsonLdError(JsonLdErrorCode.INVALID_TERM_DEFINITION);
        }

        // 10.
        final TermDefinition definition = new TermDefinition(false, protectedFlag, false);

        // 11.
        if (valueObject.containsKey(Keywords.PROTECTED)) {

            if (activeContext.inMode(JsonLdVersion.V1_0)) {
                throw new JsonLdError(JsonLdErrorCode.INVALID_TERM_DEFINITION);
            }

            final JsonValue protectedValue = valueObject.get(Keywords.PROTECTED);

            if (JsonUtils.isNotBoolean(protectedValue)) {
                throw new JsonLdError(JsonLdErrorCode.INVALID_KEYWORD_PROTECTED_VALUE);
            }

            definition.setProtected(JsonUtils.isTrue(protectedValue));
        }

        // 12.
        if (valueObject.containsKey(Keywords.TYPE)) {

            // 12.1.
            final JsonValue type = valueObject.get(Keywords.TYPE);

            if (JsonUtils.isNotString(type)) {
                throw new JsonLdError(JsonLdErrorCode.INVALID_TYPE_MAPPING);
            }

            // 12.2.
            final String expandedTypeString =
                    activeContext
                            .uriExpansion()
                            .localContext(localContext)
                            .defined(defined)
                            .vocab(true)
                            .expand(((JsonString) type).getString());

            if (expandedTypeString == null) {
                throw new JsonLdError(JsonLdErrorCode.INVALID_TYPE_MAPPING);
            }

            // 12.3.
            if (((Keywords.JSON.equals(expandedTypeString) || Keywords.NONE.equals(expandedTypeString))
                    && activeContext.inMode(JsonLdVersion.V1_0))
                    // 12.4.
                    || (Keywords.noneMatch(expandedTypeString, Keywords.ID, Keywords.JSON, Keywords.NONE, Keywords.VOCAB)
                    && UriUtils.isNotAbsoluteUri(expandedTypeString, true))) {
                throw new JsonLdError(JsonLdErrorCode.INVALID_TYPE_MAPPING);
            }

            // 12.5.
            definition.setTypeMapping(expandedTypeString);
        }

        // 13.
        if (valueObject.containsKey(Keywords.REVERSE)) {

            // 13.1.
            if (valueObject.containsKey(Keywords.ID) || valueObject.containsKey(Keywords.NEST)) {
                throw new JsonLdError(JsonLdErrorCode.INVALID_REVERSE_PROPERTY);
            }

            final JsonValue reverse = valueObject.get(Keywords.REVERSE);

            // 13.2.
            if (JsonUtils.isNotString(reverse)) {
                throw new JsonLdError(JsonLdErrorCode.INVALID_IRI_MAPPING);
            }

            final String reverseString = ((JsonString) reverse).getString();

            // 13.3.
            if (Keywords.matchForm(reverseString)) {
                LOGGER.log(Level.WARNING, "The value [{0}] associated with @reverse cannot have form of a keyword.", reverseString);
                if (activeContext.getOptions().isExceptionOnWarning()) {
                    throw new JsonLdError(JsonLdErrorCode.UNSPECIFIED, "The value [" + reverseString + "] associated with @reverse cannot have form of a keyword.");
                }
                return;
            }

            // 13.4.
            definition.setUriMapping(
                    activeContext
                            .uriExpansion()
                            .localContext(localContext)
                            .defined(defined)
                            .vocab(true)
                            .expand(reverseString));

            if (UriUtils.isNotURI(definition.getUriMapping())) {
                throw new JsonLdError(JsonLdErrorCode.INVALID_IRI_MAPPING);
            }

            // 13.5.
            if (valueObject.containsKey(Keywords.CONTAINER)) {

                final JsonValue container = valueObject.get(Keywords.CONTAINER);

                if (JsonUtils.isNotString(container) && JsonUtils.isNotNull(container)) {
                    throw new JsonLdError(JsonLdErrorCode.INVALID_REVERSE_PROPERTY);
                }

                if (JsonUtils.isString(container)) {

                    final String containerString = ((JsonString) container).getString();

                    if (Keywords.anyMatch(containerString, Keywords.SET, Keywords.INDEX)) {
                        definition.addContainerMapping(containerString);

                    } else {
                        throw new JsonLdError(JsonLdErrorCode.INVALID_REVERSE_PROPERTY);
                    }
                }
            }

            // 13.6.
            definition.setReverseProperty(true);

            // 13.7.
            activeContext.setTerm(term, definition);
            defined.put(term, true);
            return;
        }

        // 14.
        if (idValue != null && (JsonUtils.isNotString(idValue) || !term.equals(((JsonString) idValue).getString()))) {

            // 14.1.
            if (JsonUtils.isNotNull(idValue)) {

                // 14.2.1
                if (JsonUtils.isNotString(idValue)) {
                    throw new JsonLdError(JsonLdErrorCode.INVALID_IRI_MAPPING);
                }

                final String idValueString = ((JsonString) idValue).getString();

                // 14.2.2
                if (!Keywords.contains(idValueString) && Keywords.matchForm(idValueString)) {
                    LOGGER.log(Level.WARNING, "The value [{0}] associated with @id has form of a keyword but is not keyword.", idValueString);
                    if (activeContext.getOptions().isExceptionOnWarning()) {
                        throw new JsonLdError(JsonLdErrorCode.UNSPECIFIED, "The value [" + idValueString + "] associated with @id has form of a keyword but is not keyword.");
                    }
                    return;
                }

                // 14.2.3
                definition.setUriMapping(
                        activeContext
                                .uriExpansion()
                                .localContext(localContext)
                                .defined(defined)
                                .vocab(true)
                                .expand(idValueString));

                if (Keywords.CONTEXT.equals(definition.getUriMapping())) {
                    throw new JsonLdError(JsonLdErrorCode.INVALID_KEYWORD_ALIAS);
                }

                if (!Keywords.contains(definition.getUriMapping()) && UriUtils.isNotURI(definition.getUriMapping())
                        && !BlankNode.hasPrefix(definition.getUriMapping())) {

                    throw new JsonLdError(JsonLdErrorCode.INVALID_IRI_MAPPING);
                }

                // 14.2.4
                if (term.substring(0, term.length() - 1).indexOf(':', 1) != -1 || term.contains("/")) {

                    // 14.2.4.1
                    defined.put(term, true);

                    // 14.2.4.2
                    final String expandedTerm =
                            activeContext
                                    .uriExpansion()
                                    .localContext(localContext)
                                    .defined(defined)
                                    .vocab(true)
                                    .expand(term);

                    if (expandedTerm == null || !expandedTerm.equals(definition.getUriMapping())) {
                        throw new JsonLdError(JsonLdErrorCode.INVALID_IRI_MAPPING);
                    }
                }

                // 14.2.5
                if (definition.isNotPrefix()) {
                    definition.setPrefix(!term.contains(":")
                                    && !term.contains("/")
                                    && simpleTerm
                                    && (definition.getUriMapping() != null
                                    && ((
                                    UriUtils.endsWithGenDelim(definition.getUriMapping())
                                            && UriUtils.isURI(definition.getUriMapping().substring(0, definition.getUriMapping().length() - 1))
                            )
                                    || BlankNode.hasPrefix(definition.getUriMapping()))
                            )
                    );
                }
            }

            // 15.
        } else if (term.indexOf(':', 1) != -1) {

            final CompactUri compactUri = CompactUri.create(term);

            // 15.1.
            if (compactUri != null && compactUri.isNotBlank() && localContext.containsKey(compactUri.getPrefix())) {

                activeContext.newTerm(localContext, defined).create(compactUri.getPrefix());
            }
            // 15.2.
            if (compactUri != null && compactUri.isNotBlank() && activeContext.containsTerm(compactUri.getPrefix())) {

                TermDefinition term1 = activeContext.getTermNullable(compactUri.getPrefix());
                if (term1 != null && term1.getUriMapping() != null) {
                    definition.setUriMapping(term1.getUriMapping() + compactUri.getSuffix());
                } else {
                    definition.setUriMapping(null);
                }


                // 15.3.
            } else if (UriUtils.isURI(term) || BlankNode.hasPrefix(term)) {
                definition.setUriMapping(term);
            }

            // 16.
        } else if (term.contains("/")) {

            definition.setUriMapping(
                    activeContext
                            .uriExpansion()
                            .localContext(localContext)
                            .defined(defined)
                            .vocab(true)
                            .expand(term));

            if (UriUtils.isNotURI(definition.getUriMapping())) {
                throw new JsonLdError(JsonLdErrorCode.INVALID_IRI_MAPPING);
            }

            // 17.
        } else if (Keywords.TYPE.equals(term)) {
            definition.setUriMapping(Keywords.TYPE);

            // 18.
        } else if (activeContext.getVocabularyMapping() == null) {
            throw new JsonLdError(JsonLdErrorCode.INVALID_IRI_MAPPING);

        } else {
            definition.setUriMapping(activeContext.getVocabularyMapping().concat(term));
        }

        // 19.
        if (valueObject.containsKey(Keywords.CONTAINER)) {

            // 19.1.
            final JsonValue containerValue = valueObject.get(Keywords.CONTAINER);

            if (!validateContainer(containerValue)) {
                throw new JsonLdError(JsonLdErrorCode.INVALID_CONTAINER_MAPPING);
            }

            // 19.3.
            JsonUtils.withStrings(containerValue, definition::addContainerMapping);

            // 19.4.
            if (definition.getContainerMapping().contains(Keywords.TYPE)) {

                // 19.4.1.
                if (definition.getTypeMapping() == null) {
                    definition.setTypeMapping(Keywords.ID);
                }

                if (!Keywords.ID.equals(definition.getTypeMapping())
                        && !Keywords.VOCAB.equals(definition.getTypeMapping())) {

                    throw new JsonLdError(JsonLdErrorCode.INVALID_TYPE_MAPPING);
                }
            }
        }

        // 20.
        if (valueObject.containsKey(Keywords.INDEX)) {

            // 20.1.
            if (activeContext.inMode(JsonLdVersion.V1_0) || !definition.getContainerMapping().contains(Keywords.INDEX)) {
                throw new JsonLdError(JsonLdErrorCode.INVALID_TERM_DEFINITION);
            }

            // 20.2.
            final JsonValue index = valueObject.get(Keywords.INDEX);

            if (JsonUtils.isNotString(index)) {
                throw new JsonLdError(JsonLdErrorCode.INVALID_TERM_DEFINITION);
            }

            final String indexString = ((JsonString) index).getString();

            final String expandedIndex =
                    activeContext
                            .uriExpansion()
                            .localContext(localContext)
                            .defined(defined)
                            .vocab(true)
                            .expand(indexString);

            if (expandedIndex == null || UriUtils.isNotURI(expandedIndex)) {
                throw new JsonLdError(JsonLdErrorCode.INVALID_TERM_DEFINITION);
            }

            definition.setIndexMapping(indexString);
        }

        // 21.
        if (valueObject.containsKey(Keywords.CONTEXT)) {

            // 21.1.
            if (activeContext.inMode(JsonLdVersion.V1_0)) {
                throw new JsonLdError(JsonLdErrorCode.INVALID_TERM_DEFINITION);
            }

            // 21.2.
            final JsonValue context = valueObject.get(Keywords.CONTEXT);

            // 21.3.
            try {
                activeContext
                        .newContext()
                        .overrideProtected(true)
                        .remoteContexts(new ArrayList<>(remoteContexts))
                        .validateScopedContext(false)
                        .create(context, baseUrl);

            } catch (JsonLdError e) {
                throw new JsonLdError(JsonLdErrorCode.INVALID_SCOPED_CONTEXT, e);
            }

            // 21.4.
            definition.setLocalContext(context);
            definition.setBaseUrl(baseUrl);
        }

        // 22.
        if (valueObject.containsKey(Keywords.LANGUAGE) && !valueObject.containsKey(Keywords.TYPE)) {

            // 22.1. - 2.
            final JsonValue language = valueObject.get(Keywords.LANGUAGE);

            if (JsonUtils.isNull(language) || JsonUtils.isString(language)) {

                if (JsonUtils.isString(language) && !LanguageTag.isWellFormed(((JsonString) language).getString())) {
                    LOGGER.log(Level.WARNING, "Language tag [{0}] is not well formed.", ((JsonString) language).getString());
                    if (activeContext.getOptions().isExceptionOnWarning()) {
                        throw new JsonLdError(JsonLdErrorCode.INVALID_LANGUAGE_TAGGED_STRING, "Language tag '" + ((JsonString) language).getString() + "' is not well formed.");
                    }
                }

                definition.setLanguageMapping(language);

            } else {
                throw new JsonLdError(JsonLdErrorCode.INVALID_LANGUAGE_MAPPING);
            }
        }

        // 23.
        if (valueObject.containsKey(Keywords.DIRECTION) && !valueObject.containsKey(Keywords.TYPE)) {

            final JsonValue direction = valueObject.get(Keywords.DIRECTION);

            if (JsonUtils.isNull(direction)) {
                definition.setDirectionMapping(DirectionType.NULL);

            } else if (JsonUtils.isString(direction)) {

                String directionString = ((JsonString) direction).getString();

                if ("ltr".equals(directionString)) {
                    definition.setDirectionMapping(DirectionType.LTR);

                } else if ("rtl".equals(directionString)) {
                    definition.setDirectionMapping(DirectionType.RTL);

                } else {
                    throw new JsonLdError(JsonLdErrorCode.INVALID_BASE_DIRECTION);
                }

            } else {
                throw new JsonLdError(JsonLdErrorCode.INVALID_BASE_DIRECTION);
            }
        }

        // 24.
        if (valueObject.containsKey(Keywords.NEST)) {

            // 24.1
            if (activeContext.inMode(JsonLdVersion.V1_0)) {
                throw new JsonLdError(JsonLdErrorCode.INVALID_TERM_DEFINITION);
            }

            final JsonValue nest = valueObject.get(Keywords.NEST);

            if (JsonUtils.isNotString(nest)) {
                throw new JsonLdError(JsonLdErrorCode.INVALID_KEYWORD_NEST_VALUE);
            }

            final String nestString = ((JsonString) nest).getString();

            if (Keywords.contains(nestString) && !Keywords.NEST.equals(nestString)) {
                throw new JsonLdError(JsonLdErrorCode.INVALID_KEYWORD_NEST_VALUE);
            }

            definition.setNestValue(nestString);
        }

        // 25.
        if (valueObject.containsKey(Keywords.PREFIX)) {

            // 25.1.
            if (activeContext.inMode(JsonLdVersion.V1_0) || term.contains(":") || term.contains("/")) {
                throw new JsonLdError(JsonLdErrorCode.INVALID_TERM_DEFINITION);
            }

            // 25.2.
            final JsonValue prefix = valueObject.get(Keywords.PREFIX);

            if (JsonUtils.isTrue(prefix)) {
                definition.setPrefix(true);

            } else if (JsonUtils.isFalse(prefix)) {
                definition.setPrefix(false);

            } else {
                throw new JsonLdError(JsonLdErrorCode.INVALID_KEYWORD_PREFIX_VALUE);
            }

            // 25.3
            if (definition.isPrefix() && Keywords.contains(definition.getUriMapping())) {
                throw new JsonLdError(JsonLdErrorCode.INVALID_TERM_DEFINITION);
            }
        }

        // 26.
        if (Keywords.notAllMatch(valueObject.keySet(), KEYWORDS)) {
            throw new JsonLdError(JsonLdErrorCode.INVALID_TERM_DEFINITION);
        }

        // 27.
        if (!overrideProtectedFlag && previousDefinition != null && previousDefinition.isProtected()) {

            // 27.1.
            if (definition.isNotSameExcept(previousDefinition)) {
                throw new JsonLdError(JsonLdErrorCode.PROTECTED_TERM_REDEFINITION);
            }

            // 27.2.
            activeContext.setTerm(term, previousDefinition);

        } else {
            activeContext.setTerm(term, definition);
        }

        defined.put(term, true);
    }

    private boolean validateContainer(final JsonValue value) {

        JsonValue container = value;

        if (JsonUtils.isNull(container)) {
            return false;
        }

        if (activeContext.inMode(JsonLdVersion.V1_0)) {

            return JsonUtils.isString(container)
                    && Keywords.noneMatch(
                    ((JsonString) container).getString(),
                    Keywords.GRAPH,
                    Keywords.ID,
                    Keywords.TYPE
            );
        }

        if (JsonUtils.isArray(container) && container.asJsonArray().size() == 1) {
            container = container.asJsonArray().get(0);
        }

        if (JsonUtils.isString(container)) {

            return Keywords.anyMatch(((JsonString) container).getString(),
                    Keywords.GRAPH,
                    Keywords.ID,
                    Keywords.INDEX,
                    Keywords.LANGUAGE,
                    Keywords.LIST,
                    Keywords.SET,
                    Keywords.TYPE);
        }

        return JsonUtils.isArray(container) && validateContainerArray(container.asJsonArray());
    }

    private static boolean validateContainerArray(final JsonArray containers) {

        if (containers.size() > 3) {
            return false;
        }

        if (JsonUtils.contains(Keywords.GRAPH, containers)
                && (JsonUtils.contains(Keywords.ID, containers)
                || JsonUtils.contains(Keywords.INDEX, containers))
        ) {

            return containers.size() == 2 || JsonUtils.contains(Keywords.SET, containers);
        }

        return containers.size() == 2
                && JsonUtils.contains(Keywords.SET, containers)
                && (JsonUtils.contains(Keywords.GRAPH, containers)
                || JsonUtils.contains(Keywords.ID, containers)
                || JsonUtils.contains(Keywords.INDEX, containers)
                || JsonUtils.contains(Keywords.LANGUAGE, containers)
                || JsonUtils.contains(Keywords.TYPE, containers)
        );
    }

}




© 2015 - 2024 Weber Informatics LLC | Privacy Policy