cordova-plugin-webserver/src/ios/Webserver.swift

151 lines
5.6 KiB
Swift
Raw Normal View History

@objc(Webserver) class Webserver : CDVPlugin {
// Timeout in seconds
let TIMEOUT: Int = 60 * 3 * 1000000
var webServer: GCDWebServer = GCDWebServer()
var responses = SynchronizedDictionary<AnyHashable,Any?>()
var onRequestCommand: CDVInvokedUrlCommand? = nil
2017-07-07 15:23:55 +02:00
override func pluginInitialize() {
self.webServer = GCDWebServer()
self.onRequestCommand = nil
self.responses = SynchronizedDictionary<AnyHashable,Any?>()
self.initHTTPRequestHandlers()
}
2017-07-12 10:45:05 +02:00
func requestToRequestDict(requestUUID: String, request: GCDWebServerRequest) -> Dictionary<String, Any> {
let dataRequest = request as! GCDWebServerDataRequest
var body = ""
if dataRequest.hasBody() {
body = String(data: dataRequest.data, encoding: String.Encoding(rawValue: String.Encoding.utf8.rawValue)) ?? ""
}
return [
"requestId": requestUUID,
2017-07-12 10:45:05 +02:00
"body": body,
"headers": dataRequest.headers,
"method": dataRequest.method,
"path": dataRequest.url.path,
"query": dataRequest.url.query ?? ""
]
}
2019-05-10 20:27:08 +02:00
func fileRequest(path: String) -> GCDWebServerResponse {
// Check if file exists, given its path
if !(FileManager.default.fileExists(atPath: path)) {
return GCDWebServerResponse(statusCode: 404);
}
return GCDWebServerFileResponse(file: path)!
}
func processRequest(request: GCDWebServerRequest, completionBlock: GCDWebServerCompletionBlock) {
var timeout = 0
// Fetch data as GCDWebserverDataRequest
let requestUUID = UUID().uuidString
// Transform it into an dictionary for the javascript plugin
2017-07-12 10:45:05 +02:00
let requestDict = self.requestToRequestDict(requestUUID: requestUUID, request: request)
// Do a call to the onRequestCommand to inform the JS plugin
if (self.onRequestCommand != nil) {
let pluginResult = CDVPluginResult(status: CDVCommandStatus_OK, messageAs: requestDict)
pluginResult?.setKeepCallbackAs(true)
self.commandDelegate.send(
pluginResult,
callbackId: self.onRequestCommand?.callbackId
)
}
// Here we have to wait until the javascript block fetches the message and do a response
while self.responses[requestUUID] == nil {
2017-07-10 14:48:40 +02:00
timeout += 1000
usleep(1000)
}
2017-07-10 14:48:40 +02:00
// We got the dict so put information in the response
let responseDict = self.responses[requestUUID] as! Dictionary<AnyHashable, Any>
2019-05-10 20:27:08 +02:00
// Check if a file path is provided else use regular data response
let response = responseDict["path"] != nil
? fileRequest(path: responseDict["path"] as! String)
: GCDWebServerDataResponse(text: responseDict["body"] as! String)
if responseDict["status"] != nil {
response?.statusCode = responseDict["status"] as! Int
}
for (key, value) in (responseDict["headers"] as! Dictionary<String, String>) {
response?.setValue(value, forAdditionalHeader: key)
}
2017-07-10 14:48:40 +02:00
// Remove the handled response
self.responses.removeValue(forKey: requestUUID)
2017-07-10 14:48:40 +02:00
// Complete the async response
completionBlock(response!)
}
2019-05-09 15:32:48 +02:00
@objc(onRequest:)
func onRequest(_ command: CDVInvokedUrlCommand) {
self.onRequestCommand = command
let pluginResult = CDVPluginResult(status: CDVCommandStatus_NO_RESULT)
pluginResult?.setKeepCallbackAs(true)
self.commandDelegate.send(
pluginResult,
callbackId: self.onRequestCommand?.callbackId
)
}
func initHTTPRequestHandlers() {
self.webServer.addHandler(
match: {
(requestMethod, requestURL, requestHeaders, urlPath, urlQuery) -> GCDWebServerRequest? in
return GCDWebServerDataRequest(method: requestMethod, url: requestURL, headers: requestHeaders, path: urlPath, query: urlQuery)
},
asyncProcessBlock: self.processRequest
)
2017-07-07 15:23:55 +02:00
}
2019-05-09 15:32:48 +02:00
@objc(sendResponse:)
func sendResponse(_ command: CDVInvokedUrlCommand) {
self.responses[command.argument(at: 0) as! String] = command.argument(at: 1)
}
2019-05-09 15:32:48 +02:00
@objc(start:)
2017-07-07 15:23:55 +02:00
func start(_ command: CDVInvokedUrlCommand) {
var port = 8080
let portArgument = command.argument(at: 0)
if portArgument != nil {
port = portArgument as! Int
2017-07-07 15:23:55 +02:00
}
if self.webServer.isRunning{
self.commandDelegate!.send(CDVPluginResult(status: CDVCommandStatus_ERROR, messageAs: "Server already running"), callbackId: command.callbackId)
return
}
2018-05-02 16:18:19 +02:00
do {
try self.webServer.start(options:[GCDWebServerOption_AutomaticallySuspendInBackground : false, GCDWebServerOption_Port: UInt(port)])
} catch let error {
print(error.localizedDescription)
self.commandDelegate!.send(CDVPluginResult(status: CDVCommandStatus_ERROR, messageAs: error.localizedDescription), callbackId: command.callbackId)
return
2018-05-02 16:18:19 +02:00
}
let pluginResult = CDVPluginResult(status: CDVCommandStatus_OK)
self.commandDelegate!.send(pluginResult, callbackId: command.callbackId)
}
2019-05-09 15:32:48 +02:00
@objc(stop:)
func stop(_ command: CDVInvokedUrlCommand) {
if self.webServer.isRunning {
self.webServer.stop()
}
print("Stopping webserver")
2017-07-07 15:23:55 +02:00
let pluginResult = CDVPluginResult(status: CDVCommandStatus_OK)
self.commandDelegate!.send(pluginResult, callbackId: command.callbackId)
}
}