net.luminis.tls.extension.ServerNameExtension Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of agent15 Show documentation
Show all versions of agent15 Show documentation
A (partial) TLS 1.3 implementation in Java, suitable and intended for use in a QUIC implementation.
/*
* Copyright © 2019, 2020, 2021, 2022, 2023 Peter Doornbosch
*
* This file is part of Agent15, an implementation of TLS 1.3 in Java.
*
* Agent15 is free software: you can redistribute it and/or modify it under
* the terms of the GNU Lesser General Public License as published by the
* Free Software Foundation, either version 3 of the License, or (at your option)
* any later version.
*
* Agent15 is distributed in the hope that it will be useful, but
* WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
* FITNESS FOR A PARTICULAR PURPOSE. See the GNU Lesser General Public License for
* more details.
*
* You should have received a copy of the GNU Lesser General Public License
* along with this program. If not, see .
*/
package net.luminis.tls.extension;
import net.luminis.tls.alert.DecodeErrorException;
import net.luminis.tls.TlsConstants;
import java.nio.ByteBuffer;
import java.nio.charset.Charset;
/**
* TLS server name extension: RFC 6066
* https://tools.ietf.org/html/rfc6066#section-3
*/
public class ServerNameExtension extends Extension {
private final String serverName;
public ServerNameExtension(String serverName) {
this.serverName = serverName;
}
public ServerNameExtension(ByteBuffer buffer) throws DecodeErrorException {
int extensionDataLength = parseExtensionHeader(buffer, TlsConstants.ExtensionType.server_name, 0);
if (extensionDataLength > 0) {
if (extensionDataLength < 2) {
throw new DecodeErrorException("incorrect extension length");
}
int serverNameListLength = buffer.getShort();
if (extensionDataLength != serverNameListLength + 2) {
throw new DecodeErrorException("inconsistent length");
}
int startPosition = buffer.position();
serverName = parseServerName(buffer);
if (buffer.position() - startPosition != serverNameListLength) {
throw new DecodeErrorException("inconsistent length");
}
}
else {
// https://tools.ietf.org/html/rfc6066#section-3
// " A server that receives a client hello containing the "server_name" extension (...). In this event,
// the server SHALL include an extension of type "server_name" in the (extended) server hello.
// The "extension_data" field of this extension SHALL be empty."
serverName = null;
}
}
private String parseServerName(ByteBuffer buffer) throws DecodeErrorException {
int nameType = buffer.get();
switch (nameType) {
case 0:
// host_name
int hostNameLength = buffer.getShort() & 0xffff;
if (hostNameLength > buffer.remaining()) {
throw new DecodeErrorException("extension underflow");
}
byte[] hostNameBytes = new byte[hostNameLength];
buffer.get(hostNameBytes);
// "The hostname is represented as a byte string using ASCII encoding without a trailing dot. "
return new String(hostNameBytes, Charset.forName("ASCII"));
}
// unsupported type, RFC 6066 only defines hostname
throw new DecodeErrorException("invalid NameType");
}
@Override
public byte[] getBytes() {
short hostnameLength = (short) serverName.length();
short extensionLength = (short) (hostnameLength + 2 + 1 + 2);
ByteBuffer buffer = ByteBuffer.allocate(4 + extensionLength);
buffer.putShort(TlsConstants.ExtensionType.server_name.value);
buffer.putShort(extensionLength); // Extension data length (in bytes)
// https://tools.ietf.org/html/rfc6066#section-3
buffer.putShort((short) (hostnameLength + 1 + 2)); // Length of server_name_list
buffer.put((byte) 0x00); // list entry is type 0x00 "DNS hostname"
buffer.putShort(hostnameLength); // Length of hostname
buffer.put(serverName.getBytes(Charset.forName("ASCII")));
return buffer.array();
}
public String getHostName() {
return serverName;
}
}