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

software.amazon.awssdk.services.outposts.endpoints.internal.Template Maven / Gradle / Ivy

/*
 * Copyright Amazon.com, Inc. or its affiliates. All Rights Reserved.
 * 
 * Licensed under the Apache License, Version 2.0 (the "License"). You may not use this file except in compliance with
 * the License. A copy of the License is located at
 * 
 * http://aws.amazon.com/apache2.0
 * 
 * or in the "license" file accompanying this file. This file 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 software.amazon.awssdk.services.outposts.endpoints.internal;

import java.util.ArrayList;
import java.util.List;
import java.util.Optional;
import java.util.stream.Collectors;
import java.util.stream.Stream;
import software.amazon.awssdk.annotations.SdkInternalApi;

/**
 * Template represents a "Template Literal". This is a literal string within the rules language. A template can contain
 * 0 or more dynamic sections. The dynamic sections use getAttr short-form:
 * 

* `https://{Region}.{partition#dnsSuffix}` -------- ------------ | | Dynamic getAttr short form */ @SdkInternalApi public class Template { private final List parts; Template(String template) { this.parts = RuleError.ctx("when parsing template", () -> parseTemplate(template)); } public Stream accept(TemplateVisitor visitor) { if (isStatic()) { return Stream.of(visitor.visitStaticTemplate(expectLiteral())); } if (parts.size() == 1) { // must be dynamic because previous branch handled single-element static template return Stream.of(visitor.visitSingleDynamicTemplate(((Dynamic) parts.get(0)).expr)); } Stream start = Stream.of(visitor.startMultipartTemplate()); Stream components = parts.stream().map(part -> part.accept(visitor)); Stream end = Stream.of(visitor.finishMultipartTemplate()); return Stream.concat(start, Stream.concat(components, end)); } public List getParts() { return parts; } public boolean isStatic() { return this.parts.stream().allMatch(it -> it instanceof Literal); } public String expectLiteral() { assert isStatic(); return this.parts.stream().map(Part::toString).collect(Collectors.joining()); } @Override public boolean equals(Object o) { if (this == o) { return true; } if (o == null || getClass() != o.getClass()) { return false; } Template template = (Template) o; return parts != null ? parts.equals(template.parts) : template.parts == null; } @Override public int hashCode() { return parts != null ? parts.hashCode() : 0; } public static Template fromString(String s) { return new Template(s); } @Override public String toString() { return String.format("\"%s\"", this.parts.stream().map(Part::toString).collect(Collectors.joining())); } public Value eval(Scope scope) { return Value.fromStr(parts.stream().map(part -> part.eval(scope)).collect(Collectors.joining())); } private List parseTemplate(String template) { List out = new ArrayList<>(); Optional templateStart = Optional.empty(); int depth = 0; int templateEnd = 0; for (int i = 0; i < template.length(); i++) { if (template.substring(i).startsWith("{{")) { i++; continue; } if (template.substring(i).startsWith("}}")) { i++; continue; } if (template.charAt(i) == '{') { if (depth == 0) { if (templateEnd != i) { out.add(Literal.unescape(template.substring(templateEnd, i))); } templateStart = Optional.of(i + 1); } depth++; } if (template.charAt(i) == '}') { depth--; if (depth < 0) { throw new InnerParseError("unmatched `}` in template"); } if (depth == 0) { out.add(Dynamic.parse(template.substring(templateStart.get(), i))); templateStart = Optional.empty(); } templateEnd = i + 1; } } if (depth != 0) { throw new InnerParseError("unmatched `{` in template"); } if (templateEnd < template.length()) { out.add(Literal.unescape(template.substring(templateEnd))); } return out; } public abstract static class Part { abstract String eval(Scope scope); abstract T accept(TemplateVisitor visitor); } public static class Literal extends Part { private final String value; public Literal(String value) { if (value.isEmpty()) { throw new RuntimeException("value cannot blank"); } this.value = value; } public static Literal unescape(String value) { return new Literal(value.replace("{{", "{").replace("}}", "}")); } public String getValue() { return value; } @Override public String toString() { return this.value; } @Override String eval(Scope scope) { return this.value; } @Override T accept(TemplateVisitor visitor) { return visitor.visitStaticElement(this.value); } @Override public boolean equals(Object o) { if (this == o) { return true; } if (o == null || getClass() != o.getClass()) { return false; } if (!super.equals(o)) { return false; } Literal literal = (Literal) o; return value != null ? value.equals(literal.value) : literal.value == null; } @Override public int hashCode() { return value != null ? value.hashCode() : 0; } } public static class Dynamic extends Part { private final String raw; private final Expr expr; private Dynamic(String raw, Expr expr) { this.raw = raw; this.expr = expr; } @Override public String toString() { return String.format("{dyn %s}", this.raw); } @Override String eval(Scope scope) { return RuleError.ctx("while evaluating " + this, () -> expr.eval(scope).expectString()); } @Override T accept(TemplateVisitor visitor) { return visitor.visitDynamicElement(this.expr); } public Expr getExpr() { return expr; } public static Dynamic parse(String value) { return new Dynamic(value, Expr.parseShortform(value)); } @Override public boolean equals(Object o) { if (this == o) { return true; } if (o == null || getClass() != o.getClass()) { return false; } if (!super.equals(o)) { return false; } Dynamic dynamic = (Dynamic) o; if (raw != null ? !raw.equals(dynamic.raw) : dynamic.raw != null) { return false; } return expr != null ? expr.equals(dynamic.expr) : dynamic.expr == null; } @Override public int hashCode() { int result = raw != null ? raw.hashCode() : 0; result = 31 * result + (expr != null ? expr.hashCode() : 0); return result; } } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy