2012-02-21 31 views
0

我遇到了這個問題,這個基本的,不成熟的我用Java編寫的Web服務器。出於某種原因,不是僅向瀏覽器發送「200 OK」或「404 Not Found」,而是將它們寫入正在檢索的任何文件中。例如,如果我試圖讓index.html文件,我回到:錯誤返回文件w/Java Web服務器

example

....而不是瀏覽器試圖編譯HTML。 嘗試獲取圖像時,情況會更糟糕,因爲該服務器嘗試向其添加「200 OK」&內容類型,該文件已損壞。任何人都可以提供任何有關如何發送狀態行& content-type與實際的HTML/JPG文件分開的任何見解?當我註釋掉「os.writeBytes(statusLine)」等等時,錯誤完全消失,但我仍然想將這些消息發送給瀏覽器...只是不合並文件。任何幫助將不勝感激。

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

final class HttpRequest implements Runnable { 
    final static String CRLF = "\r\n"; 
    Socket socket; 

    // Constructor 
    public HttpRequest(Socket socket) throws Exception { 
     this.socket = socket; 
    } 

    // Implement the run() method of the Runnable interface. 
    public void run() { 
     try { 
      processRequest(); 
     } catch (Exception e) { 
      System.out.println(e); 
     } 
    } 

    private static void sendBytes(FileInputStream fis, OutputStream os) 
    throws Exception { 
    // Construct a 1K buffer to hold bytes on their way to the socket. 
    byte[] buffer = new byte[1024]; 
    int bytes = 0; 

    // Copy requested file into the socket's output stream. 
    while((bytes = fis.read(buffer)) != -1) { 
     os.write(buffer, 0, bytes); 
     } 
    } 

    private static String contentType(String fileName) { 
    if(fileName.endsWith(".htm") || fileName.endsWith(".html")) { 
     return "text/html"; 
    } 
    if(fileName.endsWith(".jpeg") || fileName.endsWith(".jpg")) { 
    return "image/jpeg"; 
    } 
    if(fileName.endsWith(".gif")) { 
    return "image/gif"; 
    } 
    return "application/octet-stream"; 
    } 

    private void processRequest() throws Exception { 
     // Get a reference to the socket's input and output streams. 
     InputStream is = socket.getInputStream(); 
     DataOutputStream os = new DataOutputStream(socket.getOutputStream()); 

     // Set up input stream filters. 
     BufferedReader br = new BufferedReader(new InputStreamReader(is)); 

     // Get the request line of the HTTP request message. 
     String requestLine = new String(br.readLine()); 

     // Display the request line. 
     System.out.println(); 
     System.out.println(requestLine); 

     // Get and display the header lines. 
     String headerLine = null; 
     while ((headerLine = br.readLine()).length() != 0) { 
      System.out.println(headerLine); 
     } 

    // Extract the filename from the request line. 
    StringTokenizer tokens = new StringTokenizer(requestLine); 
    tokens.nextToken(); // skip over the method, which should be "GET" 
    String fileName = tokens.nextToken(); 
    // Prepend a "." so that file request is within the current directory. 
    fileName = "." + fileName; 

    // Open the requested file. 
    FileInputStream fis = null; 
    boolean fileExists = true; 
    try { 
    fis = new FileInputStream(fileName); 
    } catch (FileNotFoundException e) { 
    fileExists = false; 
    } 

    // Construct the response message. 
    String statusLine = null; 
    String contentTypeLine = null; 
    String entityBody = null; 
    if (fileExists) { 
    statusLine = "200 OK" + CRLF; 
    contentTypeLine = "Content-type: " + 
     contentType(fileName) + CRLF; 
    } else { 
    statusLine = "404 NOT FOUND" + CRLF; 
    contentTypeLine = "Content Not Found!" + CRLF; 
    entityBody = "<HTML>" + 
     "<HEAD><TITLE>Not Found</TITLE></HEAD>" + 
     "<BODY>Not Found</BODY></HTML>"; 
    } 

    // Send the status line. 
    os.writeBytes(statusLine); 

    // Send the content type line. 
    os.writeBytes(contentTypeLine); 

    // Send a blank line to indicate the end of the header lines. 
    os.writeBytes(CRLF); 

    // Send the entity body. 
    if (fileExists) { 
    sendBytes(fis, os); 
    fis.close(); 
    } else { 
    os.writeBytes("File DNE: Content Not Found!"); 
    } 

     // Close streams and socket. 
     os.close(); 
     br.close(); 
     socket.close(); 
    } 
    public static void main(String[] args) throws Exception { 
     final ServerSocket ss = new ServerSocket(6789); 
     while (true) 
      new HttpRequest(ss.accept()).run(); 
    } 
} 

回答

2

我相信你錯過了CRLF。是不是有一個額外的一個需要,在狀態欄的末尾

200 OK 的Content-Type:XXXX

當然,你可以讀取HTTP規範,或者做一個Wireshark的跟蹤:)

更新:是的,你有很多問題。閱讀Easy Http獲取簡單的 摘要。

但無論如何,你需要說「HTTP/1.0 200 OK」,例如

http://www.somehost.com/path/file.html 

首先打開一個套接字主機www.somehost.com,端口80(使用默認的80端口,因爲沒有在URL中指定)。然後,送東西想通過插座下面:

GET /path/file.html HTTP/1.0 
From: [email protected] 
User-Agent: HTTPTool/1.0 
[blank line here] 

服務器應具有類似下面的響應,發回通過相同的插座:

HTTP/1.0 200 OK 
Date: Fri, 31 Dec 1999 23:59:59 GMT 
Content-Type: text/html 
Content-Length: 1354 

<html> 
<body> 
<h1>Happy New Millennium!</h1> 
(more file contents) 
    . 
    . 
    . 
</body> 
</html> 
+0

我不太看到我可能錯過了CRLF ...我有一個在狀態線本身,以及在發送實體主體之前。你的意思是? – 2012-02-21 04:24:42

+0

請看我收錄的鏈接。簡而言之,您應該擁有HTTP/1.0 200 OK 內容類型:文本/ html 嗨!雖然有比規範更多的東西。 – MJB 2012-02-21 04:25:30

0

在狀態行=「200 OK」 + CRLF;

你闖到大具有正確的請求,你丟失了一些HTTP/1.1

權代碼:

if (fileExists) { 
 
    statusLine = "HTTP/1.1 200 OK" + CRLF; 
 
    contentTypeLine = "Content-type: " + 
 
     contentType(fileName) + CRLF + CRLF; 
 
    } else { 
 
    statusLine = "404 NOT FOUND" + CRLF; 
 
    contentTypeLine = "Content Not Found!" + CRLF; 
 
    entityBody = "<HTML>" + 
 
     "<HEAD><TITLE>Not Found</TITLE></HEAD>" + 
 
     "<BODY>Not Found</BODY></HTML>"; 
 
    }