org.raml.v2.internal.impl.commons.grammar.BaseRamlGrammar.java Source code

Java tutorial

Introduction

Here is the source code for org.raml.v2.internal.impl.commons.grammar.BaseRamlGrammar.java

Source

/*
 * Copyright 2013 (c) MuleSoft, Inc.
 *
 * 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.
 */
package org.raml.v2.internal.impl.commons.grammar;

import static org.raml.v2.internal.impl.commons.phase.StringTemplateExpressionTransformer.TEMPLATE_PATTERN;

import com.google.common.base.Function;
import com.google.common.collect.Range;
import com.google.common.collect.Sets;

import java.util.concurrent.Callable;

import javax.annotation.Nonnull;
import javax.annotation.Nullable;

import org.raml.v2.internal.impl.commons.nodes.BodyNode;
import org.raml.v2.internal.impl.commons.nodes.MethodNode;
import org.raml.v2.internal.impl.commons.nodes.ParametrizedResourceTypeRefNode;
import org.raml.v2.internal.impl.commons.nodes.ParametrizedSecuritySchemeRefNode;
import org.raml.v2.internal.impl.commons.nodes.ParametrizedTraitRefNode;
import org.raml.v2.internal.impl.commons.nodes.RamlDocumentNode;
import org.raml.v2.internal.impl.commons.nodes.ResourceNode;
import org.raml.v2.internal.impl.commons.nodes.ResourceTypeNode;
import org.raml.v2.internal.impl.commons.nodes.ResourceTypeRefNode;
import org.raml.v2.internal.impl.commons.nodes.SecuritySchemeNode;
import org.raml.v2.internal.impl.commons.nodes.SecuritySchemeRefNode;
import org.raml.v2.internal.impl.commons.nodes.TraitNode;
import org.raml.v2.internal.impl.commons.nodes.TraitRefNode;
import org.raml.v2.internal.impl.commons.rule.NodeReferenceFactory;
import org.raml.v2.internal.impl.commons.rule.NodeReferenceRule;
import org.raml.v2.internal.impl.commons.rule.ParametrizedNodeReferenceRule;
import org.raml.v2.internal.impl.v10.nodes.factory.EmptyObjectNodeFactory;
import org.raml.v2.internal.impl.v10.nodes.factory.OverlayableSimpleTypeFactory;
import org.raml.v2.internal.impl.v10.nodes.factory.TypeExpressionReferenceFactory;
import org.raml.yagi.framework.grammar.BaseGrammar;
import org.raml.yagi.framework.grammar.ExclusiveSiblingRule;
import org.raml.yagi.framework.grammar.RuleFactory;
import org.raml.yagi.framework.grammar.RuleTraverser;
import org.raml.yagi.framework.grammar.rule.AnyOfRule;
import org.raml.yagi.framework.grammar.rule.ArrayRule;
import org.raml.yagi.framework.grammar.rule.ArrayWrapperFactory;
import org.raml.yagi.framework.grammar.rule.KeyValueRule;
import org.raml.yagi.framework.grammar.rule.NodeFactory;
import org.raml.yagi.framework.grammar.rule.ObjectRule;
import org.raml.yagi.framework.grammar.rule.RegexValueRule;
import org.raml.yagi.framework.grammar.rule.Rule;
import org.raml.yagi.framework.grammar.rule.StringValueRule;
import org.raml.yagi.framework.nodes.Node;

public abstract class BaseRamlGrammar extends BaseGrammar {

    public static final String USES_KEY_NAME = "uses";
    public static final String RESOURCE_TYPES_KEY_NAME = "resourceTypes";
    public static final String TRAITS_KEY_NAME = "traits";
    public static final String TYPES_KEY_NAME = "types";
    public static final String SCHEMAS_KEY_NAME = "schemas";
    public static final String QUERY_PARAMETERS_KEY_NAME = "queryParameters";
    public static final String QUERY_STRING_KEY_NAME = "queryString";
    public static final String SECURITY_SCHEMES_KEY_NAME = "securitySchemes";
    public static final String MIME_TYPE_REGEX = "([\\w\\d\\.\\-\\_\\+]+|\\*)\\/([\\w\\d\\.\\-\\_\\+]+|\\*);?([\\w\\d\\.\\-\\_\\+]+=[\\w\\d\\.\\-\\_\\+]+)?(\\s+[\\w\\d\\.\\-\\_\\+]+=[\\w\\d\\.\\-\\_\\+]+)*";
    public static final String OAUTH_1_0 = "OAuth 1.0";
    public static final String OAUTH_2_0 = "OAuth 2.0";

    public ObjectRule raml() {
        return untitledRaml().with(titleField().description("Short plain-text label for the API."));
    }

    protected NodeReferenceRule nodeRef(String referenceKey) {
        return new NodeReferenceRule(referenceKey);
    }

    public ObjectRule untitledRaml() {
        return objectType().with(descriptionField()).with(schemasField()).with(traitsField())
                .with(resourceTypesField()).with(securitySchemesField()).with(versionField()).with(baseUriField())
                .with(baseUriParametersField()).with(protocolsField()).with(mediaTypeField())
                .with(securedByField()
                        .description("The security schemes that apply to every resource and method in the API."))
                .with(resourceField()).with(fieldWithRequiredValue(documentationKey(), documentations()))
                .then(RamlDocumentNode.class);
    }

    protected KeyValueRule baseUriParametersField() {
        return field(baseUriParametersKey(), parameters());
    }

    protected KeyValueRule baseUriField() {
        return field(baseUriKey(), new UriTemplateValidationRule(ramlScalarValue()));
    }

    protected KeyValueRule docTitleField() {
        return requiredField(titleKey(), allOf(minLength(1), ramlScalarValue()));
    }

    protected KeyValueRule titleField() {
        return requiredField(titleKey(), titleValue());
    }

    protected Rule titleValue() {
        return allOf(ramlScalarValue(), minLength(1));
    }

    protected KeyValueRule resourceField() {
        return new KeyValueRule(resourceKey(),
                anyOf(resourceValue(), nullValue().then(new EmptyObjectNodeFactory()))).then(ResourceNode.class);
    }

    protected KeyValueRule versionField() {
        return field(versionKey(), ramlScalarValue());
    }

    protected Rule ramlScalarValue() {
        return scalarType();
    }

    protected KeyValueRule mediaTypeField() {
        return field(mediaTypeKey(), mimeTypeRegex());
    }

    public ObjectRule resourceType() {
        final ObjectRule resourceType = inNewContext(new Callable<ObjectRule>() {

            @Override
            public ObjectRule call() {
                return new RuleTraverser().traverse(
                        baseResourceValue().with(field(anyResourceTypeMethod(), methodValue())),
                        ruleParameterModifier());
            }
        });
        return resourceType.with(0, usageField()); // match anything but nested resources
    }

    /**
     * Returns a function that modifies a rule to support parameter
     * @return The function that injects the parameters
     */
    private Function<Rule, Boolean> ruleParameterModifier() {
        return new Function<Rule, Boolean>() {
            @Nullable
            @Override
            public Boolean apply(@Nonnull Rule input) {
                // We should change that any object now should have a parametrized key value pair
                // This mean that any key can be parametrized and we don't know anything about the value
                // And any value can be parametrized of any known key
                if (input instanceof ObjectRule) {
                    ((ObjectRule) input).with(0, fieldParametrizedKey());
                } else if (input instanceof KeyValueRule) {
                    final KeyValueRule keyValueRule = (KeyValueRule) input;
                    keyValueRule.setValueRule(anyOf(parametrizedValueRule(), keyValueRule.getValueRule()));
                    keyValueRule.cleanDefaultValue();
                } else if (input instanceof ArrayRule) {
                    final ArrayRule arrayRule = (ArrayRule) input;
                    arrayRule.of(anyOf(parametrizedValueRule(), arrayRule.of()));
                }
                // We keep reference factories so that we can validate reference consistency
                // and we keep overlay factories so traits/rt can be overlaid
                if (!isReferenceFactory(input) && !isOverlayFactory(input) && input.getFactory() != null) {
                    input.cleanFactory();
                }
                return true;
            }
        };
    }

    private boolean isOverlayFactory(Rule input) {
        return input.getFactory() instanceof OverlayableSimpleTypeFactory;
    }

    private boolean isReferenceFactory(@Nonnull Rule input) {
        return input.getFactory() instanceof NodeReferenceFactory
                || input.getFactory() instanceof TypeExpressionReferenceFactory;
    }

    public ObjectRule resourceTypeParamsResolved() {
        return baseResourceValue().with(usageField())
                .with(field(anyResourceTypeMethod(), methodValue()).then(MethodNode.class));
    }

    public Rule traitParamsResolved() {
        return objectType().with(descriptionField()).with(field(displayNameKey(), ramlScalarValue()))
                .with(queryParametersField()).with(headersField()).with(responsesField()).with(bodyField())
                .with(protocolsField().description(
                        "A method can override the protocols specified in the resource or at the API root, by employing this property."))
                .with(isField().description("A list of the traits to apply to this method."))
                .with(securedByField().description("The security schemes that apply to this method."))
                .with(usageField());

    }

    // Documentation
    protected Rule documentations() {
        return array(documentation());
    }

    public ObjectRule documentation() {
        return objectType().with(docTitleField().description("Title of documentation section."))
                .with(contentField().description("Content of documentation section."));
    }

    protected KeyValueRule contentField() {
        return requiredField(string("content"), ramlScalarValue());
    }

    // Security scheme

    protected Rule securitySchemes() {
        return objectType().with(field(scalarType(), securityScheme()).then(SecuritySchemeNode.class));
    }

    public ObjectRule securityScheme() {
        return objectType().with(descriptionField()).with(requiredField(securitySchemeTypeKey(), anyOf(
                string(OAUTH_1_0)
                        .description("The API's authentication uses OAuth 1.0 as described in RFC5849 [RFC5849]"),
                string(OAUTH_2_0)
                        .description("The API's authentication uses OAuth 2.0 as described in RFC6749 [RFC6749]"),
                string("Basic Authentication").description(
                        "The API's authentication uses Basic Authentication as described in RFC2617 [RFC2617]"),
                string("Digest Authentication").description(
                        "The API's authentication uses Digest Authentication as described in RFC2617 [RFC2617]"),
                string("Pass Through").description(
                        "Headers or Query Parameters are passed through to the API based on a defined object."),
                regex("x-.+")
                        .description("The API's authentication uses an authentication method not listed above."))))
                .with(displayNameField()).with(field(string("describedBy"), securitySchemePart()))
                .with(when("type", is(anyOf(string(OAUTH_1_0), string(OAUTH_2_0)))
                        .add(requiredField(string("settings"), securitySchemeSettings()))));
    }

    protected ObjectRule securitySchemePart() {
        return objectType().with(displayNameField()).with(descriptionField())
                .with(field(headersKey(), parameters())).with(queryParametersField()).with(responsesField());
    }

    protected ObjectRule securitySchemeSettings() {
        return objectType().with(field(string("requestTokenUri"), ramlScalarValue()))
                .with(field(string("tokenCredentialsUri"), ramlScalarValue()))
                .with(field(string("accessTokenUri"), ramlScalarValue()))
                .with(field(string("authorizationGrants"),
                        anyOf(authorizationGrantsValue(), array(authorizationGrantsValue()))))
                .with(field(string("scopes"), anyOf(scalarType(), array(scalarType()))));
    }

    protected Rule authorizationGrantsValue() {
        return scalarType();
    }

    // Traits
    protected Rule traitsValue() {
        return objectType().with(field(scalarType(), trait()).then(TraitNode.class));
    }

    public ObjectRule trait() {
        final ObjectRule trait = inNewContext(new Callable<ObjectRule>() {

            @Override
            public ObjectRule call() {
                return new RuleTraverser().traverse(methodValue(), ruleParameterModifier());
            }
        });
        return trait.with(0, usageField());
    }

    private RegexValueRule parametrizedValueRule() {
        return regex(TEMPLATE_PATTERN).fullMatch(false);
    }

    private KeyValueRule fieldParametrizedKey() {
        return field(parametrizedValueRule(), any());
    }

    // Resource Types
    protected Rule resourceTypes() {
        return objectType().with(field(scalarType(), resourceType()).then(ResourceTypeNode.class));
    }

    // Resources
    protected ObjectRule resourceValue() {
        return named("resourceValue", new RuleFactory<ObjectRule>() {
            @Override
            public ObjectRule create() {
                return baseResourceValue().with(methodField()).with(resourceField());
            }
        });
    }

    protected KeyValueRule methodField() {
        return new KeyValueRule(anyMethod(), anyOf(methodValue(), nullValue().then(new EmptyObjectNodeFactory())))
                .then(MethodNode.class);
    }

    protected ObjectRule baseResourceValue() {
        return objectType().with(displayNameField()).with(descriptionField()).with(isField().description(
                "A list of the traits to apply to all methods declared (implicitly or explicitly) for this resource. "))
                .with(resourceTypeReferenceField())
                .with(securedByField().description(
                        "The security schemes that apply to all methods declared (implicitly or explicitly) for this resource."))
                .with(field(uriParametersKey(), parameters()));
    }

    // Method
    protected ObjectRule methodValue() {
        return objectType().with(descriptionField()).with(displayNameField()).with(queryParametersField())
                .with(headersField()).with(responsesField()).with(bodyField())
                .with(protocolsField().description(
                        "A method can override the protocols specified in the resource or at the API root, by employing this property."))
                .with(isField().description("A list of the traits to apply to this method."))
                .with(securedByField().description("The security schemes that apply to this method."));
    }

    private KeyValueRule queryParametersField() {
        return field(queryParametersKey(), parameters());
    }

    private KeyValueRule responsesField() {
        return field(responseKey(), responses());
    }

    protected StringValueRule responseKey() {
        return string("responses").description("Information about the expected responses to a request");
    }

    protected StringValueRule queryStringKey() {
        return string(QUERY_STRING_KEY_NAME).description(
                "Specifies the query string needed by this method." + " Mutually exclusive with queryParameters.");
    }

    protected StringValueRule queryParametersKey() {
        return string(QUERY_PARAMETERS_KEY_NAME)
                .description("Detailed information about any query parameters needed by this method. "
                        + "Mutually exclusive with queryString.");
    }

    protected Rule responses() {
        return objectType().with(responseField());
    }

    protected KeyValueRule responseField() {
        return field(responseCodes(), response());
    }

    protected ObjectRule response() {
        return objectType().with(displayNameField()).with(descriptionField()).with(headersField())
                .with(bodyField());
    }

    protected Rule body() {
        return objectType().with(mimeTypeField());
    }

    public KeyValueRule mimeTypeField() {
        return field(mimeTypeRegex(), mimeType());
    }

    public RegexValueRule mimeTypeRegex() {
        return regex(MIME_TYPE_REGEX).suggest("application/json").suggest("application/xml");
    }

    protected abstract Rule mimeType();

    protected Rule parameters() {
        return objectType().with(field(scalarType(), parameter()));
    }

    /**
     * Describes the rule for a parameter.
     */
    protected abstract Rule parameter();

    protected ExclusiveSiblingRule exclusiveWith(String key, String... notAllowedSiblings) {
        return new ExclusiveSiblingRule(key, Sets.newHashSet(notAllowedSiblings));
    }

    protected KeyValueRule securitySchemesField() {
        return field(securitySchemesKey(), securitySchemesValue());
    }

    protected abstract Rule securitySchemesValue();

    protected StringValueRule securitySchemesKey() {
        return string(SECURITY_SCHEMES_KEY_NAME)
                .description("Declarations of security schemes for use within this API.");
    }

    protected KeyValueRule schemasField() {
        return field(schemasKey(), schemasValue());
    }

    protected abstract Rule schemasValue();

    protected StringValueRule schemasKey() {
        return string("schemas").description(schemasDescription());
    }

    @Nonnull
    protected abstract String schemasDescription();

    protected KeyValueRule resourceTypesField() {
        return field(resourceTypesKey(), resourceTypesValue());
    }

    protected Rule resourceTypesValue() {
        return anyOf(array(resourceTypes()), resourceTypes()).then(new ArrayWrapperFactory());
    }

    protected StringValueRule resourceTypesKey() {
        return string(RESOURCE_TYPES_KEY_NAME)
                .description("Declarations of resource types for use within this API.");
    }

    protected KeyValueRule traitsField() {
        return field(traitsKey(), traitsValue());
    }

    protected StringValueRule traitsKey() {
        return string(TRAITS_KEY_NAME).description("Declarations of traits for use within this API.");
    }

    protected KeyValueRule protocolsField() {
        return field(protocolsKey(), protocols());
    }

    protected StringValueRule protocolsKey() {
        return string("protocols").description("The protocols supported by the API.");
    }

    protected KeyValueRule bodyField() {
        return field(bodyKey(),
                firstOf(whenChildIs(mimeTypeField(), body()), whenPresent("/mediaType", mimeType())))
                        .then(BodyNode.class);
    }

    protected StringValueRule bodyKey() {
        return string("body")
                .description("Some methods admit request bodies, which are described by this property.");
    }

    protected KeyValueRule headersField() {
        return field(headersKey(), parameters());
    }

    protected StringValueRule headersKey() {
        return string("headers")
                .description("Detailed information about any request headers needed by this method.");
    }

    protected KeyValueRule descriptionField() {
        return field(descriptionKey(), descriptionValue());
    }

    protected Rule descriptionValue() {
        return ramlScalarValue();
    }

    protected KeyValueRule displayNameField() {
        return field(displayNameKey(), ramlScalarValue()).defaultValue(parentKey());
    }

    protected KeyValueRule securedByField() {
        Rule securitySchemeRef = anyOf(nullValue(), anyTypeReference(SECURITY_SCHEMES_KEY_NAME,
                SecuritySchemeRefNode.class, ParametrizedSecuritySchemeRefNode.class));
        return field(securedByKey(),
                anyOf(securitySchemeRef, array(securitySchemeRef)).then(new ArrayWrapperFactory()));
    }

    protected KeyValueRule usageField() {
        return field(string("usage"), ramlScalarValue());
    }

    protected KeyValueRule isField() {
        Rule traitRef = anyTypeReference(TRAITS_KEY_NAME, TraitRefNode.class, ParametrizedTraitRefNode.class);
        return field(isKey(), anyOf(traitRef, array(traitRef)).then(new ArrayWrapperFactory()));
    }

    protected KeyValueRule resourceTypeReferenceField() {
        return field(resourceTypeRefKey(), anyTypeReference(RESOURCE_TYPES_KEY_NAME, ResourceTypeRefNode.class,
                ParametrizedResourceTypeRefNode.class));
    }

    protected Rule anyTypeReference(String referenceKey, Class<? extends Node> simpleClass,
            Class<? extends Node> parametrisedClass) {
        final KeyValueRule paramsRule = field(scalarType(), any());
        final KeyValueRule typeWithParams = field(scalarType(), objectType().with(paramsRule));
        final NodeFactory factory = new NodeReferenceFactory(simpleClass);
        final NodeFactory parametrisedFactory = new NodeReferenceFactory(parametrisedClass);
        return anyOf(nodeRef(referenceKey).then(factory),
                new ParametrizedNodeReferenceRule(referenceKey).with(typeWithParams).then(parametrisedFactory));
    }

    protected StringValueRule titleKey() {
        return string("title");
    }

    // Repeated keys

    protected Rule resourceKey() {
        RegexValueRule rule = regex("/.*").label("/Resource").suggest("/<cursor>")
                .description("The resources of the API, identified as relative URIs that begin with a slash (/). "
                        + "Every property whose key begins with a slash (/), and is either at the root of the API definition "
                        + "or is the child property of a resource property, is a resource property, e.g.: /users, /{groupId}, etc");

        return new UriTemplateValidationRule(rule);
    }

    protected StringValueRule usesKey() {
        return string(USES_KEY_NAME).description("Importing libraries.");
    }

    protected StringValueRule uriParametersKey() {
        return string("uriParameters")
                .description("Detailed information about any URI parameters of this resource");
    }

    protected StringValueRule securedByKey() {
        return string("securedBy");
    }

    protected StringValueRule typeKey() {
        return string("type").description("The resource type which this resource inherits.");
    }

    protected StringValueRule securitySchemeTypeKey() {
        return string("type").description("Specify the security mechanism.");
    }

    protected StringValueRule resourceTypeRefKey() {
        return string("type").description("The resource type which this resource inherits.");
    }

    protected StringValueRule isKey() {
        return string("is");
    }

    protected StringValueRule displayNameKey() {
        return string("displayName")
                .description("An alternate, human-friendly name for the method (in the resource's context).");
    }

    protected StringValueRule descriptionKey() {
        return string("description").description("A longer, human-friendly description of the API");
    }

    protected StringValueRule documentationKey() {
        return string("documentation").description("Additional overall documentation for the API.");
    }

    protected StringValueRule mediaTypeKey() {
        return string("mediaType").description(
                "The default media type to use for request and response bodies (payloads), e.g. \"application/json\".");
    }

    protected StringValueRule baseUriParametersKey() {
        return string("baseUriParameters").description("Named parameters used in the baseUri (template).");
    }

    protected StringValueRule baseUriKey() {
        return string("baseUri").description("A URI that's to be used as the base of all the resources' URIs."
                + " Often used as the base of the URL of each resource, containing the location of the API. "
                + "Can be a template URI.");
    }

    protected StringValueRule versionKey() {
        return string("version").description("The version of the API, e.g. \"v1\".");
    }

    // Enum of values

    protected AnyOfRule anyMethod() {
        return anyOf(string("get"), string("patch"), string("put"), string("post"), string("delete"),
                string("options"), string("head"));
    }

    protected AnyOfRule anyOptionalMethod() {
        return anyOf(string("get?"), string("patch?"), string("put?"), string("post?"), string("delete?"),
                string("options?"), string("head?"));
    }

    protected AnyOfRule anyResourceTypeMethod() {
        return anyOf(anyMethod(), anyOptionalMethod());
    }

    protected Rule protocols() {
        AnyOfRule protocols = anyOf(string("HTTP"), string("HTTPS"));
        return anyOf(protocols, array(protocols)).then(new ArrayWrapperFactory());
    }

    protected Rule responseCodes() {
        return range(Range.closed(100L, 599L));
    }
}