HttpServer.swift 6.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145
  1. //
  2. // HttpServer.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. class HttpServer
  9. {
  10. typealias Handler = HttpRequest -> HttpResponse
  11. var handlers: [(expression: NSRegularExpression, handler: Handler)] = []
  12. var acceptSocket: CInt = -1
  13. let matchingOptions = NSMatchingOptions(0)
  14. let expressionOptions = NSRegularExpressionOptions(0)
  15. subscript (path: String) -> Handler? {
  16. get {
  17. return nil
  18. }
  19. set ( newValue ) {
  20. if let regex: NSRegularExpression = NSRegularExpression(pattern: path, options: expressionOptions, error: nil) {
  21. if let newHandler = newValue {
  22. handlers.append(expression: regex, handler: newHandler)
  23. }
  24. }
  25. }
  26. }
  27. // Uncommenting this will cause following compilation errors:
  28. //
  29. // Cannot invoke 'subscript' with an argument list of type '($T5, Builtin.RawPointer)'
  30. // Cannot invoke 'subscript' with an argument list of type '($T5, Builtin.RawPointer)'
  31. //
  32. // Swift stopped to support subscripts with multiple outputs.
  33. //
  34. // subscript (asdasd: String) -> String {
  35. // get {
  36. // return asdasd
  37. // }
  38. // set ( directoryPath ) {
  39. // if let regex = NSRegularExpression(pattern: asdasd, options: expressionOptions, error: nil) {
  40. // handlers.append(expression: regex, handler: { request in
  41. // let result = regex.firstMatchInString(request.url, options: self.matchingOptions, range: NSMakeRange(0, request.url.lengthOfBytesUsingEncoding(NSASCIIStringEncoding)))
  42. // let nsPath: NSString = request.url
  43. // let filesPath = directoryPath.stringByExpandingTildeInPath.stringByAppendingPathComponent(nsPath.substringWithRange(result!.rangeAtIndex(1)))
  44. // if let fileBody = String(contentsOfFile: filesPath, encoding: NSUTF8StringEncoding, error: nil) {
  45. // return HttpResponse.OK(.RAW(fileBody))
  46. // }
  47. // return HttpResponse.NotFound
  48. // })
  49. // }
  50. // }
  51. // }
  52. func routes() -> Array<String> {
  53. var results = [String]()
  54. for (expression,_) in handlers { results.append(expression.pattern) }
  55. return results
  56. }
  57. func start(listenPort: in_port_t = 8080, error:NSErrorPointer = nil) -> Bool {
  58. releaseAcceptSocket()
  59. if let socket = Socket.tcpForListen(port: listenPort, error: error) {
  60. acceptSocket = socket
  61. dispatch_async(dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_BACKGROUND, 0), {
  62. while let socket = Socket.acceptClientSocket(self.acceptSocket) {
  63. dispatch_async(dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_BACKGROUND, 0), {
  64. let parser = HttpParser()
  65. while let request = parser.nextHttpRequest(socket) {
  66. let keepAlive = parser.supportsKeepAlive(request.headers)
  67. if let (expression, handler) = self.findHandler(request.url) {
  68. let capturedUrlsGroups = self.captureExpressionGroups(expression, value: request.url)
  69. let updatedRequest = HttpRequest(url: request.url, method: request.method, headers: request.headers, body: request.body, capturedUrlGroups: capturedUrlsGroups)
  70. HttpServer.writeResponse(socket, response: handler(updatedRequest), keepAlive: keepAlive)
  71. } else {
  72. HttpServer.writeResponse(socket, response: HttpResponse.NotFound, keepAlive: keepAlive)
  73. }
  74. if !keepAlive { break }
  75. }
  76. Socket.release(socket)
  77. });
  78. }
  79. self.releaseAcceptSocket()
  80. });
  81. return true
  82. }
  83. return false
  84. }
  85. func findHandler(url:String) -> (NSRegularExpression, Handler)? {
  86. return filter(self.handlers, { (expression: NSRegularExpression, handler) -> Bool in
  87. return expression.numberOfMatchesInString(url, options: self.matchingOptions, range: NSMakeRange(0, url.lengthOfBytesUsingEncoding(NSASCIIStringEncoding))) > 0
  88. }).first
  89. }
  90. func captureExpressionGroups(expression: NSRegularExpression, value: String) -> [String] {
  91. var capturedGroups = [String]()
  92. if let result = expression.firstMatchInString(value, options: matchingOptions, range: NSMakeRange(0, value.lengthOfBytesUsingEncoding(NSASCIIStringEncoding))) {
  93. let nsValue: NSString = value
  94. for var i = 1 ; i < result.numberOfRanges ; ++i {
  95. capturedGroups.append(nsValue.substringWithRange(result.rangeAtIndex(i)))
  96. }
  97. }
  98. return capturedGroups
  99. }
  100. class func writeResponse(socket: CInt, response: HttpResponse, keepAlive: Bool) {
  101. Socket.writeStringUTF8(socket, string: "HTTP/1.1 \(response.statusCode()) \(response.reasonPhrase())\r\n")
  102. let messageBody = response.body()
  103. if let body = messageBody {
  104. if let nsdata = body.dataUsingEncoding(NSUTF8StringEncoding) {
  105. Socket.writeStringUTF8(socket, string: "Content-Length: \(nsdata.length)\r\n")
  106. }
  107. } else {
  108. Socket.writeStringUTF8(socket, string: "Content-Length: 0\r\n")
  109. }
  110. if keepAlive {
  111. Socket.writeStringUTF8(socket, string: "Connection: keep-alive\r\n")
  112. }
  113. //Socket.writeStringUTF8(socket, string: "Content-Type: text/html; charset=UTF-8\r\n")
  114. for (name, value) in response.headers() {
  115. Socket.writeStringUTF8(socket, string: "\(name): \(value)\r\n")
  116. }
  117. Socket.writeStringUTF8(socket, string: "\r\n")
  118. if let body = messageBody {
  119. Socket.writeStringUTF8(socket, string: body)
  120. }
  121. }
  122. func stop() {
  123. releaseAcceptSocket()
  124. }
  125. func releaseAcceptSocket() {
  126. if ( acceptSocket != -1 ) {
  127. Socket.release(acceptSocket)
  128. acceptSocket = -1
  129. }
  130. }
  131. }