HttpParser.swift 4.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103
  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) -> HttpRequest? { //(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 != nil { 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. var responseString = ""
  24. while let line = nextLine(socket, error: error)
  25. {
  26. if line.isEmpty {
  27. break
  28. }
  29. responseString += line
  30. }
  31. println(responseString)
  32. let responseData = responseString.dataUsingEncoding(NSUTF8StringEncoding, allowLossyConversion: true)
  33. return HttpRequest(url: path, method: method, headers: headers, responseData: responseData)
  34. }
  35. }
  36. return nil
  37. }
  38. func nextHeaders(socket: CInt, error:NSErrorPointer) -> Dictionary<String, String>? {
  39. var headers = Dictionary<String, String>()
  40. while let headerLine = nextLine(socket, error: error) {
  41. if ( headerLine.isEmpty ) {
  42. return headers
  43. }
  44. let headerTokens = split(headerLine, { $0 == ":" })
  45. if ( headerTokens.count >= 2 ) {
  46. // RFC 2616 - "Hypertext Transfer Protocol -- HTTP/1.1", paragraph 4.2, "Message Headers":
  47. // "Each header field consists of a name followed by a colon (":") and the field value. Field names are case-insensitive."
  48. // We can keep lower case version.
  49. let headerName = headerTokens[0].lowercaseString
  50. let headerValue = headerTokens[1]
  51. if ( !headerName.isEmpty && !headerValue.isEmpty ) {
  52. headers.updateValue(headerValue, forKey: headerName)
  53. }
  54. }
  55. }
  56. return nil
  57. }
  58. var recvBuffer = [UInt8](count: 1024, repeatedValue: 0)
  59. var recvBufferSize: Int = 0
  60. var recvBufferOffset: Int = 0
  61. func nextUInt8(socket: CInt) -> Int {
  62. if ( recvBufferSize == 0 || recvBufferOffset == recvBuffer.count ) {
  63. recvBufferOffset = 0
  64. recvBufferSize = recv(socket, &recvBuffer, UInt(recvBuffer.count), MSG_DONTWAIT)
  65. if ( recvBufferSize <= 0 ) { return recvBufferSize }
  66. if recvBufferSize < recvBuffer.count
  67. {
  68. recvBuffer[recvBufferSize] = 0
  69. }
  70. }
  71. let returnValue = recvBuffer[recvBufferOffset]
  72. recvBufferOffset++
  73. return Int(returnValue)
  74. }
  75. func nextLine(socket: CInt, error:NSErrorPointer) -> String? {
  76. var characters: String = ""
  77. var n = 0
  78. do {
  79. n = nextUInt8(socket)
  80. if ( n > 13 /* CR */ ) { characters.append(Character(UnicodeScalar(n))) }
  81. } while ( n > 0 && n != 10 /* NL */)
  82. if ( n == -1 && characters.isEmpty ) {
  83. if error != nil { error.memory = Socket.socketLastError("recv(...) failed.") }
  84. return nil
  85. }
  86. return characters
  87. }
  88. func supportsKeepAlive(headers: Dictionary<String, String>) -> Bool {
  89. if let value = headers["connection"] {
  90. return "keep-alive" == value.stringByTrimmingCharactersInSet(NSCharacterSet.whitespaceAndNewlineCharacterSet()).lowercaseString
  91. }
  92. return false
  93. }
  94. }