1
0

HttpServer.swift 6.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168
  1. //
  2. // HttpServer.swift
  3. // Swifter
  4. // Copyright (c) 2015 Damian Kołakowski. All rights reserved.
  5. //
  6. import Foundation
  7. public class HttpServer
  8. {
  9. static let VERSION = "1.0.2";
  10. typealias HandlerLine = (rawExpression: String, expression: NSRegularExpression, handler: Handler)
  11. public typealias Handler = HttpRequest -> HttpResponse
  12. private(set) var handlers: [HandlerLine] = []
  13. private(set) var acceptSocket: Socket!
  14. private(set) var clientSockets: Set<Socket> = []
  15. private let clientSocketsLock = 0
  16. let matchingOptions = NSMatchingOptions(rawValue: 0)
  17. let expressionOptions = NSRegularExpressionOptions(rawValue: 0)
  18. public init() { }
  19. public subscript (path: String) -> Handler? {
  20. get {
  21. return nil
  22. }
  23. set {
  24. do {
  25. let regex = try NSRegularExpression(pattern: path, options: self.expressionOptions)
  26. if let newHandler = newValue {
  27. self.handlers.append(rawExpression: path, expression: regex, handler: newHandler)
  28. }
  29. } catch {
  30. print("Could not register handler for: \(path), error: \(error)")
  31. }
  32. }
  33. }
  34. public var routes:[String] {
  35. return self.handlers.map { $0.expression.pattern }
  36. }
  37. public func start(listenPort: in_port_t = 8080) throws {
  38. self.stop()
  39. self.acceptSocket = try Socket(port:listenPort)
  40. dispatch_async(dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_BACKGROUND, 0)) {
  41. while let socket = try? self.acceptSocket.acceptClientSocket() {
  42. HttpServer.lock(self.clientSocketsLock) {
  43. self.clientSockets.insert(socket)
  44. }
  45. let socketAddress = try? self.acceptSocket.peername()
  46. dispatch_async(dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_BACKGROUND, 0)) {
  47. let parser = HttpParser()
  48. while let request = try? parser.nextHttpRequest(socket) {
  49. let keepAlive = parser.supportsKeepAlive(request.headers)
  50. let response: HttpResponse
  51. if let (_, expression, handler) = self.findHandler(request.url) {
  52. let capturedUrlsGroups = self.captureExpressionGroups(expression, value: request.url)
  53. let updatedRequest = HttpRequest(url: request.url, urlParams: request.urlParams, method: request.method, headers: request.headers, body: request.body, capturedUrlGroups: capturedUrlsGroups, address: socketAddress)
  54. response = handler(updatedRequest)
  55. } else {
  56. print("handler not found")
  57. response = HttpResponse.NotFound
  58. }
  59. do {
  60. try HttpServer.respond(socket, response: response, keepAlive: keepAlive)
  61. } catch {
  62. print("Failed to send response: \(error)")
  63. break
  64. }
  65. if !keepAlive { break }
  66. }
  67. HttpServer.lock(self.clientSocketsLock) {
  68. self.clientSockets.remove(socket)
  69. }
  70. }
  71. }
  72. self.stop()
  73. }
  74. }
  75. func findHandler(url:String) -> HandlerLine? {
  76. if let u = NSURL(string: url), path = u.path {
  77. var res = self.handlers.filter { $0.expression.numberOfMatchesInString(path, options: self.matchingOptions, range: HttpServer.asciiRange(path)) > 0 }
  78. if res.count > 1 {
  79. // we eliminate first the "/" route
  80. res = res.filter { $0.rawExpression != "/" }
  81. // if there is still the conflict, we take the better matching route
  82. if res.count > 1 {
  83. print("conflict between routes")
  84. let weight = res.map { $0.expression.numberOfMatchesInString(path, options: self.matchingOptions, range: HttpServer.asciiRange(path)) }
  85. res = [res[weight.indexOf(weight.maxElement()!)!]]
  86. }
  87. }
  88. if res.count > 0 {
  89. return res[0]
  90. }
  91. }
  92. return nil
  93. }
  94. func captureExpressionGroups(expression: NSRegularExpression, value: String) -> [String] {
  95. guard let u = NSURL(string: value), path = u.path else {
  96. return []
  97. }
  98. var capturedGroups = [String]()
  99. if let result = expression.firstMatchInString(path, options: matchingOptions, range: HttpServer.asciiRange(path)) {
  100. let nsValue: NSString = path
  101. for i in 1..<result.numberOfRanges {
  102. if let group = nsValue.substringWithRange(result.rangeAtIndex(i)).stringByRemovingPercentEncoding {
  103. capturedGroups.append(group)
  104. }
  105. }
  106. }
  107. return capturedGroups
  108. }
  109. public func stop() {
  110. // self.acceptSocket?.release()
  111. self.acceptSocket = nil
  112. HttpServer.lock(self.clientSocketsLock) {
  113. // for clientSocket in self.clientSockets {
  114. // clientSocket.release()
  115. // }
  116. self.clientSockets.removeAll(keepCapacity: true)
  117. }
  118. }
  119. private class func asciiRange(value: String) -> NSRange {
  120. return NSMakeRange(0, value.lengthOfBytesUsingEncoding(NSASCIIStringEncoding))
  121. }
  122. private class func lock(handle: AnyObject, closure: () -> ()) {
  123. objc_sync_enter(handle)
  124. closure()
  125. objc_sync_exit(handle)
  126. }
  127. private class func respond(socket: Socket, response: HttpResponse, keepAlive: Bool) throws {
  128. try socket.writeUTF8("HTTP/1.1 \(response.statusCode()) \(response.reasonPhrase())\r\n")
  129. let length = response.body()?.length ?? 0
  130. try socket.writeASCII("Content-Length: \(length)\r\n")
  131. if keepAlive {
  132. try socket.writeASCII("Connection: keep-alive\r\n")
  133. }
  134. for (name, value) in response.headers() {
  135. try socket.writeASCII("\(name): \(value)\r\n")
  136. }
  137. try socket.writeASCII("\r\n")
  138. if let body = response.body() {
  139. try socket.writeData(body)
  140. }
  141. }
  142. }