HttpParser.swift 3.4 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091
  1. //
  2. // HttpParser.swift
  3. //
  4. // Created by Damian Kolakowski on 05/06/14.
  5. // Copyright (c) 2014 Damian Kołakowski. All rights reserved.
  6. //
  7. import Foundation
  8. class HttpParser {
  9. class func err(reason:String) -> NSError {
  10. return NSError.errorWithDomain("HTTP_PARSER", code: 0, userInfo:[NSLocalizedFailureReasonErrorKey : reason])
  11. }
  12. func nextHttpRequest(socket: CInt, error:NSErrorPointer = nil) -> (String, String, Dictionary<String, String>)? {
  13. if let statusLine = nextLine(socket, error: error) {
  14. let statusTokens = split(statusLine, { $0 == " " })
  15. println(statusTokens)
  16. if ( statusTokens.count < 3 ) {
  17. if error { error.memory = HttpParser.err("Invalid status line: \(statusLine)") }
  18. return nil
  19. }
  20. let method = statusTokens[0]
  21. let path = statusTokens[1]
  22. if let headers = nextHeaders(socket, error: error) {
  23. return (path, method, headers)
  24. }
  25. }
  26. return nil
  27. }
  28. func nextHeaders(socket: CInt, error:NSErrorPointer) -> Dictionary<String, String>? {
  29. var headers = Dictionary<String, String>()
  30. while let headerLine = nextLine(socket, error: error) {
  31. if ( headerLine.isEmpty ) {
  32. return headers
  33. }
  34. let headerTokens = split(headerLine, { $0 == ":" })
  35. if ( headerTokens.count >= 2 ) {
  36. // RFC 2616 - "Hypertext Transfer Protocol -- HTTP/1.1", paragraph 4.2, "Message Headers":
  37. // "Each header field consists of a name followed by a colon (":") and the field value. Field names are case-insensitive."
  38. // We can keep lower case version.
  39. let headerName = headerTokens[0].lowercaseString
  40. let headerValue = headerTokens[1]
  41. if ( !headerName.isEmpty && !headerValue.isEmpty ) {
  42. headers.updateValue(headerValue, forKey: headerName)
  43. }
  44. }
  45. }
  46. return nil
  47. }
  48. var recvBuffer: UInt8[] = UInt8[](count: 1024, repeatedValue: 0)
  49. var recvBufferSize: Int = 0
  50. var recvBufferOffset: Int = 0
  51. func nextUInt8(socket: CInt) -> Int {
  52. if ( recvBufferSize == 0 || recvBufferSize == recvBufferOffset ) {
  53. recvBufferOffset = 0
  54. recvBufferSize = recv(socket, &recvBuffer, UInt(recvBuffer.count), 0)
  55. if ( recvBufferSize <= 0 ) { return recvBufferSize }
  56. }
  57. let returnValue = recvBuffer[recvBufferOffset]
  58. recvBufferOffset++
  59. return Int(returnValue)
  60. }
  61. func nextLine(socket: CInt, error:NSErrorPointer) -> String? {
  62. var characters: String = ""
  63. var n = 0
  64. do {
  65. n = nextUInt8(socket)
  66. if ( n > 13 /* CR */ ) {
  67. characters += Character(UnicodeScalar(n))
  68. }
  69. } while ( n > 0 && n != 10 /* NL */ );
  70. if ( n == -1 ) {
  71. if error { error.memory = Socket.socketLastError("recv(...) failed.") }
  72. return nil
  73. }
  74. return characters
  75. }
  76. func supportsKeepAlive(headers: Dictionary<String, String>) -> Bool {
  77. if let value = headers["connection"] {
  78. return "keep-alive" == value.stringByTrimmingCharactersInSet(NSCharacterSet.whitespaceAndNewlineCharacterSet()).lowercaseString
  79. }
  80. return false
  81. }
  82. }