HttpParser.swift 2.6 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071
  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. /* HTTP stream parser */
  9. class HttpParser {
  10. func parseHttpHeader(socket: CInt) -> (String, Dictionary<String, String>)? {
  11. if let statusLine = parseLine(socket) {
  12. let statusTokens = split(statusLine, { $0 == " " })
  13. if ( statusTokens.count >= 3 ) {
  14. let path = statusTokens[1]
  15. if let headers = parseHeaders(socket) {
  16. return (path, headers)
  17. }
  18. }
  19. }
  20. return nil
  21. }
  22. func parseHeaders(socket: CInt) -> Dictionary<String, String>? {
  23. var headers = Dictionary<String, String>()
  24. while let headerLine = parseLine(socket) {
  25. if ( headerLine.isEmpty ) {
  26. return headers
  27. }
  28. let headerTokens = split(headerLine, { $0 == ":" })
  29. if ( headerTokens.count >= 2 ) {
  30. // RFC 2616 - "Hypertext Transfer Protocol -- HTTP/1.1", paragraph 4.2, "Message Headers":
  31. // "Each header field consists of a name followed by a colon (":") and the field value. Field names are case-insensitive."
  32. // We can keep lower case version.
  33. let headerName = headerTokens[0].lowercaseString
  34. let headerValue = headerTokens[1]
  35. if ( !headerName.isEmpty && !headerValue.isEmpty ) {
  36. headers.updateValue(headerValue, forKey: headerName)
  37. }
  38. }
  39. }
  40. return nil
  41. }
  42. func parseLine(socket: CInt) -> String? {
  43. // TODO - read more bytes than one. It makes the server very slow.
  44. // TODO - check if there is a nicer way to manipulate bytes with Swift ( recv(...) -> String )
  45. var characters: String = ""
  46. var buff: UInt8[] = UInt8[](count: 1, repeatedValue: 0), n: Int = 1
  47. do {
  48. n = recv(socket, &buff, 1, 0);
  49. if ( n > 0 && buff[0] > 13 /* CR */ ) {
  50. characters += Character(UnicodeScalar(UInt32(buff[0])))
  51. }
  52. } while ( n > 0 && buff[0] != 10 /* NL */ )
  53. if ( n == -1 ) {
  54. return nil
  55. }
  56. return characters
  57. }
  58. func supportsKeepAlive(headers: Dictionary<String, String>) -> Bool {
  59. if let value = headers["connection"] {
  60. return "keep-alive" == value.stringByTrimmingCharactersInSet(NSCharacterSet.whitespaceAndNewlineCharacterSet()).lowercaseString
  61. }
  62. return false
  63. }
  64. }