// // WebSearchService.swift // oAI // // DuckDuckGo web search for non-OpenRouter providers // import Foundation import os struct SearchResult: Sendable { let title: String let url: String let snippet: String } final class WebSearchService: Sendable { nonisolated static let shared = WebSearchService() private let session: URLSession nonisolated private init() { let config = URLSessionConfiguration.default config.timeoutIntervalForRequest = 10 session = URLSession(configuration: config) } /// Search DuckDuckGo HTML interface (no API key needed) nonisolated func search(query: String, maxResults: Int = 5) async -> [SearchResult] { Log.search.info("Web search: \(query)") guard let encoded = query.addingPercentEncoding(withAllowedCharacters: .urlQueryAllowed), let url = URL(string: "https://html.duckduckgo.com/html/?q=\(encoded)") else { return [] } var request = URLRequest(url: url) request.httpMethod = "GET" request.setValue( "Mozilla/5.0 (Macintosh; Intel Mac OS X 10_15_7) AppleWebKit/537.36", forHTTPHeaderField: "User-Agent" ) do { let (data, _) = try await session.data(for: request) guard let html = String(data: data, encoding: .utf8) else { return [] } return parseResults(from: html, maxResults: maxResults) } catch { Log.search.error("Web search failed: \(error.localizedDescription)") return [] } } /// Format search results as markdown for prompt injection nonisolated func formatResults(_ results: [SearchResult], maxLength: Int = 2000) -> String { if results.isEmpty { return "No search results found." } var formatted = "**Web Search Results:**\n\n" for (i, result) in results.enumerated() { var entry = "\(i + 1). **\(result.title)**\n" entry += " URL: \(result.url)\n" if !result.snippet.isEmpty { entry += " \(result.snippet)\n" } entry += "\n" if formatted.count + entry.count > maxLength { formatted += "... (\(results.count - i) more results truncated)\n" break } formatted += entry } return formatted.trimmingCharacters(in: .whitespacesAndNewlines) } // MARK: - HTML Parsing private nonisolated func parseResults(from html: String, maxResults: Int) -> [SearchResult] { var results: [SearchResult] = [] // Match result blocks: