forked from intitni/CopilotForXcode
-
Notifications
You must be signed in to change notification settings - Fork 0
Expand file tree
/
Copy pathPythonThread.swift
More file actions
90 lines (79 loc) · 2.27 KB
/
PythonThread.swift
File metadata and controls
90 lines (79 loc) · 2.27 KB
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
import Foundation
import PythonKit
final class PythonThread: Thread {
static let shared = {
let thread = PythonThread(
target: PythonThread.self,
selector: #selector(PythonThread.setup),
object: nil
)
thread.name = "Python Thread"
thread.stackSize = 1_048_576 // so that langchain can be correctly imported.
return thread
}()
@objc static func setup() {
CFRunLoopRun()
}
@objc static func runPythonJob(_ job: PythonJob) {
job.run()
}
func runPython(_ closure: @escaping () -> Void) {
if !isExecuting {
start()
}
if Thread.current.isEqual(self) {
closure()
} else {
PythonThread.perform(
#selector(PythonThread.runPythonJob),
on: self,
with: PythonJob(closure: closure),
waitUntilDone: false
)
}
}
func runPythonAndWait<T>(_ closure: @escaping () throws -> T) throws -> T {
if !isExecuting {
start()
}
if Thread.current.isEqual(self) {
return try closure()
} else {
let job = PythonJob(closure: closure)
PythonThread.perform(
#selector(PythonThread.runPythonJob),
on: self,
with: job,
waitUntilDone: true
)
guard let result = job.result else {
throw FailedToGetPythonJobResultError()
}
switch result {
case let .success(value):
if let value = value as? T {
return value
} else {
throw FailedToGetPythonJobResultError()
}
case let .failure(error):
throw error
}
}
}
}
struct FailedToGetPythonJobResultError: Error, LocalizedError {
var errorDescription: String? {
"Failed to get PythonJob result."
}
}
final class PythonJob: NSObject {
let closure: () throws -> Any
var result: Result<Any, Error>?
init(closure: @escaping () throws -> Any) {
self.closure = closure
}
func run() {
result = Result(catching: closure)
}
}