init
This commit is contained in:
131
test/org/apache/tomcat/websocket/server/TesterWsClient.java
Normal file
131
test/org/apache/tomcat/websocket/server/TesterWsClient.java
Normal file
@@ -0,0 +1,131 @@
|
||||
/*
|
||||
* Licensed to the Apache Software Foundation (ASF) under one or more
|
||||
* contributor license agreements. See the NOTICE file distributed with
|
||||
* this work for additional information regarding copyright ownership.
|
||||
* The ASF licenses this file to You 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 org.apache.tomcat.websocket.server;
|
||||
|
||||
import java.io.BufferedReader;
|
||||
import java.io.IOException;
|
||||
import java.io.InputStreamReader;
|
||||
import java.net.Socket;
|
||||
import java.nio.charset.StandardCharsets;
|
||||
|
||||
import javax.websocket.CloseReason.CloseCode;
|
||||
|
||||
/**
|
||||
* A client for testing Websocket behavior that differs from standard client
|
||||
* behavior.
|
||||
*/
|
||||
public class TesterWsClient {
|
||||
|
||||
private static final byte[] maskingKey = new byte[] { 0x12, 0x34, 0x56,
|
||||
0x78 };
|
||||
|
||||
private final Socket socket;
|
||||
|
||||
public TesterWsClient(String host, int port) throws Exception {
|
||||
this.socket = new Socket(host, port);
|
||||
// Set read timeout in case of failure so test doesn't hang
|
||||
socket.setSoTimeout(2000);
|
||||
// Disable Nagle's algorithm to ensure packets sent immediately
|
||||
// TODO: Hoping this causes writes to wait for a TCP ACK for TCP RST
|
||||
// test cases but I'm not sure?
|
||||
socket.setTcpNoDelay(true);
|
||||
}
|
||||
|
||||
public void httpUpgrade(String path) throws IOException {
|
||||
String req = createUpgradeRequest(path);
|
||||
write(req.getBytes(StandardCharsets.UTF_8));
|
||||
readUpgradeResponse();
|
||||
}
|
||||
|
||||
public void sendTextMessage(String text) throws IOException {
|
||||
sendTextMessage(text.getBytes(StandardCharsets.UTF_8));
|
||||
}
|
||||
|
||||
public void sendTextMessage(byte[] utf8Bytes) throws IOException {
|
||||
write(createFrame(true, 1, utf8Bytes));
|
||||
}
|
||||
|
||||
public void sendCloseFrame(CloseCode closeCode) throws IOException {
|
||||
int code = closeCode.getCode();
|
||||
byte[] codeBytes = new byte[2];
|
||||
codeBytes[0] = (byte) (code >> 8);
|
||||
codeBytes[1] = (byte) code;
|
||||
write(createFrame(true, 8, codeBytes));
|
||||
}
|
||||
|
||||
private void readUpgradeResponse() throws IOException {
|
||||
BufferedReader in = new BufferedReader(new InputStreamReader(
|
||||
socket.getInputStream()));
|
||||
String line = in.readLine();
|
||||
while (line != null && !line.isEmpty()) {
|
||||
line = in.readLine();
|
||||
}
|
||||
}
|
||||
|
||||
public void closeSocket() throws IOException {
|
||||
// Enable SO_LINGER to ensure close() only returns when TCP closing
|
||||
// handshake completes
|
||||
socket.setSoLinger(true, 65535);
|
||||
socket.close();
|
||||
}
|
||||
|
||||
/*
|
||||
* Send a TCP RST instead of a TCP closing handshake
|
||||
*/
|
||||
public void forceCloseSocket() throws IOException {
|
||||
// SO_LINGER sends a TCP RST when timeout expires
|
||||
socket.setSoLinger(true, 0);
|
||||
socket.close();
|
||||
}
|
||||
|
||||
private void write(byte[] bytes) throws IOException {
|
||||
socket.getOutputStream().write(bytes);
|
||||
socket.getOutputStream().flush();
|
||||
}
|
||||
|
||||
private static String createUpgradeRequest(String path) {
|
||||
String[] upgradeRequestLines = { "GET " + path + " HTTP/1.1",
|
||||
"Connection: Upgrade", "Host: localhost:8080",
|
||||
"Origin: localhost:8080",
|
||||
"Sec-WebSocket-Key: OEvAoAKn5jsuqv2/YJ1Wfg==",
|
||||
"Sec-WebSocket-Version: 13", "Upgrade: websocket" };
|
||||
StringBuffer sb = new StringBuffer();
|
||||
for (String line : upgradeRequestLines) {
|
||||
sb.append(line);
|
||||
sb.append("\r\n");
|
||||
}
|
||||
sb.append("\r\n");
|
||||
return sb.toString();
|
||||
}
|
||||
|
||||
private static byte[] createFrame(boolean fin, int opCode, byte[] payload) {
|
||||
byte[] frame = new byte[6 + payload.length];
|
||||
frame[0] = (byte) (opCode | (fin ? 1 << 7 : 0));
|
||||
frame[1] = (byte) (0x80 | payload.length);
|
||||
|
||||
frame[2] = maskingKey[0];
|
||||
frame[3] = maskingKey[1];
|
||||
frame[4] = maskingKey[2];
|
||||
frame[5] = maskingKey[3];
|
||||
|
||||
for (int i = 0; i < payload.length; i++) {
|
||||
frame[i + 6] = (byte) (payload[i] ^ maskingKey[i % 4]);
|
||||
}
|
||||
|
||||
return frame;
|
||||
}
|
||||
}
|
||||
Reference in New Issue
Block a user