import Foundation import LanguageClient import LanguageServerProtocol import Logger import SuggestionModel public protocol CodeiumSuggestionServiceType { func getCompletions( fileURL: URL, content: String, cursorPosition: CursorPosition, tabSize: Int, indentSize: Int, usesTabsForIndentation: Bool, ignoreSpaceOnlySuggestions: Bool ) async throws -> [CodeSuggestion] } enum CodeiumError: Error, LocalizedError { case languageServerNotInstalled var errorDescription: String? { switch self { case .languageServerNotInstalled: return "Language server is not installed." } } } public class CodeiumSuggestionService: CodeiumSuggestionServiceType { static let sessionId = UUID().uuidString let projectRootURL: URL var server: CodeiumLSP var heartbeatTask: Task? var requestCounter: UInt64 = 0 init(designatedServer: CodeiumLSP) { projectRootURL = URL(fileURLWithPath: "/") server = designatedServer } public init(projectRootURL: URL) throws { self.projectRootURL = projectRootURL let urls = try CodeiumSuggestionService.createFoldersIfNeeded() let languageServerURL = urls.executableURL.appendingPathComponent("language_server") guard FileManager.default.fileExists(atPath: languageServerURL.path) else { throw CodeiumError.languageServerNotInstalled } let tempFolderURL = FileManager.default.temporaryDirectory let managerDirectoryURL = tempFolderURL .appendingPathComponent("com.intii.CopilotForXcode") .appendingPathComponent(UUID().uuidString) if !FileManager.default.fileExists(atPath: managerDirectoryURL.path) { try FileManager.default.createDirectory( at: managerDirectoryURL, withIntermediateDirectories: true ) } let server = CodeiumLanguageServer( languageServerExecutableURL: languageServerURL, managerDirectoryURL: managerDirectoryURL, supportURL: urls.supportURL ) self.server = server server.terminationHandler = { [weak self] in Logger.codeium.info("Language server is terminated") guard let self else { return } } server.launchHandler = { [weak self] in guard let self else { return } let metadata = self.getMetadata() self.heartbeatTask = Task { [weak self] in while true { try Task.checkCancellation() _ = try? await self?.server.sendRequest( CodeiumRequest.Heartbeat(requestBody: .init(metadata: metadata)) ) try await Task.sleep(nanoseconds: 5_000_000_000) } } } server.start() } public func getCompletions( fileURL: URL, content: String, cursorPosition: CursorPosition, tabSize: Int, indentSize: Int, usesTabsForIndentation: Bool, ignoreSpaceOnlySuggestions: Bool ) async throws -> [CodeSuggestion] { requestCounter += 1 let languageId = languageIdentifierFromFileURL(fileURL) let relativePath = { let filePath = fileURL.path let rootPath = projectRootURL.path if let range = filePath.range(of: rootPath), range.lowerBound == filePath.startIndex { let relativePath = filePath.replacingCharacters( in: filePath.startIndex.. ( applicationSupportURL: URL, gitHubCopilotURL: URL, executableURL: URL, supportURL: URL ) { let supportURL = FileManager.default.urls( for: .applicationSupportDirectory, in: .userDomainMask ).first!.appendingPathComponent( Bundle.main .object(forInfoDictionaryKey: "APPLICATION_SUPPORT_FOLDER") as! String ) if !FileManager.default.fileExists(atPath: supportURL.path) { try? FileManager.default .createDirectory(at: supportURL, withIntermediateDirectories: false) } let gitHubCopilotFolderURL = supportURL.appendingPathComponent("Codeium") if !FileManager.default.fileExists(atPath: gitHubCopilotFolderURL.path) { try? FileManager.default .createDirectory(at: gitHubCopilotFolderURL, withIntermediateDirectories: false) } let supportFolderURL = gitHubCopilotFolderURL.appendingPathComponent("support") if !FileManager.default.fileExists(atPath: supportFolderURL.path) { try? FileManager.default .createDirectory(at: supportFolderURL, withIntermediateDirectories: false) } let executableFolderURL = gitHubCopilotFolderURL.appendingPathComponent("executable") if !FileManager.default.fileExists(atPath: executableFolderURL.path) { try? FileManager.default .createDirectory(at: executableFolderURL, withIntermediateDirectories: false) } return (supportURL, gitHubCopilotFolderURL, executableFolderURL, supportFolderURL) } } extension CodeiumSuggestionService { func getMetadata() -> Metadata { Metadata( ide_name: "jetbrains", ide_version: "14.3", extension_name: "Copilot for Xcode", extension_version: "14.0.0", api_key: token, session_id: CodeiumSuggestionService.sessionId, request_id: requestCounter ) } }