1
0

HttpServer.swift 5.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149
  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. public typealias Handler = HttpRequest -> HttpResponse
  11. private var handlers: [(expression: NSRegularExpression, handler: Handler)] = []
  12. private var listenSocket: Socket = Socket(socketFileDescriptor: -1)
  13. private var clientSockets: Set<Socket> = []
  14. private let clientSocketsLock = 0
  15. public init() { }
  16. public subscript (path: String) -> Handler? {
  17. set {
  18. do {
  19. let regex = try NSRegularExpression(pattern: path, options: self.expressionOptions)
  20. if let newHandler = newValue {
  21. self.handlers.append(expression: regex, handler: newHandler)
  22. // Longer patterns will have higher priority.
  23. self.handlers = self.handlers.sort { $0.0.pattern > $1.0.pattern }
  24. }
  25. } catch {
  26. print("Could not register handler for: \(path), error: \(error)")
  27. }
  28. }
  29. get { return nil }
  30. }
  31. public var routes:[String] {
  32. return self.handlers.map { $0.expression.pattern }
  33. }
  34. public func start(listenPort: in_port_t = 8080) throws {
  35. self.stop()
  36. self.listenSocket = try Socket.tcpSocketForListen(listenPort)
  37. dispatch_async(dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_BACKGROUND, 0)) {
  38. while let socket = try? self.listenSocket.acceptClientSocket() {
  39. HttpServer.lock(self.clientSocketsLock) {
  40. self.clientSockets.insert(socket)
  41. }
  42. dispatch_async(dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_BACKGROUND, 0)) {
  43. let socketAddress = try? socket.peername()
  44. let httpParser = HttpParser()
  45. while let request = try? httpParser.readHttpRequest(socket) {
  46. let keepAlive = httpParser.supportsKeepAlive(request.headers)
  47. let response: HttpResponse
  48. if let (expression, handler) = self.findHandler(request.url) {
  49. let capturedUrlsGroups = self.captureExpressionGroups(expression, value: request.url)
  50. let updatedRequest = HttpRequest(url: request.url, urlParams: request.urlParams, method: request.method, headers: request.headers, body: request.body, capturedUrlGroups: capturedUrlsGroups, address: socketAddress)
  51. response = handler(updatedRequest)
  52. } else {
  53. response = HttpResponse.NotFound
  54. }
  55. do {
  56. try HttpServer.respond(socket, response: response, keepAlive: keepAlive)
  57. } catch {
  58. print("Failed to send response: \(error)")
  59. break
  60. }
  61. if !keepAlive { break }
  62. }
  63. socket.release()
  64. HttpServer.lock(self.clientSocketsLock) {
  65. self.clientSockets.remove(socket)
  66. }
  67. }
  68. }
  69. self.stop()
  70. }
  71. }
  72. public func stop() {
  73. self.listenSocket.release()
  74. HttpServer.lock(self.clientSocketsLock) {
  75. for socket in self.clientSockets {
  76. socket.shutdown()
  77. }
  78. self.clientSockets.removeAll(keepCapacity: true)
  79. }
  80. }
  81. private let matchingOptions = NSMatchingOptions(rawValue: 0)
  82. private let expressionOptions = NSRegularExpressionOptions(rawValue: 0)
  83. private func findHandler(url:String) -> (NSRegularExpression, Handler)? {
  84. if let u = NSURL(string: url), path = u.path {
  85. for handler in self.handlers {
  86. if handler.expression.numberOfMatchesInString(path, options: self.matchingOptions, range: HttpServer.asciiRange(path)) > 0 {
  87. return handler
  88. }
  89. }
  90. }
  91. return nil
  92. }
  93. private func captureExpressionGroups(expression: NSRegularExpression, value: String) -> [String] {
  94. guard let u = NSURL(string: value), path = u.path else {
  95. return []
  96. }
  97. var capturedGroups = [String]()
  98. if let result = expression.firstMatchInString(path, options: matchingOptions, range: HttpServer.asciiRange(path)) {
  99. let nsValue: NSString = path
  100. for i in 1..<result.numberOfRanges {
  101. if let group = nsValue.substringWithRange(result.rangeAtIndex(i)).stringByRemovingPercentEncoding {
  102. capturedGroups.append(group)
  103. }
  104. }
  105. }
  106. return capturedGroups
  107. }
  108. private class func asciiRange(value: String) -> NSRange {
  109. return NSMakeRange(0, value.lengthOfBytesUsingEncoding(NSASCIIStringEncoding))
  110. }
  111. private class func lock(handle: AnyObject, closure: () -> ()) {
  112. objc_sync_enter(handle)
  113. closure()
  114. objc_sync_exit(handle)
  115. }
  116. private class func respond(socket: Socket, response: HttpResponse, keepAlive: Bool) throws {
  117. try socket.writeUTF8("HTTP/1.1 \(response.statusCode()) \(response.reasonPhrase())\r\n")
  118. let length = response.body()?.count ?? 0
  119. try socket.writeUTF8("Content-Length: \(length)\r\n")
  120. if keepAlive {
  121. try socket.writeUTF8("Connection: keep-alive\r\n")
  122. }
  123. for (name, value) in response.headers() {
  124. try socket.writeUTF8("\(name): \(value)\r\n")
  125. }
  126. try socket.writeUTF8("\r\n")
  127. if let body = response.body() {
  128. try socket.writeUInt8(body)
  129. }
  130. }
  131. }