Files
hikari-desktop/vitest.setup.ts
T
hikari 452fe185df
Security Scan and Upload / Security & DefectDojo Upload (push) Successful in 1m21s
CI / Lint & Test (push) Has started running
CI / Build Linux (push) Has been cancelled
CI / Build Windows (cross-compile) (push) Has been cancelled
feat: CLI v2.1.68–v2.1.74 compatibility updates (#221)
## Summary

This PR brings Hikari Desktop up to full compatibility with Claude Code CLI versions v2.1.68 through v2.1.74, implementing all changelog items audited in issues #200–#218.

## Changes

### Bug Fixes
- Remove deprecated Claude Opus 4.0 and 4.1 models from the model selector
- Auto-migrate users pinned to deprecated models to Opus 4.6

### New Features
- Add cron tool support (`CronCreate`, `CronDelete`, `CronList`) with character state mapping and `CLAUDE_CODE_DISABLE_CRON` settings toggle
- Handle `EnterWorktree` and `ExitWorktree` tools in character state mapping and tool display
- Add CLI update check with npm registry indicator in the version bar
- Add `agent_type` field and support the Agent tool rename from CLI v2.1.69
- Consume `worktree` field from status line hook events
- Display per-agent model override in the agent monitor tree
- Expose Claude Code CLI built-in slash commands (`/simplify`, `/loop`, `/batch`, `/memory`, `/context`) in the command menu with CLI badges
- Add `includeGitInstructions` toggle in settings
- Add `ENABLE_CLAUDEAI_MCP_SERVERS` opt-out setting
- Linkify MCP binary file paths (PDFs, audio, Office docs) in markdown output
- Add auto-memory panel, `/memory` slash command shortcut, and unified toast notification system
- Toast notifications for `WorktreeCreate` and `WorktreeRemove` hook events
- Sort session resume list by most recent activity, with most recent user message as preview
- Convert WSL Linux paths to Windows UNC paths when opening binary files via `open_binary_file` command
- Expose `autoMemoryDirectory` setting in ConfigSidebar (Agent Settings section)
- Add `/context` as a CLI built-in in the slash command menu
- Expose `modelOverrides` setting as a JSON textarea in ConfigSidebar (for AWS Bedrock, Google Vertex, etc.)

> **Note:** The CLI update check commit does not have a corresponding issue — it was a bonus addition during the audit sprint.

## Closes

Closes #200
Closes #201
Closes #202
Closes #205
Closes #206
Closes #207
Closes #208
Closes #209
Closes #210
Closes #211
Closes #212
Closes #213
Closes #214
Closes #215
Closes #216
Closes #217
Closes #218

Reviewed-on: #221
Co-authored-by: Hikari <hikari@nhcarrigan.com>
Co-committed-by: Hikari <hikari@nhcarrigan.com>
2026-03-13 01:34:44 -07:00

295 lines
8.0 KiB
TypeScript

import "@testing-library/jest-dom/vitest";
import { vi, beforeEach } from "vitest";
// Mock Tauri invoke API
const mockInvokeResults: Record<string, unknown> = {};
export function setMockInvokeResult(command: string, result: unknown) {
mockInvokeResults[command] = result;
}
export function clearMockInvokeResults() {
Object.keys(mockInvokeResults).forEach((key) => delete mockInvokeResults[key]);
}
vi.mock("@tauri-apps/api/core", () => ({
invoke: vi.fn((command: string) => {
if (command in mockInvokeResults) {
const result = mockInvokeResults[command];
if (result instanceof Error) {
const err = result;
return Promise.resolve().then(() => {
throw err;
});
}
return Promise.resolve(result);
}
// Default return values for common commands
switch (command) {
case "get_config":
return Promise.resolve({
model: null,
api_key: null,
custom_instructions: null,
mcp_servers_json: null,
auto_granted_tools: [],
theme: "dark",
greeting_enabled: true,
greeting_custom_prompt: null,
notifications_enabled: true,
notification_volume: 0.7,
always_on_top: false,
update_checks_enabled: true,
character_panel_width: null,
font_size: 14,
streamer_mode: false,
streamer_hide_paths: false,
compact_mode: false,
profile_name: null,
profile_avatar_path: null,
profile_bio: null,
custom_theme_colors: {},
auto_memory_directory: null,
model_overrides: null,
});
case "list_quick_actions":
return Promise.resolve([]);
case "list_snippets":
return Promise.resolve([]);
case "list_sessions":
return Promise.resolve([]);
case "get_usage_stats":
return Promise.resolve({
total_messages: 0,
total_sessions: 0,
total_tokens: 0,
total_cost: 0,
});
case "get_persisted_stats":
return Promise.resolve({
lifetime_messages: 0,
lifetime_sessions: 0,
lifetime_tokens: 0,
lifetime_cost: 0,
achievements: [],
unlocked_achievements: [],
});
case "load_saved_achievements":
return Promise.resolve([]);
case "list_clipboard_entries":
return Promise.resolve([]);
case "list_drafts":
return Promise.resolve([]);
case "cleanup_temp_files":
return Promise.resolve();
case "validate_directory":
return Promise.resolve(true);
case "git_status":
return Promise.resolve({
branch: "main",
files: [],
ahead: 0,
behind: 0,
});
default:
return Promise.resolve(null);
}
}),
}));
// Mock Tauri event API
const eventListeners: Map<string, Array<(event: { payload: unknown }) => void>> = new Map();
export function emitMockEvent(eventName: string, payload: unknown) {
const listeners = eventListeners.get(eventName) || [];
listeners.forEach((listener) => listener({ payload }));
}
export function clearMockEventListeners() {
eventListeners.clear();
}
vi.mock("@tauri-apps/api/event", () => ({
listen: vi.fn((eventName: string, handler: (event: { payload: unknown }) => void) => {
const listeners = eventListeners.get(eventName) || [];
listeners.push(handler);
eventListeners.set(eventName, listeners);
// Return an unlisten function
return Promise.resolve(() => {
const currentListeners = eventListeners.get(eventName) || [];
const index = currentListeners.indexOf(handler);
if (index > -1) {
currentListeners.splice(index, 1);
}
});
}),
emit: vi.fn(),
}));
// Mock Tauri plugins
vi.mock("@tauri-apps/plugin-dialog", () => ({
save: vi.fn(() => Promise.resolve(null)),
open: vi.fn(() => Promise.resolve(null)),
message: vi.fn(() => Promise.resolve()),
ask: vi.fn(() => Promise.resolve(true)),
confirm: vi.fn(() => Promise.resolve(true)),
}));
vi.mock("@tauri-apps/plugin-fs", () => ({
writeTextFile: vi.fn(() => Promise.resolve()),
readTextFile: vi.fn(() => Promise.resolve("{}")),
exists: vi.fn(() => Promise.resolve(false)),
mkdir: vi.fn(() => Promise.resolve()),
remove: vi.fn(() => Promise.resolve()),
readDir: vi.fn(() => Promise.resolve([])),
}));
vi.mock("@tauri-apps/plugin-opener", () => ({
openPath: vi.fn(() => Promise.resolve()),
openUrl: vi.fn(() => Promise.resolve()),
}));
vi.mock("@tauri-apps/plugin-notification", () => ({
sendNotification: vi.fn(() => Promise.resolve()),
requestPermission: vi.fn(() => Promise.resolve("granted")),
isPermissionGranted: vi.fn(() => Promise.resolve(true)),
}));
vi.mock("@tauri-apps/plugin-clipboard-manager", () => ({
writeText: vi.fn(() => Promise.resolve()),
readText: vi.fn(() => Promise.resolve("")),
writeImage: vi.fn(() => Promise.resolve()),
readImage: vi.fn(() => Promise.resolve(null)),
}));
vi.mock("@tauri-apps/plugin-os", () => ({
platform: vi.fn(() => Promise.resolve("linux")),
arch: vi.fn(() => Promise.resolve("x86_64")),
type: vi.fn(() => Promise.resolve("Linux")),
version: vi.fn(() => Promise.resolve("1.0.0")),
}));
vi.mock("@tauri-apps/plugin-http", () => ({
fetch: vi.fn(() =>
Promise.resolve({
ok: true,
status: 200,
json: () => Promise.resolve({}),
text: () => Promise.resolve(""),
})
),
}));
// Mock browser APIs
class MockAudioElement {
src = "";
volume = 1;
loop = false;
currentTime = 0;
paused = true;
preload = "auto";
onloadeddata: (() => void) | null = null;
onended: (() => void) | null = null;
onerror: ((e: Event) => void) | null = null;
play() {
this.paused = false;
return Promise.resolve();
}
pause() {
this.paused = true;
}
load() {
if (this.onloadeddata) {
setTimeout(() => this.onloadeddata?.(), 0);
}
}
addEventListener(event: string, handler: () => void) {
if (event === "loadeddata") this.onloadeddata = handler;
if (event === "ended") this.onended = handler;
}
removeEventListener() {
// No-op for tests
}
}
// @ts-expect-error - Mock Audio constructor
globalThis.Audio = MockAudioElement;
// Mock localStorage
const localStorageStore: Record<string, string> = {};
Object.defineProperty(globalThis, "localStorage", {
value: {
getItem: vi.fn((key: string) => localStorageStore[key] || null),
setItem: vi.fn((key: string, value: string) => {
localStorageStore[key] = value;
}),
removeItem: vi.fn((key: string) => {
delete localStorageStore[key];
}),
clear: vi.fn(() => {
Object.keys(localStorageStore).forEach((key) => delete localStorageStore[key]);
}),
key: vi.fn((index: number) => Object.keys(localStorageStore)[index] || null),
get length() {
return Object.keys(localStorageStore).length;
},
},
writable: true,
});
// Mock matchMedia
Object.defineProperty(globalThis, "matchMedia", {
writable: true,
value: vi.fn((query: string) => ({
matches: query.includes("dark"),
media: query,
onchange: null,
addListener: vi.fn(),
removeListener: vi.fn(),
addEventListener: vi.fn(),
removeEventListener: vi.fn(),
dispatchEvent: vi.fn(),
})),
});
// Mock ResizeObserver
class MockResizeObserver {
observe = vi.fn();
unobserve = vi.fn();
disconnect = vi.fn();
}
globalThis.ResizeObserver = MockResizeObserver as unknown as typeof ResizeObserver;
// Mock IntersectionObserver
class MockIntersectionObserver {
observe = vi.fn();
unobserve = vi.fn();
disconnect = vi.fn();
}
globalThis.IntersectionObserver =
MockIntersectionObserver as unknown as typeof IntersectionObserver;
// Mock requestAnimationFrame
globalThis.requestAnimationFrame = vi.fn((callback) => {
return setTimeout(callback, 0) as unknown as number;
});
globalThis.cancelAnimationFrame = vi.fn((id) => {
clearTimeout(id);
});
// Reset all mocks before each test
beforeEach(() => {
vi.clearAllMocks();
clearMockInvokeResults();
clearMockEventListeners();
});