HttpParser.swift 2.5 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980
  1. //
  2. // HttpParser.swift
  3. // Swifter
  4. //
  5. // Copyright (c) 2014-2016 Damian Kołakowski. All rights reserved.
  6. //
  7. #if os(Linux)
  8. import Glibc
  9. #else
  10. import Foundation
  11. #endif
  12. enum HttpParserError: ErrorType {
  13. case InvalidStatusLine(String)
  14. }
  15. class HttpParser {
  16. func readHttpRequest(socket: Socket) throws -> HttpRequest {
  17. let statusLine = try socket.readLine()
  18. let statusLineTokens = statusLine.split(" ")
  19. if statusLineTokens.count < 3 {
  20. throw HttpParserError.InvalidStatusLine(statusLine)
  21. }
  22. let request = HttpRequest()
  23. request.method = statusLineTokens[0]
  24. request.path = statusLineTokens[1]
  25. request.queryParams = extractQueryParams(request.path)
  26. request.headers = try readHeaders(socket)
  27. if let contentLength = request.headers["content-length"], let contentLengthValue = Int(contentLength) {
  28. request.body = try readBody(socket, size: contentLengthValue)
  29. }
  30. return request
  31. }
  32. private func extractQueryParams(url: String) -> [(String, String)] {
  33. guard let query = url.split("?").last else {
  34. return []
  35. }
  36. return query.split("&").reduce([(String, String)]()) { (c, s) -> [(String, String)] in
  37. let tokens = s.split(1, separator: "=")
  38. if let name = tokens.first, value = tokens.last {
  39. return c + [(name.removePercentEncoding(), value.removePercentEncoding())]
  40. }
  41. return c
  42. }
  43. }
  44. private func readBody(socket: Socket, size: Int) throws -> [UInt8] {
  45. var body = [UInt8]()
  46. var counter = 0
  47. while counter < size {
  48. body.append(try socket.read())
  49. counter += 1
  50. }
  51. return body
  52. }
  53. private func readHeaders(socket: Socket) throws -> [String: String] {
  54. var requestHeaders = [String: String]()
  55. repeat {
  56. let headerLine = try socket.readLine()
  57. if headerLine.isEmpty {
  58. return requestHeaders
  59. }
  60. let headerTokens = headerLine.split(1, separator: ":")
  61. if let name = headerTokens.first, value = headerTokens.last {
  62. requestHeaders[name.lowercaseString] = value.trim()
  63. }
  64. } while true
  65. }
  66. func supportsKeepAlive(headers: [String: String]) -> Bool {
  67. if let value = headers["connection"] {
  68. return "keep-alive" == value.trim()
  69. }
  70. return false
  71. }
  72. }