1
0

HttpServerIO.swift 6.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196
  1. //
  2. // HttpServer.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. public class HttpServerIO {
  13. private var socket = Socket(socketFileDescriptor: -1)
  14. private var sockets = Set<Socket>()
  15. private var stateValue: Int32 = HttpServerIOState.Stopped.rawValue
  16. public private(set) var state: HttpServerIOState {
  17. get {
  18. return HttpServerIOState(rawValue: stateValue)!
  19. }
  20. set(state) {
  21. OSAtomicCompareAndSwapInt(self.state.rawValue, state.rawValue, &stateValue)
  22. }
  23. }
  24. public var operating: Bool { get { return self.state == .Running } }
  25. private let queue = dispatch_queue_create("swifter.httpserverio.clientsockets", DISPATCH_QUEUE_SERIAL)
  26. public func port() throws -> Int {
  27. return Int(try socket.port())
  28. }
  29. public func isIPv4() throws -> Bool {
  30. return try socket.isIPv4()
  31. }
  32. deinit {
  33. stop()
  34. }
  35. public func start(port: in_port_t = 8080, forceIPv4: Bool = false, priority: Int = DISPATCH_QUEUE_PRIORITY_BACKGROUND) throws {
  36. guard !self.operating else { return }
  37. stop()
  38. self.state = .Starting
  39. self.socket = try Socket.tcpSocketForListen(port, forceIPv4: forceIPv4)
  40. dispatch_async(dispatch_get_global_queue(priority, 0)) { [weak self] in
  41. guard let `self` = self else { return }
  42. guard self.operating else { return }
  43. while let socket = try? self.socket.acceptClientSocket() {
  44. dispatch_async(dispatch_get_global_queue(priority, 0), { [weak self] in
  45. guard let `self` = self else { return }
  46. guard self.operating else { return }
  47. dispatch_async(self.queue) {
  48. self.sockets.insert(socket)
  49. }
  50. self.handleConnection(socket)
  51. dispatch_async(self.queue) {
  52. self.sockets.remove(socket)
  53. }
  54. })
  55. }
  56. self.stop()
  57. }
  58. self.state = .Running
  59. }
  60. public func stop() {
  61. guard self.operating else { return }
  62. self.state = .Stopping
  63. // Shutdown connected peers because they can live in 'keep-alive' or 'websocket' loops.
  64. for socket in self.sockets {
  65. socket.shutdwn()
  66. }
  67. dispatch_sync(queue) {
  68. self.sockets.removeAll(keepCapacity: true)
  69. }
  70. socket.release()
  71. self.state = .Stopped
  72. }
  73. public func dispatch(request: HttpRequest) -> ([String: String], HttpRequest -> HttpResponse) {
  74. return ([:], { _ in HttpResponse.NotFound })
  75. }
  76. private func handleConnection(socket: Socket) {
  77. let parser = HttpParser()
  78. while self.operating, let request = try? parser.readHttpRequest(socket) {
  79. let request = request
  80. request.address = try? socket.peername()
  81. let (params, handler) = self.dispatch(request)
  82. request.params = params
  83. let response = handler(request)
  84. var keepConnection = parser.supportsKeepAlive(request.headers)
  85. do {
  86. if self.operating {
  87. keepConnection = try self.respond(socket, response: response, keepAlive: keepConnection)
  88. }
  89. } catch {
  90. print("Failed to send response: \(error)")
  91. break
  92. }
  93. if let session = response.socketSession() {
  94. session(socket)
  95. break
  96. }
  97. if !keepConnection { break }
  98. }
  99. socket.release()
  100. }
  101. private struct InnerWriteContext: HttpResponseBodyWriter {
  102. let socket: Socket
  103. func write(file: File) throws {
  104. try socket.writeFile(file)
  105. }
  106. func write(data: [UInt8]) throws {
  107. try write(ArraySlice(data))
  108. }
  109. func write(data: ArraySlice<UInt8>) throws {
  110. try socket.writeUInt8(data)
  111. }
  112. func write(data: NSData) throws {
  113. try socket.writeData(data)
  114. }
  115. }
  116. private func respond(socket: Socket, response: HttpResponse, keepAlive: Bool) throws -> Bool {
  117. guard self.operating else { return false }
  118. try socket.writeUTF8("HTTP/1.1 \(response.statusCode()) \(response.reasonPhrase())\r\n")
  119. let content = response.content()
  120. if content.length >= 0 {
  121. try socket.writeUTF8("Content-Length: \(content.length)\r\n")
  122. }
  123. if keepAlive && content.length != -1 {
  124. try socket.writeUTF8("Connection: keep-alive\r\n")
  125. }
  126. for (name, value) in response.headers() {
  127. try socket.writeUTF8("\(name): \(value)\r\n")
  128. }
  129. try socket.writeUTF8("\r\n")
  130. if let writeClosure = content.write {
  131. let context = InnerWriteContext(socket: socket)
  132. try writeClosure(context)
  133. }
  134. return keepAlive && content.length != -1;
  135. }
  136. }
  137. public enum HttpServerIOState: Int32 {
  138. case Starting
  139. case Running
  140. case Stopping
  141. case Stopped
  142. }
  143. #if os(Linux)
  144. let DISPATCH_QUEUE_PRIORITY_BACKGROUND = 0
  145. private class dispatch_context {
  146. let block: ((Void) -> Void)
  147. init(_ block: ((Void) -> Void)) {
  148. self.block = block
  149. }
  150. }
  151. func dispatch_get_global_queue(queueId: Int, _ arg: Int) -> Int { return 0 }
  152. func dispatch_async(queueId: Int, _ block: ((Void) -> Void)) {
  153. let unmanagedDispatchContext = Unmanaged.passRetained(dispatch_context(block))
  154. let context = UnsafeMutablePointer<Void>(unmanagedDispatchContext.toOpaque())
  155. var pthread: pthread_t = 0
  156. pthread_create(&pthread, nil, { (context: UnsafeMutablePointer<Void>) -> UnsafeMutablePointer<Void> in
  157. let unmanaged = Unmanaged<dispatch_context>.fromOpaque(COpaquePointer(context))
  158. unmanaged.takeUnretainedValue().block()
  159. unmanaged.release()
  160. return context
  161. }, context)
  162. }
  163. #endif