Send HTTP request manually via socket

JavaHttpSocketsNetwork ProgrammingHttp Headers

Java Problem Overview


When I send a normal HTTP request via a socket, the server does not respond with an OK response. I copied the HTTP header from Firefox. Here is the code:

Socket s = new Socket(InetAddress.getByName("stackoverflow.com"), 80);
PrintWriter pw = new PrintWriter(s.getOutputStream());
pw.print("GET / HTTP/1.1");
pw.print("Host: stackoverflow.com");
pw.flush();
BufferedReader br = new BufferedReader(new InputStreamReader(s.getInputStream()));
String t;
while((t = br.readLine()) != null) System.out.println(t);
br.close();

However, here is the response I received:

HTTP/1.0 408 Request Time-out
Cache-Control: no-cache
Connection: close
Content-Type: text/html

<html><body><h1>408 Request Time-out</h1>
Your browser didn't send a complete request in time.
</body></html>

I know that I can do this by using URL.openStream(), but why doesn't the server identify the HTTP request when I send it manually?

Java Solutions


Solution 1 - Java

Two things:

  1. You should use println instead of print to print your entries to separate lines.
  2. HTTP request should end in a blank line (link). So add pw.println("");

Solution 2 - Java

You don't follow the HTTP RFC.

  • Header lines are always ended by a CR LF (i.e. 0x0d plus 0x0a).
  • The header ends after the first double-newline. In your case, you don't include the trailing newline so the server doesn't recognize the end of the request headers.

Generally, you should always try to use existing HTTP libraries. Although HTTP seems to be a simple protocol (and it is compared to others), it has rather strict syntactic and semantic rules. If you try to implement this yourself, you should have read and understand the relevant parts of RFC 2616 (and related).

Sadly, there are already too many crappy HTTP implementations not following the standards out there making the life for everyone miserable. Save yourself the hassle and use the HTTP libraries of your chosen language.

Solution 3 - Java

The correct fix which really works and it is cross platform:

	pw.print("GET / HTTP/1.1\r\n");
	pw.print("Host: stackoverflow.com\r\n\r\n");

Solution 4 - Java

The following fix, as mentioned by the previous answers, solves the problem;

pw.print("GET / HTTP/1.1\n\r\n");
pw.print("Host: stackoverflow.com\n\r\n");

Attributions

All content for this solution is sourced from the original question on Stackoverflow.

The content on this page is licensed under the Attribution-ShareAlike 4.0 International (CC BY-SA 4.0) license.

Content TypeOriginal AuthorOriginal Content on Stackoverflow
QuestionEng.FouadView Question on Stackoverflow
Solution 1 - JavaLychaView Answer on Stackoverflow
Solution 2 - JavaHolger JustView Answer on Stackoverflow
Solution 3 - JavaTadejPView Answer on Stackoverflow
Solution 4 - JavaAimar HassanoView Answer on Stackoverflow