HttpParser.swift 4.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120
  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(domain: "HTTP_PARSER", code: 0, userInfo:[NSLocalizedFailureReasonErrorKey : reason])
  11. }
  12. func nextHttpRequest(socket: CInt, error:NSErrorPointer = nil) -> HttpRequest? {
  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. let urlParams = extractUrlParams(path)
  23. // TODO extract query parameters
  24. if let headers = nextHeaders(socket, error: error) {
  25. // TODO detect content-type and handle:
  26. // 'application/x-www-form-urlencoded' -> Dictionary
  27. // 'multipart' -> Dictionary
  28. if let contentSize = headers["content-length"]?.toInt() {
  29. let body = nextBody(socket, size: contentSize, error: error)
  30. return HttpRequest(url: path, urlParams: urlParams, method: method, headers: headers, body: body, capturedUrlGroups: [])
  31. }
  32. return HttpRequest(url: path, urlParams: urlParams, method: method, headers: headers, body: nil, capturedUrlGroups: [])
  33. }
  34. }
  35. return nil
  36. }
  37. private func extractUrlParams(url: String) -> [(String, String)] {
  38. if let query = split(url, { $0 == "?" }).last {
  39. return map(split(query, { $0 == "&" }), { (param:String) -> (String, String) in
  40. let tokens = split(param, { $0 == "=" })
  41. if tokens.count >= 2 {
  42. let key = tokens[0].stringByRemovingPercentEncoding
  43. let value = tokens[1].stringByRemovingPercentEncoding
  44. if key != nil && value != nil { return (key!, value!) }
  45. }
  46. return ("","")
  47. })
  48. }
  49. return []
  50. }
  51. private func nextBody(socket: CInt, size: Int , error:NSErrorPointer) -> String? {
  52. var body = ""
  53. var counter = 0;
  54. while ( counter < size ) {
  55. let c = nextUInt8(socket)
  56. if ( c < 0 ) {
  57. if error != nil { error.memory = HttpParser.err("IO error while reading body") }
  58. return nil
  59. }
  60. body.append(UnicodeScalar(c))
  61. counter++;
  62. }
  63. return body
  64. }
  65. private func nextHeaders(socket: CInt, error:NSErrorPointer) -> Dictionary<String, String>? {
  66. var headers = Dictionary<String, String>()
  67. while let headerLine = nextLine(socket, error: error) {
  68. if ( headerLine.isEmpty ) {
  69. return headers
  70. }
  71. let headerTokens = split(headerLine, { $0 == ":" })
  72. if ( headerTokens.count >= 2 ) {
  73. // RFC 2616 - "Hypertext Transfer Protocol -- HTTP/1.1", paragraph 4.2, "Message Headers":
  74. // "Each header field consists of a name followed by a colon (":") and the field value. Field names are case-insensitive."
  75. // We can keep lower case version.
  76. let headerName = headerTokens[0].lowercaseString
  77. let headerValue = headerTokens[1].stringByTrimmingCharactersInSet(NSCharacterSet.whitespaceAndNewlineCharacterSet())
  78. if ( !headerName.isEmpty && !headerValue.isEmpty ) {
  79. headers.updateValue(headerValue, forKey: headerName)
  80. }
  81. }
  82. }
  83. return nil
  84. }
  85. private func nextUInt8(socket: CInt) -> Int {
  86. var buffer = [UInt8](count: 1, repeatedValue: 0);
  87. let next = recv(socket, &buffer, UInt(buffer.count), 0)
  88. if next <= 0 { return next }
  89. return Int(buffer[0])
  90. }
  91. private func nextLine(socket: CInt, error:NSErrorPointer) -> String? {
  92. var characters: String = ""
  93. var n = 0
  94. do {
  95. n = nextUInt8(socket)
  96. if ( n > 13 /* CR */ ) { characters.append(Character(UnicodeScalar(n))) }
  97. } while ( n > 0 && n != 10 /* NL */)
  98. if ( n == -1 && characters.isEmpty ) {
  99. if error != nil { error.memory = Socket.socketLastError("recv(...) failed.") }
  100. return nil
  101. }
  102. return characters
  103. }
  104. func supportsKeepAlive(headers: Dictionary<String, String>) -> Bool {
  105. if let value = headers["connection"] {
  106. return "keep-alive" == value.stringByTrimmingCharactersInSet(NSCharacterSet.whitespaceAndNewlineCharacterSet()).lowercaseString
  107. }
  108. return false
  109. }
  110. }