TabFS/extension/safari/TabFS/TabFS Extension/SafariWebExtensionHandler.swift

58 lines
2.4 KiB
Swift
Raw Normal View History

//
// SafariWebExtensionHandler.swift
// TabFS Extension
//
// Created by Omar Rizwan on 1/31/21.
//
import SafariServices
import os.log
class SafariWebExtensionHandler: NSObject, NSExtensionRequestHandling {
func beginRequest(with context: NSExtensionContext) {
let item = context.inputItems[0] as! NSExtensionItem
guard let message = item.userInfo?["message"] as? [AnyHashable: Any] else { return }
2021-02-08 13:21:58 +01:00
guard message["op"] as! String == "safari_did_connect" else { return }
2021-02-08 13:21:58 +01:00
// The XPC service is a subprocess that lives outside the macOS App Sandbox.
2021-02-08 14:03:31 +01:00
// (Safari extension native code, including this file, has to live in the sandbox.)
2021-02-08 13:21:58 +01:00
// It can do forbidden things like spawn tabfs filesystem and set up WebSocket server.
2021-02-08 14:03:31 +01:00
// We only use one native message to bootstrap the XPC service, then do all communications
// to that service (which in turn talks to tabfs.c) over WebSocket instead.
// (Safari makes doing native messaging quite painful, so we try to avoid it.
// It forces the browser to pop to front if you message Safari in the obvious way,
// for instance: https://developer.apple.com/forums/thread/122232
// And with the WebSocket, the XPC service can talk straight to background.js, whereas
// native messaging would require us here to sit in the middle.)
2021-02-08 13:21:58 +01:00
let connection = NSXPCConnection(serviceName: "com.rsnous.TabFSService")
connection.remoteObjectInterface = NSXPCInterface(with: TabFSServiceProtocol.self)
connection.resume()
let service = connection.remoteObjectProxyWithErrorHandler { error in
2021-02-08 14:03:31 +01:00
os_log(.default, "Received error: %{public}@", error as CVarArg)
2021-02-08 13:21:58 +01:00
} as? TabFSServiceProtocol
2021-02-08 14:03:31 +01:00
// Need this one XPC call to actually initialize the service.
service?.start() {
os_log(.default, "Response from XPC service")
2021-02-08 13:21:58 +01:00
// FIXME: report port back?
let response = NSExtensionItem()
2021-02-08 14:03:31 +01:00
response.userInfo = [ "message": "alive" ]
// This response (over native messaging) prompts background.js to
// connect to the WebSocket server that the XPC service should now be running.
2021-02-08 13:21:58 +01:00
context.completeRequest(returningItems: [response]) { (what) in
print(what)
}
}
2021-02-08 13:21:58 +01:00
return
}
}