Solution for the Proxy Cache Lab

The following is one solution for the proxy cache lab. It contains code for all the three classes in the lab.

ProxyCache.java

  1. /**
  2. * ProxyCache.java - Simple caching proxy
  3. *
  4. * $Id: ProxyCache.java,v 1.3 2004/02/16 15:22:00 kangasha Exp $
  5. *
  6. */
  7.  
  8. import java.net.*;
  9. import java.io.*;
  10. import java.util.*;
  11.  
  12. public class ProxyCache {
  13. /** Port for the proxy */
  14. private static int port;
  15. /** Socket for client connections */
  16. private static ServerSocket socket;
  17.  
  18. /** Create the ProxyCache object and the socket */
  19. public static void init(int p) {
  20. port = p;
  21. try {
  22. socket = new ServerSocket(port);
  23. } catch (IOException e) {
  24. System.out.println("Error creating socket: " + e);
  25. System.exit(-1);
  26. }
  27. }
  28.  
  29. public static void handle(Socket client) {
  30. Socket server = null;
  31. HttpRequest request = null;
  32. HttpResponse response = null;
  33.  
  34. /* Process request. If there are any exceptions, then simply
  35. * return and end this thread. This unfortunately means the
  36. * client will hang for a while, until it timeouts. */
  37.  
  38. /* Read request */
  39. try {
  40. BufferedReader fromClient =
  41. new BufferedReader(new InputStreamReader(client.getInputStream()));
  42. System.out.println("Reading request...");
  43.  
  44. request = new HttpRequest(fromClient);
  45. System.out.println("Got request...");
  46. } catch (IOException e) {
  47. System.out.println("Error reading request from client: " + e);
  48. return;
  49. }
  50. /* Send request to server */
  51. try {
  52. /* Open socket and write request to socket */
  53. server = new Socket(request.getHost(), request.getPort());
  54. DataOutputStream toServer =
  55. new DataOutputStream(server.getOutputStream());
  56. toServer.writeBytes(request.toString());
  57. } catch (UnknownHostException e) {
  58. System.out.println("Unknown host: " + request.getHost());
  59. System.out.println(e);
  60. return;
  61. } catch (IOException e) {
  62. System.out.println("Error writing request to server: " + e);
  63. return;
  64. }
  65. /* Read response and forward it to client */
  66. try {
  67. DataInputStream fromServer =
  68. new DataInputStream(server.getInputStream());
  69. //InputStream is = server.getInputStream();
  70.  
  71. response = new HttpResponse(fromServer);
  72. //response = new HttpResponse(is);
  73. DataOutputStream toClient =
  74. new DataOutputStream(client.getOutputStream());
  75. toClient.writeBytes(response.toString());
  76. toClient.write(response.body);
  77. client.close();
  78. server.close();
  79. /* Insert object into the cache */
  80. /* Fill in (optional exercise only) */
  81. } catch (IOException e) {
  82. System.out.println("Error writing response to client: " + e);
  83. }
  84. }
  85.  
  86.  
  87. /** Read command line arguments and start proxy */
  88. public static void main(String args[]) {
  89. int myPort = 0;
  90.  
  91. try {
  92. myPort = Integer.parseInt(args[0]);
  93. } catch (ArrayIndexOutOfBoundsException e) {
  94. System.out.println("Need port number as argument");
  95. System.exit(-1);
  96. } catch (NumberFormatException e) {
  97. System.out.println("Please give port number as integer.");
  98. System.exit(-1);
  99. }
  100.  
  101. init(myPort);
  102.  
  103. /** Main loop. Listen for incoming connections and spawn a new
  104. * thread for handling them */
  105. Socket client = null;
  106.  
  107. while (true) {
  108. try {
  109. client = socket.accept();
  110. System.out.println("Got connection " + client);
  111. handle(client);
  112. //ProxyCacheThread request = new ProxyCacheThread(client);
  113. //request.run();
  114. } catch (IOException e) {
  115. System.out.println("Error reading request from client: " + e);
  116. /* Definitely cannot continue, so skip to next
  117. * iteration of while loop. */
  118. continue;
  119. }
  120. }
  121.  
  122. }
  123. }

HttpRequest.java

  1. /**
  2. * HttpRequest - HTTP request container and parser
  3. *
  4. * $Id: HttpRequest.java,v 1.3 2004/02/16 15:21:14 kangasha Exp $
  5. *
  6. */
  7.  
  8. import java.io.*;
  9. import java.net.*;
  10. import java.util.*;
  11.  
  12. public class HttpRequest {
  13. /** Help variables */
  14. final static String CRLF = "\r\n";
  15. final static int HTTP_PORT = 80;
  16. /** Store the request parameters */
  17. String method;
  18. String URI;
  19. String version;
  20. String headers = "";
  21. /** Server and port */
  22. private String host;
  23. private int port;
  24.  
  25. /** Create HttpRequest by reading it from the client socket */
  26. public HttpRequest(BufferedReader from) {
  27. String firstLine = "";
  28. try {
  29. firstLine = from.readLine();
  30. } catch (IOException e) {
  31. System.out.println("Error reading request line: " + e);
  32. }
  33.  
  34. String[] tmp = firstLine.split(" ");
  35. method = tmp[0];
  36. URI = tmp[1];
  37. version = tmp[2];
  38.  
  39. System.out.println("URI is: " + URI);
  40.  
  41. if (!method.equals("GET")) {
  42. System.out.println("Error: Method not GET");
  43. }
  44. try {
  45. String line = from.readLine();
  46. while (line.length() != 0) {
  47. headers += line + CRLF;
  48. /* We need to find host header to know which server to
  49. * contact in case the request URI is not complete. */
  50. if (line.startsWith("Host:")) {
  51. tmp = line.split(" ");
  52. if (tmp[1].indexOf(':') > 0) {
  53. String[] tmp2 = tmp[1].split(":");
  54. host = tmp2[0];
  55. port = Integer.parseInt(tmp2[1]);
  56. } else {
  57. host = tmp[1];
  58. port = HTTP_PORT;
  59. }
  60. }
  61. line = from.readLine();
  62. }
  63. } catch (IOException e) {
  64. System.out.println("Error reading from socket: " + e);
  65. return;
  66. }
  67. System.out.println("Host to contact is: " + host + " at port " + port);
  68. }
  69.  
  70. /** Return host for which this request is intended */
  71. public String getHost() {
  72. return host;
  73. }
  74.  
  75. /** Return port for server */
  76. public int getPort() {
  77. return port;
  78. }
  79.  
  80. /**
  81. * Convert request into a string for easy re-sending.
  82. */
  83. public String toString() {
  84. String req = "";
  85.  
  86. req = method + " " + URI + " " + version + CRLF;
  87. req += headers;
  88. /* This proxy does not support persistent connections */
  89. req += "Connection: close" + CRLF;
  90. req += CRLF;
  91.  
  92. return req;
  93. }
  94. }

HttpResponse.java

  1. /**
  2. * HttpResponse - Handle HTTP replies
  3. *
  4. * $Id: HttpResponse.java,v 1.3 2004/02/16 15:21:46 kangasha Exp $
  5. *
  6. */
  7.  
  8. import java.io.*;
  9. import java.net.*;
  10. import java.util.*;
  11.  
  12. public class HttpResponse {
  13. final static String CRLF = "\r\n";
  14. /** How big is the buffer used for reading the object */
  15. final static int BUF_SIZE = 8192;
  16. /** Maximum size of objects that this proxy can handle. For the
  17. * moment set to 100 KB */
  18. final static int MAX_OBJECT_SIZE = 100000;
  19. /** Reply status and headers */
  20. String version;
  21. int status;
  22. String statusLine = "";
  23. String headers = "";
  24. /* Body of reply */
  25. byte[] body = new byte[MAX_OBJECT_SIZE];
  26.  
  27. /** Read response from server. */
  28. public HttpResponse(DataInputStream fromServer) {
  29. //public HttpResponse(InputStream is) {
  30. /* Length of the object */
  31. int length = -1;
  32. boolean gotStatusLine = false;
  33.  
  34. // Test code
  35. //BufferedReader fromServer =
  36. // new BufferedReader(new InputStreamReader(is));
  37.  
  38. /* First read status line and response headers */
  39. try {
  40. String line = fromServer.readLine();
  41. while (line.length() != 0) {
  42. if (!gotStatusLine) {
  43. statusLine = line;
  44. gotStatusLine = true;
  45. } else {
  46. headers += line + CRLF;
  47. }
  48.  
  49. /* Note length of content as indicated by
  50. * Content-Length header. Unfortunately this is not
  51. * present in every response. */
  52. if (line.startsWith("Content-Length:") ||
  53. line.startsWith("Content-length:")) {
  54. String[] tmp = line.split(" ");
  55. length = Integer.parseInt(tmp[1]);
  56. }
  57. line = fromServer.readLine();
  58. }
  59. } catch (IOException e) {
  60. System.out.println("Error reading headers from server: " + e);
  61. return;
  62. }
  63.  
  64. try {
  65. int bytesRead = 0;
  66. byte buf[] = new byte[BUF_SIZE];
  67. boolean loop = false;
  68.  
  69. /* If we didn't get Content-Length header, just loop until
  70. * the connection is closed. */
  71. if (length == -1) {
  72. loop = true;
  73. }
  74.  
  75. /* Read the body in chunks of BUF_SIZE and copy the chunk
  76. * into body. Usually replies come back in smaller chunks
  77. * than BUF_SIZE. The while-loop ends when either we have
  78. * read Content-Length bytes or when the connection is
  79. * closed (when there is no Connection-Length in the
  80. * response. */
  81. while (bytesRead < length || loop) {
  82. /* Read it in as binary data */
  83. int res = fromServer.read(buf, 0, BUF_SIZE);
  84. //int res = is.read(buf, 0, BUF_SIZE);
  85. if (res == -1) {
  86. break;
  87. }
  88. /* Copy the bytes into body. Make sure we don't exceed
  89. * the maximum object size. */
  90. for (int i = 0;
  91. i < res && (i + bytesRead) < MAX_OBJECT_SIZE;
  92. i++) {
  93. body[bytesRead + i] = buf[i];
  94. }
  95. bytesRead += res;
  96. }
  97. } catch (IOException e) {
  98. System.out.println("Error reading response body: " + e);
  99. return;
  100. }
  101.  
  102.  
  103. }
  104.  
  105. /**
  106. * Convert response into a string for easy re-sending. Only
  107. * converts the response headers, body is not converted to a
  108. * string.
  109. */
  110. public String toString() {
  111. String res = "";
  112.  
  113. res = statusLine + CRLF;
  114. res += headers;
  115. res += CRLF;
  116.  
  117. return res;
  118. }
  119. }
  120.  
  121.  
  122.  
  123.  
  124.