mirror of
https://github.com/signalapp/Signal-Server
synced 2026-04-19 10:48:04 +01:00
112 lines
3.7 KiB
Java
112 lines
3.7 KiB
Java
/*
|
|
* Copyright 2013-2020 Signal Messenger, LLC
|
|
* SPDX-License-Identifier: AGPL-3.0-only
|
|
*/
|
|
package org.whispersystems.websocket;
|
|
|
|
import org.eclipse.jetty.websocket.api.RemoteEndpoint;
|
|
import org.eclipse.jetty.websocket.api.Session;
|
|
import org.eclipse.jetty.websocket.api.WebSocketException;
|
|
import org.eclipse.jetty.websocket.api.WriteCallback;
|
|
import org.slf4j.Logger;
|
|
import org.slf4j.LoggerFactory;
|
|
import org.whispersystems.websocket.messages.WebSocketMessage;
|
|
import org.whispersystems.websocket.messages.WebSocketMessageFactory;
|
|
import org.whispersystems.websocket.messages.WebSocketResponseMessage;
|
|
|
|
import java.io.IOException;
|
|
import java.nio.ByteBuffer;
|
|
import java.security.SecureRandom;
|
|
import java.util.List;
|
|
import java.util.Map;
|
|
import java.util.Optional;
|
|
import java.util.concurrent.CompletableFuture;
|
|
|
|
@SuppressWarnings("OptionalUsedAsFieldOrParameterType")
|
|
public class WebSocketClient {
|
|
|
|
private static final Logger logger = LoggerFactory.getLogger(WebSocketClient.class);
|
|
|
|
private final Session session;
|
|
private final RemoteEndpoint remoteEndpoint;
|
|
private final WebSocketMessageFactory messageFactory;
|
|
private final Map<Long, CompletableFuture<WebSocketResponseMessage>> pendingRequestMapper;
|
|
private final long created;
|
|
|
|
public WebSocketClient(Session session, RemoteEndpoint remoteEndpoint,
|
|
WebSocketMessageFactory messageFactory,
|
|
Map<Long, CompletableFuture<WebSocketResponseMessage>> pendingRequestMapper) {
|
|
this.session = session;
|
|
this.remoteEndpoint = remoteEndpoint;
|
|
this.messageFactory = messageFactory;
|
|
this.pendingRequestMapper = pendingRequestMapper;
|
|
this.created = System.currentTimeMillis();
|
|
}
|
|
|
|
public CompletableFuture<WebSocketResponseMessage> sendRequest(String verb, String path,
|
|
List<String> headers,
|
|
Optional<byte[]> body)
|
|
{
|
|
final long requestId = generateRequestId();
|
|
final CompletableFuture<WebSocketResponseMessage> future = new CompletableFuture<>();
|
|
|
|
pendingRequestMapper.put(requestId, future);
|
|
|
|
WebSocketMessage requestMessage = messageFactory.createRequest(Optional.of(requestId), verb, path, headers, body);
|
|
|
|
try {
|
|
remoteEndpoint.sendBytes(ByteBuffer.wrap(requestMessage.toByteArray()), new WriteCallback() {
|
|
@Override
|
|
public void writeFailed(Throwable x) {
|
|
logger.debug("Write failed", x);
|
|
pendingRequestMapper.remove(requestId);
|
|
future.completeExceptionally(x);
|
|
}
|
|
|
|
@Override
|
|
public void writeSuccess() {}
|
|
});
|
|
} catch (WebSocketException e) {
|
|
logger.debug("Write", e);
|
|
pendingRequestMapper.remove(requestId);
|
|
future.completeExceptionally(e);
|
|
}
|
|
|
|
return future;
|
|
}
|
|
|
|
public String getUserAgent() {
|
|
return session.getUpgradeRequest().getHeader("User-Agent");
|
|
}
|
|
|
|
public long getCreatedTimestamp() {
|
|
return this.created;
|
|
}
|
|
|
|
public boolean isOpen() {
|
|
return session.isOpen();
|
|
}
|
|
|
|
public void close(int code, String message) {
|
|
session.close(code, message);
|
|
}
|
|
|
|
public boolean shouldDeliverStories() {
|
|
String value = session.getUpgradeRequest().getHeader(Stories.X_SIGNAL_RECEIVE_STORIES);
|
|
return Stories.parseReceiveStoriesHeader(value);
|
|
}
|
|
|
|
public void hardDisconnectQuietly() {
|
|
try {
|
|
session.disconnect();
|
|
} catch (IOException e) {
|
|
// quietly we said
|
|
}
|
|
}
|
|
|
|
private long generateRequestId() {
|
|
return Math.abs(new SecureRandom().nextLong());
|
|
}
|
|
|
|
}
|