2017-07-07 12:41:22 +02:00
|
|
|
@objc(Webserver) class Webserver : CDVPlugin {
|
2017-07-10 09:58:47 +02:00
|
|
|
// Timeout in seconds
|
|
|
|
let TIMEOUT: Int = 60 * 3 * 1000000
|
|
|
|
|
2017-07-08 12:15:41 +02:00
|
|
|
var webServer: GCDWebServer = GCDWebServer()
|
2018-08-21 11:57:01 +03:00
|
|
|
var responses = SynchronizedDictionary<AnyHashable,Any?>()
|
2017-07-10 09:58:47 +02:00
|
|
|
var onRequestCommand: CDVInvokedUrlCommand? = nil
|
2017-07-07 15:23:55 +02:00
|
|
|
|
|
|
|
override func pluginInitialize() {
|
2017-07-08 12:15:41 +02:00
|
|
|
self.webServer = GCDWebServer()
|
2017-07-10 09:58:47 +02:00
|
|
|
self.onRequestCommand = nil
|
2018-08-21 11:57:01 +03:00
|
|
|
self.responses = SynchronizedDictionary<AnyHashable,Any?>()
|
2017-07-10 14:38:46 +02:00
|
|
|
self.initHTTPRequestHandlers()
|
2017-07-10 09:58:47 +02:00
|
|
|
}
|
|
|
|
|
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)) ?? ""
|
|
|
|
}
|
|
|
|
|
2017-07-10 09:58:47 +02:00
|
|
|
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 ?? ""
|
2017-07-10 09:58:47 +02:00
|
|
|
]
|
|
|
|
}
|
|
|
|
|
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)!
|
|
|
|
}
|
|
|
|
|
2017-07-10 14:38:46 +02:00
|
|
|
func processRequest(request: GCDWebServerRequest, completionBlock: GCDWebServerCompletionBlock) {
|
2017-07-10 09:58:47 +02:00
|
|
|
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)
|
2017-07-10 09:58:47 +02:00
|
|
|
|
|
|
|
// Do a call to the onRequestCommand to inform the JS plugin
|
|
|
|
if (self.onRequestCommand != nil) {
|
2017-07-10 14:38:46 +02:00
|
|
|
let pluginResult = CDVPluginResult(status: CDVCommandStatus_OK, messageAs: requestDict)
|
|
|
|
pluginResult?.setKeepCallbackAs(true)
|
2017-07-10 09:58:47 +02:00
|
|
|
self.commandDelegate.send(
|
2017-07-10 14:38:46 +02:00
|
|
|
pluginResult,
|
2017-07-10 09:58:47 +02:00
|
|
|
callbackId: self.onRequestCommand?.callbackId
|
|
|
|
)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Here we have to wait until the javascript block fetches the message and do a response
|
2017-07-10 14:38:46 +02:00
|
|
|
while self.responses[requestUUID] == nil {
|
2017-07-10 14:48:40 +02:00
|
|
|
timeout += 1000
|
|
|
|
usleep(1000)
|
2017-07-10 09:58:47 +02:00
|
|
|
}
|
|
|
|
|
2017-07-10 14:48:40 +02:00
|
|
|
// We got the dict so put information in the response
|
2017-07-10 14:38:46 +02:00
|
|
|
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
|
|
|
|
}
|
2017-07-10 14:38:46 +02:00
|
|
|
|
|
|
|
for (key, value) in (responseDict["headers"] as! Dictionary<String, String>) {
|
|
|
|
response?.setValue(value, forAdditionalHeader: key)
|
|
|
|
}
|
2017-07-10 14:48:40 +02:00
|
|
|
|
2018-08-21 11:57:01 +03:00
|
|
|
// Remove the handled response
|
|
|
|
self.responses.removeValue(forKey: requestUUID)
|
|
|
|
|
2017-07-10 14:48:40 +02:00
|
|
|
// Complete the async response
|
2017-07-10 14:38:46 +02:00
|
|
|
completionBlock(response!)
|
2017-07-08 12:15:41 +02:00
|
|
|
}
|
2017-07-10 09:58:47 +02:00
|
|
|
|
2019-05-09 15:32:48 +02:00
|
|
|
@objc(onRequest:)
|
2017-07-10 09:58:47 +02:00
|
|
|
func onRequest(_ command: CDVInvokedUrlCommand) {
|
|
|
|
self.onRequestCommand = command
|
2017-07-10 14:38:46 +02:00
|
|
|
let pluginResult = CDVPluginResult(status: CDVCommandStatus_NO_RESULT)
|
|
|
|
pluginResult?.setKeepCallbackAs(true)
|
|
|
|
self.commandDelegate.send(
|
|
|
|
pluginResult,
|
|
|
|
callbackId: self.onRequestCommand?.callbackId
|
|
|
|
)
|
2017-07-10 09:58:47 +02:00
|
|
|
}
|
|
|
|
|
2017-07-08 12:15:41 +02:00
|
|
|
func initHTTPRequestHandlers() {
|
2017-07-10 14:38:46 +02:00
|
|
|
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:)
|
2017-07-10 09:58:47 +02:00
|
|
|
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) {
|
2017-07-08 12:15:41 +02:00
|
|
|
var port = 8080
|
|
|
|
let portArgument = command.argument(at: 0)
|
2017-07-10 09:58:47 +02:00
|
|
|
|
2017-07-08 12:15:41 +02:00
|
|
|
if portArgument != nil {
|
|
|
|
port = portArgument as! Int
|
2017-07-07 15:23:55 +02:00
|
|
|
}
|
2019-05-13 15:08:15 +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)
|
2019-05-13 15:08:15 +02:00
|
|
|
self.commandDelegate!.send(CDVPluginResult(status: CDVCommandStatus_ERROR, messageAs: error.localizedDescription), callbackId: command.callbackId)
|
|
|
|
return
|
2018-05-02 16:18:19 +02:00
|
|
|
}
|
2017-07-08 12:15:41 +02:00
|
|
|
let pluginResult = CDVPluginResult(status: CDVCommandStatus_OK)
|
|
|
|
self.commandDelegate!.send(pluginResult, callbackId: command.callbackId)
|
|
|
|
}
|
2017-07-10 09:58:47 +02:00
|
|
|
|
2019-05-09 15:32:48 +02:00
|
|
|
|
|
|
|
@objc(stop:)
|
2017-07-08 12:15:41 +02:00
|
|
|
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)
|
|
|
|
}
|
2017-07-07 12:41:22 +02:00
|
|
|
}
|