服务器仅从请求中读取标头时,Http客户端未收到任何响应

时间:2019-02-22 09:29:16

标签: java sockets http

我正在弄乱Java中的HTTP和套接字,希望您能对此有所了解:

当我用Java SE 11编写的HTTP服务器没有读取整个请求然后进行响应时,客户端将无法获取该请求或发生错误。这是为什么?在服务器读取整个请求之前,客户端是否无法读取响应?如果在以下代码段中执行了对readBody的调用,则可以正常工作。如果响应具有Content-Length标头和文本正文,它也可以正常工作。这实际上让我感到困惑。

我的示例请求是带有数据 fds 的POST。邮递员说“无法获得任何请求”,而curl说“卷曲:(56)接收失败:对等重置连接”。

import java.io.*;
import java.net.Socket;
import java.util.*;

class Handler {

    public synchronized void read(Socket incoming) {
        try (incoming;
             OutputStream outputStream = incoming.getOutputStream();
             InputStream inputStream = incoming.getInputStream();
             PrintWriter pw = new PrintWriter(outputStream)) {

            writeRequest(inputStream);
            pw.print("HTTP/1.1 200 OK\r\n");
            pw.print("\r\n");
            pw.flush();
        } catch (IOException e) {
            System.out.println("ERROR: " + e.getMessage());
            e.printStackTrace();
        }
    }

    private void writeRequest(InputStream inputStream) throws IOException {
        String verbLine = readLine(inputStream);

        Map<String, String> headers = readHeaders(inputStream);

        //readBody(inputStream, headers);
    }

    private void readBody(InputStream inputStream, Map<String, String> headers) throws IOException {
        Optional<String> optKey = headers.keySet().stream()
                .filter(k -> k.equalsIgnoreCase("Content-Length"))
                .findFirst();
        if (optKey.isPresent()) {
            int contentLength = Integer.parseInt(headers.get(optKey.get()));
            byte[] bytes = inputStream.readNBytes(contentLength);
        }
    }

    private Map<String, String> readHeaders(InputStream inputStream) throws IOException {
        Map<String, String> headers = new HashMap<>();
        while (true) {
            String line = readLine(inputStream);
            if (line == null || line.isEmpty()) {
                return headers;
            }
            String key = line.split(":")[0].trim();
            String value = line.split(":")[1].trim();
            headers.put(key, value);
        }
    }

    private String readLine(InputStream inputStream) throws IOException {
        byte[] buf = new byte[200];
        int offset = 0;
        while (true) {
            int read = inputStream.read();
            if (read == -1) {
                return null;
            }
            buf[offset] = (byte) read;
            if (buf[0] == '\n' || (buf[0] == '\r' && buf[1] == '\n')) {
                return "";
            }
            if (buf[offset] == 0x0A) {
                int endOfLine = buf[offset - 1] == 0x0D ? offset - 1 : offset;
                return new String(buf, 0, endOfLine);
            } else {
                offset++;
            }
        }
    }
}

2 个答案:

答案 0 :(得分:2)

如果在服务器上仍未读取数据的情况下关闭套接字,将导致客户端连接重置错误。因为您没有阅读完整的请求,所以在这里发生。如果尚未读取服务器的完整响应,则该错误将向用户显示。

如果您发送的响应是content-length然后是完整的正文,则客户端将读取完整的响应,因此错误将被忽略。相反,如果您既不发送content-length也未使用分块编码,则客户端将期望响应以适当的TCP连接关闭而结束。在这种情况下,由于尚未(正确)读取来自服务器的完整响应,因此连接重置将传播给用户。

答案 1 :(得分:0)

您的响应需要具有Content-Length标头或Transfer-Encoding标头-告诉客户端如何发送响应并允许其确定何时接收到所有字节。否则,它将需要等待EOF并假定响应主体被EOF终止(这是为了与HTTP / 1.0兼容)。您的客户可能不支持。 了解您正在使用哪个客户端可能会有所帮助。