forked from intitni/CopilotForXcode
-
Notifications
You must be signed in to change notification settings - Fork 0
Expand file tree
/
Copy pathAcceptSuggestionCommand.swift
More file actions
62 lines (58 loc) · 2.06 KB
/
AcceptSuggestionCommand.swift
File metadata and controls
62 lines (58 loc) · 2.06 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
import Client
import SuggestionModel
import Foundation
import XcodeKit
import XPCShared
class AcceptSuggestionCommand: NSObject, XCSourceEditorCommand, CommandType {
var name: String { "Accept Suggestion" }
func perform(
with invocation: XCSourceEditorCommandInvocation,
completionHandler: @escaping (Error?) -> Void
) {
Task {
do {
try await (Task(timeout: 7) {
let service = try getService()
if let content = try await service.getSuggestionAcceptedCode(
editorContent: .init(invocation)
) {
invocation.accept(content)
}
completionHandler(nil)
}.value)
} catch is CancellationError {
completionHandler(nil)
} catch {
completionHandler(error)
}
}
}
}
/// https://gist.github.com/swhitty/9be89dfe97dbb55c6ef0f916273bbb97
extension Task where Failure == Error {
// Start a new Task with a timeout. If the timeout expires before the operation is
// completed then the task is cancelled and an error is thrown.
init(
priority: TaskPriority? = nil,
timeout: TimeInterval,
operation: @escaping @Sendable () async throws -> Success
) {
self = Task(priority: priority) {
try await withThrowingTaskGroup(of: Success.self) { group -> Success in
group.addTask(operation: operation)
group.addTask {
try await _Concurrency.Task.sleep(nanoseconds: UInt64(timeout * 1_000_000_000))
throw TimeoutError()
}
guard let success = try await group.next() else {
throw _Concurrency.CancellationError()
}
group.cancelAll()
return success
}
}
}
}
private struct TimeoutError: LocalizedError {
var errorDescription: String? = "Task timed out before completion"
}