io.helidon.http.Forwarded Maven / Gradle / Ivy
/*
* Copyright (c) 2022, 2023 Oracle and/or its affiliates.
*
* 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 io.helidon.http;
import java.util.Arrays;
import java.util.List;
import java.util.Locale;
import java.util.Optional;
import static io.helidon.http.HeaderNames.FORWARDED;
/**
* A representation of the {@link HeaderNames#FORWARDED} HTTP header.
*/
public class Forwarded {
private static final System.Logger LOGGER = System.getLogger(Forwarded.class.getName());
private final Optional by;
private final Optional forClient;
private final Optional host;
private final Optional proto;
private Forwarded(String by, String forClient, String host, String proto) {
this.by = Optional.ofNullable(by);
this.forClient = Optional.ofNullable(forClient);
this.host = Optional.ofNullable(host);
this.proto = Optional.ofNullable(proto);
}
/**
* Create forwarded from a value of a single forwarded header, such as {@code by=a.b.c;for=d.e.f;host=host;proto=https}.
*
* @param string string representation of a single forwarded header
* @return forwarded parsed from the string
* @see #create(Headers)
*/
public static Forwarded create(String string) {
// first split by semicolon, as that separates the directives
String[] directives = string.split(";");
String by = null;
String forClient = null;
String host = null;
String proto = null;
for (String directive : directives) {
int index = directive.indexOf('=');
if (index == -1 && !directive.isEmpty()) {
throw new IllegalArgumentException("Invalid Forwarded header");
}
String name = directive.substring(0, index);
String value = unquote(directive.substring(index + 1));
switch (name.toLowerCase(Locale.ROOT)) {
case "by" -> {
by = value;
}
case "for" -> {
forClient = value;
}
case "host" -> {
host = value;
}
case "proto" -> {
proto = value;
}
default -> {
if (LOGGER.isLoggable(System.Logger.Level.DEBUG)) {
String printableName = name.replaceAll("\\p{C}", "?");
String printableValue = value.replaceAll("\\p{C}", "?");
LOGGER.log(System.Logger.Level.DEBUG, "Encountered unknown directive of Forwarded header: \n"
+ printableName + "\nValue:\n" + printableValue);
}
}
}
}
return new Forwarded(by, forClient, host, proto);
}
/**
* Parse forwarded header(s) from the provided headers.
*
* @param headers header to process
* @return list of forwarded headers, will be empty if the header does not exist.
*/
public static List create(Headers headers) {
List values = headers.values(FORWARDED);
return values.isEmpty() ? List.of() : values.stream()
.flatMap(it -> Arrays.stream(it.split(",")))
.map(String::trim)
.map(Forwarded::create)
.toList();
}
/**
* {@code by} directive of the forwarded header.
*
* @return by directive
*/
public Optional by() {
return by;
}
/**
* {@code for} directive of the forwarded header.
*
* @return for directive
*/
public Optional forClient() {
return forClient;
}
/**
* {@code host} directive of the forwarded header. The host of the original request.
*
* @return host directive
*/
public Optional host() {
return host;
}
/**
* {@code proto} directive of the forwarded header. The protocol of the original request (http or https).
*
* @return proto directive
*/
public Optional proto() {
return proto;
}
private static String unquote(String string) {
if (string.indexOf('"') == 0 && string.lastIndexOf('"') == string.length() - 1) {
return string.substring(1, string.length() - 1);
}
return string;
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy