net.luminis.quic.cid.ConnectionIdRegistry Maven / Gradle / Ivy
/*
* Copyright © 2020, 2021, 2022, 2023, 2024 Peter Doornbosch
*
* This file is part of Kwik, an implementation of the QUIC protocol in Java.
*
* Kwik 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.
*
* Kwik 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.quic.cid;
import net.luminis.quic.log.Logger;
import java.security.SecureRandom;
import java.util.Arrays;
import java.util.List;
import java.util.Map;
import java.util.concurrent.ConcurrentHashMap;
import java.util.stream.Collectors;
public abstract class ConnectionIdRegistry {
public static final int DEFAULT_CID_LENGTH = 8;
/** Maps sequence number to connection ID (info) */
protected final Map connectionIds = new ConcurrentHashMap<>();
protected volatile byte[] currentConnectionId;
protected final Logger log;
private final SecureRandom randomGenerator;
private final int connectionIdLength;
public ConnectionIdRegistry(Logger log) {
this(DEFAULT_CID_LENGTH, log);
}
public ConnectionIdRegistry(Integer cidLength, Logger logger) {
connectionIdLength = cidLength != null? cidLength: DEFAULT_CID_LENGTH;
this.log = logger;
randomGenerator = new SecureRandom();
currentConnectionId = generateConnectionId();
connectionIds.put(0, new ConnectionIdInfo(0, currentConnectionId, ConnectionIdStatus.IN_USE));
}
public byte[] retireConnectionId(int sequenceNr) {
if (connectionIds.containsKey(sequenceNr)) {
ConnectionIdInfo cidInfo = connectionIds.get(sequenceNr);
if (cidInfo.getConnectionIdStatus().active()) {
cidInfo.setStatus(ConnectionIdStatus.RETIRED);
return cidInfo.getConnectionId();
}
else {
return null;
}
}
else {
return null;
}
}
/**
* @deprecated use getActive to get an active connection ID
*/
@Deprecated
public byte[] getCurrent() {
return currentConnectionId;
}
/**
* Get an active connection ID. There can be multiple active connection IDs, this method returns an arbitrary one.
* @return an active connection ID or null if non is active (which should never happen).
*/
public byte[] getActive() {
return connectionIds.entrySet().stream()
.filter(e -> e.getValue().getConnectionIdStatus().active())
.map(e -> e.getValue().getConnectionId())
.findFirst().orElse(null);
}
public Map getAll() {
return connectionIds;
}
protected int currentIndex() {
return connectionIds.entrySet().stream()
.filter(entry -> Arrays.equals(entry.getValue().getConnectionId(), currentConnectionId))
.mapToInt(entry -> entry.getKey())
.findFirst().orElseThrow();
}
protected byte[] generateConnectionId() {
byte[] connectionId = new byte[connectionIdLength];
randomGenerator.nextBytes(connectionId);
return connectionId;
}
public int getConnectionIdlength() {
return connectionIdLength;
}
public List getActiveConnectionIds() {
return connectionIds.values().stream()
.filter(cid -> cid.getConnectionIdStatus().active())
.map(info -> info.getConnectionId())
.collect(Collectors.toList());
}
}