forked from intitni/CopilotForXcode
-
Notifications
You must be signed in to change notification settings - Fork 0
Expand file tree
/
Copy pathRunPython.swift
More file actions
88 lines (80 loc) · 2.49 KB
/
RunPython.swift
File metadata and controls
88 lines (80 loc) · 2.49 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
import Foundation
import PythonKit
var gilStateEnsure: (() -> Any)!
var gilStateRelease: ((Any) -> Void)!
func gilStateGuard<T>(_ closure: @escaping () throws -> T) throws -> T {
let state = gilStateEnsure()
defer { gilStateRelease(state) }
do {
let result = try closure()
return result
} catch {
throw error
}
}
@MainActor
var isPythonInitialized = false
@MainActor
public func initializePython<GilState, ThreadState>(
sitePackagePath: String,
stdLibPath: String,
libDynloadPath: String,
Py_Initialize: () -> Void,
PyEval_SaveThread: () -> ThreadState,
PyGILState_Ensure: @escaping () -> GilState,
PyGILState_Release: @escaping (GilState) -> Void
) {
guard !isPythonInitialized else { return }
setenv("PYTHONHOME", stdLibPath, 1)
setenv("PYTHONPATH", "\(stdLibPath):\(libDynloadPath):\(sitePackagePath)", 1)
setenv("PYTHONIOENCODING", "utf-8", 1)
isPythonInitialized = true
// Initialize python
Py_Initialize()
// // Immediately release the thread, so that we can ensure the GIL state later.
// // We may not recover the thread because all future tasks will be done in the other threads.
// _ = PyEval_SaveThread()
// // Setup GIL state guard.
// gilStateEnsure = { PyGILState_Ensure() }
// gilStateRelease = { gilState in PyGILState_Release(gilState as! GilState) }
}
public func runPython<T>(
usePythonThread: Bool = false,
_ closure: @escaping () throws -> T
) throws -> T {
if usePythonThread {
return try PythonThread.shared.runPythonAndWait {
// return try gilStateGuard {
return try closure()
// }
}
} else {
// return try gilStateGuard {
return try closure()
// }
}
}
public extension PythonInterface {
func attemptImportOnPythonThread(_ name: String) throws -> PythonObject {
try PythonThread.shared.runPythonAndWait {
let module = try Python.attemptImport(name)
return module
}
}
}
public struct ReadablePythonError: Error, LocalizedError {
public var error: PythonError
public init(_ error: PythonError) {
self.error = error
}
public var errorDescription: String? {
switch error {
case let .exception(object, _):
return "\(object)"
case let .invalidCall(object):
return "Invalid call: \(object)"
case let .invalidModule(module):
return "Invalid module: \(module)"
}
}
}