Peekaboo/peekaboo-cli/Sources/peekaboo/Models.swift
Peter Steinberger a491adbdf1 Enhance error handling with specific exit codes and user-friendly messages
- Add distinct exit codes for different error conditions in Swift CLI
- Map exit codes to clear, actionable error messages in Node.js server
- Replace generic "Swift CLI execution failed" with specific guidance
- Improve permission error messages to guide users to System Settings

🤖 Generated with [Claude Code](https://claude.ai/code)

Co-Authored-By: Claude <noreply@anthropic.com>
2025-06-07 22:44:07 +01:00

161 lines
4.2 KiB
Swift

import ArgumentParser
import Foundation
// MARK: - Image Capture Models
struct SavedFile: Codable {
let path: String
let item_label: String?
let window_title: String?
let window_id: UInt32?
let window_index: Int?
let mime_type: String
}
struct ImageCaptureData: Codable {
let saved_files: [SavedFile]
}
enum CaptureMode: String, CaseIterable, ExpressibleByArgument {
case screen
case window
case multi
}
enum ImageFormat: String, CaseIterable, ExpressibleByArgument {
case png
case jpg
}
enum CaptureFocus: String, CaseIterable, ExpressibleByArgument {
case background
case foreground
}
// MARK: - Application & Window Models
struct ApplicationInfo: Codable {
let app_name: String
let bundle_id: String
let pid: Int32
let is_active: Bool
let window_count: Int
}
struct ApplicationListData: Codable {
let applications: [ApplicationInfo]
}
struct WindowInfo: Codable {
let window_title: String
let window_id: UInt32?
let window_index: Int?
let bounds: WindowBounds?
let is_on_screen: Bool?
}
struct WindowBounds: Codable {
let xCoordinate: Int
let yCoordinate: Int
let width: Int
let height: Int
}
struct TargetApplicationInfo: Codable {
let app_name: String
let bundle_id: String?
let pid: Int32
}
struct WindowListData: Codable {
let windows: [WindowInfo]
let target_application_info: TargetApplicationInfo
}
// MARK: - Window Specifier
enum WindowSpecifier {
case title(String)
case index(Int)
}
// MARK: - Window Details Options
enum WindowDetailOption: String, CaseIterable {
case off_screen
case bounds
case ids
}
// MARK: - Window Management
struct WindowData {
let windowId: UInt32
let title: String
let bounds: CGRect
let isOnScreen: Bool
let windowIndex: Int
}
// MARK: - Error Types
enum CaptureError: Error, LocalizedError {
case noDisplaysAvailable
case screenRecordingPermissionDenied
case accessibilityPermissionDenied
case invalidDisplayID
case captureCreationFailed
case windowNotFound
case windowCaptureFailed
case fileWriteError(String)
case appNotFound(String)
case invalidWindowIndex(Int)
case invalidArgument(String)
case unknownError(String)
var errorDescription: String? {
switch self {
case .noDisplaysAvailable:
"No displays available for capture."
case .screenRecordingPermissionDenied:
"Screen recording permission is required. Please grant it in System Settings > Privacy & Security > Screen Recording."
case .accessibilityPermissionDenied:
"Accessibility permission is required for some operations. Please grant it in System Settings > Privacy & Security > Accessibility."
case .invalidDisplayID:
"Invalid display ID provided."
case .captureCreationFailed:
"Failed to create the screen capture."
case .windowNotFound:
"The specified window could not be found."
case .windowCaptureFailed:
"Failed to capture the specified window."
case let .fileWriteError(path):
"Failed to write capture file to path: \(path)."
case let .appNotFound(identifier):
"Application with identifier '\(identifier)' not found or is not running."
case let .invalidWindowIndex(index):
"Invalid window index: \(index)."
case let .invalidArgument(message):
"Invalid argument: \(message)"
case let .unknownError(message):
"An unexpected error occurred: \(message)"
}
}
var exitCode: Int32 {
switch self {
case .noDisplaysAvailable: return 10
case .screenRecordingPermissionDenied: return 11
case .accessibilityPermissionDenied: return 12
case .invalidDisplayID: return 13
case .captureCreationFailed: return 14
case .windowNotFound: return 15
case .windowCaptureFailed: return 16
case .fileWriteError: return 17
case .appNotFound: return 18
case .invalidWindowIndex: return 19
case .invalidArgument: return 20
case .unknownError: return 1
}
}
}