首先,我的问题是关于Java中用于处理来自客户端的POST请求的HttpServer,而不是关于可以将文件上传到web服务器的Java客户端的而不是。
好的。我使用Java语言中的轻量级HttpServer来处理"GET“|| "POST”请求。HttpServer的源代码是从http://www.prasannatech.net/2008/11/http-web-server-java-post-file-upload.html复制的。
/*
* HTTPPOSTServer.java
* Author: S.Prasanna
* @version 1.00
*/
import java.io.*;
import java.net.*;
import java.util.*;
public class HTTPPOSTServer extends Thread {
static final String HTML_START =
"<html>" +
"<title>HTTP POST Server in java</title>" +
"<body>";
static final String HTML_END =
"</body>" +
"</html>";
Socket connectedClient = null;
BufferedReader inFromClient = null;
DataOutputStream outToClient = null;
public HTTPPOSTServer(Socket client) {
connectedClient = client;
}
public void run() {
String currentLine = null, postBoundary = null, contentength = null, filename = null, contentLength = null;
PrintWriter fout = null;
try {
System.out.println( "The Client "+
connectedClient.getInetAddress() + ":" + connectedClient.getPort() + " is connected");
inFromClient = new BufferedReader(new InputStreamReader (connectedClient.getInputStream()));
outToClient = new DataOutputStream(connectedClient.getOutputStream());
currentLine = inFromClient.readLine();
String headerLine = currentLine;
StringTokenizer tokenizer = new StringTokenizer(headerLine);
String httpMethod = tokenizer.nextToken();
String httpQueryString = tokenizer.nextToken();
System.out.println(currentLine);
if (httpMethod.equals("GET")) {
System.out.println("GET request");
if (httpQueryString.equals("/")) {
// The default home page
String responseString = HTTPPOSTServer.HTML_START +
"<form action=\"http://127.0.0.1:5000\" enctype=\"multipart/form-data\"" +
"method=\"post\">" +
"Enter the name of the File <input name=\"file\" type=\"file\"><br>" +
"<input value=\"Upload\" type=\"submit\"></form>" +
"Upload only text files." +
HTTPPOSTServer.HTML_END;
sendResponse(200, responseString , false);
} else {
sendResponse(404, "<b>The Requested resource not found ...." +
"Usage: http://127.0.0.1:5000</b>", false);
}
}
else { //POST request
System.out.println("POST request");
do {
currentLine = inFromClient.readLine();
if (currentLine.indexOf("Content-Type: multipart/form-data") != -1) {
String boundary = currentLine.split("boundary=")[1];
// The POST boundary
while (true) {
currentLine = inFromClient.readLine();
if (currentLine.indexOf("Content-Length:") != -1) {
contentLength = currentLine.split(" ")[1];
System.out.println("Content Length = " + contentLength);
break;
}
}
//Content length should be < 2MB
if (Long.valueOf(contentLength) > 2000000L) {
sendResponse(200, "File size should be < 2MB", false);
}
while (true) {
currentLine = inFromClient.readLine();
if (currentLine.indexOf("--" + boundary) != -1) {
filename = inFromClient.readLine().split("filename=")[1].replaceAll("\"", "");
String [] filelist = filename.split("\\" + System.getProperty("file.separator"));
filename = filelist[filelist.length - 1];
System.out.println("File to be uploaded = " + filename);
break;
}
}
String fileContentType = inFromClient.readLine().split(" ")[1];
System.out.println("File content type = " + fileContentType);
inFromClient.readLine(); //assert(inFromClient.readLine().equals("")) : "Expected line in POST request is "" ";
fout = new PrintWriter(filename);
String prevLine = inFromClient.readLine();
currentLine = inFromClient.readLine();
//Here we upload the actual file contents
while (true) {
if (currentLine.equals("--" + boundary + "--")) {
fout.print(prevLine);
break;
}
else {
fout.println(prevLine);
}
prevLine = currentLine;
currentLine = inFromClient.readLine();
}
sendResponse(200, "File " + filename + " Uploaded..", false);
fout.close();
} //if
}while (inFromClient.ready()); //End of do-while
}//else
} catch (Exception e) {
e.printStackTrace();
}
}
public void sendResponse (int statusCode, String responseString, boolean isFile) throws Exception {
String statusLine = null;
String serverdetails = "Server: Java HTTPServer";
String contentLengthLine = null;
String fileName = null;
String contentTypeLine = "Content-Type: text/html" + "\r\n";
FileInputStream fin = null;
if (statusCode == 200)
statusLine = "HTTP/1.1 200 OK" + "\r\n";
else
statusLine = "HTTP/1.1 404 Not Found" + "\r\n";
if (isFile) {
fileName = responseString;
fin = new FileInputStream(fileName);
contentLengthLine = "Content-Length: " + Integer.toString(fin.available()) + "\r\n";
if (!fileName.endsWith(".htm") && !fileName.endsWith(".html"))
contentTypeLine = "Content-Type: \r\n";
}
else {
responseString = HTTPPOSTServer.HTML_START + responseString + HTTPPOSTServer.HTML_END;
contentLengthLine = "Content-Length: " + responseString.length() + "\r\n";
}
outToClient.writeBytes(statusLine);
outToClient.writeBytes(serverdetails);
outToClient.writeBytes(contentTypeLine);
outToClient.writeBytes(contentLengthLine);
outToClient.writeBytes("Connection: close\r\n");
outToClient.writeBytes("\r\n");
if (isFile) sendFile(fin, outToClient);
else outToClient.writeBytes(responseString);
outToClient.close();
}
public void sendFile (FileInputStream fin, DataOutputStream out) throws Exception {
byte[] buffer = new byte[1024] ;
int bytesRead;
while ((bytesRead = fin.read(buffer)) != -1 ) {
out.write(buffer, 0, bytesRead);
}
fin.close();
}
public static void main (String args[]) throws Exception {
ServerSocket Server = new ServerSocket (5000);
System.out.println ("HTTP Server Waiting for client on port 5000");
while(true) {
Socket connected = Server.accept();
(new HTTPPOSTServer(connected)).start();
}
}
}
我通读了一下代码,我想代码应该没问题。
但是当我尝试上传一个文件时,它会打印出POST请求,然后挂在那里,永远不会收到任何字节。
如果你愿意,你可以直接运行上面的源代码。启动后,您可以在浏览器中输入127.0.0.1:5000,它将显示一个文件上传,然后如果我尝试上传一个文件,它将在打印PoST请求后挂起。
如果你对代码感到厌烦,我可以问你下面这个更简单的问题吗?
那么,Chrome或者其他网络浏览器对表单->输入type='file‘到底做了什么呢?
如果我使用ServerSocket处理HTTP请求,我只需要获取请求的InputStream,然后所有内容(包括HTTP头和上传文件的内容)都将通过该InputStream,对吗?
上面的代码可以分析标头,但之后浏览器似乎再也不会发送任何内容。
有谁能帮上忙吗?
谢谢
发布于 2011-07-23 10:32:12
它挂起是因为客户端(在我的例子中是Chrome)不提供Content-Length
。RFC 1867对此相当含糊。它在某种程度上暗示了这一点,但没有强制它,也没有例子。显然,客户并不总是会发送它。代码应该防止长度丢失。相反,它会遍历循环,直到到达文件的末尾。然后它就挂了。
使用调试器有时非常有用。
发布于 2011-07-23 10:44:43
这是很多代码:)
首先,将break;
调用移出POST部分内的if语句,从第83行开始。我们需要找出挂起的是什么,以及那些while语句可能很容易成为问题。
这将帮助我们找出问题所在。
如果你只是想让它工作,而不是真的关心bug是什么,这是一种重新发明的轮子,有一吨的库在那里使这成为一到两行的操作。
这里有一个很好的例子:http://www.servlets.com/cos/ -查看MultiPartRequest
类(源代码可从下载中获得)
发布于 2018-06-04 03:57:02
这是一个古老的问题,但以防其他人偶然发现它是为了寻找标题问题的答案,而不是细节中提到的具体bug (我有一种感觉,大多数观众都是这样的)……我建议你省去这个痛苦,不要使用上面的代码片段。
它不是HTTP服务器,除了最简单的用例之外,它在任何情况下都会失败(显然即使是这样),也不遵循基本的良好编码实践。即使不是这样,它也明确声明它只支持文本文件(实际上将整个文件读入一个字符串),而且即使这样也不能正确处理字符集编码。
也就是说,您可能在这里寻找两个答案中的一个-如何在应用程序中接受文件上传,或者HTTP服务器本身如何实现这一点。
如果您对如何出于娱乐或教育目的感兴趣,我强烈建议您阅读RFC (RFC 7230用于核心HTTP/1.1,RFC 2046第5.1节用于多部分解析),并尝试编写一个类似HTTP的简单服务器。这是一个很好的学习方法,特别是如果你有一位经验丰富的开发人员来审查你的代码,给你一些提示,并帮助你找到边缘案例。作为练习,您甚至可以从上面的代码开始,尝试修复它的缺点并使其更易于使用,尽管如此,不要将其与生产中使用的真正HTTP服务器混淆。
如果您不关心所有这些,只想在真实的应用程序中完成任务,我建议您只使用合适的HTTP服务器。有很多这样的代码可以让你用更少的代码和bug来做上面的代码片段试图做的事情。
免责声明:我是JLHTTP - The Java Lightweight HTTP Server的作者,它是一个小型的单文件服务器(或~50K/35K jar),没有依赖关系,努力符合RFC,并支持文件上传等。您可以随意使用它,或者浏览代码和文档,了解HTTP服务器和多部分解析(文件上传)如何工作的相对简单的示例,以及如何在应用程序中使用它。或者查看其他许多HTTP服务器中的任何一个,它们也可以做同样的事情。
https://stackoverflow.com/questions/6797415
复制相似问题