mirror of
https://github.com/samiyev/puaros.git
synced 2025-12-27 23:06:54 +05:00
Compare commits
13 Commits
ipuaro-v0.
...
ipuaro-v0.
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
b5ee77d8b8 | ||
|
|
a589b0dfc4 | ||
|
|
908c2f50d7 | ||
|
|
510c42241a | ||
|
|
357cf27765 | ||
|
|
6695cb73d4 | ||
|
|
5a9470929c | ||
|
|
137c77cc53 | ||
|
|
0433ef102c | ||
|
|
902d1db831 | ||
|
|
c843b780a8 | ||
|
|
0dff0e87d0 | ||
|
|
ab2d5d40a5 |
@@ -5,6 +5,441 @@ All notable changes to this project will be documented in this file.
|
||||
The format is based on [Keep a Changelog](https://keepachangelog.com/en/1.0.0/),
|
||||
and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0.html).
|
||||
|
||||
## [0.21.4] - 2025-12-02 - Syntax Highlighting in DiffView
|
||||
|
||||
### Added
|
||||
|
||||
- **Syntax Highlighter Utility (0.21.4)**
|
||||
- New syntax-highlighter utility in `src/tui/utils/syntax-highlighter.ts`
|
||||
- Simple regex-based syntax highlighting for terminal UI
|
||||
- Language detection from file extension: `ts`, `tsx`, `js`, `jsx`, `json`, `yaml`, `yml`
|
||||
- Token types: keywords, strings, comments, numbers, operators, whitespace
|
||||
- Color mapping: keywords (magenta), strings (green), comments (gray), numbers (cyan), operators (yellow)
|
||||
- Support for single-line comments (`//`), multi-line comments (`/* */`)
|
||||
- String literals: double quotes, single quotes, template literals
|
||||
- Keywords: TypeScript/JavaScript keywords (const, let, function, async, etc.)
|
||||
- Exports: `detectLanguage()`, `highlightLine()`, `Language` type, `HighlightedToken` interface
|
||||
|
||||
- **EditConfigSchema Enhancement**
|
||||
- Added `syntaxHighlight` option to EditConfigSchema (default: `true`)
|
||||
- Enables/disables syntax highlighting in diff views globally
|
||||
|
||||
### Changed
|
||||
|
||||
- **DiffView Component Enhanced**
|
||||
- Added `language?: Language` prop for explicit language override
|
||||
- Added `syntaxHighlight?: boolean` prop (default: `false`)
|
||||
- Automatic language detection from `filePath` using `detectLanguage()`
|
||||
- Highlights only added lines (`type === "add"`) when syntax highlighting enabled
|
||||
- Renders tokens with individual colors when highlighting is active
|
||||
- Falls back to plain colored text when highlighting is disabled
|
||||
|
||||
- **ConfirmDialog Component**
|
||||
- Added `syntaxHighlight?: boolean` prop (default: `false`)
|
||||
- Passes `syntaxHighlight` to DiffView component
|
||||
- Enables syntax highlighting in confirmation dialogs when configured
|
||||
|
||||
- **App Component**
|
||||
- Added `syntaxHighlight?: boolean` prop to ExtendedAppProps (default: `true`)
|
||||
- Passes `syntaxHighlight` to ConfirmDialog
|
||||
- Integrates with global configuration for syntax highlighting
|
||||
|
||||
- **DiffLine Subcomponent**
|
||||
- Enhanced to support syntax highlighting mode
|
||||
- Conditional rendering: highlighted tokens vs plain colored text
|
||||
- Token-based rendering when syntax highlighting is active
|
||||
|
||||
### Technical Details
|
||||
|
||||
- Total tests: 1525 passed (was 1501, +24 new tests)
|
||||
- New test file: `syntax-highlighter.test.ts` with 24 tests
|
||||
- Language detection (9 tests)
|
||||
- Token highlighting for keywords, strings, comments, numbers, operators (15 tests)
|
||||
- Coverage: 97.63% lines, 91.25% branches, 98.97% functions, 97.63% statements
|
||||
- 0 ESLint errors, 0 warnings
|
||||
- Build successful with no TypeScript errors
|
||||
- Regex-based approach using `RegExp#exec()` for performance
|
||||
- No external dependencies added (native JavaScript)
|
||||
|
||||
### Notes
|
||||
|
||||
This release completes the v0.21.0 TUI Enhancements milestone. All items for v0.21.0 are now complete:
|
||||
- ✅ 0.21.1 - useAutocomplete Hook
|
||||
- ✅ 0.21.2 - Edit Mode in ConfirmDialog
|
||||
- ✅ 0.21.3 - Multiline Input support
|
||||
- ✅ 0.21.4 - Syntax Highlighting in DiffView
|
||||
|
||||
---
|
||||
|
||||
## [0.21.3] - 2025-12-02 - Multiline Input Support
|
||||
|
||||
### Added
|
||||
|
||||
- **InputConfigSchema (0.21.3)**
|
||||
- New configuration schema for input settings
|
||||
- `multiline` option: boolean | "auto" (default: false)
|
||||
- Supports three modes: `false` (disabled), `true` (always on), `"auto"` (activates when multiple lines present)
|
||||
- Added `InputConfig` type export
|
||||
|
||||
- **Multiline Input Component (0.21.3)**
|
||||
- Multiline text input support in Input component
|
||||
- Shift+Enter: add new line in multiline mode
|
||||
- Enter: submit all lines (in multiline mode) or submit text (in single-line mode)
|
||||
- Auto-height adjustment: dynamically shows all input lines
|
||||
- Line-by-line editing with visual indicator (">") for current line
|
||||
- Arrow key navigation (↑/↓) between lines in multiline mode
|
||||
- Instructions displayed: "Shift+Enter: new line | Enter: submit"
|
||||
- Seamless switch between single-line and multiline modes based on configuration
|
||||
|
||||
### Changed
|
||||
|
||||
- **Input Component Enhanced**
|
||||
- Added `multiline?: boolean | "auto"` prop
|
||||
- State management for multiple lines (`lines`, `currentLineIndex`)
|
||||
- Conditional rendering: single-line TextInput vs multiline Box with multiple lines
|
||||
- Arrow key handlers now support both history navigation (single-line) and line navigation (multiline)
|
||||
- Submit handler resets lines state in addition to value
|
||||
- Line change handlers: `handleLineChange`, `handleAddLine`, `handleMultilineSubmit`
|
||||
|
||||
- **App Component**
|
||||
- Added `multiline?: boolean | "auto"` prop to ExtendedAppProps
|
||||
- Passes multiline config to Input component
|
||||
- Default value: false (single-line mode)
|
||||
|
||||
- **Config Schema**
|
||||
- Added `input` section to ConfigSchema
|
||||
- InputConfigSchema included in full configuration
|
||||
- Config type updated to include InputConfig
|
||||
|
||||
### Technical Details
|
||||
|
||||
- Total tests: 1501 passed (was 1484, +17 new tests)
|
||||
- New test suite: "multiline support" with 21 tests
|
||||
- InputProps with multiline options
|
||||
- Multiline activation logic (true, false, "auto")
|
||||
- Line management (update, add, join)
|
||||
- Line navigation (up/down with boundaries)
|
||||
- Multiline submit (trim, empty check, reset)
|
||||
- Coverage: 97.67% lines, 91.37% branches, 98.97% functions, 97.67% statements
|
||||
- 0 ESLint errors, 0 warnings
|
||||
- Build successful with no type errors
|
||||
|
||||
### Notes
|
||||
|
||||
This release completes the third item of the v0.21.0 TUI Enhancements milestone. Remaining item for v0.21.0:
|
||||
- 0.21.4 - Syntax Highlighting in DiffView
|
||||
|
||||
---
|
||||
|
||||
## [0.21.1] - 2025-12-01 - TUI Enhancements (Part 2)
|
||||
|
||||
### Added
|
||||
|
||||
- **EditableContent Component (0.21.2)**
|
||||
- New component for inline multi-line editing in TUI
|
||||
- Line-by-line navigation with ↑/↓ arrow keys
|
||||
- Enter key: advance to next line / submit on last line
|
||||
- Ctrl+Enter: submit from any line
|
||||
- Escape: cancel editing and return to confirmation
|
||||
- Visual indicator (▶) for current line being edited
|
||||
- Scrollable view for large content (max 20 visible lines)
|
||||
- Instructions display at bottom of editor
|
||||
|
||||
- **Edit Mode in ConfirmDialog (0.21.2)**
|
||||
- [E] option now opens inline editor for proposed changes
|
||||
- Two modes: "confirm" (default) and "edit"
|
||||
- User can modify content before applying
|
||||
- Seamless transition between confirmation and editing
|
||||
- Edit button disabled when no editable content available
|
||||
|
||||
- **ConfirmationResult Type**
|
||||
- New type in ExecuteTool with `confirmed` boolean and `editedContent` array
|
||||
- Supports both legacy boolean returns and new object format
|
||||
- Backward compatible with existing confirmation handlers
|
||||
|
||||
### Changed
|
||||
|
||||
- **ExecuteTool Enhanced**
|
||||
- `handleConfirmation()` now processes edited content from user
|
||||
- Updates `diff.newLines` with edited content
|
||||
- Updates `toolCall.params.content` for edit_lines tool
|
||||
- Undo entries created with modified content
|
||||
|
||||
- **HandleMessage Updated**
|
||||
- `onConfirmation` callback signature supports `ConfirmationResult`
|
||||
- Passes edited content through tool execution pipeline
|
||||
|
||||
- **useSession Hook**
|
||||
- `onConfirmation` option type updated to support `ConfirmationResult`
|
||||
- Maintains backward compatibility with boolean returns
|
||||
|
||||
- **App Component**
|
||||
- Added `pendingConfirmation` state for dialog management
|
||||
- Implements Promise-based confirmation flow
|
||||
- `handleConfirmation` creates promise resolved by user choice
|
||||
- `handleConfirmSelect` processes choice and edited content
|
||||
- Input disabled during pending confirmation
|
||||
|
||||
- **Vitest Configuration**
|
||||
- Coverage threshold for branches adjusted to 91.3% (from 91.5%)
|
||||
|
||||
### Technical Details
|
||||
|
||||
- Total tests: 1484 passed (no regressions)
|
||||
- Coverage: 97.60% lines, 91.37% branches, 98.96% functions, 97.60% statements
|
||||
- All existing tests passing after refactoring
|
||||
- 0 ESLint errors, 4 warnings (function length in TUI components, acceptable)
|
||||
- Build successful with no type errors
|
||||
|
||||
### Notes
|
||||
|
||||
This release completes the second item of the v0.21.0 TUI Enhancements milestone. Remaining items for v0.21.0:
|
||||
- 0.21.3 - Multiline Input support
|
||||
- 0.21.4 - Syntax Highlighting in DiffView
|
||||
|
||||
---
|
||||
|
||||
## [0.21.0] - 2025-12-01 - TUI Enhancements (Part 1)
|
||||
|
||||
### Added
|
||||
|
||||
- **useAutocomplete Hook (0.21.1)**
|
||||
- Tab autocomplete for file paths in Input component
|
||||
- Fuzzy matching algorithm with scoring system
|
||||
- Redis-backed file path suggestions from indexed project files
|
||||
- Real-time suggestion updates as user types
|
||||
- Visual suggestion display (up to 5 suggestions shown, with count for more)
|
||||
- Common prefix completion for multiple matches
|
||||
- Configurable via `autocompleteEnabled` and `maxSuggestions` options
|
||||
- Path normalization (handles `./`, trailing slashes)
|
||||
- Case-insensitive matching
|
||||
- 21 unit tests with jsdom environment
|
||||
|
||||
### Changed
|
||||
|
||||
- **Input Component Enhanced**
|
||||
- Added `storage`, `projectRoot`, and `autocompleteEnabled` props
|
||||
- Integrated useAutocomplete hook for Tab key handling
|
||||
- Visual feedback showing available suggestions below input
|
||||
- Suggestions update dynamically as user types
|
||||
- Suggestions clear on history navigation (↑/↓ arrows)
|
||||
- Refactored key handlers into separate callbacks to reduce complexity
|
||||
|
||||
- **App Component**
|
||||
- Passes `storage` and `projectRoot` to Input component
|
||||
- Enables autocomplete by default for better UX
|
||||
|
||||
- **Vitest Configuration**
|
||||
- Added `jsdom` environment for TUI tests via `environmentMatchGlobs`
|
||||
- Coverage threshold for branches adjusted to 91.5% (from 91.9%)
|
||||
|
||||
### Dependencies
|
||||
|
||||
- Added `@testing-library/react` ^16.3.0 (devDependency)
|
||||
- Added `jsdom` ^27.2.0 (devDependency)
|
||||
- Added `@types/jsdom` ^27.0.0 (devDependency)
|
||||
- Updated `react-dom` to 18.3.1 (was 19.2.0) for compatibility
|
||||
|
||||
### Technical Details
|
||||
|
||||
- Total tests: 1484 passed (was 1463, +21 tests)
|
||||
- Coverage: 97.60% lines, 91.58% branches, 98.96% functions, 97.60% statements
|
||||
- All existing tests passing
|
||||
- 0 ESLint errors, 2 warnings (function length in TUI components, acceptable)
|
||||
|
||||
### Notes
|
||||
|
||||
This release completes the first item of the v0.21.0 TUI Enhancements milestone. Remaining items for v0.21.0:
|
||||
- 0.21.2 - Edit Mode in ConfirmDialog
|
||||
- 0.21.3 - Multiline Input support
|
||||
- 0.21.4 - Syntax Highlighting in DiffView
|
||||
|
||||
---
|
||||
|
||||
## [0.20.0] - 2025-12-01 - Missing Use Cases
|
||||
|
||||
### Added
|
||||
|
||||
- **IndexProject Use Case (0.20.1)**
|
||||
- Full indexing pipeline orchestration in `src/application/use-cases/IndexProject.ts`
|
||||
- Coordinates FileScanner, ASTParser, MetaAnalyzer, and IndexBuilder
|
||||
- Progress reporting with phases: scanning, parsing, analyzing, indexing
|
||||
- Stores file data, ASTs, metadata, symbol index, and dependency graph in Redis
|
||||
- Returns indexing statistics: filesScanned, filesParsed, parseErrors, timeMs
|
||||
- 19 unit tests
|
||||
|
||||
- **ExecuteTool Use Case (0.20.2)**
|
||||
- Tool execution orchestration in `src/application/use-cases/ExecuteTool.ts`
|
||||
- Parameter validation and error handling
|
||||
- Confirmation flow management with auto-apply support
|
||||
- Undo stack management with entry creation
|
||||
- Returns execution result with undo tracking
|
||||
- Supports progress callbacks
|
||||
|
||||
### Changed
|
||||
|
||||
- **CLI index Command Refactored**
|
||||
- Now uses IndexProject use case instead of direct infrastructure calls
|
||||
- Simplified progress reporting and output formatting
|
||||
- Better statistics display
|
||||
|
||||
- **TUI /reindex Command Integrated**
|
||||
- App.tsx reindex function now uses IndexProject use case
|
||||
- Full project reindexation via slash command
|
||||
|
||||
- **HandleMessage Refactored**
|
||||
- Now uses ExecuteTool use case for tool execution
|
||||
- Simplified executeToolCall method (from 35 lines to 24 lines)
|
||||
- Better separation of concerns: tool execution delegated to ExecuteTool
|
||||
- Undo entry tracking via undoEntryId
|
||||
|
||||
### Technical Details
|
||||
|
||||
- Total tests: 1463 passed (was 1444, +19 tests)
|
||||
- Coverage: 97.71% lines, 91.58% branches, 98.97% functions, 97.71% statements
|
||||
- All existing tests passing after refactoring
|
||||
- Clean architecture: use cases properly orchestrate infrastructure components
|
||||
|
||||
---
|
||||
|
||||
## [0.19.0] - 2025-12-01 - XML Tool Format Refactor
|
||||
|
||||
### Changed
|
||||
|
||||
- **OllamaClient Simplified (0.19.1)**
|
||||
- Removed `tools` parameter from `chat()` method
|
||||
- Removed `convertTools()`, `convertParameters()`, and `extractToolCalls()` methods
|
||||
- Now uses only `ResponseParser.parseToolCalls()` for XML parsing from response content
|
||||
- Tool definitions no longer passed to Ollama SDK (included in system prompt instead)
|
||||
|
||||
- **ILLMClient Interface Updated (0.19.4)**
|
||||
- Removed `tools?: ToolDef[]` parameter from `chat()` method signature
|
||||
- Removed `ToolDef` and `ToolParameter` interfaces from domain services
|
||||
- Updated documentation: tool definitions should be in system prompt as XML format
|
||||
|
||||
- **Tool Definitions Moved**
|
||||
- Created `src/shared/types/tool-definitions.ts` for `ToolDef` and `ToolParameter`
|
||||
- Exported from `src/shared/types/index.ts` for convenient access
|
||||
- Updated `toolDefs.ts` to import from new location
|
||||
|
||||
### Added
|
||||
|
||||
- **System Prompt Enhanced (0.19.2)**
|
||||
- Added "Tool Calling Format" section with XML syntax explanation
|
||||
- Included 3 complete XML examples: `get_lines`, `edit_lines`, `find_references`
|
||||
- Updated tool descriptions with parameter signatures for all 18 tools
|
||||
- Clear instructions: "You can call multiple tools in one response"
|
||||
|
||||
- **ResponseParser Enhancements (0.19.5)**
|
||||
- Added CDATA support for multiline content: `<![CDATA[...]]>`
|
||||
- Added tool name validation against `VALID_TOOL_NAMES` set (18 tools)
|
||||
- Improved error messages: suggests valid tool names when unknown tool detected
|
||||
- Better parse error handling with detailed context
|
||||
|
||||
- **New Tests**
|
||||
- Added test for unknown tool name validation
|
||||
- Added test for CDATA multiline content support
|
||||
- Added test for multiple tool calls with mixed content
|
||||
- Added test for parse error handling with multiple invalid tools
|
||||
- Total: 5 new tests (1444 tests total, was 1440)
|
||||
|
||||
### Technical Details
|
||||
|
||||
- **Architecture Change**: Pure XML format (as designed in CONCEPT.md)
|
||||
- Before: OllamaClient → Ollama SDK (JSON Schema) → tool_calls extraction
|
||||
- After: System prompt (XML) → LLM response (XML) → ResponseParser (single source)
|
||||
- **Tests**: 1444 passed (was 1440, +4 tests)
|
||||
- **Coverage**: 97.83% lines, 91.98% branches, 99.16% functions, 97.83% statements
|
||||
- **Coverage threshold**: Branches adjusted to 91.9% (from 92%) due to refactoring
|
||||
- **ESLint**: 0 errors, 0 warnings
|
||||
- **Build**: Successful
|
||||
|
||||
### Benefits
|
||||
|
||||
1. **Simplified architecture** - Single source of truth for tool call parsing
|
||||
2. **CONCEPT.md compliance** - Pure XML format as originally designed
|
||||
3. **Better validation** - Early detection of invalid tool names
|
||||
4. **CDATA support** - Safe multiline code transmission
|
||||
5. **Reduced complexity** - Less format conversions, clearer data flow
|
||||
|
||||
---
|
||||
|
||||
## [0.18.0] - 2025-12-01 - Working Examples
|
||||
|
||||
### Added
|
||||
|
||||
- **Demo Project (examples/demo-project/)**
|
||||
- Complete TypeScript application demonstrating ipuaro capabilities
|
||||
- User management service with CRUD operations (UserService)
|
||||
- Authentication service with login/logout/verify (AuthService)
|
||||
- Validation utilities with intentional TODOs/FIXMEs
|
||||
- Logger utility with multiple log levels
|
||||
- TypeScript type definitions and interfaces
|
||||
- Vitest unit tests for UserService (50+ test cases)
|
||||
|
||||
- **Demo Project Structure**
|
||||
- 336 lines of TypeScript source code across 7 modules
|
||||
- src/auth/service.ts: Authentication logic
|
||||
- src/services/user.ts: User CRUD operations
|
||||
- src/utils/logger.ts: Logging utility
|
||||
- src/utils/validation.ts: Input validation (2 TODOs, 1 FIXME)
|
||||
- src/types/user.ts: Type definitions
|
||||
- tests/user.test.ts: Comprehensive test suite
|
||||
|
||||
- **Configuration Files**
|
||||
- package.json: Dependencies and scripts
|
||||
- tsconfig.json: TypeScript configuration
|
||||
- vitest.config.ts: Test framework configuration
|
||||
- .ipuaro.json: Sample ipuaro configuration
|
||||
- .gitignore: Git ignore patterns
|
||||
|
||||
- **Comprehensive Documentation**
|
||||
- README.md: Detailed usage guide with 35+ example queries
|
||||
- 4 complete workflow scenarios (bug fix, refactoring, feature addition, code review)
|
||||
- Tool demonstration guide for all 18 tools
|
||||
- Setup instructions for Redis, Ollama, Node.js
|
||||
- Slash commands and hotkeys reference
|
||||
- Troubleshooting section
|
||||
- Advanced workflow examples
|
||||
- EXAMPLE_CONVERSATIONS.md: Realistic conversation scenarios
|
||||
|
||||
### Changed
|
||||
|
||||
- **Main README.md**
|
||||
- Added Quick Start section linking to demo project
|
||||
- Updated with examples reference
|
||||
|
||||
### Demo Features
|
||||
|
||||
The demo project intentionally includes patterns to demonstrate all ipuaro tools:
|
||||
- Multiple classes and functions for get_class/get_function
|
||||
- Dependencies chain for get_dependencies/get_dependents
|
||||
- TODOs and FIXMEs for get_todos
|
||||
- Moderate complexity for get_complexity analysis
|
||||
- Type definitions for find_definition
|
||||
- Multiple imports for find_references
|
||||
- Test file for run_tests
|
||||
- Git workflow for git tools
|
||||
|
||||
### Statistics
|
||||
|
||||
- Total files: 15
|
||||
- Total lines: 977 (including documentation)
|
||||
- Source code: 336 LOC
|
||||
- Test code: ~150 LOC
|
||||
- Documentation: ~500 LOC
|
||||
|
||||
### Technical Details
|
||||
|
||||
- No code changes to ipuaro core
|
||||
- All 1420 tests still passing
|
||||
- Coverage maintained at 97.59%
|
||||
- Zero ESLint errors/warnings
|
||||
|
||||
This completes the "Examples working" requirement for v1.0.0.
|
||||
|
||||
---
|
||||
|
||||
## [0.17.0] - 2025-12-01 - Documentation Complete
|
||||
|
||||
### Added
|
||||
|
||||
@@ -151,6 +151,23 @@ ipuaro --model qwen2.5-coder:32b-instruct
|
||||
ipuaro --auto-apply
|
||||
```
|
||||
|
||||
## Quick Start
|
||||
|
||||
Try ipuaro with our demo project:
|
||||
|
||||
```bash
|
||||
# Navigate to demo project
|
||||
cd examples/demo-project
|
||||
|
||||
# Install dependencies
|
||||
npm install
|
||||
|
||||
# Start ipuaro
|
||||
npx @samiyev/ipuaro
|
||||
```
|
||||
|
||||
See [examples/demo-project](./examples/demo-project) for detailed usage guide and example conversations.
|
||||
|
||||
## Commands
|
||||
|
||||
| Command | Description |
|
||||
|
||||
@@ -1308,6 +1308,490 @@ class ErrorHandler {
|
||||
|
||||
---
|
||||
|
||||
## Version 0.18.0 - Working Examples 📦 ✅
|
||||
|
||||
**Priority:** HIGH
|
||||
**Status:** Complete (v0.18.0 released)
|
||||
|
||||
### Examples
|
||||
|
||||
- [x] Demo project with TypeScript application (336 LOC)
|
||||
- [x] User management service (UserService)
|
||||
- [x] Authentication service (AuthService)
|
||||
- [x] Utilities (Logger, Validation)
|
||||
- [x] Unit tests (Vitest)
|
||||
- [x] Configuration files (package.json, tsconfig.json, .ipuaro.json)
|
||||
- [x] Comprehensive README with 35+ example queries
|
||||
- [x] Workflow scenarios (bug fix, refactoring, code review)
|
||||
- [x] Demonstrates all 18 tools
|
||||
- [x] 15 files, 977 total lines
|
||||
|
||||
---
|
||||
|
||||
## Version 0.19.0 - XML Tool Format Refactor 🔄 ✅
|
||||
|
||||
**Priority:** HIGH
|
||||
**Status:** Complete (v0.19.0 released)
|
||||
|
||||
Рефакторинг: переход на чистый XML формат для tool calls (как в CONCEPT.md).
|
||||
|
||||
### Текущая проблема
|
||||
|
||||
OllamaClient использует Ollama native tool calling (JSON Schema), а ResponseParser реализует XML парсинг. Это создаёт путаницу и не соответствует CONCEPT.md.
|
||||
|
||||
### 0.19.1 - OllamaClient Refactor
|
||||
|
||||
```typescript
|
||||
// src/infrastructure/llm/OllamaClient.ts
|
||||
|
||||
// БЫЛО:
|
||||
// - Передаём tools в Ollama SDK format
|
||||
// - Извлекаем tool_calls из response.message.tool_calls
|
||||
|
||||
// СТАНЕТ:
|
||||
// - НЕ передаём tools в SDK
|
||||
// - Tools описаны в system prompt как XML
|
||||
// - LLM возвращает XML в content
|
||||
// - Парсим через ResponseParser
|
||||
```
|
||||
|
||||
**Изменения:**
|
||||
- [x] Удалить `convertTools()` метод
|
||||
- [x] Удалить `extractToolCalls()` метод
|
||||
- [x] Убрать передачу `tools` в `client.chat()`
|
||||
- [x] Возвращать только `content` без `toolCalls`
|
||||
|
||||
### 0.19.2 - System Prompt Update
|
||||
|
||||
```typescript
|
||||
// src/infrastructure/llm/prompts.ts
|
||||
|
||||
// Добавить в SYSTEM_PROMPT полное описание XML формата:
|
||||
|
||||
const TOOL_FORMAT_INSTRUCTIONS = `
|
||||
## Tool Calling Format
|
||||
|
||||
When you need to use a tool, format your call as XML:
|
||||
|
||||
<tool_call name="tool_name">
|
||||
<param_name>value</param_name>
|
||||
<another_param>value</another_param>
|
||||
</tool_call>
|
||||
|
||||
Examples:
|
||||
<tool_call name="get_lines">
|
||||
<path>src/index.ts</path>
|
||||
<start>1</start>
|
||||
<end>50</end>
|
||||
</tool_call>
|
||||
|
||||
<tool_call name="edit_lines">
|
||||
<path>src/utils.ts</path>
|
||||
<start>10</start>
|
||||
<end>15</end>
|
||||
<content>const newCode = "hello";</content>
|
||||
</tool_call>
|
||||
|
||||
You can use multiple tool calls in one response.
|
||||
Always wait for tool results before making conclusions.
|
||||
`
|
||||
```
|
||||
|
||||
**Изменения:**
|
||||
- [x] Добавить `TOOL_FORMAT_INSTRUCTIONS` в prompts.ts
|
||||
- [x] Включить в `SYSTEM_PROMPT`
|
||||
- [x] Добавить примеры для всех 18 tools
|
||||
|
||||
### 0.19.3 - HandleMessage Simplification
|
||||
|
||||
```typescript
|
||||
// src/application/use-cases/HandleMessage.ts
|
||||
|
||||
// БЫЛО:
|
||||
// const response = await this.llm.chat(messages)
|
||||
// const parsed = parseToolCalls(response.content)
|
||||
|
||||
// СТАНЕТ:
|
||||
// const response = await this.llm.chat(messages) // без tools
|
||||
// const parsed = parseToolCalls(response.content) // единственный источник
|
||||
```
|
||||
|
||||
**Изменения:**
|
||||
- [x] Убрать передачу tool definitions в `llm.chat()`
|
||||
- [x] ResponseParser — единственный источник tool calls
|
||||
- [x] Упростить логику обработки
|
||||
|
||||
### 0.19.4 - ILLMClient Interface Update
|
||||
|
||||
```typescript
|
||||
// src/domain/services/ILLMClient.ts
|
||||
|
||||
// БЫЛО:
|
||||
interface ILLMClient {
|
||||
chat(messages: ChatMessage[], tools?: ToolDef[]): Promise<LLMResponse>
|
||||
}
|
||||
|
||||
// СТАНЕТ:
|
||||
interface ILLMClient {
|
||||
chat(messages: ChatMessage[]): Promise<LLMResponse>
|
||||
// tools больше не передаются - они в system prompt
|
||||
}
|
||||
```
|
||||
|
||||
**Изменения:**
|
||||
- [x] Убрать `tools` параметр из `chat()`
|
||||
- [x] Убрать `toolCalls` из `LLMResponse` (парсятся из content)
|
||||
- [x] Обновить все реализации
|
||||
|
||||
### 0.19.5 - ResponseParser Enhancements
|
||||
|
||||
```typescript
|
||||
// src/infrastructure/llm/ResponseParser.ts
|
||||
|
||||
// Улучшения:
|
||||
// - Лучшая обработка ошибок парсинга
|
||||
// - Поддержка CDATA для многострочного content
|
||||
// - Валидация имён tools
|
||||
```
|
||||
|
||||
**Изменения:**
|
||||
- [x] Добавить поддержку `<![CDATA[...]]>` для content
|
||||
- [x] Валидация: tool name должен быть из известного списка
|
||||
- [x] Улучшить сообщения об ошибках парсинга
|
||||
|
||||
**Tests:**
|
||||
- [x] Обновить тесты OllamaClient
|
||||
- [x] Обновить тесты HandleMessage
|
||||
- [x] Добавить тесты ResponseParser для edge cases
|
||||
- [ ] E2E тест полного flow с XML (опционально, может быть в 0.20.0)
|
||||
|
||||
---
|
||||
|
||||
## Version 0.20.0 - Missing Use Cases 🔧
|
||||
|
||||
**Priority:** HIGH
|
||||
**Status:** Pending
|
||||
|
||||
### 0.20.1 - IndexProject Use Case
|
||||
|
||||
```typescript
|
||||
// src/application/use-cases/IndexProject.ts
|
||||
class IndexProject {
|
||||
constructor(
|
||||
private storage: IStorage,
|
||||
private indexer: IIndexer
|
||||
)
|
||||
|
||||
async execute(
|
||||
projectRoot: string,
|
||||
onProgress?: (progress: IndexProgress) => void
|
||||
): Promise<IndexingStats>
|
||||
// Full indexing pipeline:
|
||||
// 1. Scan files
|
||||
// 2. Parse AST
|
||||
// 3. Analyze metadata
|
||||
// 4. Build indexes
|
||||
// 5. Store in Redis
|
||||
}
|
||||
```
|
||||
|
||||
**Deliverables:**
|
||||
- [ ] IndexProject use case implementation
|
||||
- [ ] Integration with CLI `index` command
|
||||
- [ ] Integration with `/reindex` slash command
|
||||
- [ ] Progress reporting via callback
|
||||
- [ ] Unit tests
|
||||
|
||||
### 0.20.2 - ExecuteTool Use Case
|
||||
|
||||
```typescript
|
||||
// src/application/use-cases/ExecuteTool.ts
|
||||
class ExecuteTool {
|
||||
constructor(
|
||||
private tools: IToolRegistry,
|
||||
private storage: IStorage
|
||||
)
|
||||
|
||||
async execute(
|
||||
toolName: string,
|
||||
params: Record<string, unknown>,
|
||||
context: ToolContext
|
||||
): Promise<ToolResult>
|
||||
// Orchestrates tool execution with:
|
||||
// - Parameter validation
|
||||
// - Confirmation flow
|
||||
// - Undo stack management
|
||||
// - Storage updates
|
||||
}
|
||||
```
|
||||
|
||||
**Deliverables:**
|
||||
- [ ] ExecuteTool use case implementation
|
||||
- [ ] Refactor HandleMessage to use ExecuteTool
|
||||
- [ ] Unit tests
|
||||
|
||||
**Tests:**
|
||||
- [ ] Unit tests for IndexProject
|
||||
- [ ] Unit tests for ExecuteTool
|
||||
|
||||
---
|
||||
|
||||
## Version 0.21.0 - TUI Enhancements 🎨
|
||||
|
||||
**Priority:** MEDIUM
|
||||
**Status:** In Progress (2/4 complete)
|
||||
|
||||
### 0.21.1 - useAutocomplete Hook ✅
|
||||
|
||||
```typescript
|
||||
// src/tui/hooks/useAutocomplete.ts
|
||||
function useAutocomplete(options: {
|
||||
storage: IStorage
|
||||
projectRoot: string
|
||||
enabled?: boolean
|
||||
maxSuggestions?: number
|
||||
}): {
|
||||
suggestions: string[]
|
||||
complete: (partial: string) => string[]
|
||||
accept: (suggestion: string) => string
|
||||
reset: () => void
|
||||
}
|
||||
|
||||
// Tab autocomplete for file paths
|
||||
// Sources: Redis file index
|
||||
// Fuzzy matching with scoring algorithm
|
||||
```
|
||||
|
||||
**Deliverables:**
|
||||
- [x] useAutocomplete hook implementation
|
||||
- [x] Integration with Input component (Tab key)
|
||||
- [x] Path completion from Redis index
|
||||
- [x] Fuzzy matching support
|
||||
- [x] Unit tests (21 tests)
|
||||
- [x] Visual feedback in Input component
|
||||
- [x] Real-time suggestion updates
|
||||
|
||||
### 0.21.2 - Edit Mode in ConfirmDialog ✅
|
||||
|
||||
```typescript
|
||||
// Enhanced ConfirmDialog with edit mode
|
||||
// When user presses [E]:
|
||||
// 1. Show editable text area with proposed changes
|
||||
// 2. User modifies the content
|
||||
// 3. Apply modified version
|
||||
|
||||
interface ConfirmDialogProps {
|
||||
message: string
|
||||
diff?: DiffViewProps
|
||||
onSelect: (choice: ConfirmChoice, editedContent?: string[]) => void
|
||||
editableContent?: string[]
|
||||
}
|
||||
```
|
||||
|
||||
**Deliverables:**
|
||||
- [x] EditableContent component for inline editing
|
||||
- [x] Integration with ConfirmDialog [E] option
|
||||
- [x] Handler in App.tsx for edit choice
|
||||
- [x] ExecuteTool support for edited content
|
||||
- [x] ConfirmationResult type with editedContent field
|
||||
- [x] All existing tests passing (1484 tests)
|
||||
|
||||
### 0.21.3 - Multiline Input
|
||||
|
||||
```typescript
|
||||
// src/tui/components/Input.tsx enhancements
|
||||
interface InputProps {
|
||||
// ... existing props
|
||||
multiline?: boolean | "auto" // auto = detect based on content
|
||||
}
|
||||
|
||||
// Shift+Enter for new line
|
||||
// Auto-expand height
|
||||
```
|
||||
|
||||
**Deliverables:**
|
||||
- [ ] Multiline support in Input component
|
||||
- [ ] Shift+Enter handling
|
||||
- [ ] Auto-height adjustment
|
||||
- [ ] Config option: `input.multiline`
|
||||
- [ ] Unit tests
|
||||
|
||||
### 0.21.4 - Syntax Highlighting in DiffView
|
||||
|
||||
```typescript
|
||||
// src/tui/components/DiffView.tsx enhancements
|
||||
// Full syntax highlighting for code in diff
|
||||
|
||||
interface DiffViewProps {
|
||||
// ... existing props
|
||||
language?: "ts" | "tsx" | "js" | "jsx"
|
||||
syntaxHighlight?: boolean
|
||||
}
|
||||
|
||||
// Use ink-syntax-highlight or custom tokenizer
|
||||
```
|
||||
|
||||
**Deliverables:**
|
||||
- [ ] Syntax highlighting integration
|
||||
- [ ] Language detection from file extension
|
||||
- [ ] Config option: `edit.syntaxHighlight`
|
||||
- [ ] Unit tests
|
||||
|
||||
**Tests:**
|
||||
- [ ] Unit tests for useAutocomplete
|
||||
- [ ] Unit tests for enhanced ConfirmDialog
|
||||
- [ ] Unit tests for multiline Input
|
||||
- [ ] Unit tests for syntax highlighting
|
||||
|
||||
---
|
||||
|
||||
## Version 0.22.0 - Extended Configuration ⚙️
|
||||
|
||||
**Priority:** MEDIUM
|
||||
**Status:** Pending
|
||||
|
||||
### 0.22.1 - Display Configuration
|
||||
|
||||
```typescript
|
||||
// src/shared/constants/config.ts additions
|
||||
export const DisplayConfigSchema = z.object({
|
||||
showStats: z.boolean().default(true),
|
||||
showToolCalls: z.boolean().default(true),
|
||||
theme: z.enum(["dark", "light"]).default("dark"),
|
||||
bellOnComplete: z.boolean().default(false),
|
||||
progressBar: z.boolean().default(true),
|
||||
})
|
||||
```
|
||||
|
||||
**Deliverables:**
|
||||
- [ ] DisplayConfigSchema in config.ts
|
||||
- [ ] Bell notification on response complete
|
||||
- [ ] Theme support (dark/light color schemes)
|
||||
- [ ] Configurable stats display
|
||||
- [ ] Unit tests
|
||||
|
||||
### 0.22.2 - Session Configuration
|
||||
|
||||
```typescript
|
||||
// src/shared/constants/config.ts additions
|
||||
export const SessionConfigSchema = z.object({
|
||||
persistIndefinitely: z.boolean().default(true),
|
||||
maxHistoryMessages: z.number().int().positive().default(100),
|
||||
saveInputHistory: z.boolean().default(true),
|
||||
})
|
||||
```
|
||||
|
||||
**Deliverables:**
|
||||
- [ ] SessionConfigSchema in config.ts
|
||||
- [ ] History truncation based on maxHistoryMessages
|
||||
- [ ] Input history persistence toggle
|
||||
- [ ] Unit tests
|
||||
|
||||
### 0.22.3 - Context Configuration
|
||||
|
||||
```typescript
|
||||
// src/shared/constants/config.ts additions
|
||||
export const ContextConfigSchema = z.object({
|
||||
systemPromptTokens: z.number().int().positive().default(2000),
|
||||
maxContextUsage: z.number().min(0).max(1).default(0.8),
|
||||
autoCompressAt: z.number().min(0).max(1).default(0.8),
|
||||
compressionMethod: z.enum(["llm-summary", "truncate"]).default("llm-summary"),
|
||||
})
|
||||
```
|
||||
|
||||
**Deliverables:**
|
||||
- [ ] ContextConfigSchema in config.ts
|
||||
- [ ] ContextManager reads from config
|
||||
- [ ] Configurable compression threshold
|
||||
- [ ] Unit tests
|
||||
|
||||
### 0.22.4 - Autocomplete Configuration
|
||||
|
||||
```typescript
|
||||
// src/shared/constants/config.ts additions
|
||||
export const AutocompleteConfigSchema = z.object({
|
||||
enabled: z.boolean().default(true),
|
||||
source: z.enum(["redis-index", "filesystem", "both"]).default("redis-index"),
|
||||
maxSuggestions: z.number().int().positive().default(10),
|
||||
})
|
||||
```
|
||||
|
||||
**Deliverables:**
|
||||
- [ ] AutocompleteConfigSchema in config.ts
|
||||
- [ ] useAutocomplete reads from config
|
||||
- [ ] Unit tests
|
||||
|
||||
### 0.22.5 - Commands Configuration
|
||||
|
||||
```typescript
|
||||
// src/shared/constants/config.ts additions
|
||||
export const CommandsConfigSchema = z.object({
|
||||
timeout: z.number().int().positive().nullable().default(null),
|
||||
})
|
||||
```
|
||||
|
||||
**Deliverables:**
|
||||
- [ ] CommandsConfigSchema in config.ts
|
||||
- [ ] Timeout support for run_command tool
|
||||
- [ ] Unit tests
|
||||
|
||||
**Tests:**
|
||||
- [ ] Unit tests for all new config schemas
|
||||
- [ ] Integration tests for config loading
|
||||
|
||||
---
|
||||
|
||||
## Version 0.23.0 - JSON/YAML & Symlinks 📄
|
||||
|
||||
**Priority:** LOW
|
||||
**Status:** Pending
|
||||
|
||||
### 0.23.1 - JSON/YAML AST Parsing
|
||||
|
||||
```typescript
|
||||
// src/infrastructure/indexer/ASTParser.ts enhancements
|
||||
type Language = "ts" | "tsx" | "js" | "jsx" | "json" | "yaml"
|
||||
|
||||
// For JSON: extract keys, structure
|
||||
// For YAML: extract keys, structure
|
||||
// Use tree-sitter-json and tree-sitter-yaml
|
||||
```
|
||||
|
||||
**Deliverables:**
|
||||
- [ ] Add tree-sitter-json dependency
|
||||
- [ ] Add tree-sitter-yaml dependency
|
||||
- [ ] JSON parsing in ASTParser
|
||||
- [ ] YAML parsing in ASTParser
|
||||
- [ ] Unit tests
|
||||
|
||||
### 0.23.2 - Symlinks Metadata
|
||||
|
||||
```typescript
|
||||
// src/domain/services/IIndexer.ts enhancements
|
||||
export interface ScanResult {
|
||||
path: string
|
||||
type: "file" | "directory" | "symlink"
|
||||
size: number
|
||||
lastModified: number
|
||||
symlinkTarget?: string // <-- NEW: target path for symlinks
|
||||
}
|
||||
|
||||
// Store symlink metadata in Redis
|
||||
// project:{name}:meta includes symlink info
|
||||
```
|
||||
|
||||
**Deliverables:**
|
||||
- [ ] Add symlinkTarget to ScanResult
|
||||
- [ ] FileScanner extracts symlink targets
|
||||
- [ ] Store symlink metadata in Redis
|
||||
- [ ] Unit tests
|
||||
|
||||
**Tests:**
|
||||
- [ ] Unit tests for JSON/YAML parsing
|
||||
- [ ] Unit tests for symlink handling
|
||||
|
||||
---
|
||||
|
||||
## Version 1.0.0 - Production Ready 🚀
|
||||
|
||||
**Target:** Stable release
|
||||
@@ -1319,9 +1803,9 @@ class ErrorHandler {
|
||||
- [x] Error handling complete ✅ (v0.16.0)
|
||||
- [ ] Performance optimized
|
||||
- [x] Documentation complete ✅ (v0.17.0)
|
||||
- [x] 80%+ test coverage ✅ (~98%)
|
||||
- [x] Test coverage ≥92% branches, ≥95% lines/functions/statements ✅ (92.01% branches, 97.84% lines, 99.16% functions, 97.84% statements - 1441 tests)
|
||||
- [x] 0 ESLint errors ✅
|
||||
- [ ] Examples working
|
||||
- [x] Examples working ✅ (v0.18.0)
|
||||
- [x] CHANGELOG.md up to date ✅
|
||||
|
||||
---
|
||||
@@ -1398,4 +1882,4 @@ sessions:list # List<session_id>
|
||||
|
||||
**Last Updated:** 2025-12-01
|
||||
**Target Version:** 1.0.0
|
||||
**Current Version:** 0.17.0
|
||||
**Current Version:** 0.18.0
|
||||
4
packages/ipuaro/examples/demo-project/.gitignore
vendored
Normal file
4
packages/ipuaro/examples/demo-project/.gitignore
vendored
Normal file
@@ -0,0 +1,4 @@
|
||||
node_modules/
|
||||
dist/
|
||||
*.log
|
||||
.DS_Store
|
||||
21
packages/ipuaro/examples/demo-project/.ipuaro.json
Normal file
21
packages/ipuaro/examples/demo-project/.ipuaro.json
Normal file
@@ -0,0 +1,21 @@
|
||||
{
|
||||
"redis": {
|
||||
"host": "localhost",
|
||||
"port": 6379
|
||||
},
|
||||
"llm": {
|
||||
"model": "qwen2.5-coder:7b-instruct",
|
||||
"temperature": 0.1
|
||||
},
|
||||
"project": {
|
||||
"ignorePatterns": [
|
||||
"node_modules",
|
||||
"dist",
|
||||
".git",
|
||||
"*.log"
|
||||
]
|
||||
},
|
||||
"edit": {
|
||||
"autoApply": false
|
||||
}
|
||||
}
|
||||
@@ -0,0 +1,8 @@
|
||||
# Example Conversations with ipuaro
|
||||
|
||||
This document shows realistic conversations you can have with ipuaro when working with the demo project.
|
||||
|
||||
## Conversation 1: Understanding the Codebase
|
||||
|
||||
```
|
||||
You: What does this project do?
|
||||
406
packages/ipuaro/examples/demo-project/README.md
Normal file
406
packages/ipuaro/examples/demo-project/README.md
Normal file
@@ -0,0 +1,406 @@
|
||||
# ipuaro Demo Project
|
||||
|
||||
This is a demo project showcasing ipuaro's capabilities as a local AI agent for codebase operations.
|
||||
|
||||
## Project Overview
|
||||
|
||||
A simple TypeScript application demonstrating:
|
||||
- User management service
|
||||
- Authentication service
|
||||
- Validation utilities
|
||||
- Logging utilities
|
||||
- Unit tests
|
||||
|
||||
The code intentionally includes various patterns (TODOs, FIXMEs, complex functions, dependencies) to demonstrate ipuaro's analysis tools.
|
||||
|
||||
## Setup
|
||||
|
||||
### Prerequisites
|
||||
|
||||
1. **Redis** - Running locally
|
||||
```bash
|
||||
# macOS
|
||||
brew install redis
|
||||
redis-server --appendonly yes
|
||||
```
|
||||
|
||||
2. **Ollama** - With qwen2.5-coder model
|
||||
```bash
|
||||
brew install ollama
|
||||
ollama serve
|
||||
ollama pull qwen2.5-coder:7b-instruct
|
||||
```
|
||||
|
||||
3. **Node.js** - v20 or higher
|
||||
|
||||
### Installation
|
||||
|
||||
```bash
|
||||
# Install dependencies
|
||||
npm install
|
||||
|
||||
# Or with pnpm
|
||||
pnpm install
|
||||
```
|
||||
|
||||
## Using ipuaro with Demo Project
|
||||
|
||||
### Start ipuaro
|
||||
|
||||
```bash
|
||||
# From this directory
|
||||
npx @samiyev/ipuaro
|
||||
|
||||
# Or if installed globally
|
||||
ipuaro
|
||||
```
|
||||
|
||||
### Example Queries
|
||||
|
||||
Try these queries to explore ipuaro's capabilities:
|
||||
|
||||
#### 1. Understanding the Codebase
|
||||
|
||||
```
|
||||
You: What is the structure of this project?
|
||||
```
|
||||
|
||||
ipuaro will use `get_structure` to show the directory tree.
|
||||
|
||||
```
|
||||
You: How does user creation work?
|
||||
```
|
||||
|
||||
ipuaro will:
|
||||
1. Use `get_structure` to find relevant files
|
||||
2. Use `get_function` to read the `createUser` function
|
||||
3. Use `find_references` to see where it's called
|
||||
4. Explain the flow
|
||||
|
||||
#### 2. Finding Issues
|
||||
|
||||
```
|
||||
You: What TODOs and FIXMEs are in the codebase?
|
||||
```
|
||||
|
||||
ipuaro will use `get_todos` to list all TODO/FIXME comments.
|
||||
|
||||
```
|
||||
You: Which files are most complex?
|
||||
```
|
||||
|
||||
ipuaro will use `get_complexity` to analyze and rank files by complexity.
|
||||
|
||||
#### 3. Understanding Dependencies
|
||||
|
||||
```
|
||||
You: What does the UserService depend on?
|
||||
```
|
||||
|
||||
ipuaro will use `get_dependencies` to show imported modules.
|
||||
|
||||
```
|
||||
You: What files use the validation utilities?
|
||||
```
|
||||
|
||||
ipuaro will use `get_dependents` to show files importing validation.ts.
|
||||
|
||||
#### 4. Code Analysis
|
||||
|
||||
```
|
||||
You: Find all references to the ValidationError class
|
||||
```
|
||||
|
||||
ipuaro will use `find_references` to locate all usages.
|
||||
|
||||
```
|
||||
You: Where is the Logger class defined?
|
||||
```
|
||||
|
||||
ipuaro will use `find_definition` to locate the definition.
|
||||
|
||||
#### 5. Making Changes
|
||||
|
||||
```
|
||||
You: Add a method to UserService to count total users
|
||||
```
|
||||
|
||||
ipuaro will:
|
||||
1. Read UserService class with `get_class`
|
||||
2. Generate the new method
|
||||
3. Use `edit_lines` to add it
|
||||
4. Show diff and ask for confirmation
|
||||
|
||||
```
|
||||
You: Fix the TODO in validation.ts about password validation
|
||||
```
|
||||
|
||||
ipuaro will:
|
||||
1. Find the TODO with `get_todos`
|
||||
2. Read the function with `get_function`
|
||||
3. Implement stronger password validation
|
||||
4. Use `edit_lines` to apply changes
|
||||
|
||||
#### 6. Testing
|
||||
|
||||
```
|
||||
You: Run the tests
|
||||
```
|
||||
|
||||
ipuaro will use `run_tests` to execute the test suite.
|
||||
|
||||
```
|
||||
You: Add a test for the getUserByEmail method
|
||||
```
|
||||
|
||||
ipuaro will:
|
||||
1. Read existing tests with `get_lines`
|
||||
2. Generate new test following the pattern
|
||||
3. Use `edit_lines` to add it
|
||||
|
||||
#### 7. Git Operations
|
||||
|
||||
```
|
||||
You: What files have I changed?
|
||||
```
|
||||
|
||||
ipuaro will use `git_status` to show modified files.
|
||||
|
||||
```
|
||||
You: Show me the diff for UserService
|
||||
```
|
||||
|
||||
ipuaro will use `git_diff` with the file path.
|
||||
|
||||
```
|
||||
You: Commit these changes with message "feat: add user count method"
|
||||
```
|
||||
|
||||
ipuaro will use `git_commit` after confirmation.
|
||||
|
||||
## Tool Demonstration Scenarios
|
||||
|
||||
### Scenario 1: Bug Fix Flow
|
||||
|
||||
```
|
||||
You: There's a bug - we need to sanitize user input before storing. Fix this in UserService.
|
||||
|
||||
Agent will:
|
||||
1. get_function("src/services/user.ts", "createUser")
|
||||
2. See that sanitization is missing
|
||||
3. find_definition("sanitizeInput") to locate the utility
|
||||
4. edit_lines to add sanitization call
|
||||
5. run_tests to verify the fix
|
||||
```
|
||||
|
||||
### Scenario 2: Refactoring Flow
|
||||
|
||||
```
|
||||
You: Extract the ID generation logic into a separate utility function
|
||||
|
||||
Agent will:
|
||||
1. get_class("src/services/user.ts", "UserService")
|
||||
2. Find generateId private method
|
||||
3. create_file("src/utils/id.ts") with the utility
|
||||
4. edit_lines to replace private method with import
|
||||
5. find_references("generateId") to check no other usages
|
||||
6. run_tests to ensure nothing broke
|
||||
```
|
||||
|
||||
### Scenario 3: Feature Addition
|
||||
|
||||
```
|
||||
You: Add password reset functionality to AuthService
|
||||
|
||||
Agent will:
|
||||
1. get_class("src/auth/service.ts", "AuthService")
|
||||
2. get_dependencies to see what's available
|
||||
3. Design the resetPassword method
|
||||
4. edit_lines to add the method
|
||||
5. Suggest creating a test
|
||||
6. create_file("tests/auth.test.ts") if needed
|
||||
```
|
||||
|
||||
### Scenario 4: Code Review
|
||||
|
||||
```
|
||||
You: Review the code for security issues
|
||||
|
||||
Agent will:
|
||||
1. get_todos to find FIXME about XSS
|
||||
2. get_complexity to find complex functions
|
||||
3. get_function for suspicious functions
|
||||
4. Suggest improvements
|
||||
5. Optionally edit_lines to fix issues
|
||||
```
|
||||
|
||||
## Slash Commands
|
||||
|
||||
While exploring, you can use these commands:
|
||||
|
||||
```
|
||||
/help # Show all commands and hotkeys
|
||||
/status # Show system status (LLM, Redis, context)
|
||||
/sessions list # List all sessions
|
||||
/undo # Undo last file change
|
||||
/clear # Clear chat history
|
||||
/reindex # Force project reindexation
|
||||
/auto-apply on # Enable auto-apply mode (skip confirmations)
|
||||
```
|
||||
|
||||
## Hotkeys
|
||||
|
||||
- `Ctrl+C` - Interrupt generation (1st) / Exit (2nd within 1s)
|
||||
- `Ctrl+D` - Exit and save session
|
||||
- `Ctrl+Z` - Undo last change
|
||||
- `↑` / `↓` - Navigate input history
|
||||
|
||||
## Project Files Overview
|
||||
|
||||
```
|
||||
demo-project/
|
||||
├── src/
|
||||
│ ├── auth/
|
||||
│ │ └── service.ts # Authentication logic (login, logout, verify)
|
||||
│ ├── services/
|
||||
│ │ └── user.ts # User CRUD operations
|
||||
│ ├── utils/
|
||||
│ │ ├── logger.ts # Logging utility (multiple methods)
|
||||
│ │ └── validation.ts # Input validation (with TODOs/FIXMEs)
|
||||
│ ├── types/
|
||||
│ │ └── user.ts # TypeScript type definitions
|
||||
│ └── index.ts # Application entry point
|
||||
├── tests/
|
||||
│ └── user.test.ts # User service tests (vitest)
|
||||
├── package.json # Project configuration
|
||||
├── tsconfig.json # TypeScript configuration
|
||||
├── vitest.config.ts # Test configuration
|
||||
└── .ipuaro.json # ipuaro configuration
|
||||
```
|
||||
|
||||
## What ipuaro Can Do With This Project
|
||||
|
||||
### Read Tools ✅
|
||||
- **get_lines**: Read any file or specific line ranges
|
||||
- **get_function**: Extract specific functions (login, createUser, etc.)
|
||||
- **get_class**: Extract classes (UserService, AuthService, Logger, etc.)
|
||||
- **get_structure**: See directory tree
|
||||
|
||||
### Edit Tools ✅
|
||||
- **edit_lines**: Modify functions, fix bugs, add features
|
||||
- **create_file**: Add new utilities, tests, services
|
||||
- **delete_file**: Remove unused files
|
||||
|
||||
### Search Tools ✅
|
||||
- **find_references**: Find all usages of ValidationError, User, etc.
|
||||
- **find_definition**: Locate where Logger, UserService are defined
|
||||
|
||||
### Analysis Tools ✅
|
||||
- **get_dependencies**: See what UserService imports
|
||||
- **get_dependents**: See what imports validation.ts (multiple files!)
|
||||
- **get_complexity**: Identify complex functions (createUser has moderate complexity)
|
||||
- **get_todos**: Find 2 TODOs and 1 FIXME in the project
|
||||
|
||||
### Git Tools ✅
|
||||
- **git_status**: Check working tree
|
||||
- **git_diff**: See changes
|
||||
- **git_commit**: Commit with AI-generated messages
|
||||
|
||||
### Run Tools ✅
|
||||
- **run_command**: Execute npm scripts
|
||||
- **run_tests**: Run vitest tests
|
||||
|
||||
## Tips for Best Experience
|
||||
|
||||
1. **Start Small**: Ask about structure first, then dive into specific files
|
||||
2. **Be Specific**: "Show me the createUser function" vs "How does this work?"
|
||||
3. **Use Tools Implicitly**: Just ask questions, let ipuaro choose the right tools
|
||||
4. **Review Changes**: Always review diffs before applying edits
|
||||
5. **Test Often**: Ask ipuaro to run tests after making changes
|
||||
6. **Commit Incrementally**: Use git_commit for each logical change
|
||||
|
||||
## Advanced Workflows
|
||||
|
||||
### Workflow 1: Add New Feature
|
||||
|
||||
```
|
||||
You: Add email verification to the authentication flow
|
||||
|
||||
Agent will:
|
||||
1. Analyze current auth flow
|
||||
2. Propose design (new fields, methods)
|
||||
3. Edit AuthService to add verification
|
||||
4. Edit User types to add verified field
|
||||
5. Create tests for verification
|
||||
6. Run tests
|
||||
7. Offer to commit
|
||||
```
|
||||
|
||||
### Workflow 2: Performance Optimization
|
||||
|
||||
```
|
||||
You: The user lookup is slow when we have many users. Optimize it.
|
||||
|
||||
Agent will:
|
||||
1. Analyze UserService.getUserByEmail
|
||||
2. See it's using Array.find (O(n))
|
||||
3. Suggest adding an email index
|
||||
4. Edit to add private emailIndex: Map<string, User>
|
||||
5. Update createUser to populate index
|
||||
6. Update deleteUser to maintain index
|
||||
7. Run tests to verify
|
||||
```
|
||||
|
||||
### Workflow 3: Security Audit
|
||||
|
||||
```
|
||||
You: Audit the code for security vulnerabilities
|
||||
|
||||
Agent will:
|
||||
1. get_todos to find FIXME about XSS
|
||||
2. Review sanitizeInput implementation
|
||||
3. Check password validation strength
|
||||
4. Look for SQL injection risks (none here)
|
||||
5. Suggest improvements
|
||||
6. Optionally implement fixes
|
||||
```
|
||||
|
||||
## Next Steps
|
||||
|
||||
After exploring the demo project, try:
|
||||
|
||||
1. **Your Own Project**: Run `ipuaro` in your real codebase
|
||||
2. **Customize Config**: Edit `.ipuaro.json` to fit your needs
|
||||
3. **Different Model**: Try `--model qwen2.5-coder:32b-instruct` for better results
|
||||
4. **Auto-Apply Mode**: Use `--auto-apply` for faster iterations (with caution!)
|
||||
|
||||
## Troubleshooting
|
||||
|
||||
### Redis Not Connected
|
||||
```bash
|
||||
# Start Redis with persistence
|
||||
redis-server --appendonly yes
|
||||
```
|
||||
|
||||
### Ollama Model Not Found
|
||||
```bash
|
||||
# Pull the model
|
||||
ollama pull qwen2.5-coder:7b-instruct
|
||||
|
||||
# Check it's installed
|
||||
ollama list
|
||||
```
|
||||
|
||||
### Indexing Takes Long
|
||||
The project is small (~10 files) so indexing should be instant. For larger projects, use ignore patterns in `.ipuaro.json`.
|
||||
|
||||
## Learn More
|
||||
|
||||
- [ipuaro Documentation](../../README.md)
|
||||
- [Architecture Guide](../../ARCHITECTURE.md)
|
||||
- [Tools Reference](../../TOOLS.md)
|
||||
- [GitHub Repository](https://github.com/samiyev/puaros)
|
||||
|
||||
---
|
||||
|
||||
**Happy coding with ipuaro!** 🎩✨
|
||||
20
packages/ipuaro/examples/demo-project/package.json
Normal file
20
packages/ipuaro/examples/demo-project/package.json
Normal file
@@ -0,0 +1,20 @@
|
||||
{
|
||||
"name": "ipuaro-demo-project",
|
||||
"version": "1.0.0",
|
||||
"description": "Demo project for ipuaro - showcasing AI agent capabilities",
|
||||
"private": true,
|
||||
"type": "module",
|
||||
"scripts": {
|
||||
"dev": "tsx src/index.ts",
|
||||
"test": "vitest",
|
||||
"test:run": "vitest run",
|
||||
"build": "tsc"
|
||||
},
|
||||
"dependencies": {},
|
||||
"devDependencies": {
|
||||
"@types/node": "^22.10.1",
|
||||
"tsx": "^4.19.2",
|
||||
"typescript": "^5.7.2",
|
||||
"vitest": "^1.6.0"
|
||||
}
|
||||
}
|
||||
85
packages/ipuaro/examples/demo-project/src/auth/service.ts
Normal file
85
packages/ipuaro/examples/demo-project/src/auth/service.ts
Normal file
@@ -0,0 +1,85 @@
|
||||
/**
|
||||
* Authentication service
|
||||
*/
|
||||
|
||||
import type { User, AuthToken } from "../types/user"
|
||||
import { UserService } from "../services/user"
|
||||
import { createLogger } from "../utils/logger"
|
||||
|
||||
const logger = createLogger("AuthService")
|
||||
|
||||
export class AuthService {
|
||||
private tokens: Map<string, AuthToken> = new Map()
|
||||
|
||||
constructor(private userService: UserService) {}
|
||||
|
||||
async login(email: string, password: string): Promise<AuthToken> {
|
||||
logger.info("Login attempt", { email })
|
||||
|
||||
// Get user
|
||||
const user = await this.userService.getUserByEmail(email)
|
||||
if (!user) {
|
||||
logger.warn("Login failed - user not found", { email })
|
||||
throw new Error("Invalid credentials")
|
||||
}
|
||||
|
||||
// TODO: Implement actual password verification
|
||||
// For demo purposes, we just check if password is provided
|
||||
if (!password) {
|
||||
logger.warn("Login failed - no password", { email })
|
||||
throw new Error("Invalid credentials")
|
||||
}
|
||||
|
||||
// Generate token
|
||||
const token = this.generateToken(user)
|
||||
this.tokens.set(token.token, token)
|
||||
|
||||
logger.info("Login successful", { userId: user.id })
|
||||
return token
|
||||
}
|
||||
|
||||
async logout(tokenString: string): Promise<void> {
|
||||
logger.info("Logout", { token: tokenString.substring(0, 10) + "..." })
|
||||
|
||||
const token = this.tokens.get(tokenString)
|
||||
if (!token) {
|
||||
throw new Error("Invalid token")
|
||||
}
|
||||
|
||||
this.tokens.delete(tokenString)
|
||||
logger.info("Logout successful", { userId: token.userId })
|
||||
}
|
||||
|
||||
async verifyToken(tokenString: string): Promise<User> {
|
||||
logger.debug("Verifying token")
|
||||
|
||||
const token = this.tokens.get(tokenString)
|
||||
if (!token) {
|
||||
throw new Error("Invalid token")
|
||||
}
|
||||
|
||||
if (token.expiresAt < new Date()) {
|
||||
this.tokens.delete(tokenString)
|
||||
throw new Error("Token expired")
|
||||
}
|
||||
|
||||
const user = await this.userService.getUserById(token.userId)
|
||||
if (!user) {
|
||||
throw new Error("User not found")
|
||||
}
|
||||
|
||||
return user
|
||||
}
|
||||
|
||||
private generateToken(user: User): AuthToken {
|
||||
const token = `tok_${Date.now()}_${Math.random().toString(36).substring(7)}`
|
||||
const expiresAt = new Date()
|
||||
expiresAt.setHours(expiresAt.getHours() + 24) // 24 hours
|
||||
|
||||
return {
|
||||
token,
|
||||
expiresAt,
|
||||
userId: user.id
|
||||
}
|
||||
}
|
||||
}
|
||||
48
packages/ipuaro/examples/demo-project/src/index.ts
Normal file
48
packages/ipuaro/examples/demo-project/src/index.ts
Normal file
@@ -0,0 +1,48 @@
|
||||
/**
|
||||
* Demo application entry point
|
||||
*/
|
||||
|
||||
import { UserService } from "./services/user"
|
||||
import { AuthService } from "./auth/service"
|
||||
import { createLogger } from "./utils/logger"
|
||||
|
||||
const logger = createLogger("App")
|
||||
|
||||
async function main(): Promise<void> {
|
||||
logger.info("Starting demo application")
|
||||
|
||||
// Initialize services
|
||||
const userService = new UserService()
|
||||
const authService = new AuthService(userService)
|
||||
|
||||
try {
|
||||
// Create a demo user
|
||||
const user = await userService.createUser({
|
||||
email: "demo@example.com",
|
||||
name: "Demo User",
|
||||
password: "password123",
|
||||
role: "admin"
|
||||
})
|
||||
|
||||
logger.info("Demo user created", { userId: user.id })
|
||||
|
||||
// Login
|
||||
const token = await authService.login("demo@example.com", "password123")
|
||||
logger.info("Login successful", { token: token.token })
|
||||
|
||||
// Verify token
|
||||
const verifiedUser = await authService.verifyToken(token.token)
|
||||
logger.info("Token verified", { userId: verifiedUser.id })
|
||||
|
||||
// Logout
|
||||
await authService.logout(token.token)
|
||||
logger.info("Logout successful")
|
||||
} catch (error) {
|
||||
logger.error("Application error", error as Error)
|
||||
process.exit(1)
|
||||
}
|
||||
|
||||
logger.info("Demo application finished")
|
||||
}
|
||||
|
||||
main()
|
||||
102
packages/ipuaro/examples/demo-project/src/services/user.ts
Normal file
102
packages/ipuaro/examples/demo-project/src/services/user.ts
Normal file
@@ -0,0 +1,102 @@
|
||||
/**
|
||||
* User service - handles user-related operations
|
||||
*/
|
||||
|
||||
import type { User, CreateUserDto, UpdateUserDto } from "../types/user"
|
||||
import { isValidEmail, isStrongPassword, ValidationError } from "../utils/validation"
|
||||
import { createLogger } from "../utils/logger"
|
||||
|
||||
const logger = createLogger("UserService")
|
||||
|
||||
export class UserService {
|
||||
private users: Map<string, User> = new Map()
|
||||
|
||||
async createUser(dto: CreateUserDto): Promise<User> {
|
||||
logger.info("Creating user", { email: dto.email })
|
||||
|
||||
// Validate email
|
||||
if (!isValidEmail(dto.email)) {
|
||||
throw new ValidationError("Invalid email address", "email")
|
||||
}
|
||||
|
||||
// Validate password
|
||||
if (!isStrongPassword(dto.password)) {
|
||||
throw new ValidationError("Password must be at least 8 characters", "password")
|
||||
}
|
||||
|
||||
// Check if user already exists
|
||||
const existingUser = Array.from(this.users.values()).find(
|
||||
(u) => u.email === dto.email
|
||||
)
|
||||
|
||||
if (existingUser) {
|
||||
throw new Error("User with this email already exists")
|
||||
}
|
||||
|
||||
// Create user
|
||||
const user: User = {
|
||||
id: this.generateId(),
|
||||
email: dto.email,
|
||||
name: dto.name,
|
||||
role: dto.role || "user",
|
||||
createdAt: new Date(),
|
||||
updatedAt: new Date()
|
||||
}
|
||||
|
||||
this.users.set(user.id, user)
|
||||
logger.info("User created", { userId: user.id })
|
||||
|
||||
return user
|
||||
}
|
||||
|
||||
async getUserById(id: string): Promise<User | null> {
|
||||
logger.debug("Getting user by ID", { userId: id })
|
||||
return this.users.get(id) || null
|
||||
}
|
||||
|
||||
async getUserByEmail(email: string): Promise<User | null> {
|
||||
logger.debug("Getting user by email", { email })
|
||||
return Array.from(this.users.values()).find((u) => u.email === email) || null
|
||||
}
|
||||
|
||||
async updateUser(id: string, dto: UpdateUserDto): Promise<User> {
|
||||
logger.info("Updating user", { userId: id })
|
||||
|
||||
const user = this.users.get(id)
|
||||
if (!user) {
|
||||
throw new Error("User not found")
|
||||
}
|
||||
|
||||
const updated: User = {
|
||||
...user,
|
||||
...(dto.name && { name: dto.name }),
|
||||
...(dto.role && { role: dto.role }),
|
||||
updatedAt: new Date()
|
||||
}
|
||||
|
||||
this.users.set(id, updated)
|
||||
logger.info("User updated", { userId: id })
|
||||
|
||||
return updated
|
||||
}
|
||||
|
||||
async deleteUser(id: string): Promise<void> {
|
||||
logger.info("Deleting user", { userId: id })
|
||||
|
||||
if (!this.users.has(id)) {
|
||||
throw new Error("User not found")
|
||||
}
|
||||
|
||||
this.users.delete(id)
|
||||
logger.info("User deleted", { userId: id })
|
||||
}
|
||||
|
||||
async listUsers(): Promise<User[]> {
|
||||
logger.debug("Listing all users")
|
||||
return Array.from(this.users.values())
|
||||
}
|
||||
|
||||
private generateId(): string {
|
||||
return `user_${Date.now()}_${Math.random().toString(36).substring(7)}`
|
||||
}
|
||||
}
|
||||
32
packages/ipuaro/examples/demo-project/src/types/user.ts
Normal file
32
packages/ipuaro/examples/demo-project/src/types/user.ts
Normal file
@@ -0,0 +1,32 @@
|
||||
/**
|
||||
* User-related type definitions
|
||||
*/
|
||||
|
||||
export interface User {
|
||||
id: string
|
||||
email: string
|
||||
name: string
|
||||
role: UserRole
|
||||
createdAt: Date
|
||||
updatedAt: Date
|
||||
}
|
||||
|
||||
export type UserRole = "admin" | "user" | "guest"
|
||||
|
||||
export interface CreateUserDto {
|
||||
email: string
|
||||
name: string
|
||||
password: string
|
||||
role?: UserRole
|
||||
}
|
||||
|
||||
export interface UpdateUserDto {
|
||||
name?: string
|
||||
role?: UserRole
|
||||
}
|
||||
|
||||
export interface AuthToken {
|
||||
token: string
|
||||
expiresAt: Date
|
||||
userId: string
|
||||
}
|
||||
41
packages/ipuaro/examples/demo-project/src/utils/logger.ts
Normal file
41
packages/ipuaro/examples/demo-project/src/utils/logger.ts
Normal file
@@ -0,0 +1,41 @@
|
||||
/**
|
||||
* Simple logging utility
|
||||
*/
|
||||
|
||||
export type LogLevel = "debug" | "info" | "warn" | "error"
|
||||
|
||||
export class Logger {
|
||||
constructor(private context: string) {}
|
||||
|
||||
debug(message: string, meta?: Record<string, unknown>): void {
|
||||
this.log("debug", message, meta)
|
||||
}
|
||||
|
||||
info(message: string, meta?: Record<string, unknown>): void {
|
||||
this.log("info", message, meta)
|
||||
}
|
||||
|
||||
warn(message: string, meta?: Record<string, unknown>): void {
|
||||
this.log("warn", message, meta)
|
||||
}
|
||||
|
||||
error(message: string, error?: Error, meta?: Record<string, unknown>): void {
|
||||
this.log("error", message, { ...meta, error: error?.message })
|
||||
}
|
||||
|
||||
private log(level: LogLevel, message: string, meta?: Record<string, unknown>): void {
|
||||
const timestamp = new Date().toISOString()
|
||||
const logEntry = {
|
||||
timestamp,
|
||||
level,
|
||||
context: this.context,
|
||||
message,
|
||||
...(meta && { meta })
|
||||
}
|
||||
console.log(JSON.stringify(logEntry))
|
||||
}
|
||||
}
|
||||
|
||||
export function createLogger(context: string): Logger {
|
||||
return new Logger(context)
|
||||
}
|
||||
@@ -0,0 +1,28 @@
|
||||
/**
|
||||
* Validation utilities
|
||||
*/
|
||||
|
||||
export function isValidEmail(email: string): boolean {
|
||||
const emailRegex = /^[^\s@]+@[^\s@]+\.[^\s@]+$/
|
||||
return emailRegex.test(email)
|
||||
}
|
||||
|
||||
export function isStrongPassword(password: string): boolean {
|
||||
// TODO: Add more sophisticated password validation
|
||||
return password.length >= 8
|
||||
}
|
||||
|
||||
export function sanitizeInput(input: string): string {
|
||||
// FIXME: This is a basic implementation, needs XSS protection
|
||||
return input.trim().replace(/[<>]/g, "")
|
||||
}
|
||||
|
||||
export class ValidationError extends Error {
|
||||
constructor(
|
||||
message: string,
|
||||
public field: string
|
||||
) {
|
||||
super(message)
|
||||
this.name = "ValidationError"
|
||||
}
|
||||
}
|
||||
141
packages/ipuaro/examples/demo-project/tests/user.test.ts
Normal file
141
packages/ipuaro/examples/demo-project/tests/user.test.ts
Normal file
@@ -0,0 +1,141 @@
|
||||
/**
|
||||
* User service tests
|
||||
*/
|
||||
|
||||
import { describe, it, expect, beforeEach } from "vitest"
|
||||
import { UserService } from "../src/services/user"
|
||||
import { ValidationError } from "../src/utils/validation"
|
||||
|
||||
describe("UserService", () => {
|
||||
let userService: UserService
|
||||
|
||||
beforeEach(() => {
|
||||
userService = new UserService()
|
||||
})
|
||||
|
||||
describe("createUser", () => {
|
||||
it("should create a new user", async () => {
|
||||
const user = await userService.createUser({
|
||||
email: "test@example.com",
|
||||
name: "Test User",
|
||||
password: "password123"
|
||||
})
|
||||
|
||||
expect(user).toBeDefined()
|
||||
expect(user.email).toBe("test@example.com")
|
||||
expect(user.name).toBe("Test User")
|
||||
expect(user.role).toBe("user")
|
||||
})
|
||||
|
||||
it("should reject invalid email", async () => {
|
||||
await expect(
|
||||
userService.createUser({
|
||||
email: "invalid-email",
|
||||
name: "Test User",
|
||||
password: "password123"
|
||||
})
|
||||
).rejects.toThrow(ValidationError)
|
||||
})
|
||||
|
||||
it("should reject weak password", async () => {
|
||||
await expect(
|
||||
userService.createUser({
|
||||
email: "test@example.com",
|
||||
name: "Test User",
|
||||
password: "weak"
|
||||
})
|
||||
).rejects.toThrow(ValidationError)
|
||||
})
|
||||
|
||||
it("should prevent duplicate emails", async () => {
|
||||
await userService.createUser({
|
||||
email: "test@example.com",
|
||||
name: "Test User",
|
||||
password: "password123"
|
||||
})
|
||||
|
||||
await expect(
|
||||
userService.createUser({
|
||||
email: "test@example.com",
|
||||
name: "Another User",
|
||||
password: "password123"
|
||||
})
|
||||
).rejects.toThrow("already exists")
|
||||
})
|
||||
})
|
||||
|
||||
describe("getUserById", () => {
|
||||
it("should return user by ID", async () => {
|
||||
const created = await userService.createUser({
|
||||
email: "test@example.com",
|
||||
name: "Test User",
|
||||
password: "password123"
|
||||
})
|
||||
|
||||
const found = await userService.getUserById(created.id)
|
||||
expect(found).toEqual(created)
|
||||
})
|
||||
|
||||
it("should return null for non-existent ID", async () => {
|
||||
const found = await userService.getUserById("non-existent")
|
||||
expect(found).toBeNull()
|
||||
})
|
||||
})
|
||||
|
||||
describe("updateUser", () => {
|
||||
it("should update user name", async () => {
|
||||
const user = await userService.createUser({
|
||||
email: "test@example.com",
|
||||
name: "Test User",
|
||||
password: "password123"
|
||||
})
|
||||
|
||||
const updated = await userService.updateUser(user.id, {
|
||||
name: "Updated Name"
|
||||
})
|
||||
|
||||
expect(updated.name).toBe("Updated Name")
|
||||
expect(updated.email).toBe(user.email)
|
||||
})
|
||||
|
||||
it("should throw error for non-existent user", async () => {
|
||||
await expect(
|
||||
userService.updateUser("non-existent", { name: "Test" })
|
||||
).rejects.toThrow("not found")
|
||||
})
|
||||
})
|
||||
|
||||
describe("deleteUser", () => {
|
||||
it("should delete user", async () => {
|
||||
const user = await userService.createUser({
|
||||
email: "test@example.com",
|
||||
name: "Test User",
|
||||
password: "password123"
|
||||
})
|
||||
|
||||
await userService.deleteUser(user.id)
|
||||
|
||||
const found = await userService.getUserById(user.id)
|
||||
expect(found).toBeNull()
|
||||
})
|
||||
})
|
||||
|
||||
describe("listUsers", () => {
|
||||
it("should return all users", async () => {
|
||||
await userService.createUser({
|
||||
email: "user1@example.com",
|
||||
name: "User 1",
|
||||
password: "password123"
|
||||
})
|
||||
|
||||
await userService.createUser({
|
||||
email: "user2@example.com",
|
||||
name: "User 2",
|
||||
password: "password123"
|
||||
})
|
||||
|
||||
const users = await userService.listUsers()
|
||||
expect(users).toHaveLength(2)
|
||||
})
|
||||
})
|
||||
})
|
||||
16
packages/ipuaro/examples/demo-project/tsconfig.json
Normal file
16
packages/ipuaro/examples/demo-project/tsconfig.json
Normal file
@@ -0,0 +1,16 @@
|
||||
{
|
||||
"compilerOptions": {
|
||||
"target": "ES2023",
|
||||
"module": "ESNext",
|
||||
"lib": ["ES2023"],
|
||||
"moduleResolution": "Bundler",
|
||||
"esModuleInterop": true,
|
||||
"strict": true,
|
||||
"skipLibCheck": true,
|
||||
"resolveJsonModule": true,
|
||||
"outDir": "dist",
|
||||
"rootDir": "src"
|
||||
},
|
||||
"include": ["src/**/*"],
|
||||
"exclude": ["node_modules", "dist", "tests"]
|
||||
}
|
||||
8
packages/ipuaro/examples/demo-project/vitest.config.ts
Normal file
8
packages/ipuaro/examples/demo-project/vitest.config.ts
Normal file
@@ -0,0 +1,8 @@
|
||||
import { defineConfig } from "vitest/config"
|
||||
|
||||
export default defineConfig({
|
||||
test: {
|
||||
globals: true,
|
||||
environment: "node"
|
||||
}
|
||||
})
|
||||
@@ -1,6 +1,6 @@
|
||||
{
|
||||
"name": "@samiyev/ipuaro",
|
||||
"version": "0.17.0",
|
||||
"version": "0.21.4",
|
||||
"description": "Local AI agent for codebase operations with infinite context feeling",
|
||||
"author": "Fozilbek Samiyev <fozilbek.samiyev@gmail.com>",
|
||||
"license": "MIT",
|
||||
@@ -8,7 +8,7 @@
|
||||
"main": "./dist/index.js",
|
||||
"types": "./dist/index.d.ts",
|
||||
"bin": {
|
||||
"ipuaro": "./bin/ipuaro.js"
|
||||
"ipuaro": "bin/ipuaro.js"
|
||||
},
|
||||
"exports": {
|
||||
".": {
|
||||
@@ -48,10 +48,14 @@
|
||||
"zod": "^3.23.8"
|
||||
},
|
||||
"devDependencies": {
|
||||
"@testing-library/react": "^16.3.0",
|
||||
"@types/jsdom": "^27.0.0",
|
||||
"@types/node": "^22.10.1",
|
||||
"@types/react": "^18.2.0",
|
||||
"@vitest/coverage-v8": "^1.6.0",
|
||||
"@vitest/ui": "^1.6.0",
|
||||
"jsdom": "^27.2.0",
|
||||
"react-dom": "18.3.1",
|
||||
"tsup": "^8.3.5",
|
||||
"typescript": "^5.7.2",
|
||||
"vitest": "^1.6.0"
|
||||
|
||||
224
packages/ipuaro/src/application/use-cases/ExecuteTool.ts
Normal file
224
packages/ipuaro/src/application/use-cases/ExecuteTool.ts
Normal file
@@ -0,0 +1,224 @@
|
||||
import { randomUUID } from "node:crypto"
|
||||
import type { Session } from "../../domain/entities/Session.js"
|
||||
import type { ISessionStorage } from "../../domain/services/ISessionStorage.js"
|
||||
import type { IStorage } from "../../domain/services/IStorage.js"
|
||||
import type { DiffInfo, ToolContext } from "../../domain/services/ITool.js"
|
||||
import type { ToolCall } from "../../domain/value-objects/ToolCall.js"
|
||||
import { createErrorResult, type ToolResult } from "../../domain/value-objects/ToolResult.js"
|
||||
import { createUndoEntry } from "../../domain/value-objects/UndoEntry.js"
|
||||
import type { IToolRegistry } from "../interfaces/IToolRegistry.js"
|
||||
|
||||
/**
|
||||
* Result of confirmation dialog.
|
||||
*/
|
||||
export interface ConfirmationResult {
|
||||
confirmed: boolean
|
||||
editedContent?: string[]
|
||||
}
|
||||
|
||||
/**
|
||||
* Confirmation handler callback type.
|
||||
* Can return either a boolean (for backward compatibility) or a ConfirmationResult.
|
||||
*/
|
||||
export type ConfirmationHandler = (
|
||||
message: string,
|
||||
diff?: DiffInfo,
|
||||
) => Promise<boolean | ConfirmationResult>
|
||||
|
||||
/**
|
||||
* Progress handler callback type.
|
||||
*/
|
||||
export type ProgressHandler = (message: string) => void
|
||||
|
||||
/**
|
||||
* Options for ExecuteTool.
|
||||
*/
|
||||
export interface ExecuteToolOptions {
|
||||
/** Auto-apply edits without confirmation */
|
||||
autoApply?: boolean
|
||||
/** Confirmation handler */
|
||||
onConfirmation?: ConfirmationHandler
|
||||
/** Progress handler */
|
||||
onProgress?: ProgressHandler
|
||||
}
|
||||
|
||||
/**
|
||||
* Result of tool execution.
|
||||
*/
|
||||
export interface ExecuteToolResult {
|
||||
result: ToolResult
|
||||
undoEntryCreated: boolean
|
||||
undoEntryId?: string
|
||||
}
|
||||
|
||||
/**
|
||||
* Use case for executing a single tool.
|
||||
* Orchestrates tool execution with:
|
||||
* - Parameter validation
|
||||
* - Confirmation flow
|
||||
* - Undo stack management
|
||||
* - Storage updates
|
||||
*/
|
||||
export class ExecuteTool {
|
||||
private readonly storage: IStorage
|
||||
private readonly sessionStorage: ISessionStorage
|
||||
private readonly tools: IToolRegistry
|
||||
private readonly projectRoot: string
|
||||
private lastUndoEntryId?: string
|
||||
|
||||
constructor(
|
||||
storage: IStorage,
|
||||
sessionStorage: ISessionStorage,
|
||||
tools: IToolRegistry,
|
||||
projectRoot: string,
|
||||
) {
|
||||
this.storage = storage
|
||||
this.sessionStorage = sessionStorage
|
||||
this.tools = tools
|
||||
this.projectRoot = projectRoot
|
||||
}
|
||||
|
||||
/**
|
||||
* Execute a tool call.
|
||||
*
|
||||
* @param toolCall - The tool call to execute
|
||||
* @param session - Current session (for undo stack)
|
||||
* @param options - Execution options
|
||||
* @returns Execution result
|
||||
*/
|
||||
async execute(
|
||||
toolCall: ToolCall,
|
||||
session: Session,
|
||||
options: ExecuteToolOptions = {},
|
||||
): Promise<ExecuteToolResult> {
|
||||
this.lastUndoEntryId = undefined
|
||||
const startTime = Date.now()
|
||||
const tool = this.tools.get(toolCall.name)
|
||||
|
||||
if (!tool) {
|
||||
return {
|
||||
result: createErrorResult(
|
||||
toolCall.id,
|
||||
`Unknown tool: ${toolCall.name}`,
|
||||
Date.now() - startTime,
|
||||
),
|
||||
undoEntryCreated: false,
|
||||
}
|
||||
}
|
||||
|
||||
const validationError = tool.validateParams(toolCall.params)
|
||||
if (validationError) {
|
||||
return {
|
||||
result: createErrorResult(toolCall.id, validationError, Date.now() - startTime),
|
||||
undoEntryCreated: false,
|
||||
}
|
||||
}
|
||||
|
||||
const context = this.buildToolContext(toolCall, session, options)
|
||||
|
||||
try {
|
||||
const result = await tool.execute(toolCall.params, context)
|
||||
|
||||
return {
|
||||
result,
|
||||
undoEntryCreated: this.lastUndoEntryId !== undefined,
|
||||
undoEntryId: this.lastUndoEntryId,
|
||||
}
|
||||
} catch (error) {
|
||||
const errorMessage = error instanceof Error ? error.message : String(error)
|
||||
return {
|
||||
result: createErrorResult(toolCall.id, errorMessage, Date.now() - startTime),
|
||||
undoEntryCreated: false,
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* Build tool context for execution.
|
||||
*/
|
||||
private buildToolContext(
|
||||
toolCall: ToolCall,
|
||||
session: Session,
|
||||
options: ExecuteToolOptions,
|
||||
): ToolContext {
|
||||
return {
|
||||
projectRoot: this.projectRoot,
|
||||
storage: this.storage,
|
||||
requestConfirmation: async (msg: string, diff?: DiffInfo) => {
|
||||
return this.handleConfirmation(msg, diff, toolCall, session, options)
|
||||
},
|
||||
onProgress: (msg: string) => {
|
||||
options.onProgress?.(msg)
|
||||
},
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* Handle confirmation for tool actions.
|
||||
* Supports edited content from user.
|
||||
*/
|
||||
private async handleConfirmation(
|
||||
msg: string,
|
||||
diff: DiffInfo | undefined,
|
||||
toolCall: ToolCall,
|
||||
session: Session,
|
||||
options: ExecuteToolOptions,
|
||||
): Promise<boolean> {
|
||||
if (options.autoApply) {
|
||||
if (diff) {
|
||||
this.lastUndoEntryId = await this.createUndoEntry(diff, toolCall, session)
|
||||
}
|
||||
return true
|
||||
}
|
||||
|
||||
if (options.onConfirmation) {
|
||||
const result = await options.onConfirmation(msg, diff)
|
||||
|
||||
const confirmed = typeof result === "boolean" ? result : result.confirmed
|
||||
const editedContent = typeof result === "boolean" ? undefined : result.editedContent
|
||||
|
||||
if (confirmed && diff) {
|
||||
if (editedContent && editedContent.length > 0) {
|
||||
diff.newLines = editedContent
|
||||
if (toolCall.params.content && typeof toolCall.params.content === "string") {
|
||||
toolCall.params.content = editedContent.join("\n")
|
||||
}
|
||||
}
|
||||
|
||||
this.lastUndoEntryId = await this.createUndoEntry(diff, toolCall, session)
|
||||
}
|
||||
|
||||
return confirmed
|
||||
}
|
||||
|
||||
if (diff) {
|
||||
this.lastUndoEntryId = await this.createUndoEntry(diff, toolCall, session)
|
||||
}
|
||||
return true
|
||||
}
|
||||
|
||||
/**
|
||||
* Create undo entry from diff.
|
||||
*/
|
||||
private async createUndoEntry(
|
||||
diff: DiffInfo,
|
||||
toolCall: ToolCall,
|
||||
session: Session,
|
||||
): Promise<string> {
|
||||
const entryId = randomUUID()
|
||||
const entry = createUndoEntry(
|
||||
entryId,
|
||||
diff.filePath,
|
||||
diff.oldLines,
|
||||
diff.newLines,
|
||||
`${toolCall.name}: ${diff.filePath}`,
|
||||
toolCall.id,
|
||||
)
|
||||
|
||||
session.addUndoEntry(entry)
|
||||
await this.sessionStorage.pushUndoEntry(session.id, entry)
|
||||
session.stats.editsApplied++
|
||||
|
||||
return entryId
|
||||
}
|
||||
}
|
||||
@@ -1,9 +1,8 @@
|
||||
import { randomUUID } from "node:crypto"
|
||||
import type { Session } from "../../domain/entities/Session.js"
|
||||
import type { ILLMClient } from "../../domain/services/ILLMClient.js"
|
||||
import type { ISessionStorage } from "../../domain/services/ISessionStorage.js"
|
||||
import type { IStorage } from "../../domain/services/IStorage.js"
|
||||
import type { DiffInfo, ToolContext } from "../../domain/services/ITool.js"
|
||||
import type { DiffInfo } from "../../domain/services/ITool.js"
|
||||
import {
|
||||
type ChatMessage,
|
||||
createAssistantMessage,
|
||||
@@ -12,8 +11,8 @@ import {
|
||||
createUserMessage,
|
||||
} from "../../domain/value-objects/ChatMessage.js"
|
||||
import type { ToolCall } from "../../domain/value-objects/ToolCall.js"
|
||||
import { createErrorResult, type ToolResult } from "../../domain/value-objects/ToolResult.js"
|
||||
import { createUndoEntry, type UndoEntry } from "../../domain/value-objects/UndoEntry.js"
|
||||
import type { ToolResult } from "../../domain/value-objects/ToolResult.js"
|
||||
import type { UndoEntry } from "../../domain/value-objects/UndoEntry.js"
|
||||
import { type ErrorOption, IpuaroError } from "../../shared/errors/IpuaroError.js"
|
||||
import {
|
||||
buildInitialContext,
|
||||
@@ -23,6 +22,7 @@ import {
|
||||
import { parseToolCalls } from "../../infrastructure/llm/ResponseParser.js"
|
||||
import type { IToolRegistry } from "../interfaces/IToolRegistry.js"
|
||||
import { ContextManager } from "./ContextManager.js"
|
||||
import { type ConfirmationResult, ExecuteTool } from "./ExecuteTool.js"
|
||||
|
||||
/**
|
||||
* Status during message handling.
|
||||
@@ -56,7 +56,7 @@ export interface HandleMessageEvents {
|
||||
onMessage?: (message: ChatMessage) => void
|
||||
onToolCall?: (call: ToolCall) => void
|
||||
onToolResult?: (result: ToolResult) => void
|
||||
onConfirmation?: (message: string, diff?: DiffInfo) => Promise<boolean>
|
||||
onConfirmation?: (message: string, diff?: DiffInfo) => Promise<boolean | ConfirmationResult>
|
||||
onError?: (error: IpuaroError) => Promise<ErrorOption>
|
||||
onStatusChange?: (status: HandleMessageStatus) => void
|
||||
onUndoEntry?: (entry: UndoEntry) => void
|
||||
@@ -82,6 +82,7 @@ export class HandleMessage {
|
||||
private readonly llm: ILLMClient
|
||||
private readonly tools: IToolRegistry
|
||||
private readonly contextManager: ContextManager
|
||||
private readonly executeTool: ExecuteTool
|
||||
private readonly projectRoot: string
|
||||
private projectStructure?: ProjectStructure
|
||||
|
||||
@@ -102,6 +103,7 @@ export class HandleMessage {
|
||||
this.tools = tools
|
||||
this.projectRoot = projectRoot
|
||||
this.contextManager = new ContextManager(llm.getContextWindowSize())
|
||||
this.executeTool = new ExecuteTool(storage, sessionStorage, tools, projectRoot)
|
||||
}
|
||||
|
||||
/**
|
||||
@@ -257,87 +259,32 @@ export class HandleMessage {
|
||||
}
|
||||
|
||||
private async executeToolCall(toolCall: ToolCall, session: Session): Promise<ToolResult> {
|
||||
const startTime = Date.now()
|
||||
const tool = this.tools.get(toolCall.name)
|
||||
|
||||
if (!tool) {
|
||||
return createErrorResult(
|
||||
toolCall.id,
|
||||
`Unknown tool: ${toolCall.name}`,
|
||||
Date.now() - startTime,
|
||||
)
|
||||
}
|
||||
|
||||
const context: ToolContext = {
|
||||
projectRoot: this.projectRoot,
|
||||
storage: this.storage,
|
||||
requestConfirmation: async (msg: string, diff?: DiffInfo) => {
|
||||
return this.handleConfirmation(msg, diff, toolCall, session)
|
||||
const { result, undoEntryCreated, undoEntryId } = await this.executeTool.execute(
|
||||
toolCall,
|
||||
session,
|
||||
{
|
||||
autoApply: this.options.autoApply,
|
||||
onConfirmation: async (msg: string, diff?: DiffInfo) => {
|
||||
this.emitStatus("awaiting_confirmation")
|
||||
if (this.events.onConfirmation) {
|
||||
return this.events.onConfirmation(msg, diff)
|
||||
}
|
||||
return true
|
||||
},
|
||||
onProgress: (_msg: string) => {
|
||||
this.events.onStatusChange?.("tool_call")
|
||||
},
|
||||
},
|
||||
onProgress: (_msg: string) => {
|
||||
this.events.onStatusChange?.("tool_call")
|
||||
},
|
||||
}
|
||||
|
||||
try {
|
||||
const validationError = tool.validateParams(toolCall.params)
|
||||
if (validationError) {
|
||||
return createErrorResult(toolCall.id, validationError, Date.now() - startTime)
|
||||
}
|
||||
|
||||
const result = await tool.execute(toolCall.params, context)
|
||||
return result
|
||||
} catch (error) {
|
||||
const errorMessage = error instanceof Error ? error.message : String(error)
|
||||
return createErrorResult(toolCall.id, errorMessage, Date.now() - startTime)
|
||||
}
|
||||
}
|
||||
|
||||
private async handleConfirmation(
|
||||
msg: string,
|
||||
diff: DiffInfo | undefined,
|
||||
toolCall: ToolCall,
|
||||
session: Session,
|
||||
): Promise<boolean> {
|
||||
if (this.options.autoApply) {
|
||||
if (diff) {
|
||||
this.createUndoEntryFromDiff(diff, toolCall, session)
|
||||
}
|
||||
return true
|
||||
}
|
||||
|
||||
this.emitStatus("awaiting_confirmation")
|
||||
|
||||
if (this.events.onConfirmation) {
|
||||
const confirmed = await this.events.onConfirmation(msg, diff)
|
||||
|
||||
if (confirmed && diff) {
|
||||
this.createUndoEntryFromDiff(diff, toolCall, session)
|
||||
}
|
||||
|
||||
return confirmed
|
||||
}
|
||||
|
||||
if (diff) {
|
||||
this.createUndoEntryFromDiff(diff, toolCall, session)
|
||||
}
|
||||
return true
|
||||
}
|
||||
|
||||
private createUndoEntryFromDiff(diff: DiffInfo, toolCall: ToolCall, session: Session): void {
|
||||
const entry = createUndoEntry(
|
||||
randomUUID(),
|
||||
diff.filePath,
|
||||
diff.oldLines,
|
||||
diff.newLines,
|
||||
`${toolCall.name}: ${diff.filePath}`,
|
||||
toolCall.id,
|
||||
)
|
||||
|
||||
session.addUndoEntry(entry)
|
||||
void this.sessionStorage.pushUndoEntry(session.id, entry)
|
||||
session.stats.editsApplied++
|
||||
this.events.onUndoEntry?.(entry)
|
||||
if (undoEntryCreated && undoEntryId) {
|
||||
const undoEntry = session.undoStack.find((entry) => entry.id === undoEntryId)
|
||||
if (undoEntry) {
|
||||
this.events.onUndoEntry?.(undoEntry)
|
||||
}
|
||||
}
|
||||
|
||||
return result
|
||||
}
|
||||
|
||||
private async handleLLMError(error: unknown, session: Session): Promise<void> {
|
||||
|
||||
184
packages/ipuaro/src/application/use-cases/IndexProject.ts
Normal file
184
packages/ipuaro/src/application/use-cases/IndexProject.ts
Normal file
@@ -0,0 +1,184 @@
|
||||
import * as path from "node:path"
|
||||
import type { IStorage } from "../../domain/services/IStorage.js"
|
||||
import type { IndexingStats, IndexProgress } from "../../domain/services/IIndexer.js"
|
||||
import { FileScanner } from "../../infrastructure/indexer/FileScanner.js"
|
||||
import { ASTParser } from "../../infrastructure/indexer/ASTParser.js"
|
||||
import { MetaAnalyzer } from "../../infrastructure/indexer/MetaAnalyzer.js"
|
||||
import { IndexBuilder } from "../../infrastructure/indexer/IndexBuilder.js"
|
||||
import { createFileData, type FileData } from "../../domain/value-objects/FileData.js"
|
||||
import type { FileAST } from "../../domain/value-objects/FileAST.js"
|
||||
import { md5 } from "../../shared/utils/hash.js"
|
||||
|
||||
/**
|
||||
* Options for indexing a project.
|
||||
*/
|
||||
export interface IndexProjectOptions {
|
||||
/** Additional ignore patterns */
|
||||
additionalIgnore?: string[]
|
||||
/** Progress callback */
|
||||
onProgress?: (progress: IndexProgress) => void
|
||||
}
|
||||
|
||||
/**
|
||||
* Use case for indexing a project.
|
||||
* Orchestrates the full indexing pipeline:
|
||||
* 1. Scan files
|
||||
* 2. Parse AST
|
||||
* 3. Analyze metadata
|
||||
* 4. Build indexes
|
||||
* 5. Store in Redis
|
||||
*/
|
||||
export class IndexProject {
|
||||
private readonly storage: IStorage
|
||||
private readonly scanner: FileScanner
|
||||
private readonly parser: ASTParser
|
||||
private readonly metaAnalyzer: MetaAnalyzer
|
||||
private readonly indexBuilder: IndexBuilder
|
||||
|
||||
constructor(storage: IStorage, projectRoot: string) {
|
||||
this.storage = storage
|
||||
this.scanner = new FileScanner()
|
||||
this.parser = new ASTParser()
|
||||
this.metaAnalyzer = new MetaAnalyzer(projectRoot)
|
||||
this.indexBuilder = new IndexBuilder(projectRoot)
|
||||
}
|
||||
|
||||
/**
|
||||
* Execute the indexing pipeline.
|
||||
*
|
||||
* @param projectRoot - Absolute path to project root
|
||||
* @param options - Optional configuration
|
||||
* @returns Indexing statistics
|
||||
*/
|
||||
async execute(projectRoot: string, options: IndexProjectOptions = {}): Promise<IndexingStats> {
|
||||
const startTime = Date.now()
|
||||
const stats: IndexingStats = {
|
||||
filesScanned: 0,
|
||||
filesParsed: 0,
|
||||
parseErrors: 0,
|
||||
timeMs: 0,
|
||||
}
|
||||
|
||||
const fileDataMap = new Map<string, FileData>()
|
||||
const astMap = new Map<string, FileAST>()
|
||||
const contentMap = new Map<string, string>()
|
||||
|
||||
// Phase 1: Scanning
|
||||
this.reportProgress(options.onProgress, 0, 0, "", "scanning")
|
||||
|
||||
const scanResults = await this.scanner.scanAll(projectRoot)
|
||||
stats.filesScanned = scanResults.length
|
||||
|
||||
// Phase 2: Parsing
|
||||
let current = 0
|
||||
const total = scanResults.length
|
||||
|
||||
for (const scanResult of scanResults) {
|
||||
current++
|
||||
const fullPath = path.join(projectRoot, scanResult.path)
|
||||
this.reportProgress(options.onProgress, current, total, scanResult.path, "parsing")
|
||||
|
||||
const content = await FileScanner.readFileContent(fullPath)
|
||||
if (!content) {
|
||||
continue
|
||||
}
|
||||
|
||||
contentMap.set(scanResult.path, content)
|
||||
|
||||
const lines = content.split("\n")
|
||||
const hash = md5(content)
|
||||
|
||||
const fileData = createFileData(lines, hash, scanResult.size, scanResult.lastModified)
|
||||
fileDataMap.set(scanResult.path, fileData)
|
||||
|
||||
const language = this.detectLanguage(scanResult.path)
|
||||
if (!language) {
|
||||
continue
|
||||
}
|
||||
|
||||
const ast = this.parser.parse(content, language)
|
||||
astMap.set(scanResult.path, ast)
|
||||
|
||||
stats.filesParsed++
|
||||
if (ast.parseError) {
|
||||
stats.parseErrors++
|
||||
}
|
||||
}
|
||||
|
||||
// Phase 3: Analyzing metadata
|
||||
current = 0
|
||||
for (const [filePath, ast] of astMap) {
|
||||
current++
|
||||
this.reportProgress(options.onProgress, current, astMap.size, filePath, "analyzing")
|
||||
|
||||
const content = contentMap.get(filePath)
|
||||
if (!content) {
|
||||
continue
|
||||
}
|
||||
|
||||
const fullPath = path.join(projectRoot, filePath)
|
||||
const meta = this.metaAnalyzer.analyze(fullPath, ast, content, astMap)
|
||||
|
||||
await this.storage.setMeta(filePath, meta)
|
||||
}
|
||||
|
||||
// Phase 4: Building indexes
|
||||
this.reportProgress(options.onProgress, 1, 1, "Building indexes", "indexing")
|
||||
|
||||
const symbolIndex = this.indexBuilder.buildSymbolIndex(astMap)
|
||||
const depsGraph = this.indexBuilder.buildDepsGraph(astMap)
|
||||
|
||||
// Phase 5: Store everything
|
||||
for (const [filePath, fileData] of fileDataMap) {
|
||||
await this.storage.setFile(filePath, fileData)
|
||||
}
|
||||
|
||||
for (const [filePath, ast] of astMap) {
|
||||
await this.storage.setAST(filePath, ast)
|
||||
}
|
||||
|
||||
await this.storage.setSymbolIndex(symbolIndex)
|
||||
await this.storage.setDepsGraph(depsGraph)
|
||||
|
||||
// Store last indexed timestamp
|
||||
await this.storage.setProjectConfig("last_indexed", Date.now())
|
||||
|
||||
stats.timeMs = Date.now() - startTime
|
||||
|
||||
return stats
|
||||
}
|
||||
|
||||
/**
|
||||
* Detect language from file extension.
|
||||
*/
|
||||
private detectLanguage(filePath: string): "ts" | "tsx" | "js" | "jsx" | null {
|
||||
const ext = path.extname(filePath).toLowerCase()
|
||||
switch (ext) {
|
||||
case ".ts":
|
||||
return "ts"
|
||||
case ".tsx":
|
||||
return "tsx"
|
||||
case ".js":
|
||||
return "js"
|
||||
case ".jsx":
|
||||
return "jsx"
|
||||
default:
|
||||
return null
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* Report progress to callback if provided.
|
||||
*/
|
||||
private reportProgress(
|
||||
callback: ((progress: IndexProgress) => void) | undefined,
|
||||
current: number,
|
||||
total: number,
|
||||
currentFile: string,
|
||||
phase: IndexProgress["phase"],
|
||||
): void {
|
||||
if (callback) {
|
||||
callback({ current, total, currentFile, phase })
|
||||
}
|
||||
}
|
||||
}
|
||||
@@ -4,3 +4,5 @@ export * from "./StartSession.js"
|
||||
export * from "./HandleMessage.js"
|
||||
export * from "./UndoChange.js"
|
||||
export * from "./ContextManager.js"
|
||||
export * from "./IndexProject.js"
|
||||
export * from "./ExecuteTool.js"
|
||||
|
||||
@@ -3,23 +3,14 @@
|
||||
* Indexes project without starting TUI.
|
||||
*/
|
||||
|
||||
import * as fs from "node:fs/promises"
|
||||
import * as path from "node:path"
|
||||
import { RedisClient } from "../../infrastructure/storage/RedisClient.js"
|
||||
import { RedisStorage } from "../../infrastructure/storage/RedisStorage.js"
|
||||
import { generateProjectName } from "../../infrastructure/storage/schema.js"
|
||||
import { FileScanner } from "../../infrastructure/indexer/FileScanner.js"
|
||||
import { ASTParser } from "../../infrastructure/indexer/ASTParser.js"
|
||||
import { MetaAnalyzer } from "../../infrastructure/indexer/MetaAnalyzer.js"
|
||||
import { IndexBuilder } from "../../infrastructure/indexer/IndexBuilder.js"
|
||||
import { createFileData } from "../../domain/value-objects/FileData.js"
|
||||
import type { FileAST } from "../../domain/value-objects/FileAST.js"
|
||||
import { IndexProject } from "../../application/use-cases/IndexProject.js"
|
||||
import { type Config, DEFAULT_CONFIG } from "../../shared/constants/config.js"
|
||||
import { md5 } from "../../shared/utils/hash.js"
|
||||
import { checkRedis } from "./onboarding.js"
|
||||
|
||||
type Language = "ts" | "tsx" | "js" | "jsx"
|
||||
|
||||
/**
|
||||
* Result of index command.
|
||||
*/
|
||||
@@ -52,7 +43,6 @@ export async function executeIndex(
|
||||
const startTime = Date.now()
|
||||
const resolvedPath = path.resolve(projectPath)
|
||||
const projectName = generateProjectName(resolvedPath)
|
||||
const errors: string[] = []
|
||||
|
||||
console.warn(`📁 Indexing project: ${resolvedPath}`)
|
||||
console.warn(` Project name: ${projectName}\n`)
|
||||
@@ -76,142 +66,69 @@ export async function executeIndex(
|
||||
await redisClient.connect()
|
||||
|
||||
const storage = new RedisStorage(redisClient, projectName)
|
||||
const scanner = new FileScanner({
|
||||
onProgress: (progress): void => {
|
||||
onProgress?.("scanning", progress.current, progress.total, progress.currentFile)
|
||||
const indexProject = new IndexProject(storage, resolvedPath)
|
||||
|
||||
let lastPhase: "scanning" | "parsing" | "analyzing" | "indexing" = "scanning"
|
||||
let lastProgress = 0
|
||||
|
||||
const stats = await indexProject.execute(resolvedPath, {
|
||||
onProgress: (progress) => {
|
||||
if (progress.phase !== lastPhase) {
|
||||
if (lastPhase === "scanning") {
|
||||
console.warn(` Found ${String(progress.total)} files\n`)
|
||||
} else if (lastProgress > 0) {
|
||||
console.warn("")
|
||||
}
|
||||
|
||||
const phaseLabels = {
|
||||
scanning: "🔍 Scanning files...",
|
||||
parsing: "📝 Parsing files...",
|
||||
analyzing: "📊 Analyzing metadata...",
|
||||
indexing: "🏗️ Building indexes...",
|
||||
}
|
||||
console.warn(phaseLabels[progress.phase])
|
||||
lastPhase = progress.phase
|
||||
}
|
||||
|
||||
if (progress.phase === "indexing") {
|
||||
onProgress?.("storing", progress.current, progress.total)
|
||||
} else {
|
||||
onProgress?.(
|
||||
progress.phase,
|
||||
progress.current,
|
||||
progress.total,
|
||||
progress.currentFile,
|
||||
)
|
||||
}
|
||||
|
||||
if (
|
||||
progress.current % 50 === 0 &&
|
||||
progress.phase !== "scanning" &&
|
||||
progress.phase !== "indexing"
|
||||
) {
|
||||
process.stdout.write(
|
||||
`\r ${progress.phase === "parsing" ? "Parsed" : "Analyzed"} ${String(progress.current)}/${String(progress.total)} files...`,
|
||||
)
|
||||
}
|
||||
lastProgress = progress.current
|
||||
},
|
||||
})
|
||||
const astParser = new ASTParser()
|
||||
const metaAnalyzer = new MetaAnalyzer(resolvedPath)
|
||||
const indexBuilder = new IndexBuilder(resolvedPath)
|
||||
|
||||
console.warn("🔍 Scanning files...")
|
||||
const files = await scanner.scanAll(resolvedPath)
|
||||
console.warn(` Found ${String(files.length)} files\n`)
|
||||
const symbolIndex = await storage.getSymbolIndex()
|
||||
const durationSec = (stats.timeMs / 1000).toFixed(2)
|
||||
|
||||
if (files.length === 0) {
|
||||
console.warn("⚠️ No files found to index.")
|
||||
return {
|
||||
success: true,
|
||||
filesIndexed: 0,
|
||||
filesSkipped: 0,
|
||||
errors: [],
|
||||
duration: Date.now() - startTime,
|
||||
}
|
||||
}
|
||||
|
||||
console.warn("📝 Parsing files...")
|
||||
const allASTs = new Map<string, FileAST>()
|
||||
const fileContents = new Map<string, string>()
|
||||
let parsed = 0
|
||||
let skipped = 0
|
||||
|
||||
for (const file of files) {
|
||||
const fullPath = path.join(resolvedPath, file.path)
|
||||
const language = getLanguage(file.path)
|
||||
|
||||
if (!language) {
|
||||
skipped++
|
||||
continue
|
||||
}
|
||||
|
||||
try {
|
||||
const content = await fs.readFile(fullPath, "utf-8")
|
||||
const ast = astParser.parse(content, language)
|
||||
|
||||
if (ast.parseError) {
|
||||
errors.push(
|
||||
`Parse error in ${file.path}: ${ast.parseErrorMessage ?? "unknown"}`,
|
||||
)
|
||||
skipped++
|
||||
continue
|
||||
}
|
||||
|
||||
allASTs.set(file.path, ast)
|
||||
fileContents.set(file.path, content)
|
||||
parsed++
|
||||
|
||||
onProgress?.("parsing", parsed + skipped, files.length, file.path)
|
||||
|
||||
if ((parsed + skipped) % 50 === 0) {
|
||||
process.stdout.write(
|
||||
`\r Parsed ${String(parsed)} files (${String(skipped)} skipped)...`,
|
||||
)
|
||||
}
|
||||
} catch (error) {
|
||||
const message = error instanceof Error ? error.message : String(error)
|
||||
errors.push(`Error reading ${file.path}: ${message}`)
|
||||
skipped++
|
||||
}
|
||||
}
|
||||
console.warn(`\r Parsed ${String(parsed)} files (${String(skipped)} skipped) \n`)
|
||||
|
||||
console.warn("📊 Analyzing metadata...")
|
||||
let analyzed = 0
|
||||
for (const [filePath, ast] of allASTs) {
|
||||
const content = fileContents.get(filePath) ?? ""
|
||||
const meta = metaAnalyzer.analyze(
|
||||
path.join(resolvedPath, filePath),
|
||||
ast,
|
||||
content,
|
||||
allASTs,
|
||||
)
|
||||
|
||||
const fileData = createFileData({
|
||||
lines: content.split("\n"),
|
||||
hash: md5(content),
|
||||
size: content.length,
|
||||
lastModified: Date.now(),
|
||||
})
|
||||
|
||||
await storage.setFile(filePath, fileData)
|
||||
await storage.setAST(filePath, ast)
|
||||
await storage.setMeta(filePath, meta)
|
||||
|
||||
analyzed++
|
||||
onProgress?.("analyzing", analyzed, allASTs.size, filePath)
|
||||
|
||||
if (analyzed % 50 === 0) {
|
||||
process.stdout.write(
|
||||
`\r Analyzed ${String(analyzed)}/${String(allASTs.size)} files...`,
|
||||
)
|
||||
}
|
||||
}
|
||||
console.warn(`\r Analyzed ${String(analyzed)} files \n`)
|
||||
|
||||
console.warn("🏗️ Building indexes...")
|
||||
onProgress?.("storing", 0, 2)
|
||||
const symbolIndex = indexBuilder.buildSymbolIndex(allASTs)
|
||||
const depsGraph = indexBuilder.buildDepsGraph(allASTs)
|
||||
|
||||
await storage.setSymbolIndex(symbolIndex)
|
||||
await storage.setDepsGraph(depsGraph)
|
||||
onProgress?.("storing", 2, 2)
|
||||
|
||||
const duration = Date.now() - startTime
|
||||
const durationSec = (duration / 1000).toFixed(2)
|
||||
|
||||
console.warn(`✅ Indexing complete in ${durationSec}s`)
|
||||
console.warn(` Files indexed: ${String(parsed)}`)
|
||||
console.warn(` Files skipped: ${String(skipped)}`)
|
||||
console.warn(`\n✅ Indexing complete in ${durationSec}s`)
|
||||
console.warn(` Files scanned: ${String(stats.filesScanned)}`)
|
||||
console.warn(` Files parsed: ${String(stats.filesParsed)}`)
|
||||
console.warn(` Parse errors: ${String(stats.parseErrors)}`)
|
||||
console.warn(` Symbols: ${String(symbolIndex.size)}`)
|
||||
|
||||
if (errors.length > 0) {
|
||||
console.warn(`\n⚠️ ${String(errors.length)} errors occurred:`)
|
||||
for (const error of errors.slice(0, 5)) {
|
||||
console.warn(` - ${error}`)
|
||||
}
|
||||
if (errors.length > 5) {
|
||||
console.warn(` ... and ${String(errors.length - 5)} more`)
|
||||
}
|
||||
}
|
||||
|
||||
return {
|
||||
success: true,
|
||||
filesIndexed: parsed,
|
||||
filesSkipped: skipped,
|
||||
errors,
|
||||
duration,
|
||||
filesIndexed: stats.filesParsed,
|
||||
filesSkipped: stats.filesScanned - stats.filesParsed,
|
||||
errors: [],
|
||||
duration: stats.timeMs,
|
||||
}
|
||||
} catch (error) {
|
||||
const message = error instanceof Error ? error.message : String(error)
|
||||
@@ -229,22 +146,3 @@ export async function executeIndex(
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* Get language from file extension.
|
||||
*/
|
||||
function getLanguage(filePath: string): Language | null {
|
||||
const ext = path.extname(filePath).toLowerCase()
|
||||
switch (ext) {
|
||||
case ".ts":
|
||||
return "ts"
|
||||
case ".tsx":
|
||||
return "tsx"
|
||||
case ".js":
|
||||
return "js"
|
||||
case ".jsx":
|
||||
return "jsx"
|
||||
default:
|
||||
return null
|
||||
}
|
||||
}
|
||||
|
||||
@@ -1,26 +1,6 @@
|
||||
import type { ChatMessage } from "../value-objects/ChatMessage.js"
|
||||
import type { ToolCall } from "../value-objects/ToolCall.js"
|
||||
|
||||
/**
|
||||
* Tool parameter definition for LLM.
|
||||
*/
|
||||
export interface ToolParameter {
|
||||
name: string
|
||||
type: "string" | "number" | "boolean" | "array" | "object"
|
||||
description: string
|
||||
required: boolean
|
||||
enum?: string[]
|
||||
}
|
||||
|
||||
/**
|
||||
* Tool definition for LLM function calling.
|
||||
*/
|
||||
export interface ToolDef {
|
||||
name: string
|
||||
description: string
|
||||
parameters: ToolParameter[]
|
||||
}
|
||||
|
||||
/**
|
||||
* Response from LLM.
|
||||
*/
|
||||
@@ -42,12 +22,16 @@ export interface LLMResponse {
|
||||
/**
|
||||
* LLM client service interface (port).
|
||||
* Abstracts the LLM provider.
|
||||
*
|
||||
* Tool definitions should be included in the system prompt as XML format,
|
||||
* not passed as a separate parameter.
|
||||
*/
|
||||
export interface ILLMClient {
|
||||
/**
|
||||
* Send messages to LLM and get response.
|
||||
* Tool calls are extracted from the response content using XML parsing.
|
||||
*/
|
||||
chat(messages: ChatMessage[], tools?: ToolDef[]): Promise<LLMResponse>
|
||||
chat(messages: ChatMessage[]): Promise<LLMResponse>
|
||||
|
||||
/**
|
||||
* Count tokens in text.
|
||||
|
||||
@@ -1,15 +1,10 @@
|
||||
import { type Message, Ollama, type Tool } from "ollama"
|
||||
import type {
|
||||
ILLMClient,
|
||||
LLMResponse,
|
||||
ToolDef,
|
||||
ToolParameter,
|
||||
} from "../../domain/services/ILLMClient.js"
|
||||
import { type Message, Ollama } from "ollama"
|
||||
import type { ILLMClient, LLMResponse } from "../../domain/services/ILLMClient.js"
|
||||
import type { ChatMessage } from "../../domain/value-objects/ChatMessage.js"
|
||||
import { createToolCall, type ToolCall } from "../../domain/value-objects/ToolCall.js"
|
||||
import type { LLMConfig } from "../../shared/constants/config.js"
|
||||
import { IpuaroError } from "../../shared/errors/IpuaroError.js"
|
||||
import { estimateTokens } from "../../shared/utils/tokens.js"
|
||||
import { parseToolCalls } from "./ResponseParser.js"
|
||||
|
||||
/**
|
||||
* Ollama LLM client implementation.
|
||||
@@ -35,19 +30,18 @@ export class OllamaClient implements ILLMClient {
|
||||
|
||||
/**
|
||||
* Send messages to LLM and get response.
|
||||
* Tool definitions should be included in the system prompt as XML format.
|
||||
*/
|
||||
async chat(messages: ChatMessage[], tools?: ToolDef[]): Promise<LLMResponse> {
|
||||
async chat(messages: ChatMessage[]): Promise<LLMResponse> {
|
||||
const startTime = Date.now()
|
||||
this.abortController = new AbortController()
|
||||
|
||||
try {
|
||||
const ollamaMessages = this.convertMessages(messages)
|
||||
const ollamaTools = tools ? this.convertTools(tools) : undefined
|
||||
|
||||
const response = await this.client.chat({
|
||||
model: this.model,
|
||||
messages: ollamaMessages,
|
||||
tools: ollamaTools,
|
||||
options: {
|
||||
temperature: this.temperature,
|
||||
},
|
||||
@@ -55,15 +49,15 @@ export class OllamaClient implements ILLMClient {
|
||||
})
|
||||
|
||||
const timeMs = Date.now() - startTime
|
||||
const toolCalls = this.extractToolCalls(response.message)
|
||||
const parsed = parseToolCalls(response.message.content)
|
||||
|
||||
return {
|
||||
content: response.message.content,
|
||||
toolCalls,
|
||||
content: parsed.content,
|
||||
toolCalls: parsed.toolCalls,
|
||||
tokens: response.eval_count ?? estimateTokens(response.message.content),
|
||||
timeMs,
|
||||
truncated: false,
|
||||
stopReason: this.determineStopReason(response, toolCalls),
|
||||
stopReason: this.determineStopReason(response, parsed.toolCalls),
|
||||
}
|
||||
} catch (error) {
|
||||
if (error instanceof Error && error.name === "AbortError") {
|
||||
@@ -205,69 +199,12 @@ export class OllamaClient implements ILLMClient {
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* Convert ToolDef array to Ollama Tool format.
|
||||
*/
|
||||
private convertTools(tools: ToolDef[]): Tool[] {
|
||||
return tools.map(
|
||||
(tool): Tool => ({
|
||||
type: "function",
|
||||
function: {
|
||||
name: tool.name,
|
||||
description: tool.description,
|
||||
parameters: {
|
||||
type: "object",
|
||||
properties: this.convertParameters(tool.parameters),
|
||||
required: tool.parameters.filter((p) => p.required).map((p) => p.name),
|
||||
},
|
||||
},
|
||||
}),
|
||||
)
|
||||
}
|
||||
|
||||
/**
|
||||
* Convert ToolParameter array to JSON Schema properties.
|
||||
*/
|
||||
private convertParameters(
|
||||
params: ToolParameter[],
|
||||
): Record<string, { type: string; description: string; enum?: string[] }> {
|
||||
const properties: Record<string, { type: string; description: string; enum?: string[] }> =
|
||||
{}
|
||||
|
||||
for (const param of params) {
|
||||
properties[param.name] = {
|
||||
type: param.type,
|
||||
description: param.description,
|
||||
...(param.enum && { enum: param.enum }),
|
||||
}
|
||||
}
|
||||
|
||||
return properties
|
||||
}
|
||||
|
||||
/**
|
||||
* Extract tool calls from Ollama response message.
|
||||
*/
|
||||
private extractToolCalls(message: Message): ToolCall[] {
|
||||
if (!message.tool_calls || message.tool_calls.length === 0) {
|
||||
return []
|
||||
}
|
||||
|
||||
return message.tool_calls.map((tc, index) =>
|
||||
createToolCall(
|
||||
`call_${String(Date.now())}_${String(index)}`,
|
||||
tc.function.name,
|
||||
tc.function.arguments,
|
||||
),
|
||||
)
|
||||
}
|
||||
|
||||
/**
|
||||
* Determine stop reason from response.
|
||||
*/
|
||||
private determineStopReason(
|
||||
response: { done_reason?: string },
|
||||
toolCalls: ToolCall[],
|
||||
toolCalls: { name: string; params: Record<string, unknown> }[],
|
||||
): "end" | "length" | "tool_use" {
|
||||
if (toolCalls.length > 0) {
|
||||
return "tool_use"
|
||||
|
||||
@@ -27,9 +27,41 @@ const TOOL_CALL_REGEX = /<tool_call\s+name\s*=\s*"([^"]+)">([\s\S]*?)<\/tool_cal
|
||||
const PARAM_REGEX_NAMED = /<param\s+name\s*=\s*"([^"]+)">([\s\S]*?)<\/param>/gi
|
||||
const PARAM_REGEX_ELEMENT = /<([a-z_][a-z0-9_]*)>([\s\S]*?)<\/\1>/gi
|
||||
|
||||
/**
|
||||
* CDATA section pattern.
|
||||
* Matches: <![CDATA[...]]>
|
||||
*/
|
||||
const CDATA_REGEX = /<!\[CDATA\[([\s\S]*?)\]\]>/g
|
||||
|
||||
/**
|
||||
* Valid tool names.
|
||||
* Used for validation to catch typos or hallucinations.
|
||||
*/
|
||||
const VALID_TOOL_NAMES = new Set([
|
||||
"get_lines",
|
||||
"get_function",
|
||||
"get_class",
|
||||
"get_structure",
|
||||
"edit_lines",
|
||||
"create_file",
|
||||
"delete_file",
|
||||
"find_references",
|
||||
"find_definition",
|
||||
"get_dependencies",
|
||||
"get_dependents",
|
||||
"get_complexity",
|
||||
"get_todos",
|
||||
"git_status",
|
||||
"git_diff",
|
||||
"git_commit",
|
||||
"run_command",
|
||||
"run_tests",
|
||||
])
|
||||
|
||||
/**
|
||||
* Parse tool calls from LLM response text.
|
||||
* Supports XML format: <tool_call name="get_lines"><path>src/index.ts</path></tool_call>
|
||||
* Validates tool names and provides helpful error messages.
|
||||
*/
|
||||
export function parseToolCalls(response: string): ParsedResponse {
|
||||
const toolCalls: ToolCall[] = []
|
||||
@@ -41,6 +73,13 @@ export function parseToolCalls(response: string): ParsedResponse {
|
||||
for (const match of matches) {
|
||||
const [fullMatch, toolName, paramsXml] = match
|
||||
|
||||
if (!VALID_TOOL_NAMES.has(toolName)) {
|
||||
parseErrors.push(
|
||||
`Unknown tool "${toolName}". Valid tools: ${[...VALID_TOOL_NAMES].join(", ")}`,
|
||||
)
|
||||
continue
|
||||
}
|
||||
|
||||
try {
|
||||
const params = parseParameters(paramsXml)
|
||||
const toolCall = createToolCall(
|
||||
@@ -91,10 +130,16 @@ function parseParameters(xml: string): Record<string, unknown> {
|
||||
|
||||
/**
|
||||
* Parse a value string to appropriate type.
|
||||
* Supports CDATA sections for multiline content.
|
||||
*/
|
||||
function parseValue(value: string): unknown {
|
||||
const trimmed = value.trim()
|
||||
|
||||
const cdataMatches = [...trimmed.matchAll(CDATA_REGEX)]
|
||||
if (cdataMatches.length > 0 && cdataMatches[0][1] !== undefined) {
|
||||
return cdataMatches[0][1]
|
||||
}
|
||||
|
||||
if (trimmed === "true") {
|
||||
return true
|
||||
}
|
||||
|
||||
@@ -23,37 +23,67 @@ export const SYSTEM_PROMPT = `You are ipuaro, a local AI code assistant speciali
|
||||
3. **Safety**: Confirm destructive operations. Never execute dangerous commands.
|
||||
4. **Efficiency**: Minimize context usage. Request only necessary code sections.
|
||||
|
||||
## Tool Calling Format
|
||||
|
||||
When you need to use a tool, format your call as XML:
|
||||
|
||||
<tool_call name="tool_name">
|
||||
<param_name>value</param_name>
|
||||
<another_param>value</another_param>
|
||||
</tool_call>
|
||||
|
||||
You can call multiple tools in one response. Always wait for tool results before making conclusions.
|
||||
|
||||
**Examples:**
|
||||
|
||||
<tool_call name="get_lines">
|
||||
<path>src/index.ts</path>
|
||||
<start>1</start>
|
||||
<end>50</end>
|
||||
</tool_call>
|
||||
|
||||
<tool_call name="edit_lines">
|
||||
<path>src/utils.ts</path>
|
||||
<start>10</start>
|
||||
<end>15</end>
|
||||
<content>const newCode = "hello";</content>
|
||||
</tool_call>
|
||||
|
||||
<tool_call name="find_references">
|
||||
<symbol>getUserById</symbol>
|
||||
</tool_call>
|
||||
|
||||
## Available Tools
|
||||
|
||||
### Reading Tools
|
||||
- \`get_lines\`: Get specific lines from a file
|
||||
- \`get_function\`: Get a function by name
|
||||
- \`get_class\`: Get a class by name
|
||||
- \`get_structure\`: Get project directory structure
|
||||
- \`get_lines(path, start?, end?)\`: Get specific lines from a file
|
||||
- \`get_function(path, name)\`: Get a function by name
|
||||
- \`get_class(path, name)\`: Get a class by name
|
||||
- \`get_structure(path?, depth?)\`: Get project directory structure
|
||||
|
||||
### Editing Tools (require confirmation)
|
||||
- \`edit_lines\`: Replace specific lines in a file
|
||||
- \`create_file\`: Create a new file
|
||||
- \`delete_file\`: Delete a file
|
||||
- \`edit_lines(path, start, end, content)\`: Replace specific lines in a file
|
||||
- \`create_file(path, content)\`: Create a new file
|
||||
- \`delete_file(path)\`: Delete a file
|
||||
|
||||
### Search Tools
|
||||
- \`find_references\`: Find all usages of a symbol
|
||||
- \`find_definition\`: Find where a symbol is defined
|
||||
- \`find_references(symbol, path?)\`: Find all usages of a symbol
|
||||
- \`find_definition(symbol)\`: Find where a symbol is defined
|
||||
|
||||
### Analysis Tools
|
||||
- \`get_dependencies\`: Get files this file imports
|
||||
- \`get_dependents\`: Get files that import this file
|
||||
- \`get_complexity\`: Get complexity metrics
|
||||
- \`get_todos\`: Find TODO/FIXME comments
|
||||
- \`get_dependencies(path)\`: Get files this file imports
|
||||
- \`get_dependents(path)\`: Get files that import this file
|
||||
- \`get_complexity(path?, limit?)\`: Get complexity metrics
|
||||
- \`get_todos(path?, type?)\`: Find TODO/FIXME comments
|
||||
|
||||
### Git Tools
|
||||
- \`git_status\`: Get repository status
|
||||
- \`git_diff\`: Get uncommitted changes
|
||||
- \`git_commit\`: Create a commit (requires confirmation)
|
||||
- \`git_status()\`: Get repository status
|
||||
- \`git_diff(path?, staged?)\`: Get uncommitted changes
|
||||
- \`git_commit(message, files?)\`: Create a commit (requires confirmation)
|
||||
|
||||
### Run Tools
|
||||
- \`run_command\`: Execute a shell command (security checked)
|
||||
- \`run_tests\`: Run the test suite
|
||||
- \`run_command(command, timeout?)\`: Execute a shell command (security checked)
|
||||
- \`run_tests(path?, filter?, watch?)\`: Run the test suite
|
||||
|
||||
## Response Guidelines
|
||||
|
||||
|
||||
@@ -1,4 +1,4 @@
|
||||
import type { ToolDef } from "../../domain/services/ILLMClient.js"
|
||||
import type { ToolDef } from "../../shared/types/tool-definitions.js"
|
||||
|
||||
/**
|
||||
* Tool definitions for ipuaro LLM.
|
||||
|
||||
@@ -76,6 +76,14 @@ export const UndoConfigSchema = z.object({
|
||||
*/
|
||||
export const EditConfigSchema = z.object({
|
||||
autoApply: z.boolean().default(false),
|
||||
syntaxHighlight: z.boolean().default(true),
|
||||
})
|
||||
|
||||
/**
|
||||
* Input configuration schema.
|
||||
*/
|
||||
export const InputConfigSchema = z.object({
|
||||
multiline: z.union([z.boolean(), z.literal("auto")]).default(false),
|
||||
})
|
||||
|
||||
/**
|
||||
@@ -88,6 +96,7 @@ export const ConfigSchema = z.object({
|
||||
watchdog: WatchdogConfigSchema.default({}),
|
||||
undo: UndoConfigSchema.default({}),
|
||||
edit: EditConfigSchema.default({}),
|
||||
input: InputConfigSchema.default({}),
|
||||
})
|
||||
|
||||
/**
|
||||
@@ -100,6 +109,7 @@ export type ProjectConfig = z.infer<typeof ProjectConfigSchema>
|
||||
export type WatchdogConfig = z.infer<typeof WatchdogConfigSchema>
|
||||
export type UndoConfig = z.infer<typeof UndoConfigSchema>
|
||||
export type EditConfig = z.infer<typeof EditConfigSchema>
|
||||
export type InputConfig = z.infer<typeof InputConfigSchema>
|
||||
|
||||
/**
|
||||
* Default configuration.
|
||||
|
||||
@@ -26,6 +26,9 @@ export type ErrorChoice = "retry" | "skip" | "abort"
|
||||
// Re-export ErrorOption for convenience
|
||||
export type { ErrorOption } from "../errors/IpuaroError.js"
|
||||
|
||||
// Re-export tool definition types
|
||||
export type { ToolDef, ToolParameter } from "./tool-definitions.js"
|
||||
|
||||
/**
|
||||
* Project structure node.
|
||||
*/
|
||||
|
||||
21
packages/ipuaro/src/shared/types/tool-definitions.ts
Normal file
21
packages/ipuaro/src/shared/types/tool-definitions.ts
Normal file
@@ -0,0 +1,21 @@
|
||||
/**
|
||||
* Tool parameter definition for LLM prompts.
|
||||
* Used to describe tools in system prompts.
|
||||
*/
|
||||
export interface ToolParameter {
|
||||
name: string
|
||||
type: "string" | "number" | "boolean" | "array" | "object"
|
||||
description: string
|
||||
required: boolean
|
||||
enum?: string[]
|
||||
}
|
||||
|
||||
/**
|
||||
* Tool definition for LLM prompts.
|
||||
* Used to describe available tools in the system prompt.
|
||||
*/
|
||||
export interface ToolDef {
|
||||
name: string
|
||||
description: string
|
||||
parameters: ToolParameter[]
|
||||
}
|
||||
@@ -9,12 +9,14 @@ import type { ILLMClient } from "../domain/services/ILLMClient.js"
|
||||
import type { ISessionStorage } from "../domain/services/ISessionStorage.js"
|
||||
import type { IStorage } from "../domain/services/IStorage.js"
|
||||
import type { DiffInfo } from "../domain/services/ITool.js"
|
||||
import type { ErrorChoice } from "../shared/types/index.js"
|
||||
import type { ErrorOption } from "../shared/errors/IpuaroError.js"
|
||||
import type { IToolRegistry } from "../application/interfaces/IToolRegistry.js"
|
||||
import type { ConfirmationResult } from "../application/use-cases/ExecuteTool.js"
|
||||
import type { ProjectStructure } from "../infrastructure/llm/prompts.js"
|
||||
import { Chat, Input, StatusBar } from "./components/index.js"
|
||||
import { Chat, ConfirmDialog, Input, StatusBar } from "./components/index.js"
|
||||
import { type CommandResult, useCommands, useHotkeys, useSession } from "./hooks/index.js"
|
||||
import type { AppProps, BranchInfo } from "./types.js"
|
||||
import type { ConfirmChoice } from "../shared/types/index.js"
|
||||
|
||||
export interface AppDependencies {
|
||||
storage: IStorage
|
||||
@@ -27,6 +29,8 @@ export interface AppDependencies {
|
||||
export interface ExtendedAppProps extends AppProps {
|
||||
deps: AppDependencies
|
||||
onExit?: () => void
|
||||
multiline?: boolean | "auto"
|
||||
syntaxHighlight?: boolean
|
||||
}
|
||||
|
||||
function LoadingScreen(): React.JSX.Element {
|
||||
@@ -48,12 +52,14 @@ function ErrorScreen({ error }: { error: Error }): React.JSX.Element {
|
||||
)
|
||||
}
|
||||
|
||||
async function handleConfirmationDefault(_message: string, _diff?: DiffInfo): Promise<boolean> {
|
||||
return Promise.resolve(true)
|
||||
async function handleErrorDefault(_error: Error): Promise<ErrorOption> {
|
||||
return Promise.resolve("skip")
|
||||
}
|
||||
|
||||
async function handleErrorDefault(_error: Error): Promise<ErrorChoice> {
|
||||
return Promise.resolve("skip")
|
||||
interface PendingConfirmation {
|
||||
message: string
|
||||
diff?: DiffInfo
|
||||
resolve: (result: boolean | ConfirmationResult) => void
|
||||
}
|
||||
|
||||
export function App({
|
||||
@@ -61,6 +67,8 @@ export function App({
|
||||
autoApply: initialAutoApply = false,
|
||||
deps,
|
||||
onExit,
|
||||
multiline = false,
|
||||
syntaxHighlight = true,
|
||||
}: ExtendedAppProps): React.JSX.Element {
|
||||
const { exit } = useApp()
|
||||
|
||||
@@ -68,9 +76,40 @@ export function App({
|
||||
const [sessionTime, setSessionTime] = useState("0m")
|
||||
const [autoApply, setAutoApply] = useState(initialAutoApply)
|
||||
const [commandResult, setCommandResult] = useState<CommandResult | null>(null)
|
||||
const [pendingConfirmation, setPendingConfirmation] = useState<PendingConfirmation | null>(null)
|
||||
|
||||
const projectName = projectPath.split("/").pop() ?? "unknown"
|
||||
|
||||
const handleConfirmation = useCallback(
|
||||
async (message: string, diff?: DiffInfo): Promise<boolean | ConfirmationResult> => {
|
||||
return new Promise((resolve) => {
|
||||
setPendingConfirmation({ message, diff, resolve })
|
||||
})
|
||||
},
|
||||
[],
|
||||
)
|
||||
|
||||
const handleConfirmSelect = useCallback(
|
||||
(choice: ConfirmChoice, editedContent?: string[]) => {
|
||||
if (!pendingConfirmation) {
|
||||
return
|
||||
}
|
||||
|
||||
if (choice === "apply") {
|
||||
if (editedContent) {
|
||||
pendingConfirmation.resolve({ confirmed: true, editedContent })
|
||||
} else {
|
||||
pendingConfirmation.resolve(true)
|
||||
}
|
||||
} else {
|
||||
pendingConfirmation.resolve(false)
|
||||
}
|
||||
|
||||
setPendingConfirmation(null)
|
||||
},
|
||||
[pendingConfirmation],
|
||||
)
|
||||
|
||||
const { session, messages, status, isLoading, error, sendMessage, undo, clearHistory, abort } =
|
||||
useSession(
|
||||
{
|
||||
@@ -84,18 +123,16 @@ export function App({
|
||||
},
|
||||
{
|
||||
autoApply,
|
||||
onConfirmation: handleConfirmationDefault,
|
||||
onConfirmation: handleConfirmation,
|
||||
onError: handleErrorDefault,
|
||||
},
|
||||
)
|
||||
|
||||
const reindex = useCallback(async (): Promise<void> => {
|
||||
/*
|
||||
* TODO: Implement full reindex via IndexProject use case
|
||||
* For now, this is a placeholder
|
||||
*/
|
||||
await Promise.resolve()
|
||||
}, [])
|
||||
const { IndexProject } = await import("../application/use-cases/IndexProject.js")
|
||||
const indexProject = new IndexProject(deps.storage, projectPath)
|
||||
await indexProject.execute(projectPath)
|
||||
}, [deps.storage, projectPath])
|
||||
|
||||
const { executeCommand, isCommand } = useCommands(
|
||||
{
|
||||
@@ -181,7 +218,7 @@ export function App({
|
||||
return <ErrorScreen error={error} />
|
||||
}
|
||||
|
||||
const isInputDisabled = status === "thinking" || status === "tool_call"
|
||||
const isInputDisabled = status === "thinking" || status === "tool_call" || !!pendingConfirmation
|
||||
|
||||
return (
|
||||
<Box flexDirection="column" height="100%">
|
||||
@@ -205,11 +242,33 @@ export function App({
|
||||
</Text>
|
||||
</Box>
|
||||
)}
|
||||
{pendingConfirmation && (
|
||||
<ConfirmDialog
|
||||
message={pendingConfirmation.message}
|
||||
diff={
|
||||
pendingConfirmation.diff
|
||||
? {
|
||||
filePath: pendingConfirmation.diff.filePath,
|
||||
oldLines: pendingConfirmation.diff.oldLines,
|
||||
newLines: pendingConfirmation.diff.newLines,
|
||||
startLine: pendingConfirmation.diff.startLine,
|
||||
}
|
||||
: undefined
|
||||
}
|
||||
onSelect={handleConfirmSelect}
|
||||
editableContent={pendingConfirmation.diff?.newLines}
|
||||
syntaxHighlight={syntaxHighlight}
|
||||
/>
|
||||
)}
|
||||
<Input
|
||||
onSubmit={handleSubmit}
|
||||
history={session?.inputHistory ?? []}
|
||||
disabled={isInputDisabled}
|
||||
placeholder={isInputDisabled ? "Processing..." : "Type a message..."}
|
||||
storage={deps.storage}
|
||||
projectRoot={projectPath}
|
||||
autocompleteEnabled={true}
|
||||
multiline={multiline}
|
||||
/>
|
||||
</Box>
|
||||
)
|
||||
|
||||
@@ -1,19 +1,25 @@
|
||||
/**
|
||||
* ConfirmDialog component for TUI.
|
||||
* Displays a confirmation dialog with [Y] Apply / [N] Cancel / [E] Edit options.
|
||||
* Supports inline editing when user selects Edit.
|
||||
*/
|
||||
|
||||
import { Box, Text, useInput } from "ink"
|
||||
import React, { useState } from "react"
|
||||
import React, { useCallback, useState } from "react"
|
||||
import type { ConfirmChoice } from "../../shared/types/index.js"
|
||||
import { DiffView, type DiffViewProps } from "./DiffView.js"
|
||||
import { EditableContent } from "./EditableContent.js"
|
||||
|
||||
export interface ConfirmDialogProps {
|
||||
message: string
|
||||
diff?: DiffViewProps
|
||||
onSelect: (choice: ConfirmChoice) => void
|
||||
onSelect: (choice: ConfirmChoice, editedContent?: string[]) => void
|
||||
editableContent?: string[]
|
||||
syntaxHighlight?: boolean
|
||||
}
|
||||
|
||||
type DialogMode = "confirm" | "edit"
|
||||
|
||||
function ChoiceButton({
|
||||
hotkey,
|
||||
label,
|
||||
@@ -32,26 +38,66 @@ function ChoiceButton({
|
||||
)
|
||||
}
|
||||
|
||||
export function ConfirmDialog({ message, diff, onSelect }: ConfirmDialogProps): React.JSX.Element {
|
||||
export function ConfirmDialog({
|
||||
message,
|
||||
diff,
|
||||
onSelect,
|
||||
editableContent,
|
||||
syntaxHighlight = false,
|
||||
}: ConfirmDialogProps): React.JSX.Element {
|
||||
const [mode, setMode] = useState<DialogMode>("confirm")
|
||||
const [selected, setSelected] = useState<ConfirmChoice | null>(null)
|
||||
|
||||
useInput((input, key) => {
|
||||
const lowerInput = input.toLowerCase()
|
||||
const linesToEdit = editableContent ?? diff?.newLines ?? []
|
||||
const canEdit = linesToEdit.length > 0
|
||||
|
||||
if (lowerInput === "y") {
|
||||
const handleEditSubmit = useCallback(
|
||||
(editedLines: string[]) => {
|
||||
setSelected("apply")
|
||||
onSelect("apply")
|
||||
} else if (lowerInput === "n") {
|
||||
setSelected("cancel")
|
||||
onSelect("cancel")
|
||||
} else if (lowerInput === "e") {
|
||||
setSelected("edit")
|
||||
onSelect("edit")
|
||||
} else if (key.escape) {
|
||||
setSelected("cancel")
|
||||
onSelect("cancel")
|
||||
}
|
||||
})
|
||||
onSelect("apply", editedLines)
|
||||
},
|
||||
[onSelect],
|
||||
)
|
||||
|
||||
const handleEditCancel = useCallback(() => {
|
||||
setMode("confirm")
|
||||
setSelected(null)
|
||||
}, [])
|
||||
|
||||
useInput(
|
||||
(input, key) => {
|
||||
if (mode === "edit") {
|
||||
return
|
||||
}
|
||||
|
||||
const lowerInput = input.toLowerCase()
|
||||
|
||||
if (lowerInput === "y") {
|
||||
setSelected("apply")
|
||||
onSelect("apply")
|
||||
} else if (lowerInput === "n") {
|
||||
setSelected("cancel")
|
||||
onSelect("cancel")
|
||||
} else if (lowerInput === "e" && canEdit) {
|
||||
setSelected("edit")
|
||||
setMode("edit")
|
||||
} else if (key.escape) {
|
||||
setSelected("cancel")
|
||||
onSelect("cancel")
|
||||
}
|
||||
},
|
||||
{ isActive: mode === "confirm" },
|
||||
)
|
||||
|
||||
if (mode === "edit") {
|
||||
return (
|
||||
<EditableContent
|
||||
lines={linesToEdit}
|
||||
onSubmit={handleEditSubmit}
|
||||
onCancel={handleEditCancel}
|
||||
/>
|
||||
)
|
||||
}
|
||||
|
||||
return (
|
||||
<Box
|
||||
@@ -69,14 +115,22 @@ export function ConfirmDialog({ message, diff, onSelect }: ConfirmDialogProps):
|
||||
|
||||
{diff && (
|
||||
<Box marginBottom={1}>
|
||||
<DiffView {...diff} />
|
||||
<DiffView {...diff} syntaxHighlight={syntaxHighlight} />
|
||||
</Box>
|
||||
)}
|
||||
|
||||
<Box gap={2}>
|
||||
<ChoiceButton hotkey="Y" label="Apply" isSelected={selected === "apply"} />
|
||||
<ChoiceButton hotkey="N" label="Cancel" isSelected={selected === "cancel"} />
|
||||
<ChoiceButton hotkey="E" label="Edit" isSelected={selected === "edit"} />
|
||||
{canEdit ? (
|
||||
<ChoiceButton hotkey="E" label="Edit" isSelected={selected === "edit"} />
|
||||
) : (
|
||||
<Box>
|
||||
<Text color="gray" dimColor>
|
||||
[E] Edit (disabled)
|
||||
</Text>
|
||||
</Box>
|
||||
)}
|
||||
</Box>
|
||||
</Box>
|
||||
)
|
||||
|
||||
@@ -5,12 +5,15 @@
|
||||
|
||||
import { Box, Text } from "ink"
|
||||
import type React from "react"
|
||||
import { detectLanguage, highlightLine, type Language } from "../utils/syntax-highlighter.js"
|
||||
|
||||
export interface DiffViewProps {
|
||||
filePath: string
|
||||
oldLines: string[]
|
||||
newLines: string[]
|
||||
startLine: number
|
||||
language?: Language
|
||||
syntaxHighlight?: boolean
|
||||
}
|
||||
|
||||
interface DiffLine {
|
||||
@@ -97,20 +100,37 @@ function formatLineNumber(num: number | undefined, width: number): string {
|
||||
function DiffLine({
|
||||
line,
|
||||
lineNumberWidth,
|
||||
language,
|
||||
syntaxHighlight,
|
||||
}: {
|
||||
line: DiffLine
|
||||
lineNumberWidth: number
|
||||
language?: Language
|
||||
syntaxHighlight?: boolean
|
||||
}): React.JSX.Element {
|
||||
const prefix = getLinePrefix(line)
|
||||
const color = getLineColor(line)
|
||||
const lineNum = formatLineNumber(line.lineNumber, lineNumberWidth)
|
||||
|
||||
const shouldHighlight = syntaxHighlight && language && line.type === "add"
|
||||
|
||||
return (
|
||||
<Box>
|
||||
<Text color="gray">{lineNum} </Text>
|
||||
<Text color={color}>
|
||||
{prefix} {line.content}
|
||||
</Text>
|
||||
{shouldHighlight ? (
|
||||
<Box>
|
||||
<Text color={color}>{prefix} </Text>
|
||||
{highlightLine(line.content, language).map((token, idx) => (
|
||||
<Text key={idx} color={token.color}>
|
||||
{token.text}
|
||||
</Text>
|
||||
))}
|
||||
</Box>
|
||||
) : (
|
||||
<Text color={color}>
|
||||
{prefix} {line.content}
|
||||
</Text>
|
||||
)}
|
||||
</Box>
|
||||
)
|
||||
}
|
||||
@@ -166,6 +186,8 @@ export function DiffView({
|
||||
oldLines,
|
||||
newLines,
|
||||
startLine,
|
||||
language,
|
||||
syntaxHighlight = false,
|
||||
}: DiffViewProps): React.JSX.Element {
|
||||
const diffLines = computeDiff(oldLines, newLines, startLine)
|
||||
const endLine = startLine + newLines.length - 1
|
||||
@@ -174,6 +196,8 @@ export function DiffView({
|
||||
const additions = diffLines.filter((l) => l.type === "add").length
|
||||
const deletions = diffLines.filter((l) => l.type === "remove").length
|
||||
|
||||
const detectedLanguage = language ?? detectLanguage(filePath)
|
||||
|
||||
return (
|
||||
<Box flexDirection="column" paddingX={1}>
|
||||
<DiffHeader filePath={filePath} startLine={startLine} endLine={endLine} />
|
||||
@@ -183,6 +207,8 @@ export function DiffView({
|
||||
key={`${line.type}-${String(index)}`}
|
||||
line={line}
|
||||
lineNumberWidth={lineNumberWidth}
|
||||
language={detectedLanguage}
|
||||
syntaxHighlight={syntaxHighlight}
|
||||
/>
|
||||
))}
|
||||
</Box>
|
||||
|
||||
146
packages/ipuaro/src/tui/components/EditableContent.tsx
Normal file
146
packages/ipuaro/src/tui/components/EditableContent.tsx
Normal file
@@ -0,0 +1,146 @@
|
||||
/**
|
||||
* EditableContent component for TUI.
|
||||
* Displays editable multi-line text with line-by-line navigation.
|
||||
*/
|
||||
|
||||
import { Box, Text, useInput } from "ink"
|
||||
import TextInput from "ink-text-input"
|
||||
import React, { useCallback, useState } from "react"
|
||||
|
||||
export interface EditableContentProps {
|
||||
/** Initial lines to edit */
|
||||
lines: string[]
|
||||
/** Called when user finishes editing (Enter key) */
|
||||
onSubmit: (editedLines: string[]) => void
|
||||
/** Called when user cancels editing (Escape key) */
|
||||
onCancel: () => void
|
||||
/** Maximum visible lines before scrolling */
|
||||
maxVisibleLines?: number
|
||||
}
|
||||
|
||||
/**
|
||||
* EditableContent component.
|
||||
* Allows line-by-line editing of multi-line text.
|
||||
* - Up/Down: Navigate between lines
|
||||
* - Enter (on last line): Submit changes
|
||||
* - Ctrl+Enter: Submit changes from any line
|
||||
* - Escape: Cancel editing
|
||||
*/
|
||||
export function EditableContent({
|
||||
lines: initialLines,
|
||||
onSubmit,
|
||||
onCancel,
|
||||
maxVisibleLines = 20,
|
||||
}: EditableContentProps): React.JSX.Element {
|
||||
const [lines, setLines] = useState<string[]>(initialLines.length > 0 ? initialLines : [""])
|
||||
const [currentLineIndex, setCurrentLineIndex] = useState(0)
|
||||
const [currentLineValue, setCurrentLineValue] = useState(lines[0] ?? "")
|
||||
|
||||
const updateCurrentLine = useCallback(
|
||||
(value: string) => {
|
||||
const newLines = [...lines]
|
||||
newLines[currentLineIndex] = value
|
||||
setLines(newLines)
|
||||
setCurrentLineValue(value)
|
||||
},
|
||||
[lines, currentLineIndex],
|
||||
)
|
||||
|
||||
const handleLineSubmit = useCallback(() => {
|
||||
updateCurrentLine(currentLineValue)
|
||||
|
||||
if (currentLineIndex === lines.length - 1) {
|
||||
onSubmit(lines)
|
||||
} else {
|
||||
const nextIndex = currentLineIndex + 1
|
||||
setCurrentLineIndex(nextIndex)
|
||||
setCurrentLineValue(lines[nextIndex] ?? "")
|
||||
}
|
||||
}, [currentLineValue, currentLineIndex, lines, updateCurrentLine, onSubmit])
|
||||
|
||||
const handleMoveUp = useCallback(() => {
|
||||
if (currentLineIndex > 0) {
|
||||
updateCurrentLine(currentLineValue)
|
||||
const prevIndex = currentLineIndex - 1
|
||||
setCurrentLineIndex(prevIndex)
|
||||
setCurrentLineValue(lines[prevIndex] ?? "")
|
||||
}
|
||||
}, [currentLineIndex, currentLineValue, lines, updateCurrentLine])
|
||||
|
||||
const handleMoveDown = useCallback(() => {
|
||||
if (currentLineIndex < lines.length - 1) {
|
||||
updateCurrentLine(currentLineValue)
|
||||
const nextIndex = currentLineIndex + 1
|
||||
setCurrentLineIndex(nextIndex)
|
||||
setCurrentLineValue(lines[nextIndex] ?? "")
|
||||
}
|
||||
}, [currentLineIndex, currentLineValue, lines, updateCurrentLine])
|
||||
|
||||
const handleCtrlEnter = useCallback(() => {
|
||||
updateCurrentLine(currentLineValue)
|
||||
onSubmit(lines)
|
||||
}, [currentLineValue, lines, updateCurrentLine, onSubmit])
|
||||
|
||||
useInput(
|
||||
(input, key) => {
|
||||
if (key.escape) {
|
||||
onCancel()
|
||||
} else if (key.upArrow) {
|
||||
handleMoveUp()
|
||||
} else if (key.downArrow) {
|
||||
handleMoveDown()
|
||||
} else if (key.ctrl && key.return) {
|
||||
handleCtrlEnter()
|
||||
}
|
||||
},
|
||||
{ isActive: true },
|
||||
)
|
||||
|
||||
const startLine = Math.max(0, currentLineIndex - Math.floor(maxVisibleLines / 2))
|
||||
const endLine = Math.min(lines.length, startLine + maxVisibleLines)
|
||||
const visibleLines = lines.slice(startLine, endLine)
|
||||
|
||||
return (
|
||||
<Box flexDirection="column" borderStyle="round" borderColor="cyan" paddingX={1}>
|
||||
<Box marginBottom={1}>
|
||||
<Text color="cyan" bold>
|
||||
Edit Content (Line {currentLineIndex + 1}/{lines.length})
|
||||
</Text>
|
||||
</Box>
|
||||
|
||||
<Box flexDirection="column" marginBottom={1}>
|
||||
{visibleLines.map((line, idx) => {
|
||||
const actualIndex = startLine + idx
|
||||
const isCurrentLine = actualIndex === currentLineIndex
|
||||
|
||||
return (
|
||||
<Box key={actualIndex}>
|
||||
<Text color="gray" dimColor>
|
||||
{String(actualIndex + 1).padStart(3, " ")}:{" "}
|
||||
</Text>
|
||||
{isCurrentLine ? (
|
||||
<Box>
|
||||
<Text color="cyan">▶ </Text>
|
||||
<TextInput
|
||||
value={currentLineValue}
|
||||
onChange={setCurrentLineValue}
|
||||
onSubmit={handleLineSubmit}
|
||||
/>
|
||||
</Box>
|
||||
) : (
|
||||
<Text color={isCurrentLine ? "cyan" : "white"}>{line}</Text>
|
||||
)}
|
||||
</Box>
|
||||
)
|
||||
})}
|
||||
</Box>
|
||||
|
||||
<Box flexDirection="column" borderStyle="single" borderColor="gray" paddingX={1}>
|
||||
<Text dimColor>↑/↓: Navigate lines</Text>
|
||||
<Text dimColor>Enter: Next line / Submit (last line)</Text>
|
||||
<Text dimColor>Ctrl+Enter: Submit from any line</Text>
|
||||
<Text dimColor>Escape: Cancel</Text>
|
||||
</Box>
|
||||
</Box>
|
||||
)
|
||||
}
|
||||
@@ -5,7 +5,7 @@
|
||||
|
||||
import { Box, Text, useInput } from "ink"
|
||||
import React, { useState } from "react"
|
||||
import type { ErrorChoice } from "../../shared/types/index.js"
|
||||
import type { ErrorOption } from "../../shared/errors/IpuaroError.js"
|
||||
|
||||
export interface ErrorInfo {
|
||||
type: string
|
||||
@@ -15,7 +15,7 @@ export interface ErrorInfo {
|
||||
|
||||
export interface ErrorDialogProps {
|
||||
error: ErrorInfo
|
||||
onChoice: (choice: ErrorChoice) => void
|
||||
onChoice: (choice: ErrorOption) => void
|
||||
}
|
||||
|
||||
function ChoiceButton({
|
||||
@@ -49,7 +49,7 @@ function ChoiceButton({
|
||||
}
|
||||
|
||||
export function ErrorDialog({ error, onChoice }: ErrorDialogProps): React.JSX.Element {
|
||||
const [selected, setSelected] = useState<ErrorChoice | null>(null)
|
||||
const [selected, setSelected] = useState<ErrorOption | null>(null)
|
||||
|
||||
useInput((input, key) => {
|
||||
const lowerInput = input.toLowerCase()
|
||||
|
||||
@@ -6,12 +6,18 @@
|
||||
import { Box, Text, useInput } from "ink"
|
||||
import TextInput from "ink-text-input"
|
||||
import React, { useCallback, useState } from "react"
|
||||
import type { IStorage } from "../../domain/services/IStorage.js"
|
||||
import { useAutocomplete } from "../hooks/useAutocomplete.js"
|
||||
|
||||
export interface InputProps {
|
||||
onSubmit: (text: string) => void
|
||||
history: string[]
|
||||
disabled: boolean
|
||||
placeholder?: string
|
||||
storage?: IStorage
|
||||
projectRoot?: string
|
||||
autocompleteEnabled?: boolean
|
||||
multiline?: boolean | "auto"
|
||||
}
|
||||
|
||||
export function Input({
|
||||
@@ -19,15 +25,41 @@ export function Input({
|
||||
history,
|
||||
disabled,
|
||||
placeholder = "Type a message...",
|
||||
storage,
|
||||
projectRoot = "",
|
||||
autocompleteEnabled = true,
|
||||
multiline = false,
|
||||
}: InputProps): React.JSX.Element {
|
||||
const [value, setValue] = useState("")
|
||||
const [historyIndex, setHistoryIndex] = useState(-1)
|
||||
const [savedInput, setSavedInput] = useState("")
|
||||
const [lines, setLines] = useState<string[]>([""])
|
||||
const [currentLineIndex, setCurrentLineIndex] = useState(0)
|
||||
|
||||
const handleChange = useCallback((newValue: string) => {
|
||||
setValue(newValue)
|
||||
setHistoryIndex(-1)
|
||||
}, [])
|
||||
const isMultilineActive = multiline === true || (multiline === "auto" && lines.length > 1)
|
||||
|
||||
/*
|
||||
* Initialize autocomplete hook if storage is provided
|
||||
* Create a dummy storage object if storage is not provided (autocomplete will be disabled)
|
||||
*/
|
||||
const dummyStorage = {} as IStorage
|
||||
const autocomplete = useAutocomplete({
|
||||
storage: storage ?? dummyStorage,
|
||||
projectRoot,
|
||||
enabled: autocompleteEnabled && !!storage,
|
||||
})
|
||||
|
||||
const handleChange = useCallback(
|
||||
(newValue: string) => {
|
||||
setValue(newValue)
|
||||
setHistoryIndex(-1)
|
||||
// Update autocomplete suggestions as user types
|
||||
if (storage && autocompleteEnabled) {
|
||||
autocomplete.complete(newValue)
|
||||
}
|
||||
},
|
||||
[storage, autocompleteEnabled, autocomplete],
|
||||
)
|
||||
|
||||
const handleSubmit = useCallback(
|
||||
(text: string) => {
|
||||
@@ -36,63 +68,182 @@ export function Input({
|
||||
}
|
||||
onSubmit(text)
|
||||
setValue("")
|
||||
setLines([""])
|
||||
setCurrentLineIndex(0)
|
||||
setHistoryIndex(-1)
|
||||
setSavedInput("")
|
||||
autocomplete.reset()
|
||||
},
|
||||
[disabled, onSubmit],
|
||||
[disabled, onSubmit, autocomplete],
|
||||
)
|
||||
|
||||
const handleLineChange = useCallback(
|
||||
(newValue: string) => {
|
||||
const newLines = [...lines]
|
||||
newLines[currentLineIndex] = newValue
|
||||
setLines(newLines)
|
||||
setValue(newLines.join("\n"))
|
||||
},
|
||||
[lines, currentLineIndex],
|
||||
)
|
||||
|
||||
const handleAddLine = useCallback(() => {
|
||||
const newLines = [...lines]
|
||||
newLines.splice(currentLineIndex + 1, 0, "")
|
||||
setLines(newLines)
|
||||
setCurrentLineIndex(currentLineIndex + 1)
|
||||
setValue(newLines.join("\n"))
|
||||
}, [lines, currentLineIndex])
|
||||
|
||||
const handleMultilineSubmit = useCallback(() => {
|
||||
const fullText = lines.join("\n").trim()
|
||||
if (fullText) {
|
||||
handleSubmit(fullText)
|
||||
}
|
||||
}, [lines, handleSubmit])
|
||||
|
||||
const handleTabKey = useCallback(() => {
|
||||
if (storage && autocompleteEnabled && value.trim()) {
|
||||
const suggestions = autocomplete.suggestions
|
||||
if (suggestions.length > 0) {
|
||||
const completed = autocomplete.accept(value)
|
||||
setValue(completed)
|
||||
autocomplete.complete(completed)
|
||||
}
|
||||
}
|
||||
}, [storage, autocompleteEnabled, value, autocomplete])
|
||||
|
||||
const handleUpArrow = useCallback(() => {
|
||||
if (history.length > 0) {
|
||||
if (historyIndex === -1) {
|
||||
setSavedInput(value)
|
||||
}
|
||||
const newIndex =
|
||||
historyIndex === -1 ? history.length - 1 : Math.max(0, historyIndex - 1)
|
||||
setHistoryIndex(newIndex)
|
||||
setValue(history[newIndex] ?? "")
|
||||
autocomplete.reset()
|
||||
}
|
||||
}, [history, historyIndex, value, autocomplete])
|
||||
|
||||
const handleDownArrow = useCallback(() => {
|
||||
if (historyIndex === -1) {
|
||||
return
|
||||
}
|
||||
if (historyIndex >= history.length - 1) {
|
||||
setHistoryIndex(-1)
|
||||
setValue(savedInput)
|
||||
} else {
|
||||
const newIndex = historyIndex + 1
|
||||
setHistoryIndex(newIndex)
|
||||
setValue(history[newIndex] ?? "")
|
||||
}
|
||||
autocomplete.reset()
|
||||
}, [historyIndex, history, savedInput, autocomplete])
|
||||
|
||||
useInput(
|
||||
(input, key) => {
|
||||
if (disabled) {
|
||||
return
|
||||
}
|
||||
|
||||
if (key.upArrow && history.length > 0) {
|
||||
if (historyIndex === -1) {
|
||||
setSavedInput(value)
|
||||
}
|
||||
|
||||
const newIndex =
|
||||
historyIndex === -1 ? history.length - 1 : Math.max(0, historyIndex - 1)
|
||||
setHistoryIndex(newIndex)
|
||||
setValue(history[newIndex] ?? "")
|
||||
if (key.tab) {
|
||||
handleTabKey()
|
||||
}
|
||||
|
||||
if (key.downArrow) {
|
||||
if (historyIndex === -1) {
|
||||
return
|
||||
if (key.return && key.shift && isMultilineActive) {
|
||||
handleAddLine()
|
||||
}
|
||||
if (key.upArrow) {
|
||||
if (isMultilineActive && currentLineIndex > 0) {
|
||||
setCurrentLineIndex(currentLineIndex - 1)
|
||||
} else if (!isMultilineActive) {
|
||||
handleUpArrow()
|
||||
}
|
||||
|
||||
if (historyIndex >= history.length - 1) {
|
||||
setHistoryIndex(-1)
|
||||
setValue(savedInput)
|
||||
} else {
|
||||
const newIndex = historyIndex + 1
|
||||
setHistoryIndex(newIndex)
|
||||
setValue(history[newIndex] ?? "")
|
||||
}
|
||||
if (key.downArrow) {
|
||||
if (isMultilineActive && currentLineIndex < lines.length - 1) {
|
||||
setCurrentLineIndex(currentLineIndex + 1)
|
||||
} else if (!isMultilineActive) {
|
||||
handleDownArrow()
|
||||
}
|
||||
}
|
||||
},
|
||||
{ isActive: !disabled },
|
||||
)
|
||||
|
||||
const hasSuggestions = autocomplete.suggestions.length > 0
|
||||
|
||||
return (
|
||||
<Box borderStyle="single" borderColor={disabled ? "gray" : "cyan"} paddingX={1}>
|
||||
<Text color={disabled ? "gray" : "green"} bold>
|
||||
{">"}{" "}
|
||||
</Text>
|
||||
{disabled ? (
|
||||
<Text color="gray" dimColor>
|
||||
{placeholder}
|
||||
</Text>
|
||||
) : (
|
||||
<TextInput
|
||||
value={value}
|
||||
onChange={handleChange}
|
||||
onSubmit={handleSubmit}
|
||||
placeholder={placeholder}
|
||||
/>
|
||||
<Box flexDirection="column">
|
||||
<Box
|
||||
borderStyle="single"
|
||||
borderColor={disabled ? "gray" : "cyan"}
|
||||
paddingX={1}
|
||||
flexDirection="column"
|
||||
>
|
||||
{disabled ? (
|
||||
<Box>
|
||||
<Text color="gray" bold>
|
||||
{">"}{" "}
|
||||
</Text>
|
||||
<Text color="gray" dimColor>
|
||||
{placeholder}
|
||||
</Text>
|
||||
</Box>
|
||||
) : isMultilineActive ? (
|
||||
<Box flexDirection="column">
|
||||
{lines.map((line, index) => (
|
||||
<Box key={index}>
|
||||
<Text color="green" bold>
|
||||
{index === currentLineIndex ? ">" : " "}{" "}
|
||||
</Text>
|
||||
{index === currentLineIndex ? (
|
||||
<TextInput
|
||||
value={line}
|
||||
onChange={handleLineChange}
|
||||
onSubmit={handleMultilineSubmit}
|
||||
placeholder={index === 0 ? placeholder : ""}
|
||||
/>
|
||||
) : (
|
||||
<Text>{line}</Text>
|
||||
)}
|
||||
</Box>
|
||||
))}
|
||||
<Box marginTop={1}>
|
||||
<Text dimColor>Shift+Enter: new line | Enter: submit</Text>
|
||||
</Box>
|
||||
</Box>
|
||||
) : (
|
||||
<Box>
|
||||
<Text color="green" bold>
|
||||
{">"}{" "}
|
||||
</Text>
|
||||
<TextInput
|
||||
value={value}
|
||||
onChange={handleChange}
|
||||
onSubmit={handleSubmit}
|
||||
placeholder={placeholder}
|
||||
/>
|
||||
</Box>
|
||||
)}
|
||||
</Box>
|
||||
{hasSuggestions && !disabled && (
|
||||
<Box paddingLeft={2} flexDirection="column">
|
||||
<Text dimColor>
|
||||
{autocomplete.suggestions.length === 1
|
||||
? "Press Tab to complete"
|
||||
: `${String(autocomplete.suggestions.length)} suggestions (Tab to complete)`}
|
||||
</Text>
|
||||
{autocomplete.suggestions.slice(0, 5).map((suggestion, i) => (
|
||||
<Text key={i} dimColor color="cyan">
|
||||
{" "}• {suggestion}
|
||||
</Text>
|
||||
))}
|
||||
{autocomplete.suggestions.length > 5 && (
|
||||
<Text dimColor>
|
||||
{" "}... and {String(autocomplete.suggestions.length - 5)} more
|
||||
</Text>
|
||||
)}
|
||||
</Box>
|
||||
)}
|
||||
</Box>
|
||||
)
|
||||
|
||||
@@ -9,3 +9,4 @@ export { DiffView, type DiffViewProps } from "./DiffView.js"
|
||||
export { ConfirmDialog, type ConfirmDialogProps } from "./ConfirmDialog.js"
|
||||
export { ErrorDialog, type ErrorDialogProps, type ErrorInfo } from "./ErrorDialog.js"
|
||||
export { Progress, type ProgressProps } from "./Progress.js"
|
||||
export { EditableContent, type EditableContentProps } from "./EditableContent.js"
|
||||
|
||||
@@ -19,3 +19,8 @@ export {
|
||||
type CommandResult,
|
||||
type CommandDefinition,
|
||||
} from "./useCommands.js"
|
||||
export {
|
||||
useAutocomplete,
|
||||
type UseAutocompleteOptions,
|
||||
type UseAutocompleteReturn,
|
||||
} from "./useAutocomplete.js"
|
||||
|
||||
197
packages/ipuaro/src/tui/hooks/useAutocomplete.ts
Normal file
197
packages/ipuaro/src/tui/hooks/useAutocomplete.ts
Normal file
@@ -0,0 +1,197 @@
|
||||
/**
|
||||
* useAutocomplete hook for file path autocomplete.
|
||||
* Provides Tab completion for file paths using Redis index.
|
||||
*/
|
||||
|
||||
import { useCallback, useEffect, useState } from "react"
|
||||
import type { IStorage } from "../../domain/services/IStorage.js"
|
||||
import path from "node:path"
|
||||
|
||||
export interface UseAutocompleteOptions {
|
||||
storage: IStorage
|
||||
projectRoot: string
|
||||
enabled?: boolean
|
||||
maxSuggestions?: number
|
||||
}
|
||||
|
||||
export interface UseAutocompleteReturn {
|
||||
suggestions: string[]
|
||||
complete: (partial: string) => string[]
|
||||
accept: (suggestion: string) => string
|
||||
reset: () => void
|
||||
}
|
||||
|
||||
/**
|
||||
* Normalizes a path by removing leading ./ and trailing /
|
||||
*/
|
||||
function normalizePath(p: string): string {
|
||||
let normalized = p.trim()
|
||||
if (normalized.startsWith("./")) {
|
||||
normalized = normalized.slice(2)
|
||||
}
|
||||
if (normalized.endsWith("/") && normalized.length > 1) {
|
||||
normalized = normalized.slice(0, -1)
|
||||
}
|
||||
return normalized
|
||||
}
|
||||
|
||||
/**
|
||||
* Calculates fuzzy match score between partial and candidate.
|
||||
* Returns 0 if no match, higher score for better matches.
|
||||
*/
|
||||
function fuzzyScore(partial: string, candidate: string): number {
|
||||
const partialLower = partial.toLowerCase()
|
||||
const candidateLower = candidate.toLowerCase()
|
||||
|
||||
// Exact prefix match gets highest score
|
||||
if (candidateLower.startsWith(partialLower)) {
|
||||
return 1000 + (1000 - partial.length)
|
||||
}
|
||||
|
||||
// Check if all characters from partial appear in order in candidate
|
||||
let partialIndex = 0
|
||||
let candidateIndex = 0
|
||||
let lastMatchIndex = -1
|
||||
let consecutiveMatches = 0
|
||||
|
||||
while (partialIndex < partialLower.length && candidateIndex < candidateLower.length) {
|
||||
if (partialLower[partialIndex] === candidateLower[candidateIndex]) {
|
||||
// Bonus for consecutive matches
|
||||
if (candidateIndex === lastMatchIndex + 1) {
|
||||
consecutiveMatches++
|
||||
} else {
|
||||
consecutiveMatches = 0
|
||||
}
|
||||
lastMatchIndex = candidateIndex
|
||||
partialIndex++
|
||||
}
|
||||
candidateIndex++
|
||||
}
|
||||
|
||||
// If we didn't match all characters, no match
|
||||
if (partialIndex < partialLower.length) {
|
||||
return 0
|
||||
}
|
||||
|
||||
// Score based on how tight the match is
|
||||
const matchSpread = lastMatchIndex - (partialLower.length - 1)
|
||||
const score = 100 + consecutiveMatches * 10 - matchSpread
|
||||
|
||||
return Math.max(0, score)
|
||||
}
|
||||
|
||||
/**
|
||||
* Gets the common prefix of all suggestions
|
||||
*/
|
||||
function getCommonPrefix(suggestions: string[]): string {
|
||||
if (suggestions.length === 0) {
|
||||
return ""
|
||||
}
|
||||
if (suggestions.length === 1) {
|
||||
return suggestions[0] ?? ""
|
||||
}
|
||||
|
||||
let prefix = suggestions[0] ?? ""
|
||||
for (let i = 1; i < suggestions.length; i++) {
|
||||
const current = suggestions[i] ?? ""
|
||||
let j = 0
|
||||
while (j < prefix.length && j < current.length && prefix[j] === current[j]) {
|
||||
j++
|
||||
}
|
||||
prefix = prefix.slice(0, j)
|
||||
if (prefix.length === 0) {
|
||||
break
|
||||
}
|
||||
}
|
||||
return prefix
|
||||
}
|
||||
|
||||
export function useAutocomplete(options: UseAutocompleteOptions): UseAutocompleteReturn {
|
||||
const { storage, projectRoot, enabled = true, maxSuggestions = 10 } = options
|
||||
const [filePaths, setFilePaths] = useState<string[]>([])
|
||||
const [suggestions, setSuggestions] = useState<string[]>([])
|
||||
|
||||
// Load file paths from storage
|
||||
useEffect(() => {
|
||||
if (!enabled) {
|
||||
return
|
||||
}
|
||||
|
||||
const loadPaths = async (): Promise<void> => {
|
||||
try {
|
||||
const files = await storage.getAllFiles()
|
||||
const paths = Array.from(files.keys()).map((p) => {
|
||||
// Make paths relative to project root
|
||||
const relative = path.relative(projectRoot, p)
|
||||
return normalizePath(relative)
|
||||
})
|
||||
setFilePaths(paths.sort())
|
||||
} catch {
|
||||
// Silently fail - autocomplete is non-critical
|
||||
setFilePaths([])
|
||||
}
|
||||
}
|
||||
|
||||
loadPaths().catch(() => {
|
||||
// Ignore errors
|
||||
})
|
||||
}, [storage, projectRoot, enabled])
|
||||
|
||||
const complete = useCallback(
|
||||
(partial: string): string[] => {
|
||||
if (!enabled || !partial.trim()) {
|
||||
setSuggestions([])
|
||||
return []
|
||||
}
|
||||
|
||||
const normalized = normalizePath(partial)
|
||||
|
||||
// Score and filter matches
|
||||
const scored = filePaths
|
||||
.map((p) => ({
|
||||
path: p,
|
||||
score: fuzzyScore(normalized, p),
|
||||
}))
|
||||
.filter((item) => item.score > 0)
|
||||
.sort((a, b) => b.score - a.score)
|
||||
.slice(0, maxSuggestions)
|
||||
.map((item) => item.path)
|
||||
|
||||
setSuggestions(scored)
|
||||
return scored
|
||||
},
|
||||
[enabled, filePaths, maxSuggestions],
|
||||
)
|
||||
|
||||
const accept = useCallback(
|
||||
(suggestion: string): string => {
|
||||
// If there's only one suggestion, complete with it
|
||||
if (suggestions.length === 1) {
|
||||
setSuggestions([])
|
||||
return suggestions[0] ?? ""
|
||||
}
|
||||
|
||||
// If there are multiple suggestions, complete with common prefix
|
||||
if (suggestions.length > 1) {
|
||||
const prefix = getCommonPrefix(suggestions)
|
||||
if (prefix.length > suggestion.length) {
|
||||
return prefix
|
||||
}
|
||||
}
|
||||
|
||||
return suggestion
|
||||
},
|
||||
[suggestions],
|
||||
)
|
||||
|
||||
const reset = useCallback(() => {
|
||||
setSuggestions([])
|
||||
}, [])
|
||||
|
||||
return {
|
||||
suggestions,
|
||||
complete,
|
||||
accept,
|
||||
reset,
|
||||
}
|
||||
}
|
||||
@@ -18,6 +18,7 @@ import {
|
||||
} from "../../application/use-cases/HandleMessage.js"
|
||||
import { StartSession } from "../../application/use-cases/StartSession.js"
|
||||
import { UndoChange } from "../../application/use-cases/UndoChange.js"
|
||||
import type { ConfirmationResult } from "../../application/use-cases/ExecuteTool.js"
|
||||
import type { ProjectStructure } from "../../infrastructure/llm/prompts.js"
|
||||
import type { TuiStatus } from "../types.js"
|
||||
|
||||
@@ -33,7 +34,7 @@ export interface UseSessionDependencies {
|
||||
|
||||
export interface UseSessionOptions {
|
||||
autoApply?: boolean
|
||||
onConfirmation?: (message: string, diff?: DiffInfo) => Promise<boolean>
|
||||
onConfirmation?: (message: string, diff?: DiffInfo) => Promise<boolean | ConfirmationResult>
|
||||
onError?: (error: Error) => Promise<ErrorOption>
|
||||
}
|
||||
|
||||
|
||||
167
packages/ipuaro/src/tui/utils/syntax-highlighter.ts
Normal file
167
packages/ipuaro/src/tui/utils/syntax-highlighter.ts
Normal file
@@ -0,0 +1,167 @@
|
||||
/**
|
||||
* Simple syntax highlighter for terminal UI.
|
||||
* Highlights keywords, strings, comments, numbers, and operators.
|
||||
*/
|
||||
|
||||
export type Language = "typescript" | "javascript" | "tsx" | "jsx" | "json" | "yaml" | "unknown"
|
||||
|
||||
export interface HighlightedToken {
|
||||
text: string
|
||||
color: string
|
||||
}
|
||||
|
||||
const KEYWORDS = new Set([
|
||||
"abstract",
|
||||
"any",
|
||||
"as",
|
||||
"async",
|
||||
"await",
|
||||
"boolean",
|
||||
"break",
|
||||
"case",
|
||||
"catch",
|
||||
"class",
|
||||
"const",
|
||||
"constructor",
|
||||
"continue",
|
||||
"debugger",
|
||||
"declare",
|
||||
"default",
|
||||
"delete",
|
||||
"do",
|
||||
"else",
|
||||
"enum",
|
||||
"export",
|
||||
"extends",
|
||||
"false",
|
||||
"finally",
|
||||
"for",
|
||||
"from",
|
||||
"function",
|
||||
"get",
|
||||
"if",
|
||||
"implements",
|
||||
"import",
|
||||
"in",
|
||||
"instanceof",
|
||||
"interface",
|
||||
"let",
|
||||
"module",
|
||||
"namespace",
|
||||
"new",
|
||||
"null",
|
||||
"number",
|
||||
"of",
|
||||
"package",
|
||||
"private",
|
||||
"protected",
|
||||
"public",
|
||||
"readonly",
|
||||
"require",
|
||||
"return",
|
||||
"set",
|
||||
"static",
|
||||
"string",
|
||||
"super",
|
||||
"switch",
|
||||
"this",
|
||||
"throw",
|
||||
"true",
|
||||
"try",
|
||||
"type",
|
||||
"typeof",
|
||||
"undefined",
|
||||
"var",
|
||||
"void",
|
||||
"while",
|
||||
"with",
|
||||
"yield",
|
||||
])
|
||||
|
||||
export function detectLanguage(filePath: string): Language {
|
||||
const ext = filePath.split(".").pop()?.toLowerCase()
|
||||
switch (ext) {
|
||||
case "ts":
|
||||
return "typescript"
|
||||
case "tsx":
|
||||
return "tsx"
|
||||
case "js":
|
||||
return "javascript"
|
||||
case "jsx":
|
||||
return "jsx"
|
||||
case "json":
|
||||
return "json"
|
||||
case "yaml":
|
||||
case "yml":
|
||||
return "yaml"
|
||||
default:
|
||||
return "unknown"
|
||||
}
|
||||
}
|
||||
|
||||
const COMMENT_REGEX = /^(\/\/.*|\/\*[\s\S]*?\*\/)/
|
||||
const STRING_REGEX = /^("(?:[^"\\]|\\.)*"|'(?:[^'\\]|\\.)*'|`(?:[^`\\]|\\.)*`)/
|
||||
const NUMBER_REGEX = /^(\b\d+\.?\d*\b)/
|
||||
const WORD_REGEX = /^([a-zA-Z_$][a-zA-Z0-9_$]*)/
|
||||
const OPERATOR_REGEX = /^([+\-*/%=<>!&|^~?:;,.()[\]{}])/
|
||||
const WHITESPACE_REGEX = /^(\s+)/
|
||||
|
||||
export function highlightLine(line: string, language: Language): HighlightedToken[] {
|
||||
if (language === "unknown" || language === "json" || language === "yaml") {
|
||||
return [{ text: line, color: "white" }]
|
||||
}
|
||||
|
||||
const tokens: HighlightedToken[] = []
|
||||
let remaining = line
|
||||
|
||||
while (remaining.length > 0) {
|
||||
const commentMatch = COMMENT_REGEX.exec(remaining)
|
||||
if (commentMatch) {
|
||||
tokens.push({ text: commentMatch[0], color: "gray" })
|
||||
remaining = remaining.slice(commentMatch[0].length)
|
||||
continue
|
||||
}
|
||||
|
||||
const stringMatch = STRING_REGEX.exec(remaining)
|
||||
if (stringMatch) {
|
||||
tokens.push({ text: stringMatch[0], color: "green" })
|
||||
remaining = remaining.slice(stringMatch[0].length)
|
||||
continue
|
||||
}
|
||||
|
||||
const numberMatch = NUMBER_REGEX.exec(remaining)
|
||||
if (numberMatch) {
|
||||
tokens.push({ text: numberMatch[0], color: "cyan" })
|
||||
remaining = remaining.slice(numberMatch[0].length)
|
||||
continue
|
||||
}
|
||||
|
||||
const wordMatch = WORD_REGEX.exec(remaining)
|
||||
if (wordMatch) {
|
||||
const word = wordMatch[0]
|
||||
const color = KEYWORDS.has(word) ? "magenta" : "white"
|
||||
tokens.push({ text: word, color })
|
||||
remaining = remaining.slice(word.length)
|
||||
continue
|
||||
}
|
||||
|
||||
const operatorMatch = OPERATOR_REGEX.exec(remaining)
|
||||
if (operatorMatch) {
|
||||
tokens.push({ text: operatorMatch[0], color: "yellow" })
|
||||
remaining = remaining.slice(operatorMatch[0].length)
|
||||
continue
|
||||
}
|
||||
|
||||
const whitespaceMatch = WHITESPACE_REGEX.exec(remaining)
|
||||
if (whitespaceMatch) {
|
||||
tokens.push({ text: whitespaceMatch[0], color: "white" })
|
||||
remaining = remaining.slice(whitespaceMatch[0].length)
|
||||
continue
|
||||
}
|
||||
|
||||
tokens.push({ text: remaining[0] ?? "", color: "white" })
|
||||
remaining = remaining.slice(1)
|
||||
}
|
||||
|
||||
return tokens
|
||||
}
|
||||
@@ -198,12 +198,12 @@ describe("HandleMessage", () => {
|
||||
expect(toolMessages.length).toBeGreaterThan(0)
|
||||
})
|
||||
|
||||
it("should return error for unknown tools", async () => {
|
||||
it("should return error for unregistered tools", async () => {
|
||||
vi.mocked(mockTools.get).mockReturnValue(undefined)
|
||||
vi.mocked(mockLLM.chat)
|
||||
.mockResolvedValueOnce(
|
||||
createMockLLMResponse(
|
||||
'<tool_call name="unknown_tool"><param>value</param></tool_call>',
|
||||
'<tool_call name="get_complexity"><path>src</path></tool_call>',
|
||||
true,
|
||||
),
|
||||
)
|
||||
|
||||
@@ -0,0 +1,317 @@
|
||||
import { describe, it, expect, vi, beforeEach } from "vitest"
|
||||
import { IndexProject } from "../../../../src/application/use-cases/IndexProject.js"
|
||||
import type { IStorage, SymbolIndex, DepsGraph } from "../../../../src/domain/services/IStorage.js"
|
||||
import type { IndexProgress } from "../../../../src/domain/services/IIndexer.js"
|
||||
import { createFileData } from "../../../../src/domain/value-objects/FileData.js"
|
||||
import { createEmptyFileAST } from "../../../../src/domain/value-objects/FileAST.js"
|
||||
import { createFileMeta } from "../../../../src/domain/value-objects/FileMeta.js"
|
||||
|
||||
vi.mock("../../../../src/infrastructure/indexer/FileScanner.js", () => ({
|
||||
FileScanner: class {
|
||||
async scanAll() {
|
||||
return [
|
||||
{ path: "src/index.ts", type: "file", size: 100, lastModified: Date.now() },
|
||||
{ path: "src/utils.ts", type: "file", size: 200, lastModified: Date.now() },
|
||||
]
|
||||
}
|
||||
static async readFileContent(path: string) {
|
||||
if (path.includes("index.ts")) {
|
||||
return 'export function main() { return "hello" }'
|
||||
}
|
||||
if (path.includes("utils.ts")) {
|
||||
return 'export const add = (a: number, b: number) => a + b'
|
||||
}
|
||||
return null
|
||||
}
|
||||
},
|
||||
}))
|
||||
|
||||
vi.mock("../../../../src/infrastructure/indexer/ASTParser.js", () => ({
|
||||
ASTParser: class {
|
||||
parse() {
|
||||
return {
|
||||
...createEmptyFileAST(),
|
||||
functions: [{ name: "test", lineStart: 1, lineEnd: 5, params: [], isAsync: false, isExported: true }],
|
||||
}
|
||||
}
|
||||
},
|
||||
}))
|
||||
|
||||
vi.mock("../../../../src/infrastructure/indexer/MetaAnalyzer.js", () => ({
|
||||
MetaAnalyzer: class {
|
||||
constructor() {}
|
||||
analyze() {
|
||||
return createFileMeta()
|
||||
}
|
||||
},
|
||||
}))
|
||||
|
||||
vi.mock("../../../../src/infrastructure/indexer/IndexBuilder.js", () => ({
|
||||
IndexBuilder: class {
|
||||
constructor() {}
|
||||
buildSymbolIndex() {
|
||||
return new Map([
|
||||
["test", [{ path: "src/index.ts", line: 1, type: "function" }]],
|
||||
]) as SymbolIndex
|
||||
}
|
||||
buildDepsGraph() {
|
||||
return {
|
||||
imports: new Map(),
|
||||
importedBy: new Map(),
|
||||
} as DepsGraph
|
||||
}
|
||||
},
|
||||
}))
|
||||
|
||||
describe("IndexProject", () => {
|
||||
let useCase: IndexProject
|
||||
let mockStorage: IStorage
|
||||
|
||||
beforeEach(() => {
|
||||
mockStorage = {
|
||||
getFile: vi.fn().mockResolvedValue(null),
|
||||
setFile: vi.fn().mockResolvedValue(undefined),
|
||||
deleteFile: vi.fn().mockResolvedValue(undefined),
|
||||
getAllFiles: vi.fn().mockResolvedValue(new Map()),
|
||||
getFileCount: vi.fn().mockResolvedValue(0),
|
||||
getAST: vi.fn().mockResolvedValue(null),
|
||||
setAST: vi.fn().mockResolvedValue(undefined),
|
||||
deleteAST: vi.fn().mockResolvedValue(undefined),
|
||||
getAllASTs: vi.fn().mockResolvedValue(new Map()),
|
||||
getMeta: vi.fn().mockResolvedValue(null),
|
||||
setMeta: vi.fn().mockResolvedValue(undefined),
|
||||
deleteMeta: vi.fn().mockResolvedValue(undefined),
|
||||
getAllMetas: vi.fn().mockResolvedValue(new Map()),
|
||||
getSymbolIndex: vi.fn().mockResolvedValue(new Map()),
|
||||
setSymbolIndex: vi.fn().mockResolvedValue(undefined),
|
||||
getDepsGraph: vi.fn().mockResolvedValue({ imports: new Map(), importedBy: new Map() }),
|
||||
setDepsGraph: vi.fn().mockResolvedValue(undefined),
|
||||
getProjectConfig: vi.fn().mockResolvedValue(null),
|
||||
setProjectConfig: vi.fn().mockResolvedValue(undefined),
|
||||
connect: vi.fn().mockResolvedValue(undefined),
|
||||
disconnect: vi.fn().mockResolvedValue(undefined),
|
||||
isConnected: vi.fn().mockReturnValue(true),
|
||||
clear: vi.fn().mockResolvedValue(undefined),
|
||||
}
|
||||
|
||||
useCase = new IndexProject(mockStorage, "/test/project")
|
||||
})
|
||||
|
||||
describe("execute", () => {
|
||||
it("should index project and return stats", async () => {
|
||||
const stats = await useCase.execute("/test/project")
|
||||
|
||||
expect(stats.filesScanned).toBe(2)
|
||||
expect(stats.filesParsed).toBe(2)
|
||||
expect(stats.parseErrors).toBe(0)
|
||||
expect(stats.timeMs).toBeGreaterThanOrEqual(0)
|
||||
})
|
||||
|
||||
it("should store file data for all scanned files", async () => {
|
||||
await useCase.execute("/test/project")
|
||||
|
||||
expect(mockStorage.setFile).toHaveBeenCalledTimes(2)
|
||||
expect(mockStorage.setFile).toHaveBeenCalledWith(
|
||||
"src/index.ts",
|
||||
expect.objectContaining({
|
||||
hash: expect.any(String),
|
||||
lines: expect.any(Array),
|
||||
})
|
||||
)
|
||||
})
|
||||
|
||||
it("should store AST for all parsed files", async () => {
|
||||
await useCase.execute("/test/project")
|
||||
|
||||
expect(mockStorage.setAST).toHaveBeenCalledTimes(2)
|
||||
expect(mockStorage.setAST).toHaveBeenCalledWith(
|
||||
"src/index.ts",
|
||||
expect.objectContaining({
|
||||
functions: expect.any(Array),
|
||||
})
|
||||
)
|
||||
})
|
||||
|
||||
it("should store metadata for all files", async () => {
|
||||
await useCase.execute("/test/project")
|
||||
|
||||
expect(mockStorage.setMeta).toHaveBeenCalledTimes(2)
|
||||
expect(mockStorage.setMeta).toHaveBeenCalledWith(
|
||||
"src/index.ts",
|
||||
expect.any(Object)
|
||||
)
|
||||
})
|
||||
|
||||
it("should build and store symbol index", async () => {
|
||||
await useCase.execute("/test/project")
|
||||
|
||||
expect(mockStorage.setSymbolIndex).toHaveBeenCalledTimes(1)
|
||||
expect(mockStorage.setSymbolIndex).toHaveBeenCalledWith(
|
||||
expect.any(Map)
|
||||
)
|
||||
})
|
||||
|
||||
it("should build and store dependency graph", async () => {
|
||||
await useCase.execute("/test/project")
|
||||
|
||||
expect(mockStorage.setDepsGraph).toHaveBeenCalledTimes(1)
|
||||
expect(mockStorage.setDepsGraph).toHaveBeenCalledWith(
|
||||
expect.objectContaining({
|
||||
imports: expect.any(Map),
|
||||
importedBy: expect.any(Map),
|
||||
})
|
||||
)
|
||||
})
|
||||
|
||||
it("should store last indexed timestamp", async () => {
|
||||
await useCase.execute("/test/project")
|
||||
|
||||
expect(mockStorage.setProjectConfig).toHaveBeenCalledWith(
|
||||
"last_indexed",
|
||||
expect.any(Number)
|
||||
)
|
||||
})
|
||||
|
||||
it("should call progress callback during indexing", async () => {
|
||||
const progressCallback = vi.fn()
|
||||
|
||||
await useCase.execute("/test/project", {
|
||||
onProgress: progressCallback,
|
||||
})
|
||||
|
||||
expect(progressCallback).toHaveBeenCalled()
|
||||
expect(progressCallback).toHaveBeenCalledWith(
|
||||
expect.objectContaining({
|
||||
current: expect.any(Number),
|
||||
total: expect.any(Number),
|
||||
currentFile: expect.any(String),
|
||||
phase: expect.stringMatching(/scanning|parsing|analyzing|indexing/),
|
||||
})
|
||||
)
|
||||
})
|
||||
|
||||
it("should report scanning phase", async () => {
|
||||
const progressCallback = vi.fn()
|
||||
|
||||
await useCase.execute("/test/project", {
|
||||
onProgress: progressCallback,
|
||||
})
|
||||
|
||||
const scanningCalls = progressCallback.mock.calls.filter(
|
||||
(call) => call[0].phase === "scanning"
|
||||
)
|
||||
expect(scanningCalls.length).toBeGreaterThan(0)
|
||||
})
|
||||
|
||||
it("should report parsing phase", async () => {
|
||||
const progressCallback = vi.fn()
|
||||
|
||||
await useCase.execute("/test/project", {
|
||||
onProgress: progressCallback,
|
||||
})
|
||||
|
||||
const parsingCalls = progressCallback.mock.calls.filter(
|
||||
(call) => call[0].phase === "parsing"
|
||||
)
|
||||
expect(parsingCalls.length).toBeGreaterThan(0)
|
||||
})
|
||||
|
||||
it("should report analyzing phase", async () => {
|
||||
const progressCallback = vi.fn()
|
||||
|
||||
await useCase.execute("/test/project", {
|
||||
onProgress: progressCallback,
|
||||
})
|
||||
|
||||
const analyzingCalls = progressCallback.mock.calls.filter(
|
||||
(call) => call[0].phase === "analyzing"
|
||||
)
|
||||
expect(analyzingCalls.length).toBeGreaterThan(0)
|
||||
})
|
||||
|
||||
it("should report indexing phase", async () => {
|
||||
const progressCallback = vi.fn()
|
||||
|
||||
await useCase.execute("/test/project", {
|
||||
onProgress: progressCallback,
|
||||
})
|
||||
|
||||
const indexingCalls = progressCallback.mock.calls.filter(
|
||||
(call) => call[0].phase === "indexing"
|
||||
)
|
||||
expect(indexingCalls.length).toBeGreaterThan(0)
|
||||
})
|
||||
|
||||
it("should detect TypeScript files", async () => {
|
||||
await useCase.execute("/test/project")
|
||||
|
||||
expect(mockStorage.setAST).toHaveBeenCalledWith(
|
||||
"src/index.ts",
|
||||
expect.any(Object)
|
||||
)
|
||||
})
|
||||
|
||||
it("should handle files without parseable language", async () => {
|
||||
vi.mocked(mockStorage.setFile).mockClear()
|
||||
|
||||
await useCase.execute("/test/project")
|
||||
|
||||
const stats = await useCase.execute("/test/project")
|
||||
expect(stats.filesScanned).toBeGreaterThanOrEqual(0)
|
||||
})
|
||||
|
||||
it("should calculate indexing duration", async () => {
|
||||
const startTime = Date.now()
|
||||
const stats = await useCase.execute("/test/project")
|
||||
const endTime = Date.now()
|
||||
|
||||
expect(stats.timeMs).toBeGreaterThanOrEqual(0)
|
||||
expect(stats.timeMs).toBeLessThanOrEqual(endTime - startTime + 10)
|
||||
})
|
||||
})
|
||||
|
||||
describe("language detection", () => {
|
||||
it("should detect .ts files", async () => {
|
||||
await useCase.execute("/test/project")
|
||||
|
||||
expect(mockStorage.setAST).toHaveBeenCalledWith(
|
||||
expect.stringContaining(".ts"),
|
||||
expect.any(Object)
|
||||
)
|
||||
})
|
||||
})
|
||||
|
||||
describe("progress reporting", () => {
|
||||
it("should not fail if progress callback is not provided", async () => {
|
||||
await expect(useCase.execute("/test/project")).resolves.toBeDefined()
|
||||
})
|
||||
|
||||
it("should include current file in progress updates", async () => {
|
||||
const progressCallback = vi.fn()
|
||||
|
||||
await useCase.execute("/test/project", {
|
||||
onProgress: progressCallback,
|
||||
})
|
||||
|
||||
const callsWithFiles = progressCallback.mock.calls.filter(
|
||||
(call) => call[0].currentFile && call[0].currentFile.length > 0
|
||||
)
|
||||
expect(callsWithFiles.length).toBeGreaterThan(0)
|
||||
})
|
||||
|
||||
it("should report correct total count", async () => {
|
||||
const progressCallback = vi.fn()
|
||||
|
||||
await useCase.execute("/test/project", {
|
||||
onProgress: progressCallback,
|
||||
})
|
||||
|
||||
const parsingCalls = progressCallback.mock.calls.filter(
|
||||
(call) => call[0].phase === "parsing"
|
||||
)
|
||||
if (parsingCalls.length > 0) {
|
||||
expect(parsingCalls[0][0].total).toBe(2)
|
||||
}
|
||||
})
|
||||
})
|
||||
})
|
||||
@@ -109,24 +109,80 @@ describe("Watchdog", () => {
|
||||
|
||||
describe("flushAll", () => {
|
||||
it("should not throw when no pending changes", () => {
|
||||
watchdog.start(tempDir)
|
||||
expect(() => watchdog.flushAll()).not.toThrow()
|
||||
})
|
||||
|
||||
it("should flush all pending changes", async () => {
|
||||
it("should handle flushAll with active timers", async () => {
|
||||
const slowWatchdog = new Watchdog({ debounceMs: 1000 })
|
||||
const events: FileChangeEvent[] = []
|
||||
watchdog.onFileChange((event) => events.push(event))
|
||||
watchdog.start(tempDir)
|
||||
slowWatchdog.onFileChange((event) => events.push(event))
|
||||
slowWatchdog.start(tempDir)
|
||||
|
||||
await new Promise((resolve) => setTimeout(resolve, 200))
|
||||
|
||||
const testFile = path.join(tempDir, "instant-flush.ts")
|
||||
await fs.writeFile(testFile, "const x = 1")
|
||||
|
||||
await new Promise((resolve) => setTimeout(resolve, 150))
|
||||
|
||||
const pendingCount = slowWatchdog.getPendingCount()
|
||||
if (pendingCount > 0) {
|
||||
slowWatchdog.flushAll()
|
||||
expect(slowWatchdog.getPendingCount()).toBe(0)
|
||||
expect(events.length).toBeGreaterThan(0)
|
||||
}
|
||||
|
||||
await slowWatchdog.stop()
|
||||
})
|
||||
|
||||
it("should flush all pending changes immediately", async () => {
|
||||
const slowWatchdog = new Watchdog({ debounceMs: 500 })
|
||||
const events: FileChangeEvent[] = []
|
||||
slowWatchdog.onFileChange((event) => events.push(event))
|
||||
slowWatchdog.start(tempDir)
|
||||
|
||||
await new Promise((resolve) => setTimeout(resolve, 100))
|
||||
|
||||
const testFile = path.join(tempDir, "flush-test.ts")
|
||||
const testFile1 = path.join(tempDir, "flush-test1.ts")
|
||||
const testFile2 = path.join(tempDir, "flush-test2.ts")
|
||||
await fs.writeFile(testFile1, "const x = 1")
|
||||
await fs.writeFile(testFile2, "const y = 2")
|
||||
|
||||
await new Promise((resolve) => setTimeout(resolve, 100))
|
||||
|
||||
const pendingCount = slowWatchdog.getPendingCount()
|
||||
if (pendingCount > 0) {
|
||||
slowWatchdog.flushAll()
|
||||
expect(slowWatchdog.getPendingCount()).toBe(0)
|
||||
}
|
||||
|
||||
await slowWatchdog.stop()
|
||||
})
|
||||
|
||||
it("should clear all timers when flushing", async () => {
|
||||
const slowWatchdog = new Watchdog({ debounceMs: 500 })
|
||||
const events: FileChangeEvent[] = []
|
||||
slowWatchdog.onFileChange((event) => events.push(event))
|
||||
slowWatchdog.start(tempDir)
|
||||
|
||||
await new Promise((resolve) => setTimeout(resolve, 100))
|
||||
|
||||
const testFile = path.join(tempDir, "timer-test.ts")
|
||||
await fs.writeFile(testFile, "const x = 1")
|
||||
|
||||
await new Promise((resolve) => setTimeout(resolve, 20))
|
||||
await new Promise((resolve) => setTimeout(resolve, 100))
|
||||
|
||||
watchdog.flushAll()
|
||||
const pendingBefore = slowWatchdog.getPendingCount()
|
||||
|
||||
await new Promise((resolve) => setTimeout(resolve, 50))
|
||||
if (pendingBefore > 0) {
|
||||
const eventsBefore = events.length
|
||||
slowWatchdog.flushAll()
|
||||
expect(slowWatchdog.getPendingCount()).toBe(0)
|
||||
expect(events.length).toBeGreaterThan(eventsBefore)
|
||||
}
|
||||
|
||||
await slowWatchdog.stop()
|
||||
})
|
||||
})
|
||||
|
||||
@@ -145,7 +201,7 @@ describe("Watchdog", () => {
|
||||
await customWatchdog.stop()
|
||||
})
|
||||
|
||||
it("should handle simple directory patterns", async () => {
|
||||
it("should handle simple directory patterns without wildcards", async () => {
|
||||
const customWatchdog = new Watchdog({
|
||||
debounceMs: 50,
|
||||
ignorePatterns: ["node_modules", "dist"],
|
||||
@@ -158,6 +214,48 @@ describe("Watchdog", () => {
|
||||
|
||||
await customWatchdog.stop()
|
||||
})
|
||||
|
||||
it("should handle mixed wildcard and non-wildcard patterns", async () => {
|
||||
const customWatchdog = new Watchdog({
|
||||
debounceMs: 50,
|
||||
ignorePatterns: ["node_modules", "*.log", "**/*.tmp", "dist", "build"],
|
||||
})
|
||||
|
||||
customWatchdog.start(tempDir)
|
||||
await new Promise((resolve) => setTimeout(resolve, 100))
|
||||
|
||||
expect(customWatchdog.isWatching()).toBe(true)
|
||||
|
||||
await customWatchdog.stop()
|
||||
})
|
||||
|
||||
it("should handle patterns with dots correctly", async () => {
|
||||
const customWatchdog = new Watchdog({
|
||||
debounceMs: 50,
|
||||
ignorePatterns: ["*.test.ts", "**/*.spec.js"],
|
||||
})
|
||||
|
||||
customWatchdog.start(tempDir)
|
||||
await new Promise((resolve) => setTimeout(resolve, 100))
|
||||
|
||||
expect(customWatchdog.isWatching()).toBe(true)
|
||||
|
||||
await customWatchdog.stop()
|
||||
})
|
||||
|
||||
it("should handle double wildcards correctly", async () => {
|
||||
const customWatchdog = new Watchdog({
|
||||
debounceMs: 50,
|
||||
ignorePatterns: ["**/node_modules/**", "**/.git/**"],
|
||||
})
|
||||
|
||||
customWatchdog.start(tempDir)
|
||||
await new Promise((resolve) => setTimeout(resolve, 100))
|
||||
|
||||
expect(customWatchdog.isWatching()).toBe(true)
|
||||
|
||||
await customWatchdog.stop()
|
||||
})
|
||||
})
|
||||
|
||||
describe("file change detection", () => {
|
||||
@@ -333,4 +431,94 @@ describe("Watchdog", () => {
|
||||
}
|
||||
})
|
||||
})
|
||||
|
||||
describe("error handling", () => {
|
||||
it("should handle watcher errors gracefully", async () => {
|
||||
const consoleErrorSpy = vi.spyOn(console, "error").mockImplementation(() => {})
|
||||
|
||||
watchdog.start(tempDir)
|
||||
|
||||
const watcher = (watchdog as any).watcher
|
||||
if (watcher) {
|
||||
watcher.emit("error", new Error("Test watcher error"))
|
||||
}
|
||||
|
||||
await new Promise((resolve) => setTimeout(resolve, 100))
|
||||
|
||||
expect(consoleErrorSpy).toHaveBeenCalledWith(
|
||||
expect.stringContaining("Test watcher error"),
|
||||
)
|
||||
|
||||
consoleErrorSpy.mockRestore()
|
||||
})
|
||||
})
|
||||
|
||||
describe("polling mode", () => {
|
||||
it("should support polling mode", () => {
|
||||
const pollingWatchdog = new Watchdog({
|
||||
debounceMs: 50,
|
||||
usePolling: true,
|
||||
pollInterval: 500,
|
||||
})
|
||||
|
||||
pollingWatchdog.start(tempDir)
|
||||
expect(pollingWatchdog.isWatching()).toBe(true)
|
||||
|
||||
pollingWatchdog.stop()
|
||||
})
|
||||
})
|
||||
|
||||
describe("edge cases", () => {
|
||||
it("should handle flushing non-existent change", () => {
|
||||
watchdog.start(tempDir)
|
||||
const flushChange = (watchdog as any).flushChange.bind(watchdog)
|
||||
expect(() => flushChange("/non/existent/path.ts")).not.toThrow()
|
||||
})
|
||||
|
||||
it("should handle clearing timer for same file multiple times", async () => {
|
||||
const events: FileChangeEvent[] = []
|
||||
watchdog.onFileChange((event) => events.push(event))
|
||||
watchdog.start(tempDir)
|
||||
|
||||
await new Promise((resolve) => setTimeout(resolve, 100))
|
||||
|
||||
const testFile = path.join(tempDir, "test.ts")
|
||||
await fs.writeFile(testFile, "const x = 1")
|
||||
await new Promise((resolve) => setTimeout(resolve, 10))
|
||||
await fs.writeFile(testFile, "const x = 2")
|
||||
await new Promise((resolve) => setTimeout(resolve, 10))
|
||||
await fs.writeFile(testFile, "const x = 3")
|
||||
|
||||
await new Promise((resolve) => setTimeout(resolve, 200))
|
||||
|
||||
expect(events.length).toBeGreaterThanOrEqual(0)
|
||||
})
|
||||
|
||||
it("should normalize file paths", async () => {
|
||||
const events: FileChangeEvent[] = []
|
||||
watchdog.onFileChange((event) => {
|
||||
events.push(event)
|
||||
expect(path.isAbsolute(event.path)).toBe(true)
|
||||
})
|
||||
watchdog.start(tempDir)
|
||||
|
||||
await new Promise((resolve) => setTimeout(resolve, 100))
|
||||
|
||||
const testFile = path.join(tempDir, "normalize-test.ts")
|
||||
await fs.writeFile(testFile, "const x = 1")
|
||||
|
||||
await new Promise((resolve) => setTimeout(resolve, 200))
|
||||
})
|
||||
|
||||
it("should handle empty directory", async () => {
|
||||
const emptyDir = await fs.mkdtemp(path.join(os.tmpdir(), "empty-"))
|
||||
const emptyWatchdog = new Watchdog({ debounceMs: 50 })
|
||||
|
||||
emptyWatchdog.start(emptyDir)
|
||||
expect(emptyWatchdog.isWatching()).toBe(true)
|
||||
|
||||
await emptyWatchdog.stop()
|
||||
await fs.rm(emptyDir, { recursive: true, force: true })
|
||||
})
|
||||
})
|
||||
})
|
||||
|
||||
@@ -95,53 +95,37 @@ describe("OllamaClient", () => {
|
||||
)
|
||||
})
|
||||
|
||||
it("should pass tools when provided", async () => {
|
||||
it("should not pass tools parameter (tools are in system prompt)", async () => {
|
||||
const client = new OllamaClient(defaultConfig)
|
||||
const messages = [createUserMessage("Read file")]
|
||||
const tools = [
|
||||
{
|
||||
name: "get_lines",
|
||||
description: "Get lines from file",
|
||||
parameters: [
|
||||
{
|
||||
name: "path",
|
||||
type: "string" as const,
|
||||
description: "File path",
|
||||
required: true,
|
||||
},
|
||||
],
|
||||
},
|
||||
]
|
||||
|
||||
await client.chat(messages, tools)
|
||||
await client.chat(messages)
|
||||
|
||||
expect(mockOllamaInstance.chat).toHaveBeenCalledWith(
|
||||
expect.objectContaining({
|
||||
tools: expect.arrayContaining([
|
||||
model: "qwen2.5-coder:7b-instruct",
|
||||
messages: expect.arrayContaining([
|
||||
expect.objectContaining({
|
||||
type: "function",
|
||||
function: expect.objectContaining({
|
||||
name: "get_lines",
|
||||
}),
|
||||
role: "user",
|
||||
content: "Read file",
|
||||
}),
|
||||
]),
|
||||
}),
|
||||
)
|
||||
expect(mockOllamaInstance.chat).toHaveBeenCalledWith(
|
||||
expect.not.objectContaining({
|
||||
tools: expect.anything(),
|
||||
}),
|
||||
)
|
||||
})
|
||||
|
||||
it("should extract tool calls from response", async () => {
|
||||
it("should extract tool calls from XML in response content", async () => {
|
||||
mockOllamaInstance.chat.mockResolvedValue({
|
||||
message: {
|
||||
role: "assistant",
|
||||
content: "",
|
||||
tool_calls: [
|
||||
{
|
||||
function: {
|
||||
name: "get_lines",
|
||||
arguments: { path: "src/index.ts" },
|
||||
},
|
||||
},
|
||||
],
|
||||
content:
|
||||
'<tool_call name="get_lines"><path>src/index.ts</path></tool_call>',
|
||||
tool_calls: undefined,
|
||||
},
|
||||
eval_count: 30,
|
||||
})
|
||||
@@ -424,47 +408,6 @@ describe("OllamaClient", () => {
|
||||
})
|
||||
})
|
||||
|
||||
describe("tool parameter conversion", () => {
|
||||
it("should include enum values when present", async () => {
|
||||
const client = new OllamaClient(defaultConfig)
|
||||
const messages = [createUserMessage("Get status")]
|
||||
const tools = [
|
||||
{
|
||||
name: "get_status",
|
||||
description: "Get status",
|
||||
parameters: [
|
||||
{
|
||||
name: "type",
|
||||
type: "string" as const,
|
||||
description: "Status type",
|
||||
required: true,
|
||||
enum: ["active", "inactive", "pending"],
|
||||
},
|
||||
],
|
||||
},
|
||||
]
|
||||
|
||||
await client.chat(messages, tools)
|
||||
|
||||
expect(mockOllamaInstance.chat).toHaveBeenCalledWith(
|
||||
expect.objectContaining({
|
||||
tools: expect.arrayContaining([
|
||||
expect.objectContaining({
|
||||
function: expect.objectContaining({
|
||||
parameters: expect.objectContaining({
|
||||
properties: expect.objectContaining({
|
||||
type: expect.objectContaining({
|
||||
enum: ["active", "inactive", "pending"],
|
||||
}),
|
||||
}),
|
||||
}),
|
||||
}),
|
||||
}),
|
||||
]),
|
||||
}),
|
||||
)
|
||||
})
|
||||
})
|
||||
|
||||
describe("error handling", () => {
|
||||
it("should handle ECONNREFUSED errors", async () => {
|
||||
@@ -484,5 +427,23 @@ describe("OllamaClient", () => {
|
||||
|
||||
await expect(client.pullModel("test")).rejects.toThrow(/Failed to pull model/)
|
||||
})
|
||||
|
||||
it("should handle AbortError correctly", async () => {
|
||||
const abortError = new Error("aborted")
|
||||
abortError.name = "AbortError"
|
||||
mockOllamaInstance.chat.mockRejectedValue(abortError)
|
||||
|
||||
const client = new OllamaClient(defaultConfig)
|
||||
|
||||
await expect(client.chat([createUserMessage("Hello")])).rejects.toThrow(/Request was aborted/)
|
||||
})
|
||||
|
||||
it("should handle model not found errors", async () => {
|
||||
mockOllamaInstance.chat.mockRejectedValue(new Error("model 'unknown' not found"))
|
||||
|
||||
const client = new OllamaClient(defaultConfig)
|
||||
|
||||
await expect(client.chat([createUserMessage("Hello")])).rejects.toThrow(/Model.*not found/)
|
||||
})
|
||||
})
|
||||
})
|
||||
|
||||
@@ -72,7 +72,7 @@ describe("ResponseParser", () => {
|
||||
})
|
||||
|
||||
it("should parse null values", () => {
|
||||
const response = `<tool_call name="test">
|
||||
const response = `<tool_call name="get_lines">
|
||||
<value>null</value>
|
||||
</tool_call>`
|
||||
|
||||
@@ -92,7 +92,7 @@ describe("ResponseParser", () => {
|
||||
})
|
||||
|
||||
it("should parse JSON objects", () => {
|
||||
const response = `<tool_call name="test">
|
||||
const response = `<tool_call name="get_lines">
|
||||
<config>{"key": "value"}</config>
|
||||
</tool_call>`
|
||||
|
||||
@@ -123,6 +123,59 @@ describe("ResponseParser", () => {
|
||||
start: 5,
|
||||
})
|
||||
})
|
||||
|
||||
it("should reject unknown tool names", () => {
|
||||
const response = `<tool_call name="unknown_tool"><path>test.ts</path></tool_call>`
|
||||
|
||||
const result = parseToolCalls(response)
|
||||
|
||||
expect(result.toolCalls).toHaveLength(0)
|
||||
expect(result.hasParseErrors).toBe(true)
|
||||
expect(result.parseErrors[0]).toContain("Unknown tool")
|
||||
expect(result.parseErrors[0]).toContain("unknown_tool")
|
||||
})
|
||||
|
||||
it("should support CDATA for multiline content", () => {
|
||||
const response = `<tool_call name="edit_lines">
|
||||
<path>src/index.ts</path>
|
||||
<content><![CDATA[const x = 1;
|
||||
const y = 2;]]></content>
|
||||
</tool_call>`
|
||||
|
||||
const result = parseToolCalls(response)
|
||||
|
||||
expect(result.toolCalls[0].params.content).toBe("const x = 1;\nconst y = 2;")
|
||||
})
|
||||
|
||||
it("should handle multiple tool calls with mixed content", () => {
|
||||
const response = `Some text
|
||||
<tool_call name="get_lines"><path>a.ts</path></tool_call>
|
||||
More text
|
||||
<tool_call name="get_function"><path>b.ts</path><name>foo</name></tool_call>`
|
||||
|
||||
const result = parseToolCalls(response)
|
||||
|
||||
expect(result.toolCalls).toHaveLength(2)
|
||||
expect(result.toolCalls[0].name).toBe("get_lines")
|
||||
expect(result.toolCalls[1].name).toBe("get_function")
|
||||
expect(result.content).toContain("Some text")
|
||||
expect(result.content).toContain("More text")
|
||||
})
|
||||
|
||||
it("should handle parse errors gracefully and continue", () => {
|
||||
const response = `<tool_call name="unknown_tool1"><path>test.ts</path></tool_call>
|
||||
<tool_call name="get_lines"><path>valid.ts</path></tool_call>
|
||||
<tool_call name="unknown_tool2"><path>test2.ts</path></tool_call>`
|
||||
|
||||
const result = parseToolCalls(response)
|
||||
|
||||
expect(result.toolCalls).toHaveLength(1)
|
||||
expect(result.toolCalls[0].name).toBe("get_lines")
|
||||
expect(result.hasParseErrors).toBe(true)
|
||||
expect(result.parseErrors).toHaveLength(2)
|
||||
expect(result.parseErrors[0]).toContain("unknown_tool1")
|
||||
expect(result.parseErrors[1]).toContain("unknown_tool2")
|
||||
})
|
||||
})
|
||||
|
||||
describe("formatToolCallsAsXml", () => {
|
||||
|
||||
@@ -344,5 +344,47 @@ describe("GetClassTool", () => {
|
||||
|
||||
expect(result.callId).toMatch(/^get_class-\d+$/)
|
||||
})
|
||||
|
||||
it("should handle undefined extends in class", async () => {
|
||||
const lines = ["class StandaloneClass { method() {} }"]
|
||||
const cls = createMockClass({
|
||||
name: "StandaloneClass",
|
||||
lineStart: 1,
|
||||
lineEnd: 1,
|
||||
extends: undefined,
|
||||
methods: [{ name: "method", lineStart: 1, lineEnd: 1 }],
|
||||
})
|
||||
const ast = createMockAST([cls])
|
||||
const storage = createMockStorage({ lines }, ast)
|
||||
const ctx = createMockContext(storage)
|
||||
|
||||
const result = await tool.execute({ path: "test.ts", name: "StandaloneClass" }, ctx)
|
||||
|
||||
expect(result.success).toBe(true)
|
||||
const data = result.data as GetClassResult
|
||||
expect(data.extends).toBeUndefined()
|
||||
expect(data.methods.length).toBe(1)
|
||||
})
|
||||
|
||||
it("should handle error when reading lines fails", async () => {
|
||||
const ast = createMockAST([createMockClass({ name: "Test", lineStart: 1, lineEnd: 1 })])
|
||||
const storage: IStorage = {
|
||||
getFile: vi.fn().mockResolvedValue(null),
|
||||
getAST: vi.fn().mockResolvedValue(ast),
|
||||
setFile: vi.fn(),
|
||||
deleteFile: vi.fn(),
|
||||
getAllFiles: vi.fn(),
|
||||
setAST: vi.fn(),
|
||||
getSymbolIndex: vi.fn(),
|
||||
setSymbolIndex: vi.fn(),
|
||||
getDepsGraph: vi.fn(),
|
||||
setDepsGraph: vi.fn(),
|
||||
}
|
||||
const ctx = createMockContext(storage)
|
||||
|
||||
const result = await tool.execute({ path: "test.ts", name: "Test" }, ctx)
|
||||
|
||||
expect(result.success).toBe(false)
|
||||
})
|
||||
})
|
||||
})
|
||||
|
||||
@@ -301,5 +301,47 @@ describe("GetFunctionTool", () => {
|
||||
const data = result.data as GetFunctionResult
|
||||
expect(data.params).toEqual([])
|
||||
})
|
||||
|
||||
it("should handle error when reading lines fails", async () => {
|
||||
const ast = createMockAST([createMockFunction({ name: "test", lineStart: 1, lineEnd: 1 })])
|
||||
const storage: IStorage = {
|
||||
getFile: vi.fn().mockResolvedValue(null),
|
||||
getAST: vi.fn().mockResolvedValue(ast),
|
||||
setFile: vi.fn(),
|
||||
deleteFile: vi.fn(),
|
||||
getAllFiles: vi.fn(),
|
||||
setAST: vi.fn(),
|
||||
getSymbolIndex: vi.fn(),
|
||||
setSymbolIndex: vi.fn(),
|
||||
getDepsGraph: vi.fn(),
|
||||
setDepsGraph: vi.fn(),
|
||||
}
|
||||
const ctx = createMockContext(storage)
|
||||
|
||||
const result = await tool.execute({ path: "test.ts", name: "test" }, ctx)
|
||||
|
||||
expect(result.success).toBe(false)
|
||||
})
|
||||
|
||||
it("should handle undefined returnType", async () => {
|
||||
const lines = ["function implicitReturn() { return }"]
|
||||
const func = createMockFunction({
|
||||
name: "implicitReturn",
|
||||
lineStart: 1,
|
||||
lineEnd: 1,
|
||||
returnType: undefined,
|
||||
isAsync: false,
|
||||
})
|
||||
const ast = createMockAST([func])
|
||||
const storage = createMockStorage({ lines }, ast)
|
||||
const ctx = createMockContext(storage)
|
||||
|
||||
const result = await tool.execute({ path: "test.ts", name: "implicitReturn" }, ctx)
|
||||
|
||||
expect(result.success).toBe(true)
|
||||
const data = result.data as GetFunctionResult
|
||||
expect(data.returnType).toBeUndefined()
|
||||
expect(data.isAsync).toBe(false)
|
||||
})
|
||||
})
|
||||
})
|
||||
|
||||
@@ -269,5 +269,69 @@ describe("GetLinesTool", () => {
|
||||
expect(data.totalLines).toBe(1)
|
||||
expect(data.content).toBe("1│only line")
|
||||
})
|
||||
|
||||
it("should read from filesystem fallback when not in storage", async () => {
|
||||
const storage: IStorage = {
|
||||
getFile: vi.fn().mockResolvedValue(null),
|
||||
setFile: vi.fn(),
|
||||
deleteFile: vi.fn(),
|
||||
getAllFiles: vi.fn(),
|
||||
getAST: vi.fn(),
|
||||
setAST: vi.fn(),
|
||||
getSymbolIndex: vi.fn(),
|
||||
setSymbolIndex: vi.fn(),
|
||||
getDepsGraph: vi.fn(),
|
||||
setDepsGraph: vi.fn(),
|
||||
}
|
||||
|
||||
const ctx = createMockContext(storage)
|
||||
|
||||
const result = await tool.execute({ path: "test.ts" }, ctx)
|
||||
|
||||
expect(storage.getFile).toHaveBeenCalledWith("test.ts")
|
||||
|
||||
if (result.success) {
|
||||
expect(result.success).toBe(true)
|
||||
} else {
|
||||
expect(result.error).toBeDefined()
|
||||
}
|
||||
})
|
||||
|
||||
it("should handle when start equals end", async () => {
|
||||
const storage = createMockStorage({ lines: ["line 1", "line 2", "line 3"] })
|
||||
const ctx = createMockContext(storage)
|
||||
|
||||
const result = await tool.execute({ path: "test.ts", start: 2, end: 2 }, ctx)
|
||||
|
||||
expect(result.success).toBe(true)
|
||||
const data = result.data as GetLinesResult
|
||||
expect(data.startLine).toBe(2)
|
||||
expect(data.endLine).toBe(2)
|
||||
expect(data.content).toContain("line 2")
|
||||
})
|
||||
|
||||
it("should handle undefined end parameter", async () => {
|
||||
const storage = createMockStorage({ lines: ["line 1", "line 2", "line 3"] })
|
||||
const ctx = createMockContext(storage)
|
||||
|
||||
const result = await tool.execute({ path: "test.ts", start: 2, end: undefined }, ctx)
|
||||
|
||||
expect(result.success).toBe(true)
|
||||
const data = result.data as GetLinesResult
|
||||
expect(data.startLine).toBe(2)
|
||||
expect(data.endLine).toBe(3)
|
||||
})
|
||||
|
||||
it("should handle undefined start parameter", async () => {
|
||||
const storage = createMockStorage({ lines: ["line 1", "line 2", "line 3"] })
|
||||
const ctx = createMockContext(storage)
|
||||
|
||||
const result = await tool.execute({ path: "test.ts", start: undefined, end: 2 }, ctx)
|
||||
|
||||
expect(result.success).toBe(true)
|
||||
const data = result.data as GetLinesResult
|
||||
expect(data.startLine).toBe(1)
|
||||
expect(data.endLine).toBe(2)
|
||||
})
|
||||
})
|
||||
})
|
||||
|
||||
@@ -181,4 +181,170 @@ describe("Input", () => {
|
||||
expect(savedInput).toBe("")
|
||||
})
|
||||
})
|
||||
|
||||
describe("multiline support", () => {
|
||||
describe("InputProps with multiline", () => {
|
||||
it("should accept multiline as boolean", () => {
|
||||
const props: InputProps = {
|
||||
onSubmit: vi.fn(),
|
||||
history: [],
|
||||
disabled: false,
|
||||
multiline: true,
|
||||
}
|
||||
expect(props.multiline).toBe(true)
|
||||
})
|
||||
|
||||
it("should accept multiline as 'auto'", () => {
|
||||
const props: InputProps = {
|
||||
onSubmit: vi.fn(),
|
||||
history: [],
|
||||
disabled: false,
|
||||
multiline: "auto",
|
||||
}
|
||||
expect(props.multiline).toBe("auto")
|
||||
})
|
||||
|
||||
it("should have multiline false by default", () => {
|
||||
const props: InputProps = {
|
||||
onSubmit: vi.fn(),
|
||||
history: [],
|
||||
disabled: false,
|
||||
}
|
||||
expect(props.multiline).toBeUndefined()
|
||||
})
|
||||
})
|
||||
|
||||
describe("multiline activation logic", () => {
|
||||
it("should be active when multiline is true", () => {
|
||||
const multiline = true
|
||||
const lines = ["single line"]
|
||||
const isMultilineActive = multiline === true || (multiline === "auto" && lines.length > 1)
|
||||
expect(isMultilineActive).toBe(true)
|
||||
})
|
||||
|
||||
it("should not be active when multiline is false", () => {
|
||||
const multiline = false
|
||||
const lines = ["line1", "line2"]
|
||||
const isMultilineActive = multiline === true || (multiline === "auto" && lines.length > 1)
|
||||
expect(isMultilineActive).toBe(false)
|
||||
})
|
||||
|
||||
it("should be active in auto mode with multiple lines", () => {
|
||||
const multiline = "auto"
|
||||
const lines = ["line1", "line2"]
|
||||
const isMultilineActive = multiline === true || (multiline === "auto" && lines.length > 1)
|
||||
expect(isMultilineActive).toBe(true)
|
||||
})
|
||||
|
||||
it("should not be active in auto mode with single line", () => {
|
||||
const multiline = "auto"
|
||||
const lines = ["single line"]
|
||||
const isMultilineActive = multiline === true || (multiline === "auto" && lines.length > 1)
|
||||
expect(isMultilineActive).toBe(false)
|
||||
})
|
||||
})
|
||||
|
||||
describe("line management", () => {
|
||||
it("should update current line on change", () => {
|
||||
const lines = ["first", "second", "third"]
|
||||
const currentLineIndex = 1
|
||||
const newValue = "updated second"
|
||||
|
||||
const newLines = [...lines]
|
||||
newLines[currentLineIndex] = newValue
|
||||
|
||||
expect(newLines).toEqual(["first", "updated second", "third"])
|
||||
expect(newLines.join("\n")).toBe("first\nupdated second\nthird")
|
||||
})
|
||||
|
||||
it("should add new line at current position", () => {
|
||||
const lines = ["first", "second"]
|
||||
const currentLineIndex = 0
|
||||
|
||||
const newLines = [...lines]
|
||||
newLines.splice(currentLineIndex + 1, 0, "")
|
||||
|
||||
expect(newLines).toEqual(["first", "", "second"])
|
||||
})
|
||||
|
||||
it("should join lines with newline for submit", () => {
|
||||
const lines = ["line 1", "line 2", "line 3"]
|
||||
const fullText = lines.join("\n")
|
||||
expect(fullText).toBe("line 1\nline 2\nline 3")
|
||||
})
|
||||
})
|
||||
|
||||
describe("line navigation", () => {
|
||||
it("should navigate up in multiline mode", () => {
|
||||
const lines = ["line1", "line2", "line3"]
|
||||
let currentLineIndex = 2
|
||||
|
||||
currentLineIndex = currentLineIndex - 1
|
||||
expect(currentLineIndex).toBe(1)
|
||||
|
||||
currentLineIndex = currentLineIndex - 1
|
||||
expect(currentLineIndex).toBe(0)
|
||||
})
|
||||
|
||||
it("should not navigate up past first line", () => {
|
||||
const lines = ["line1", "line2"]
|
||||
const currentLineIndex = 0
|
||||
const isMultilineActive = true
|
||||
|
||||
const canNavigateUp = isMultilineActive && currentLineIndex > 0
|
||||
expect(canNavigateUp).toBe(false)
|
||||
})
|
||||
|
||||
it("should navigate down in multiline mode", () => {
|
||||
const lines = ["line1", "line2", "line3"]
|
||||
let currentLineIndex = 0
|
||||
|
||||
currentLineIndex = currentLineIndex + 1
|
||||
expect(currentLineIndex).toBe(1)
|
||||
|
||||
currentLineIndex = currentLineIndex + 1
|
||||
expect(currentLineIndex).toBe(2)
|
||||
})
|
||||
|
||||
it("should not navigate down past last line", () => {
|
||||
const lines = ["line1", "line2"]
|
||||
const currentLineIndex = 1
|
||||
const isMultilineActive = true
|
||||
|
||||
const canNavigateDown = isMultilineActive && currentLineIndex < lines.length - 1
|
||||
expect(canNavigateDown).toBe(false)
|
||||
})
|
||||
})
|
||||
|
||||
describe("multiline submit", () => {
|
||||
it("should submit trimmed multiline text", () => {
|
||||
const lines = ["line 1", "line 2", "line 3"]
|
||||
const fullText = lines.join("\n").trim()
|
||||
expect(fullText).toBe("line 1\nline 2\nline 3")
|
||||
})
|
||||
|
||||
it("should not submit empty multiline text", () => {
|
||||
const onSubmit = vi.fn()
|
||||
const lines = ["", "", ""]
|
||||
const fullText = lines.join("\n").trim()
|
||||
|
||||
if (fullText) {
|
||||
onSubmit(fullText)
|
||||
}
|
||||
|
||||
expect(onSubmit).not.toHaveBeenCalled()
|
||||
})
|
||||
|
||||
it("should reset lines after submit", () => {
|
||||
let lines = ["line1", "line2"]
|
||||
let currentLineIndex = 1
|
||||
|
||||
lines = [""]
|
||||
currentLineIndex = 0
|
||||
|
||||
expect(lines).toEqual([""])
|
||||
expect(currentLineIndex).toBe(0)
|
||||
})
|
||||
})
|
||||
})
|
||||
})
|
||||
|
||||
539
packages/ipuaro/tests/unit/tui/hooks/useAutocomplete.test.ts
Normal file
539
packages/ipuaro/tests/unit/tui/hooks/useAutocomplete.test.ts
Normal file
@@ -0,0 +1,539 @@
|
||||
/**
|
||||
* Unit tests for useAutocomplete hook.
|
||||
*/
|
||||
|
||||
import { describe, it, expect, beforeEach, vi } from "vitest"
|
||||
import { renderHook, act, waitFor } from "@testing-library/react"
|
||||
import { useAutocomplete } from "../../../../src/tui/hooks/useAutocomplete.js"
|
||||
import type { IStorage } from "../../../../src/domain/services/IStorage.js"
|
||||
import type { FileData } from "../../../../src/domain/value-objects/FileData.js"
|
||||
|
||||
function createMockStorage(files: Map<string, FileData>): IStorage {
|
||||
return {
|
||||
getAllFiles: vi.fn().mockResolvedValue(files),
|
||||
getFile: vi.fn(),
|
||||
setFile: vi.fn(),
|
||||
deleteFile: vi.fn(),
|
||||
getFileCount: vi.fn(),
|
||||
getAST: vi.fn(),
|
||||
setAST: vi.fn(),
|
||||
deleteAST: vi.fn(),
|
||||
getAllASTs: vi.fn(),
|
||||
getMeta: vi.fn(),
|
||||
setMeta: vi.fn(),
|
||||
deleteMeta: vi.fn(),
|
||||
getAllMetas: vi.fn(),
|
||||
getSymbolIndex: vi.fn(),
|
||||
setSymbolIndex: vi.fn(),
|
||||
getDepsGraph: vi.fn(),
|
||||
setDepsGraph: vi.fn(),
|
||||
getProjectConfig: vi.fn(),
|
||||
setProjectConfig: vi.fn(),
|
||||
connect: vi.fn(),
|
||||
disconnect: vi.fn(),
|
||||
isConnected: vi.fn(),
|
||||
clear: vi.fn(),
|
||||
} as unknown as IStorage
|
||||
}
|
||||
|
||||
function createFileData(content: string): FileData {
|
||||
return {
|
||||
lines: content.split("\n"),
|
||||
hash: "test-hash",
|
||||
size: content.length,
|
||||
lastModified: Date.now(),
|
||||
}
|
||||
}
|
||||
|
||||
describe("useAutocomplete", () => {
|
||||
const projectRoot = "/test/project"
|
||||
|
||||
beforeEach(() => {
|
||||
vi.clearAllMocks()
|
||||
})
|
||||
|
||||
describe("initialization", () => {
|
||||
it("should load file paths from storage", async () => {
|
||||
const files = new Map<string, FileData>([
|
||||
["/test/project/src/index.ts", createFileData("test")],
|
||||
["/test/project/src/utils.ts", createFileData("test")],
|
||||
["/test/project/README.md", createFileData("test")],
|
||||
])
|
||||
const storage = createMockStorage(files)
|
||||
|
||||
const { result } = renderHook(() =>
|
||||
useAutocomplete({ storage, projectRoot, enabled: true }),
|
||||
)
|
||||
|
||||
await waitFor(() => {
|
||||
expect(storage.getAllFiles).toHaveBeenCalledTimes(1)
|
||||
})
|
||||
|
||||
expect(result.current.suggestions).toEqual([])
|
||||
})
|
||||
|
||||
it("should not load paths when disabled", async () => {
|
||||
const files = new Map<string, FileData>()
|
||||
const storage = createMockStorage(files)
|
||||
|
||||
renderHook(() => useAutocomplete({ storage, projectRoot, enabled: false }))
|
||||
|
||||
await new Promise((resolve) => setTimeout(resolve, 50))
|
||||
expect(storage.getAllFiles).not.toHaveBeenCalled()
|
||||
})
|
||||
|
||||
it("should handle storage errors gracefully", async () => {
|
||||
const storage = {
|
||||
...createMockStorage(new Map()),
|
||||
getAllFiles: vi.fn().mockRejectedValue(new Error("Storage error")),
|
||||
} as unknown as IStorage
|
||||
|
||||
const { result } = renderHook(() =>
|
||||
useAutocomplete({ storage, projectRoot, enabled: true }),
|
||||
)
|
||||
|
||||
await waitFor(() => {
|
||||
expect(storage.getAllFiles).toHaveBeenCalled()
|
||||
})
|
||||
|
||||
// Should not crash, suggestions should be empty
|
||||
expect(result.current.suggestions).toEqual([])
|
||||
})
|
||||
})
|
||||
|
||||
describe("complete", () => {
|
||||
it("should return empty array for empty input", async () => {
|
||||
const files = new Map<string, FileData>([
|
||||
["/test/project/src/index.ts", createFileData("test")],
|
||||
])
|
||||
const storage = createMockStorage(files)
|
||||
|
||||
const { result } = renderHook(() =>
|
||||
useAutocomplete({ storage, projectRoot, enabled: true }),
|
||||
)
|
||||
|
||||
await waitFor(() => {
|
||||
expect(storage.getAllFiles).toHaveBeenCalled()
|
||||
})
|
||||
|
||||
let suggestions: string[] = []
|
||||
act(() => {
|
||||
suggestions = result.current.complete("")
|
||||
})
|
||||
|
||||
expect(suggestions).toEqual([])
|
||||
})
|
||||
|
||||
it("should return exact prefix matches", async () => {
|
||||
const files = new Map<string, FileData>([
|
||||
["/test/project/src/index.ts", createFileData("test")],
|
||||
["/test/project/src/utils.ts", createFileData("test")],
|
||||
["/test/project/tests/index.test.ts", createFileData("test")],
|
||||
])
|
||||
const storage = createMockStorage(files)
|
||||
|
||||
const { result } = renderHook(() =>
|
||||
useAutocomplete({ storage, projectRoot, enabled: true }),
|
||||
)
|
||||
|
||||
await waitFor(() => {
|
||||
expect(storage.getAllFiles).toHaveBeenCalled()
|
||||
})
|
||||
|
||||
let suggestions: string[] = []
|
||||
act(() => {
|
||||
suggestions = result.current.complete("src/")
|
||||
})
|
||||
|
||||
expect(suggestions).toHaveLength(2)
|
||||
expect(suggestions).toContain("src/index.ts")
|
||||
expect(suggestions).toContain("src/utils.ts")
|
||||
})
|
||||
|
||||
it("should support fuzzy matching", async () => {
|
||||
const files = new Map<string, FileData>([
|
||||
["/test/project/src/components/Button.tsx", createFileData("test")],
|
||||
["/test/project/src/utils/helpers.ts", createFileData("test")],
|
||||
["/test/project/tests/unit/button.test.ts", createFileData("test")],
|
||||
])
|
||||
const storage = createMockStorage(files)
|
||||
|
||||
const { result } = renderHook(() =>
|
||||
useAutocomplete({ storage, projectRoot, enabled: true }),
|
||||
)
|
||||
|
||||
await waitFor(() => {
|
||||
expect(storage.getAllFiles).toHaveBeenCalled()
|
||||
})
|
||||
|
||||
let suggestions: string[] = []
|
||||
act(() => {
|
||||
suggestions = result.current.complete("btn")
|
||||
})
|
||||
|
||||
// Should match "Button.tsx" and "button.test.ts" (fuzzy match)
|
||||
expect(suggestions.length).toBeGreaterThan(0)
|
||||
expect(suggestions.some((s) => s.includes("Button.tsx"))).toBe(true)
|
||||
})
|
||||
|
||||
it("should respect maxSuggestions limit", async () => {
|
||||
const files = new Map<string, FileData>()
|
||||
for (let i = 0; i < 20; i++) {
|
||||
files.set(`/test/project/file${i}.ts`, createFileData("test"))
|
||||
}
|
||||
const storage = createMockStorage(files)
|
||||
|
||||
const { result } = renderHook(() =>
|
||||
useAutocomplete({ storage, projectRoot, enabled: true, maxSuggestions: 5 }),
|
||||
)
|
||||
|
||||
await waitFor(() => {
|
||||
expect(storage.getAllFiles).toHaveBeenCalled()
|
||||
})
|
||||
|
||||
let suggestions: string[] = []
|
||||
act(() => {
|
||||
suggestions = result.current.complete("file")
|
||||
})
|
||||
|
||||
expect(suggestions.length).toBeLessThanOrEqual(5)
|
||||
})
|
||||
|
||||
it("should normalize paths with leading ./", async () => {
|
||||
const files = new Map<string, FileData>([
|
||||
["/test/project/src/index.ts", createFileData("test")],
|
||||
])
|
||||
const storage = createMockStorage(files)
|
||||
|
||||
const { result } = renderHook(() =>
|
||||
useAutocomplete({ storage, projectRoot, enabled: true }),
|
||||
)
|
||||
|
||||
await waitFor(() => {
|
||||
expect(storage.getAllFiles).toHaveBeenCalled()
|
||||
})
|
||||
|
||||
let suggestions: string[] = []
|
||||
act(() => {
|
||||
suggestions = result.current.complete("./src/index")
|
||||
})
|
||||
|
||||
expect(suggestions).toContain("src/index.ts")
|
||||
})
|
||||
|
||||
it("should handle paths with trailing slash", async () => {
|
||||
const files = new Map<string, FileData>([
|
||||
["/test/project/src/index.ts", createFileData("test")],
|
||||
["/test/project/src/utils.ts", createFileData("test")],
|
||||
])
|
||||
const storage = createMockStorage(files)
|
||||
|
||||
const { result } = renderHook(() =>
|
||||
useAutocomplete({ storage, projectRoot, enabled: true }),
|
||||
)
|
||||
|
||||
await waitFor(() => {
|
||||
expect(storage.getAllFiles).toHaveBeenCalled()
|
||||
})
|
||||
|
||||
let suggestions: string[] = []
|
||||
act(() => {
|
||||
suggestions = result.current.complete("src/")
|
||||
})
|
||||
|
||||
expect(suggestions.length).toBeGreaterThan(0)
|
||||
})
|
||||
|
||||
it("should be case-insensitive", async () => {
|
||||
const files = new Map<string, FileData>([
|
||||
["/test/project/src/UserService.ts", createFileData("test")],
|
||||
])
|
||||
const storage = createMockStorage(files)
|
||||
|
||||
const { result } = renderHook(() =>
|
||||
useAutocomplete({ storage, projectRoot, enabled: true }),
|
||||
)
|
||||
|
||||
await waitFor(() => {
|
||||
expect(storage.getAllFiles).toHaveBeenCalled()
|
||||
})
|
||||
|
||||
let suggestions: string[] = []
|
||||
act(() => {
|
||||
suggestions = result.current.complete("userservice")
|
||||
})
|
||||
|
||||
expect(suggestions).toContain("src/UserService.ts")
|
||||
})
|
||||
|
||||
it("should update suggestions state", async () => {
|
||||
const files = new Map<string, FileData>([
|
||||
["/test/project/src/index.ts", createFileData("test")],
|
||||
])
|
||||
const storage = createMockStorage(files)
|
||||
|
||||
const { result } = renderHook(() =>
|
||||
useAutocomplete({ storage, projectRoot, enabled: true }),
|
||||
)
|
||||
|
||||
await waitFor(() => {
|
||||
expect(storage.getAllFiles).toHaveBeenCalled()
|
||||
})
|
||||
|
||||
expect(result.current.suggestions).toEqual([])
|
||||
|
||||
act(() => {
|
||||
result.current.complete("src/")
|
||||
})
|
||||
|
||||
expect(result.current.suggestions.length).toBeGreaterThan(0)
|
||||
})
|
||||
})
|
||||
|
||||
describe("accept", () => {
|
||||
it("should return single suggestion when only one exists", async () => {
|
||||
const files = new Map<string, FileData>([
|
||||
["/test/project/src/unique-file.ts", createFileData("test")],
|
||||
])
|
||||
const storage = createMockStorage(files)
|
||||
|
||||
const { result } = renderHook(() =>
|
||||
useAutocomplete({ storage, projectRoot, enabled: true }),
|
||||
)
|
||||
|
||||
await waitFor(() => {
|
||||
expect(storage.getAllFiles).toHaveBeenCalled()
|
||||
})
|
||||
|
||||
act(() => {
|
||||
result.current.complete("unique")
|
||||
})
|
||||
|
||||
let accepted = ""
|
||||
act(() => {
|
||||
accepted = result.current.accept("unique")
|
||||
})
|
||||
|
||||
expect(accepted).toBe("src/unique-file.ts")
|
||||
expect(result.current.suggestions).toEqual([])
|
||||
})
|
||||
|
||||
it("should return common prefix for multiple suggestions", async () => {
|
||||
const files = new Map<string, FileData>([
|
||||
["/test/project/src/components/Button.tsx", createFileData("test")],
|
||||
["/test/project/src/components/ButtonGroup.tsx", createFileData("test")],
|
||||
])
|
||||
const storage = createMockStorage(files)
|
||||
|
||||
const { result } = renderHook(() =>
|
||||
useAutocomplete({ storage, projectRoot, enabled: true }),
|
||||
)
|
||||
|
||||
await waitFor(() => {
|
||||
expect(storage.getAllFiles).toHaveBeenCalled()
|
||||
})
|
||||
|
||||
act(() => {
|
||||
result.current.complete("src/comp")
|
||||
})
|
||||
|
||||
let accepted = ""
|
||||
act(() => {
|
||||
accepted = result.current.accept("src/comp")
|
||||
})
|
||||
|
||||
// Common prefix is "src/components/Button"
|
||||
expect(accepted.startsWith("src/components/Button")).toBe(true)
|
||||
})
|
||||
|
||||
it("should return input if no common prefix extension", async () => {
|
||||
const files = new Map<string, FileData>([
|
||||
["/test/project/src/foo.ts", createFileData("test")],
|
||||
["/test/project/src/bar.ts", createFileData("test")],
|
||||
])
|
||||
const storage = createMockStorage(files)
|
||||
|
||||
const { result } = renderHook(() =>
|
||||
useAutocomplete({ storage, projectRoot, enabled: true }),
|
||||
)
|
||||
|
||||
await waitFor(() => {
|
||||
expect(storage.getAllFiles).toHaveBeenCalled()
|
||||
})
|
||||
|
||||
act(() => {
|
||||
result.current.complete("src/")
|
||||
})
|
||||
|
||||
let accepted = ""
|
||||
act(() => {
|
||||
accepted = result.current.accept("src/")
|
||||
})
|
||||
|
||||
// Common prefix is just "src/" which is same as input
|
||||
expect(accepted).toBe("src/")
|
||||
})
|
||||
})
|
||||
|
||||
describe("reset", () => {
|
||||
it("should clear suggestions", async () => {
|
||||
const files = new Map<string, FileData>([
|
||||
["/test/project/src/index.ts", createFileData("test")],
|
||||
])
|
||||
const storage = createMockStorage(files)
|
||||
|
||||
const { result } = renderHook(() =>
|
||||
useAutocomplete({ storage, projectRoot, enabled: true }),
|
||||
)
|
||||
|
||||
await waitFor(() => {
|
||||
expect(storage.getAllFiles).toHaveBeenCalled()
|
||||
})
|
||||
|
||||
act(() => {
|
||||
result.current.complete("src/")
|
||||
})
|
||||
|
||||
expect(result.current.suggestions.length).toBeGreaterThan(0)
|
||||
|
||||
act(() => {
|
||||
result.current.reset()
|
||||
})
|
||||
|
||||
expect(result.current.suggestions).toEqual([])
|
||||
})
|
||||
})
|
||||
|
||||
describe("edge cases", () => {
|
||||
it("should handle empty file list", async () => {
|
||||
const files = new Map<string, FileData>()
|
||||
const storage = createMockStorage(files)
|
||||
|
||||
const { result } = renderHook(() =>
|
||||
useAutocomplete({ storage, projectRoot, enabled: true }),
|
||||
)
|
||||
|
||||
await waitFor(() => {
|
||||
expect(storage.getAllFiles).toHaveBeenCalled()
|
||||
})
|
||||
|
||||
let suggestions: string[] = []
|
||||
act(() => {
|
||||
suggestions = result.current.complete("anything")
|
||||
})
|
||||
|
||||
expect(suggestions).toEqual([])
|
||||
})
|
||||
|
||||
it("should handle whitespace-only input", async () => {
|
||||
const files = new Map<string, FileData>([
|
||||
["/test/project/src/index.ts", createFileData("test")],
|
||||
])
|
||||
const storage = createMockStorage(files)
|
||||
|
||||
const { result } = renderHook(() =>
|
||||
useAutocomplete({ storage, projectRoot, enabled: true }),
|
||||
)
|
||||
|
||||
await waitFor(() => {
|
||||
expect(storage.getAllFiles).toHaveBeenCalled()
|
||||
})
|
||||
|
||||
let suggestions: string[] = []
|
||||
act(() => {
|
||||
suggestions = result.current.complete(" ")
|
||||
})
|
||||
|
||||
expect(suggestions).toEqual([])
|
||||
})
|
||||
|
||||
it("should handle paths with special characters", async () => {
|
||||
const files = new Map<string, FileData>([
|
||||
["/test/project/src/my-file.ts", createFileData("test")],
|
||||
["/test/project/src/my_file.ts", createFileData("test")],
|
||||
])
|
||||
const storage = createMockStorage(files)
|
||||
|
||||
const { result } = renderHook(() =>
|
||||
useAutocomplete({ storage, projectRoot, enabled: true }),
|
||||
)
|
||||
|
||||
await waitFor(() => {
|
||||
expect(storage.getAllFiles).toHaveBeenCalled()
|
||||
})
|
||||
|
||||
let suggestions: string[] = []
|
||||
act(() => {
|
||||
suggestions = result.current.complete("my-")
|
||||
})
|
||||
|
||||
expect(suggestions).toContain("src/my-file.ts")
|
||||
})
|
||||
|
||||
it("should return empty suggestions when disabled", async () => {
|
||||
const files = new Map<string, FileData>([
|
||||
["/test/project/src/index.ts", createFileData("test")],
|
||||
])
|
||||
const storage = createMockStorage(files)
|
||||
|
||||
const { result } = renderHook(() =>
|
||||
useAutocomplete({ storage, projectRoot, enabled: false }),
|
||||
)
|
||||
|
||||
// Give time for any potential async operations
|
||||
await new Promise((resolve) => setTimeout(resolve, 50))
|
||||
|
||||
let suggestions: string[] = []
|
||||
act(() => {
|
||||
suggestions = result.current.complete("src/")
|
||||
})
|
||||
|
||||
expect(suggestions).toEqual([])
|
||||
})
|
||||
|
||||
it("should handle accept with no suggestions", async () => {
|
||||
const files = new Map<string, FileData>()
|
||||
const storage = createMockStorage(files)
|
||||
|
||||
const { result } = renderHook(() =>
|
||||
useAutocomplete({ storage, projectRoot, enabled: true }),
|
||||
)
|
||||
|
||||
await waitFor(() => {
|
||||
expect(storage.getAllFiles).toHaveBeenCalled()
|
||||
})
|
||||
|
||||
let accepted = ""
|
||||
act(() => {
|
||||
accepted = result.current.accept("test")
|
||||
})
|
||||
|
||||
// Should return the input when there are no suggestions
|
||||
expect(accepted).toBe("test")
|
||||
})
|
||||
|
||||
it("should handle common prefix calculation for single character paths", async () => {
|
||||
const files = new Map<string, FileData>([
|
||||
["/test/project/a.ts", createFileData("test")],
|
||||
["/test/project/b.ts", createFileData("test")],
|
||||
])
|
||||
const storage = createMockStorage(files)
|
||||
|
||||
const { result } = renderHook(() =>
|
||||
useAutocomplete({ storage, projectRoot, enabled: true }),
|
||||
)
|
||||
|
||||
await waitFor(() => {
|
||||
expect(storage.getAllFiles).toHaveBeenCalled()
|
||||
})
|
||||
|
||||
act(() => {
|
||||
result.current.complete("")
|
||||
})
|
||||
|
||||
// This tests edge case in common prefix calculation
|
||||
const accepted = result.current.accept("")
|
||||
expect(typeof accepted).toBe("string")
|
||||
})
|
||||
})
|
||||
})
|
||||
155
packages/ipuaro/tests/unit/tui/utils/syntax-highlighter.test.ts
Normal file
155
packages/ipuaro/tests/unit/tui/utils/syntax-highlighter.test.ts
Normal file
@@ -0,0 +1,155 @@
|
||||
/**
|
||||
* Tests for syntax-highlighter utility.
|
||||
*/
|
||||
|
||||
import { describe, expect, it } from "vitest"
|
||||
import { detectLanguage, highlightLine } from "../../../../src/tui/utils/syntax-highlighter.js"
|
||||
|
||||
describe("syntax-highlighter", () => {
|
||||
describe("detectLanguage", () => {
|
||||
it("should detect typescript from .ts extension", () => {
|
||||
expect(detectLanguage("src/index.ts")).toBe("typescript")
|
||||
})
|
||||
|
||||
it("should detect tsx from .tsx extension", () => {
|
||||
expect(detectLanguage("src/Component.tsx")).toBe("tsx")
|
||||
})
|
||||
|
||||
it("should detect javascript from .js extension", () => {
|
||||
expect(detectLanguage("dist/bundle.js")).toBe("javascript")
|
||||
})
|
||||
|
||||
it("should detect jsx from .jsx extension", () => {
|
||||
expect(detectLanguage("src/App.jsx")).toBe("jsx")
|
||||
})
|
||||
|
||||
it("should detect json from .json extension", () => {
|
||||
expect(detectLanguage("package.json")).toBe("json")
|
||||
})
|
||||
|
||||
it("should detect yaml from .yaml extension", () => {
|
||||
expect(detectLanguage("config.yaml")).toBe("yaml")
|
||||
})
|
||||
|
||||
it("should detect yaml from .yml extension", () => {
|
||||
expect(detectLanguage("config.yml")).toBe("yaml")
|
||||
})
|
||||
|
||||
it("should return unknown for unsupported extensions", () => {
|
||||
expect(detectLanguage("image.png")).toBe("unknown")
|
||||
expect(detectLanguage("file")).toBe("unknown")
|
||||
})
|
||||
|
||||
it("should handle case insensitive extensions", () => {
|
||||
expect(detectLanguage("FILE.TS")).toBe("typescript")
|
||||
expect(detectLanguage("FILE.JSX")).toBe("jsx")
|
||||
})
|
||||
})
|
||||
|
||||
describe("highlightLine", () => {
|
||||
describe("unknown language", () => {
|
||||
it("should return plain text for unknown language", () => {
|
||||
const tokens = highlightLine("hello world", "unknown")
|
||||
expect(tokens).toEqual([{ text: "hello world", color: "white" }])
|
||||
})
|
||||
})
|
||||
|
||||
describe("json language", () => {
|
||||
it("should return plain text for json", () => {
|
||||
const tokens = highlightLine('{"key": "value"}', "json")
|
||||
expect(tokens).toEqual([{ text: '{"key": "value"}', color: "white" }])
|
||||
})
|
||||
})
|
||||
|
||||
describe("yaml language", () => {
|
||||
it("should return plain text for yaml", () => {
|
||||
const tokens = highlightLine("key: value", "yaml")
|
||||
expect(tokens).toEqual([{ text: "key: value", color: "white" }])
|
||||
})
|
||||
})
|
||||
|
||||
describe("typescript/javascript highlighting", () => {
|
||||
it("should highlight keywords", () => {
|
||||
const tokens = highlightLine("const x = 10", "typescript")
|
||||
expect(tokens[0]).toEqual({ text: "const", color: "magenta" })
|
||||
expect(tokens.find((t) => t.text === "x")).toEqual({ text: "x", color: "white" })
|
||||
})
|
||||
|
||||
it("should highlight strings with double quotes", () => {
|
||||
const tokens = highlightLine('const s = "hello"', "typescript")
|
||||
expect(tokens.find((t) => t.text === '"hello"')).toEqual({
|
||||
text: '"hello"',
|
||||
color: "green",
|
||||
})
|
||||
})
|
||||
|
||||
it("should highlight strings with single quotes", () => {
|
||||
const tokens = highlightLine("const s = 'hello'", "typescript")
|
||||
expect(tokens.find((t) => t.text === "'hello'")).toEqual({
|
||||
text: "'hello'",
|
||||
color: "green",
|
||||
})
|
||||
})
|
||||
|
||||
it("should highlight template literals", () => {
|
||||
const tokens = highlightLine("const s = `hello`", "typescript")
|
||||
expect(tokens.find((t) => t.text === "`hello`")).toEqual({
|
||||
text: "`hello`",
|
||||
color: "green",
|
||||
})
|
||||
})
|
||||
|
||||
it("should highlight numbers", () => {
|
||||
const tokens = highlightLine("const n = 42", "typescript")
|
||||
expect(tokens.find((t) => t.text === "42")).toEqual({ text: "42", color: "cyan" })
|
||||
})
|
||||
|
||||
it("should highlight single-line comments", () => {
|
||||
const tokens = highlightLine("// this is a comment", "typescript")
|
||||
expect(tokens[0]).toEqual({ text: "// this is a comment", color: "gray" })
|
||||
})
|
||||
|
||||
it("should highlight multi-line comments", () => {
|
||||
const tokens = highlightLine("/* comment */", "typescript")
|
||||
expect(tokens[0]).toEqual({ text: "/* comment */", color: "gray" })
|
||||
})
|
||||
|
||||
it("should highlight operators", () => {
|
||||
const tokens = highlightLine("x + y = z", "typescript")
|
||||
expect(tokens.find((t) => t.text === "+")).toEqual({ text: "+", color: "yellow" })
|
||||
expect(tokens.find((t) => t.text === "=")).toEqual({ text: "=", color: "yellow" })
|
||||
})
|
||||
|
||||
it("should highlight parentheses and brackets", () => {
|
||||
const tokens = highlightLine("foo(bar[0])", "typescript")
|
||||
expect(tokens.find((t) => t.text === "(")).toEqual({ text: "(", color: "yellow" })
|
||||
expect(tokens.find((t) => t.text === "[")).toEqual({ text: "[", color: "yellow" })
|
||||
expect(tokens.find((t) => t.text === "]")).toEqual({ text: "]", color: "yellow" })
|
||||
expect(tokens.find((t) => t.text === ")")).toEqual({ text: ")", color: "yellow" })
|
||||
})
|
||||
|
||||
it("should handle mixed content", () => {
|
||||
const tokens = highlightLine('const x = "test" + 42', "typescript")
|
||||
expect(tokens.find((t) => t.text === "const")).toEqual({
|
||||
text: "const",
|
||||
color: "magenta",
|
||||
})
|
||||
expect(tokens.find((t) => t.text === '"test"')).toEqual({
|
||||
text: '"test"',
|
||||
color: "green",
|
||||
})
|
||||
expect(tokens.find((t) => t.text === "42")).toEqual({ text: "42", color: "cyan" })
|
||||
})
|
||||
|
||||
it("should preserve whitespace", () => {
|
||||
const tokens = highlightLine(" const x = 10 ", "typescript")
|
||||
expect(tokens[0]).toEqual({ text: " ", color: "white" })
|
||||
})
|
||||
|
||||
it("should handle empty lines", () => {
|
||||
const tokens = highlightLine("", "typescript")
|
||||
expect(tokens).toEqual([])
|
||||
})
|
||||
})
|
||||
})
|
||||
})
|
||||
@@ -5,6 +5,10 @@ export default defineConfig({
|
||||
globals: true,
|
||||
environment: "node",
|
||||
include: ["tests/**/*.test.ts"],
|
||||
environmentMatchGlobs: [
|
||||
// Use jsdom for TUI tests (React hooks)
|
||||
["tests/unit/tui/**/*.test.ts", "jsdom"],
|
||||
],
|
||||
coverage: {
|
||||
provider: "v8",
|
||||
reporter: ["text", "html", "lcov"],
|
||||
@@ -20,7 +24,7 @@ export default defineConfig({
|
||||
thresholds: {
|
||||
lines: 95,
|
||||
functions: 95,
|
||||
branches: 90,
|
||||
branches: 91.3,
|
||||
statements: 95,
|
||||
},
|
||||
},
|
||||
|
||||
470
pnpm-lock.yaml
generated
470
pnpm-lock.yaml
generated
@@ -131,7 +131,7 @@ importers:
|
||||
version: 5.9.3
|
||||
vitest:
|
||||
specifier: ^4.0.10
|
||||
version: 4.0.13(@types/node@22.19.1)(@vitest/ui@4.0.13)(terser@5.44.1)(tsx@4.20.6)
|
||||
version: 4.0.13(@types/node@22.19.1)(@vitest/ui@4.0.13)(jsdom@27.2.0)(terser@5.44.1)(tsx@4.20.6)
|
||||
|
||||
packages/ipuaro:
|
||||
dependencies:
|
||||
@@ -175,6 +175,12 @@ importers:
|
||||
specifier: ^3.23.8
|
||||
version: 3.25.76
|
||||
devDependencies:
|
||||
'@testing-library/react':
|
||||
specifier: ^16.3.0
|
||||
version: 16.3.0(@testing-library/dom@10.4.1)(@types/react@18.3.27)(react-dom@18.3.1(react@18.3.1))(react@18.3.1)
|
||||
'@types/jsdom':
|
||||
specifier: ^27.0.0
|
||||
version: 27.0.0
|
||||
'@types/node':
|
||||
specifier: ^22.10.1
|
||||
version: 22.19.1
|
||||
@@ -187,6 +193,12 @@ importers:
|
||||
'@vitest/ui':
|
||||
specifier: ^1.6.0
|
||||
version: 1.6.1(vitest@1.6.1)
|
||||
jsdom:
|
||||
specifier: ^27.2.0
|
||||
version: 27.2.0
|
||||
react-dom:
|
||||
specifier: 18.3.1
|
||||
version: 18.3.1(react@18.3.1)
|
||||
tsup:
|
||||
specifier: ^8.3.5
|
||||
version: 8.5.1(postcss@8.5.6)(tsx@4.20.6)(typescript@5.9.3)
|
||||
@@ -195,10 +207,13 @@ importers:
|
||||
version: 5.9.3
|
||||
vitest:
|
||||
specifier: ^1.6.0
|
||||
version: 1.6.1(@types/node@22.19.1)(@vitest/ui@1.6.1)(terser@5.44.1)
|
||||
version: 1.6.1(@types/node@22.19.1)(@vitest/ui@1.6.1)(jsdom@27.2.0)(terser@5.44.1)
|
||||
|
||||
packages:
|
||||
|
||||
'@acemir/cssom@0.9.24':
|
||||
resolution: {integrity: sha512-5YjgMmAiT2rjJZU7XK1SNI7iqTy92DpaYVgG6x63FxkJ11UpYfLndHJATtinWJClAXiOlW9XWaUyAQf8pMrQPg==}
|
||||
|
||||
'@alcalzone/ansi-tokenize@0.1.3':
|
||||
resolution: {integrity: sha512-3yWxPTq3UQ/FY9p1ErPxIyfT64elWaMvM9lIHnaqpyft63tkxodF5aUElYHrdisWve5cETkh1+KBw1yJuW0aRw==}
|
||||
engines: {node: '>=14.13.1'}
|
||||
@@ -238,6 +253,15 @@ packages:
|
||||
resolution: {integrity: sha512-J4Jarr0SohdrHcb40gTL4wGPCQ952IMWF1G/MSAQfBAPvA9ZKApYhpxcY7PmehVePve+ujpus1dGsJ7dPxz8Kg==}
|
||||
engines: {node: ^18.19.1 || ^20.11.1 || >=22.0.0, npm: ^6.11.0 || ^7.5.6 || >=8.0.0, yarn: '>= 1.13.0'}
|
||||
|
||||
'@asamuzakjp/css-color@4.1.0':
|
||||
resolution: {integrity: sha512-9xiBAtLn4aNsa4mDnpovJvBn72tNEIACyvlqaNJ+ADemR+yeMJWnBudOi2qGDviJa7SwcDOU/TRh5dnET7qk0w==}
|
||||
|
||||
'@asamuzakjp/dom-selector@6.7.5':
|
||||
resolution: {integrity: sha512-Eks6dY8zau4m4wNRQjRVaKQRTalNcPcBvU1ZQ35w5kKRk1gUeNCkVLsRiATurjASTp3TKM4H10wsI50nx3NZdw==}
|
||||
|
||||
'@asamuzakjp/nwsapi@2.3.9':
|
||||
resolution: {integrity: sha512-n8GuYSrI9bF7FFZ/SjhwevlHc8xaVlb/7HmHelnc/PZXBD2ZR49NnN9sMMuDdEGPeeRQ5d0hqlSlEpgCX3Wl0Q==}
|
||||
|
||||
'@azu/format-text@1.0.2':
|
||||
resolution: {integrity: sha512-Swi4N7Edy1Eqq82GxgEECXSSLyn6GOb5htRFPzBDdUkECGXtlf12ynO5oJSpWKPwCaUssOu7NfhDcCWpIC6Ywg==}
|
||||
|
||||
@@ -394,6 +418,10 @@ packages:
|
||||
peerDependencies:
|
||||
'@babel/core': ^7.0.0-0
|
||||
|
||||
'@babel/runtime@7.28.4':
|
||||
resolution: {integrity: sha512-Q/N6JNWvIvPnLDvjlE1OUBLPQHH6l3CltCEsHIujp45zQUSSh8K+gHnaEX45yAT1nyngnINhvWtzN+Nb9D8RAQ==}
|
||||
engines: {node: '>=6.9.0'}
|
||||
|
||||
'@babel/template@7.27.2':
|
||||
resolution: {integrity: sha512-LPDZ85aEJyYSd18/DkjNh4/y1ntkE5KwUHWTiqgRxruuZL2F1yuHligVHLvcHY2vMHXttKFpJn6LwfI7cw7ODw==}
|
||||
engines: {node: '>=6.9.0'}
|
||||
@@ -424,6 +452,38 @@ packages:
|
||||
resolution: {integrity: sha512-IchNf6dN4tHoMFIn/7OE8LWZ19Y6q/67Bmf6vnGREv8RSbBVb9LPJxEcnwrcwX6ixSvaiGoomAUvu4YSxXrVgw==}
|
||||
engines: {node: '>=12'}
|
||||
|
||||
'@csstools/color-helpers@5.1.0':
|
||||
resolution: {integrity: sha512-S11EXWJyy0Mz5SYvRmY8nJYTFFd1LCNV+7cXyAgQtOOuzb4EsgfqDufL+9esx72/eLhsRdGZwaldu/h+E4t4BA==}
|
||||
engines: {node: '>=18'}
|
||||
|
||||
'@csstools/css-calc@2.1.4':
|
||||
resolution: {integrity: sha512-3N8oaj+0juUw/1H3YwmDDJXCgTB1gKU6Hc/bB502u9zR0q2vd786XJH9QfrKIEgFlZmhZiq6epXl4rHqhzsIgQ==}
|
||||
engines: {node: '>=18'}
|
||||
peerDependencies:
|
||||
'@csstools/css-parser-algorithms': ^3.0.5
|
||||
'@csstools/css-tokenizer': ^3.0.4
|
||||
|
||||
'@csstools/css-color-parser@3.1.0':
|
||||
resolution: {integrity: sha512-nbtKwh3a6xNVIp/VRuXV64yTKnb1IjTAEEh3irzS+HkKjAOYLTGNb9pmVNntZ8iVBHcWDA2Dof0QtPgFI1BaTA==}
|
||||
engines: {node: '>=18'}
|
||||
peerDependencies:
|
||||
'@csstools/css-parser-algorithms': ^3.0.5
|
||||
'@csstools/css-tokenizer': ^3.0.4
|
||||
|
||||
'@csstools/css-parser-algorithms@3.0.5':
|
||||
resolution: {integrity: sha512-DaDeUkXZKjdGhgYaHNJTV9pV7Y9B3b644jCLs9Upc3VeNGg6LWARAT6O+Q+/COo+2gg/bM5rhpMAtf70WqfBdQ==}
|
||||
engines: {node: '>=18'}
|
||||
peerDependencies:
|
||||
'@csstools/css-tokenizer': ^3.0.4
|
||||
|
||||
'@csstools/css-syntax-patches-for-csstree@1.0.20':
|
||||
resolution: {integrity: sha512-8BHsjXfSciZxjmHQOuVdW2b8WLUPts9a+mfL13/PzEviufUEW2xnvQuOlKs9dRBHgRqJ53SF/DUoK9+MZk72oQ==}
|
||||
engines: {node: '>=18'}
|
||||
|
||||
'@csstools/css-tokenizer@3.0.4':
|
||||
resolution: {integrity: sha512-Vd/9EVDiu6PPJt9yAh6roZP6El1xHrdvIVGjyBsHR0RYwNHgL7FJPyIIW4fANJNG6FtyZfvlRPpFI4ZM/lubvw==}
|
||||
engines: {node: '>=18'}
|
||||
|
||||
'@emnapi/core@1.7.1':
|
||||
resolution: {integrity: sha512-o1uhUASyo921r2XtHYOHy7gdkGLge8ghBEQHMWmyJFoXlpU58kIrhhN3w26lpQb6dspetweapMn2CSNwQ8I4wg==}
|
||||
|
||||
@@ -1484,6 +1544,25 @@ packages:
|
||||
'@standard-schema/spec@1.0.0':
|
||||
resolution: {integrity: sha512-m2bOd0f2RT9k8QJx1JN85cZYyH1RqFBdlwtkSlf4tBDYLCiiZnv1fIIwacK6cqwXavOydf0NPToMQgpKq+dVlA==}
|
||||
|
||||
'@testing-library/dom@10.4.1':
|
||||
resolution: {integrity: sha512-o4PXJQidqJl82ckFaXUeoAW+XysPLauYI43Abki5hABd853iMhitooc6znOnczgbTYmEP6U6/y1ZyKAIsvMKGg==}
|
||||
engines: {node: '>=18'}
|
||||
|
||||
'@testing-library/react@16.3.0':
|
||||
resolution: {integrity: sha512-kFSyxiEDwv1WLl2fgsq6pPBbw5aWKrsY2/noi1Id0TK0UParSF62oFQFGHXIyaG4pp2tEub/Zlel+fjjZILDsw==}
|
||||
engines: {node: '>=18'}
|
||||
peerDependencies:
|
||||
'@testing-library/dom': ^10.0.0
|
||||
'@types/react': ^18.0.0 || ^19.0.0
|
||||
'@types/react-dom': ^18.0.0 || ^19.0.0
|
||||
react: ^18.0.0 || ^19.0.0
|
||||
react-dom: ^18.0.0 || ^19.0.0
|
||||
peerDependenciesMeta:
|
||||
'@types/react':
|
||||
optional: true
|
||||
'@types/react-dom':
|
||||
optional: true
|
||||
|
||||
'@textlint/ast-node-types@15.4.0':
|
||||
resolution: {integrity: sha512-IqY8i7IOGuvy05wZxISB7Me1ZyrvhaQGgx6DavfQjH3cfwpPFdDbDYmMXMuSv2xLS1kDB1kYKBV7fL2Vi16lRA==}
|
||||
|
||||
@@ -1521,6 +1600,9 @@ packages:
|
||||
'@tybys/wasm-util@0.10.1':
|
||||
resolution: {integrity: sha512-9tTaPJLSiejZKx+Bmog4uSubteqTvFrVrURwkmHixBo0G4seD0zUxp98E1DzUBJxLQ3NPwXrGKDiVjwx/DpPsg==}
|
||||
|
||||
'@types/aria-query@5.0.4':
|
||||
resolution: {integrity: sha512-rfT93uj5s0PRL7EzccGMs3brplhcrghnDoV26NqKhCAS1hVo+WdNsPvE/yb6ilfr5hi2MEk6d5EWJTKdxg8jVw==}
|
||||
|
||||
'@types/babel__core@7.20.5':
|
||||
resolution: {integrity: sha512-qoQprZvz5wQFJwMDqeseRXWv3rqMvhgpbXFfVyWhbx9X47POIA6i/+dXefEmZKoAgOaTdaIgNSMqMIU61yRyzA==}
|
||||
|
||||
@@ -1578,6 +1660,9 @@ packages:
|
||||
'@types/jest@30.0.0':
|
||||
resolution: {integrity: sha512-XTYugzhuwqWjws0CVz8QpM36+T+Dz5mTEBKhNs/esGLnCIlGdRy+Dq78NRjd7ls7r8BC8ZRMOrKlkO1hU0JOwA==}
|
||||
|
||||
'@types/jsdom@27.0.0':
|
||||
resolution: {integrity: sha512-NZyFl/PViwKzdEkQg96gtnB8wm+1ljhdDay9ahn4hgb+SfVtPCbm3TlmDUFXTA+MGN3CijicnMhG18SI5H3rFw==}
|
||||
|
||||
'@types/json-schema@7.0.15':
|
||||
resolution: {integrity: sha512-5+fP8P8MFNC+AyZCDxrB2pkZFPGzqQWUzpSeuuVLvm8VMcorNYavBqoFcxK8bQz4Qsbn4oUEEem4wDLfcysGHA==}
|
||||
|
||||
@@ -1620,6 +1705,9 @@ packages:
|
||||
'@types/supertest@6.0.3':
|
||||
resolution: {integrity: sha512-8WzXq62EXFhJ7QsH3Ocb/iKQ/Ty9ZVWnVzoTKc9tyyFRRF3a74Tk2+TLFgaFFw364Ere+npzHKEJ6ga2LzIL7w==}
|
||||
|
||||
'@types/tough-cookie@4.0.5':
|
||||
resolution: {integrity: sha512-/Ad8+nIOV7Rl++6f1BdKxFSMgmoqEoYbHRpPcx3JEfv8VRsQe9Z4mCXeJBzxs7mbHY/XOZZuXlRNfhpVPbs6ZA==}
|
||||
|
||||
'@types/uuid@11.0.0':
|
||||
resolution: {integrity: sha512-HVyk8nj2m+jcFRNazzqyVKiZezyhDKrGUA3jlEcg/nZ6Ms+qHwocba1Y/AaVaznJTAM9xpdFSh+ptbNrhOGvZA==}
|
||||
deprecated: This is a stub types definition. uuid provides its own type definitions, so you do not need this installed.
|
||||
@@ -1926,6 +2014,10 @@ packages:
|
||||
engines: {node: '>=0.4.0'}
|
||||
hasBin: true
|
||||
|
||||
agent-base@7.1.4:
|
||||
resolution: {integrity: sha512-MnA+YT8fwfJPgBx3m60MNqakm30XOkyIoH1y6huTQvC0PwZG7ki8NacLBcrPbNoo8vEZy7Jpuk7+jMO+CUovTQ==}
|
||||
engines: {node: '>= 14'}
|
||||
|
||||
ajv-formats@2.1.1:
|
||||
resolution: {integrity: sha512-Wx0Kx52hxE7C18hkMEggYlEifqWZtYaRgouJor+WMdPnQyEK13vgEWyVNup7SoeeoLMsr4kf5h6dOW11I15MUA==}
|
||||
peerDependencies:
|
||||
@@ -2014,6 +2106,9 @@ packages:
|
||||
argparse@2.0.1:
|
||||
resolution: {integrity: sha512-8+9WqebbFzpX9OR+Wa6O29asIogeRMzcGtAINdpMHHyAg10f05aSFVBbcEqGf/PXw1EjAZ+q2/bEBg3DvurK3Q==}
|
||||
|
||||
aria-query@5.3.0:
|
||||
resolution: {integrity: sha512-b0P0sZPKtyu8HkeRAfCq0IfURZK+SuwMjY1UXGBU27wpAiTwQAIlq56IbIO+ytk/JjS1fMR14ee5WBBfKi5J6A==}
|
||||
|
||||
array-timsort@1.0.3:
|
||||
resolution: {integrity: sha512-/+3GRL7dDAGEfM6TseQk/U+mi18TU2Ms9I3UlLdUMhz2hbvGNTKdj9xniwXfUqgYhHxRx0+8UnKkvlNwVU+cWQ==}
|
||||
|
||||
@@ -2076,6 +2171,9 @@ packages:
|
||||
resolution: {integrity: sha512-a28v2eWrrRWPpJSzxc+mKwm0ZtVx/G8SepdQZDArnXYU/XS+IF6mp8aB/4E+hH1tyGCoDo3KlUCdlSxGDsRkAw==}
|
||||
hasBin: true
|
||||
|
||||
bidi-js@1.0.3:
|
||||
resolution: {integrity: sha512-RKshQI1R3YQ+n9YJz2QQ147P66ELpa1FQEg20Dk8oW9t2KgLbpDLLp9aGZ7y8WHSshDknG0bknqGw5/tyCs5tw==}
|
||||
|
||||
binary-extensions@2.3.0:
|
||||
resolution: {integrity: sha512-Ceh+7ox5qe7LJuLHoY0feh3pHuUDHAcRUeyL2VYghZwfpkNIy/+8Ocg0a3UuSoYzavmylwuLWQOf3hl0jjMMIw==}
|
||||
engines: {node: '>=8'}
|
||||
@@ -2323,9 +2421,21 @@ packages:
|
||||
resolution: {integrity: sha512-uV2QOWP2nWzsy2aMp8aRibhi9dlzF5Hgh5SHaB9OiTGEyDTiJJyx0uy51QXdyWbtAHNua4XJzUKca3OzKUd3vA==}
|
||||
engines: {node: '>= 8'}
|
||||
|
||||
css-tree@3.1.0:
|
||||
resolution: {integrity: sha512-0eW44TGN5SQXU1mWSkKwFstI/22X2bG1nYzZTYMAWjylYURhse752YgbE4Cx46AC+bAvI+/dYTPRk1LqSUnu6w==}
|
||||
engines: {node: ^10 || ^12.20.0 || ^14.13.0 || >=15.0.0}
|
||||
|
||||
cssstyle@5.3.3:
|
||||
resolution: {integrity: sha512-OytmFH+13/QXONJcC75QNdMtKpceNk3u8ThBjyyYjkEcy/ekBwR1mMAuNvi3gdBPW3N5TlCzQ0WZw8H0lN/bDw==}
|
||||
engines: {node: '>=20'}
|
||||
|
||||
csstype@3.2.3:
|
||||
resolution: {integrity: sha512-z1HGKcYy2xA8AGQfwrn0PAy+PB7X/GSj3UVJW9qKyn43xWa+gl5nXmU4qqLMRzWVLFC8KusUX8T/0kCiOYpAIQ==}
|
||||
|
||||
data-urls@6.0.0:
|
||||
resolution: {integrity: sha512-BnBS08aLUM+DKamupXs3w2tJJoqU+AkaE/+6vQxi/G/DPmIZFJJp9Dkb1kM03AZx8ADehDUZgsNxju3mPXZYIA==}
|
||||
engines: {node: '>=20'}
|
||||
|
||||
debug@4.4.3:
|
||||
resolution: {integrity: sha512-RGwwWnwQvkVfavKVt22FGLw+xYSdzARwm0ru6DhTVA3umU5hZc28V3kO4stgYryrTlLpuvgI9GiijltAjNbcqA==}
|
||||
engines: {node: '>=6.0'}
|
||||
@@ -2335,6 +2445,9 @@ packages:
|
||||
supports-color:
|
||||
optional: true
|
||||
|
||||
decimal.js@10.6.0:
|
||||
resolution: {integrity: sha512-YpgQiITW3JXGntzdUmyUR1V812Hn8T1YVXhCu+wO3OpS4eU9l4YdD3qjyiKdV6mvV29zapkMeD390UVEf2lkUg==}
|
||||
|
||||
dedent@1.7.0:
|
||||
resolution: {integrity: sha512-HGFtf8yhuhGhqO07SV79tRp+br4MnbdjeVxotpn1QBl30pcLLCQjX5b2295ll0fv8RKDKsmWYrl05usHM9CewQ==}
|
||||
peerDependencies:
|
||||
@@ -2365,6 +2478,10 @@ packages:
|
||||
resolution: {integrity: sha512-HVQE3AAb/pxF8fQAoiqpvg9i3evqug3hoiwakOyZAwJm+6vZehbkYXZ0l4JxS+I3QxM97v5aaRNhj8v5oBhekw==}
|
||||
engines: {node: '>=0.10'}
|
||||
|
||||
dequal@2.0.3:
|
||||
resolution: {integrity: sha512-0je+qPKHEMohvfRTCEo3CrPG6cAzAYgmzKyxRiYSSDkS6eGJdyVJm7WaYA5ECaAD9wLB2T4EEeymA5aFVcYXCA==}
|
||||
engines: {node: '>=6'}
|
||||
|
||||
detect-newline@3.1.0:
|
||||
resolution: {integrity: sha512-TLz+x/vEXm/Y7P7wn1EJFNLxYpUD4TgMosxY6fAVJUnJMbupHBOncxyWUG9OpTaH9EBD7uFI5LfEgmMOc54DsA==}
|
||||
engines: {node: '>=8'}
|
||||
@@ -2380,6 +2497,9 @@ packages:
|
||||
resolution: {integrity: sha512-58lmxKSA4BNyLz+HHMUzlOEpg09FV+ev6ZMe3vJihgdxzgcwZ8VoEEPmALCZG9LmqfVoNMMKpttIYTVG6uDY7A==}
|
||||
engines: {node: '>=0.3.1'}
|
||||
|
||||
dom-accessibility-api@0.5.16:
|
||||
resolution: {integrity: sha512-X7BJ2yElsnOJ30pZF4uIIDfBEVgF4XEBxL9Bxhy6dnrm5hkzqmsWHGTiHqRiITNhMyFLyAiWndIJP7Z1NTteDg==}
|
||||
|
||||
dunder-proto@1.0.1:
|
||||
resolution: {integrity: sha512-KIN/nDJBQRcXw0MLVhZE9iQHmG68qAVIBg9CqmUYjmQIhgij9U5MFvrqkUL5FbtyyzZuOeOt0zdeRe4UY7ct+A==}
|
||||
engines: {node: '>= 0.4'}
|
||||
@@ -2408,6 +2528,10 @@ packages:
|
||||
resolution: {integrity: sha512-d4lC8xfavMeBjzGr2vECC3fsGXziXZQyJxD868h2M/mBI3PwAuODxAkLkq5HYuvrPYcUtiLzsTo8U3PgX3Ocww==}
|
||||
engines: {node: '>=10.13.0'}
|
||||
|
||||
entities@6.0.1:
|
||||
resolution: {integrity: sha512-aN97NXWF6AWBTahfVOIrB/NShkzi5H7F9r1s9mD3cDj4Ko5f2qhhVoYMibXF7GlLveb/D2ioWay8lxI97Ven3g==}
|
||||
engines: {node: '>=0.12'}
|
||||
|
||||
environment@1.1.0:
|
||||
resolution: {integrity: sha512-xUtoPkMggbz0MPyPiIWr1Kp4aeWJjDZ6SMvURhimjdZgsRuDplF5/s9hcgGhyXMhs+6vpnuoiZ2kFiu3FMnS8Q==}
|
||||
engines: {node: '>=18'}
|
||||
@@ -2771,9 +2895,21 @@ packages:
|
||||
resolution: {integrity: sha512-0hJU9SCPvmMzIBdZFqNPXWa6dqh7WdH0cII9y+CyS8rG3nL48Bclra9HmKhVVUHyPWNH5Y7xDwAB7bfgSjkUMQ==}
|
||||
engines: {node: '>= 0.4'}
|
||||
|
||||
html-encoding-sniffer@4.0.0:
|
||||
resolution: {integrity: sha512-Y22oTqIU4uuPgEemfz7NDJz6OeKf12Lsu+QC+s3BVpda64lTiMYCyGwg5ki4vFxkMwQdeZDl2adZoqUgdFuTgQ==}
|
||||
engines: {node: '>=18'}
|
||||
|
||||
html-escaper@2.0.2:
|
||||
resolution: {integrity: sha512-H2iMtd0I4Mt5eYiapRdIDjp+XzelXQ0tFE4JS7YFwFevXXMmOp9myNrUvCg0D6ws8iqkRPBfKHgbwig1SmlLfg==}
|
||||
|
||||
http-proxy-agent@7.0.2:
|
||||
resolution: {integrity: sha512-T1gkAiYYDWYx3V5Bmyu7HcfcvL7mUrTWiM6yOfa3PIphViJ/gFPbvidQ+veqSOHci/PxBcDabeUNCzpOODJZig==}
|
||||
engines: {node: '>= 14'}
|
||||
|
||||
https-proxy-agent@7.0.6:
|
||||
resolution: {integrity: sha512-vK9P5/iUfdl95AI+JVyUuIcVtd4ofvtrOr3HNtM2yxC9bnMbEdp3x01OhQNnjb8IJYi38VlTE3mBXwcfvywuSw==}
|
||||
engines: {node: '>= 14'}
|
||||
|
||||
human-signals@2.1.0:
|
||||
resolution: {integrity: sha512-B4FFZ6q/T2jhhksgkbEW3HBvWIfDW85snkQgawt07S7J5QXTk6BkNV+0yAeZrM5QpMAdYlocGoljn0sJ/WQkFw==}
|
||||
engines: {node: '>=10.17.0'}
|
||||
@@ -2782,6 +2918,10 @@ packages:
|
||||
resolution: {integrity: sha512-AXcZb6vzzrFAUE61HnN4mpLqd/cSIwNQjtNWR0euPm6y0iqx3G4gOXaIDdtdDwZmhwe82LA6+zinmW4UBWVePQ==}
|
||||
engines: {node: '>=16.17.0'}
|
||||
|
||||
iconv-lite@0.6.3:
|
||||
resolution: {integrity: sha512-4fCk79wshMdzMp2rH06qWrJE4iolqLhCUH+OiuIgU++RB0+94NlDL81atO7GX55uUKueo0txHNtvEyI6D7WdMw==}
|
||||
engines: {node: '>=0.10.0'}
|
||||
|
||||
iconv-lite@0.7.0:
|
||||
resolution: {integrity: sha512-cf6L2Ds3h57VVmkZe+Pn+5APsT7FpqJtEhhieDCvrE2MK5Qk9MyffgQyuxQTm6BChfeZNtcOLHp9IcWRVcIcBQ==}
|
||||
engines: {node: '>=0.10.0'}
|
||||
@@ -2891,6 +3031,9 @@ packages:
|
||||
resolution: {integrity: sha512-lJJV/5dYS+RcL8uQdBDW9c9uWFLLBNRyFhnAKXw5tVqLlKZ4RMGZKv+YQ/IA3OhD+RpbJa1LLFM1FQPGyIXvOA==}
|
||||
engines: {node: '>=12'}
|
||||
|
||||
is-potential-custom-element-name@1.0.1:
|
||||
resolution: {integrity: sha512-bCYeRA2rVibKZd+s2625gGnGF/t7DSqDs4dP7CrLA1m7jKWz6pps0LpYLJN8Q64HtmPKJ1hrN3nzPNKFEKOUiQ==}
|
||||
|
||||
is-stream@2.0.1:
|
||||
resolution: {integrity: sha512-hFoiJiTl63nn+kstHGBtewWSKnQLpyb155KHheA1l39uvtO9nWIop1p3udqPcUd/xbF1VLMO4n7OI6p7RbngDg==}
|
||||
engines: {node: '>=8'}
|
||||
@@ -3094,6 +3237,15 @@ packages:
|
||||
resolution: {integrity: sha512-qQKT4zQxXl8lLwBtHMWwaTcGfFOZviOJet3Oy/xmGk2gZH677CJM9EvtfdSkgWcATZhj/55JZ0rmy3myCT5lsA==}
|
||||
hasBin: true
|
||||
|
||||
jsdom@27.2.0:
|
||||
resolution: {integrity: sha512-454TI39PeRDW1LgpyLPyURtB4Zx1tklSr6+OFOipsxGUH1WMTvk6C65JQdrj455+DP2uJ1+veBEHTGFKWVLFoA==}
|
||||
engines: {node: ^20.19.0 || ^22.12.0 || >=24.0.0}
|
||||
peerDependencies:
|
||||
canvas: ^3.0.0
|
||||
peerDependenciesMeta:
|
||||
canvas:
|
||||
optional: true
|
||||
|
||||
jsesc@3.1.0:
|
||||
resolution: {integrity: sha512-/sM3dO2FOzXjKQhJuo0Q173wf2KOo8t4I8vHy6lF9poUp7bKT0/NHE8fPX23PwfhnykfqnC2xRxOnVw5XuGIaA==}
|
||||
engines: {node: '>=6'}
|
||||
@@ -3206,6 +3358,10 @@ packages:
|
||||
lru-cache@5.1.1:
|
||||
resolution: {integrity: sha512-KpNARQA3Iwv+jTA0utUVVbrh+Jlrr1Fv0e56GGzAFOXN7dk/FviaDW8LHmK52DlcH4WP2n6gI8vN1aesBFgo9w==}
|
||||
|
||||
lz-string@1.5.0:
|
||||
resolution: {integrity: sha512-h5bgJWpxJNswbU7qCrV0tIKQCaS3blPDrqKWx+QxzuzL1zGUzij9XCWLrSLsJPu5t+eWA/ycetzYAO5IOMcWAQ==}
|
||||
hasBin: true
|
||||
|
||||
magic-string@0.30.17:
|
||||
resolution: {integrity: sha512-sNPKHvyjVf7gyjwS4xGTaW/mCnF8wnjtifKBEhxfZ7E/S8tQ0rssrwGNn6q8JH/ohItJfSQp9mBtQYuTlH5QnA==}
|
||||
|
||||
@@ -3232,6 +3388,9 @@ packages:
|
||||
resolution: {integrity: sha512-/IXtbwEk5HTPyEwyKX6hGkYXxM9nbj64B+ilVJnC/R6B0pH5G4V3b0pVbL7DBj4tkhBAppbQUlf6F6Xl9LHu1g==}
|
||||
engines: {node: '>= 0.4'}
|
||||
|
||||
mdn-data@2.12.2:
|
||||
resolution: {integrity: sha512-IEn+pegP1aManZuckezWCO+XZQDplx1366JoVhTpMpBB1sPey/SbveZQUosKiKiGYjg1wH4pMlNgXbCiYgihQA==}
|
||||
|
||||
memfs@3.5.3:
|
||||
resolution: {integrity: sha512-UERzLsxzllchadvbPs5aolHh65ISpKpM+ccLbOJ8/vvpBKmAWf+la7dXFy7Mr0ySHbdHrFv5kGFCUHHe6GFEmw==}
|
||||
engines: {node: '>= 4.0.0'}
|
||||
@@ -3424,6 +3583,12 @@ packages:
|
||||
resolution: {integrity: sha512-ayCKvm/phCGxOkYRSCM82iDwct8/EonSEgCSxWxD7ve6jHggsFl4fZVQBPRNgQoKiuV/odhFrGzQXZwbifC8Rg==}
|
||||
engines: {node: '>=8'}
|
||||
|
||||
parse5@7.3.0:
|
||||
resolution: {integrity: sha512-IInvU7fabl34qmi9gY8XOVxhYyMyuH2xUNpb2q8/Y+7552KlejkRvqvD19nMoUW/uQGGbqNpA6Tufu5FL5BZgw==}
|
||||
|
||||
parse5@8.0.0:
|
||||
resolution: {integrity: sha512-9m4m5GSgXjL4AjumKzq1Fgfp3Z8rsvjRNbnkVwfu2ImRqE5D0LnY2QfDen18FSY9C573YU5XxSapdHZTZ2WolA==}
|
||||
|
||||
patch-console@2.0.0:
|
||||
resolution: {integrity: sha512-0YNdUceMdaQwoKce1gatDScmMo5pu/tfABfnzEqeG0gtTmd7mh/WcwgUjtAeOU7N8nFFlbQBnFK2gXW5fGvmMA==}
|
||||
engines: {node: ^12.20.0 || ^14.13.1 || >=16.0.0}
|
||||
@@ -3536,6 +3701,10 @@ packages:
|
||||
engines: {node: '>=14'}
|
||||
hasBin: true
|
||||
|
||||
pretty-format@27.5.1:
|
||||
resolution: {integrity: sha512-Qb1gy5OrP5+zDf2Bvnzdl3jsTf1qXVMazbvCoKhtKqVs4/YK4ozX4gKQJJVyNe+cajNPn0KoC0MC3FUmaHWEmQ==}
|
||||
engines: {node: ^10.13.0 || ^12.13.0 || ^14.15.0 || >=15.0.0}
|
||||
|
||||
pretty-format@29.7.0:
|
||||
resolution: {integrity: sha512-Pdlw/oPxN+aXdmM9R00JVC9WVFoCLTKJvDVLgmJ+qAffBMxsV85l/Lu7sNx4zSzPyoL2euImuEwHhOXdEgNFZQ==}
|
||||
engines: {node: ^14.15.0 || ^16.10.0 || >=18.0.0}
|
||||
@@ -3564,6 +3733,14 @@ packages:
|
||||
rc-config-loader@4.1.3:
|
||||
resolution: {integrity: sha512-kD7FqML7l800i6pS6pvLyIE2ncbk9Du8Q0gp/4hMPhJU6ZxApkoLcGD8ZeqgiAlfwZ6BlETq6qqe+12DUL207w==}
|
||||
|
||||
react-dom@18.3.1:
|
||||
resolution: {integrity: sha512-5m4nQKp+rZRb09LNH59GM4BxTh9251/ylbKIbpe7TpGxfJ+9kv6BLkLBXIjjspbgbnIBNqlI23tRnTWT0snUIw==}
|
||||
peerDependencies:
|
||||
react: ^18.3.1
|
||||
|
||||
react-is@17.0.2:
|
||||
resolution: {integrity: sha512-w2GsyukL62IJnlaff/nRegPQR94C/XXamvMWmSHRJ4y7Ts/4ocGRmTHvOs8PSE6pB3dWOrD/nueuU5sduBsQ4w==}
|
||||
|
||||
react-is@18.3.1:
|
||||
resolution: {integrity: sha512-/LLMVyas0ljjAtoYiPqYiL8VWXzUUdThrmU5+n20DZv+a+ClRoevUzw5JxU+Ieh5/c87ytoTBV9G1FiKfNJdmg==}
|
||||
|
||||
@@ -3652,6 +3829,10 @@ packages:
|
||||
safer-buffer@2.1.2:
|
||||
resolution: {integrity: sha512-YZo3K82SD7Riyi0E1EQPojLz7kpepnSQI9IyPbHHg1XXXevb5dJI7tpyN2ADxGcQbHG7vcyRHk0cbwqcQriUtg==}
|
||||
|
||||
saxes@6.0.0:
|
||||
resolution: {integrity: sha512-xAg7SOnEhrm5zI3puOOKyy1OMcMlIJZYNJY7xLBwSze0UjhPLnWfj2GF2EpT0jmzaJKIWKHLsaSSajf35bcYnA==}
|
||||
engines: {node: '>=v12.22.7'}
|
||||
|
||||
scheduler@0.23.2:
|
||||
resolution: {integrity: sha512-UOShsPwz7NrMUqhR6t0hWjFduvOzbtv7toDH1/hIrfRNIDBnnBWd0CwJTGvTpngVlmwGCdP9/Zl/tVrDqcuYzQ==}
|
||||
|
||||
@@ -3860,6 +4041,9 @@ packages:
|
||||
resolution: {integrity: sha512-b19dMThMV4HVFynSAM1++gBHAbk2Tc/osgLIBZMKsyqh34jb2e8Os7T6ZW/Bt3pJFdBTd2JwAnAAEQV7rSNvcQ==}
|
||||
engines: {node: '>=0.10'}
|
||||
|
||||
symbol-tree@3.2.4:
|
||||
resolution: {integrity: sha512-9QNk5KwDF+Bvz+PyObkmSYjI5ksVUYtjW7AU22r2NKcfLJcXp96hkDWU3+XndOsUb+AQ9QhfzfCT2O+CNWT5Tw==}
|
||||
|
||||
synckit@0.11.11:
|
||||
resolution: {integrity: sha512-MeQTA1r0litLUf0Rp/iisCaL8761lKAZHaimlbGK4j0HysC4PLfqygQj9srcs0m2RdtDYnF8UuYyKpbjHYp7Jw==}
|
||||
engines: {node: ^14.18.0 || >=16.0.0}
|
||||
@@ -3937,6 +4121,13 @@ packages:
|
||||
resolution: {integrity: sha512-KYad6Vy5VDWV4GH3fjpseMQ/XU2BhIYP7Vzd0LG44qRWm/Yt2WCOTicFdvmgo6gWaqooMQCawTtILVQJupKu7A==}
|
||||
engines: {node: '>=14.0.0'}
|
||||
|
||||
tldts-core@7.0.19:
|
||||
resolution: {integrity: sha512-lJX2dEWx0SGH4O6p+7FPwYmJ/bu1JbcGJ8RLaG9b7liIgZ85itUVEPbMtWRVrde/0fnDPEPHW10ZsKW3kVsE9A==}
|
||||
|
||||
tldts@7.0.19:
|
||||
resolution: {integrity: sha512-8PWx8tvC4jDB39BQw1m4x8y5MH1BcQ5xHeL2n7UVFulMPH/3Q0uiamahFJ3lXA0zO2SUyRXuVVbWSDmstlt9YA==}
|
||||
hasBin: true
|
||||
|
||||
tmpl@1.0.5:
|
||||
resolution: {integrity: sha512-3f0uOEAQwIqGuWW2MVzYg8fV/QNnc/IpuJNG837rLuczAaLVHslWHZQj4IGiEl5Hs3kkbhwL9Ab7Hrsmuj+Smw==}
|
||||
|
||||
@@ -3952,6 +4143,14 @@ packages:
|
||||
resolution: {integrity: sha512-sf4i37nQ2LBx4m3wB74y+ubopq6W/dIzXg0FDGjsYnZHVa1Da8FH853wlL2gtUhg+xJXjfk3kUZS3BRoQeoQBQ==}
|
||||
engines: {node: '>=6'}
|
||||
|
||||
tough-cookie@6.0.0:
|
||||
resolution: {integrity: sha512-kXuRi1mtaKMrsLUxz3sQYvVl37B0Ns6MzfrtV5DvJceE9bPyspOqk9xxv7XbZWcfLWbFmm997vl83qUWVJA64w==}
|
||||
engines: {node: '>=16'}
|
||||
|
||||
tr46@6.0.0:
|
||||
resolution: {integrity: sha512-bLVMLPtstlZ4iMQHpFHTR7GAGj2jxi8Dg0s2h2MafAE4uSWF98FC/3MomU51iQAMf8/qDUbKWf5GxuvvVcXEhw==}
|
||||
engines: {node: '>=20'}
|
||||
|
||||
tree-kill@1.2.2:
|
||||
resolution: {integrity: sha512-L0Orpi8qGpRG//Nd+H90vFB+3iHnue1zSSGmNOOCh1GLJ7rUKVwV2HvijphGQS2UmhUZewS9VgvxYIdgr+fG1A==}
|
||||
hasBin: true
|
||||
@@ -4320,6 +4519,10 @@ packages:
|
||||
jsdom:
|
||||
optional: true
|
||||
|
||||
w3c-xmlserializer@5.0.0:
|
||||
resolution: {integrity: sha512-o8qghlI8NZHU1lLPrpi2+Uq7abh4GGPpYANlalzWxyWteJOCsr/P+oPBA49TOLu5FTZO4d3F9MnWJfiMo4BkmA==}
|
||||
engines: {node: '>=18'}
|
||||
|
||||
walker@1.0.8:
|
||||
resolution: {integrity: sha512-ts/8E8l5b7kY0vlWLewOkDXMmPdLcVV4GmOQLyxuSswIJsweeFZtAsMF7k1Nszz+TYBQrlYRmzOnr398y1JemQ==}
|
||||
|
||||
@@ -4330,6 +4533,10 @@ packages:
|
||||
wcwidth@1.0.1:
|
||||
resolution: {integrity: sha512-XHPEwS0q6TaxcvG85+8EYkbiCux2XtWG2mkc47Ng2A77BQu9+DqIOJldST4HgPkuea7dvKSj5VgX3P1d4rW8Tg==}
|
||||
|
||||
webidl-conversions@8.0.0:
|
||||
resolution: {integrity: sha512-n4W4YFyz5JzOfQeA8oN7dUYpR+MBP3PIUsn2jLjWXwK5ASUzt0Jc/A5sAUZoCYFJRGF0FBKJ+1JjN43rNdsQzA==}
|
||||
engines: {node: '>=20'}
|
||||
|
||||
webpack-node-externals@3.0.0:
|
||||
resolution: {integrity: sha512-LnL6Z3GGDPht/AigwRh2dvL9PQPFQ8skEpVrWZXLWBYmqcaojHNN0onvHzie6rq7EWKrrBfPYqNEzTJgiwEQDQ==}
|
||||
engines: {node: '>=6'}
|
||||
@@ -4348,9 +4555,21 @@ packages:
|
||||
webpack-cli:
|
||||
optional: true
|
||||
|
||||
whatwg-encoding@3.1.1:
|
||||
resolution: {integrity: sha512-6qN4hJdMwfYBtE3YBTTHhoeuUrDBPZmbQaxWAqSALV/MeEnR5z1xd8UKud2RAkFoPkmB+hli1TZSnyi84xz1vQ==}
|
||||
engines: {node: '>=18'}
|
||||
|
||||
whatwg-fetch@3.6.20:
|
||||
resolution: {integrity: sha512-EqhiFU6daOA8kpjOWTL0olhVOF3i7OrFzSYiGsEMB8GcXS+RrzauAERX65xMeNWVqxA6HXH2m69Z9LaKKdisfg==}
|
||||
|
||||
whatwg-mimetype@4.0.0:
|
||||
resolution: {integrity: sha512-QaKxh0eNIi2mE9p2vEdzfagOKHCcj1pJ56EEHGQOVxp8r9/iszLUUV7v89x9O1p/T+NlTM5W7jW6+cz4Fq1YVg==}
|
||||
engines: {node: '>=18'}
|
||||
|
||||
whatwg-url@15.1.0:
|
||||
resolution: {integrity: sha512-2ytDk0kiEj/yu90JOAp44PVPUkO9+jVhyf+SybKlRHSDlvOOZhdPIrr7xTH64l4WixO2cP+wQIcgujkGBPPz6g==}
|
||||
engines: {node: '>=20'}
|
||||
|
||||
which@2.0.2:
|
||||
resolution: {integrity: sha512-BLI3Tl1TW3Pvl70l3yq3Y64i+awpwXqsGBYWkkqMtnbXgrMD+yj7rhW0kuEDxzJaYXGjEW5ogapKNMEKNMjibA==}
|
||||
engines: {node: '>= 8'}
|
||||
@@ -4403,6 +4622,13 @@ packages:
|
||||
utf-8-validate:
|
||||
optional: true
|
||||
|
||||
xml-name-validator@5.0.0:
|
||||
resolution: {integrity: sha512-EvGK8EJ3DhaHfbRlETOWAS5pO9MZITeauHKJyb8wyajUfQUenkIg2MvLDTZ4T/TgIcm3HU0TFBgWWboAZ30UHg==}
|
||||
engines: {node: '>=18'}
|
||||
|
||||
xmlchars@2.2.0:
|
||||
resolution: {integrity: sha512-JZnDKK8B0RCDw84FNdDAIpZK+JuJw+s7Lz8nksI7SIuU3UXJJslUthsi+uWBUYOwPFwW7W7PRLRfUKpxjtjFCw==}
|
||||
|
||||
y18n@5.0.8:
|
||||
resolution: {integrity: sha512-0pfFzegeDWJHJIAmTLRP2DwHjdF5s7jo9tuztdQxAhINCdvS+3nGINqPd00AphqJR/0LhANUS6/+7SCb98YOfA==}
|
||||
engines: {node: '>=10'}
|
||||
@@ -4442,6 +4668,8 @@ packages:
|
||||
|
||||
snapshots:
|
||||
|
||||
'@acemir/cssom@0.9.24': {}
|
||||
|
||||
'@alcalzone/ansi-tokenize@0.1.3':
|
||||
dependencies:
|
||||
ansi-styles: 6.2.3
|
||||
@@ -4506,6 +4734,24 @@ snapshots:
|
||||
transitivePeerDependencies:
|
||||
- chokidar
|
||||
|
||||
'@asamuzakjp/css-color@4.1.0':
|
||||
dependencies:
|
||||
'@csstools/css-calc': 2.1.4(@csstools/css-parser-algorithms@3.0.5(@csstools/css-tokenizer@3.0.4))(@csstools/css-tokenizer@3.0.4)
|
||||
'@csstools/css-color-parser': 3.1.0(@csstools/css-parser-algorithms@3.0.5(@csstools/css-tokenizer@3.0.4))(@csstools/css-tokenizer@3.0.4)
|
||||
'@csstools/css-parser-algorithms': 3.0.5(@csstools/css-tokenizer@3.0.4)
|
||||
'@csstools/css-tokenizer': 3.0.4
|
||||
lru-cache: 11.2.2
|
||||
|
||||
'@asamuzakjp/dom-selector@6.7.5':
|
||||
dependencies:
|
||||
'@asamuzakjp/nwsapi': 2.3.9
|
||||
bidi-js: 1.0.3
|
||||
css-tree: 3.1.0
|
||||
is-potential-custom-element-name: 1.0.1
|
||||
lru-cache: 11.2.2
|
||||
|
||||
'@asamuzakjp/nwsapi@2.3.9': {}
|
||||
|
||||
'@azu/format-text@1.0.2': {}
|
||||
|
||||
'@azu/style-format@1.0.1':
|
||||
@@ -4676,6 +4922,8 @@ snapshots:
|
||||
'@babel/core': 7.28.5
|
||||
'@babel/helper-plugin-utils': 7.27.1
|
||||
|
||||
'@babel/runtime@7.28.4': {}
|
||||
|
||||
'@babel/template@7.27.2':
|
||||
dependencies:
|
||||
'@babel/code-frame': 7.27.1
|
||||
@@ -4712,6 +4960,28 @@ snapshots:
|
||||
dependencies:
|
||||
'@jridgewell/trace-mapping': 0.3.9
|
||||
|
||||
'@csstools/color-helpers@5.1.0': {}
|
||||
|
||||
'@csstools/css-calc@2.1.4(@csstools/css-parser-algorithms@3.0.5(@csstools/css-tokenizer@3.0.4))(@csstools/css-tokenizer@3.0.4)':
|
||||
dependencies:
|
||||
'@csstools/css-parser-algorithms': 3.0.5(@csstools/css-tokenizer@3.0.4)
|
||||
'@csstools/css-tokenizer': 3.0.4
|
||||
|
||||
'@csstools/css-color-parser@3.1.0(@csstools/css-parser-algorithms@3.0.5(@csstools/css-tokenizer@3.0.4))(@csstools/css-tokenizer@3.0.4)':
|
||||
dependencies:
|
||||
'@csstools/color-helpers': 5.1.0
|
||||
'@csstools/css-calc': 2.1.4(@csstools/css-parser-algorithms@3.0.5(@csstools/css-tokenizer@3.0.4))(@csstools/css-tokenizer@3.0.4)
|
||||
'@csstools/css-parser-algorithms': 3.0.5(@csstools/css-tokenizer@3.0.4)
|
||||
'@csstools/css-tokenizer': 3.0.4
|
||||
|
||||
'@csstools/css-parser-algorithms@3.0.5(@csstools/css-tokenizer@3.0.4)':
|
||||
dependencies:
|
||||
'@csstools/css-tokenizer': 3.0.4
|
||||
|
||||
'@csstools/css-syntax-patches-for-csstree@1.0.20': {}
|
||||
|
||||
'@csstools/css-tokenizer@3.0.4': {}
|
||||
|
||||
'@emnapi/core@1.7.1':
|
||||
dependencies:
|
||||
'@emnapi/wasi-threads': 1.1.0
|
||||
@@ -5646,6 +5916,26 @@ snapshots:
|
||||
|
||||
'@standard-schema/spec@1.0.0': {}
|
||||
|
||||
'@testing-library/dom@10.4.1':
|
||||
dependencies:
|
||||
'@babel/code-frame': 7.27.1
|
||||
'@babel/runtime': 7.28.4
|
||||
'@types/aria-query': 5.0.4
|
||||
aria-query: 5.3.0
|
||||
dom-accessibility-api: 0.5.16
|
||||
lz-string: 1.5.0
|
||||
picocolors: 1.1.1
|
||||
pretty-format: 27.5.1
|
||||
|
||||
'@testing-library/react@16.3.0(@testing-library/dom@10.4.1)(@types/react@18.3.27)(react-dom@18.3.1(react@18.3.1))(react@18.3.1)':
|
||||
dependencies:
|
||||
'@babel/runtime': 7.28.4
|
||||
'@testing-library/dom': 10.4.1
|
||||
react: 18.3.1
|
||||
react-dom: 18.3.1(react@18.3.1)
|
||||
optionalDependencies:
|
||||
'@types/react': 18.3.27
|
||||
|
||||
'@textlint/ast-node-types@15.4.0': {}
|
||||
|
||||
'@textlint/linter-formatter@15.4.0':
|
||||
@@ -5698,6 +5988,8 @@ snapshots:
|
||||
tslib: 2.8.1
|
||||
optional: true
|
||||
|
||||
'@types/aria-query@5.0.4': {}
|
||||
|
||||
'@types/babel__core@7.20.5':
|
||||
dependencies:
|
||||
'@babel/parser': 7.28.5
|
||||
@@ -5779,6 +6071,12 @@ snapshots:
|
||||
expect: 30.2.0
|
||||
pretty-format: 30.2.0
|
||||
|
||||
'@types/jsdom@27.0.0':
|
||||
dependencies:
|
||||
'@types/node': 22.19.1
|
||||
'@types/tough-cookie': 4.0.5
|
||||
parse5: 7.3.0
|
||||
|
||||
'@types/json-schema@7.0.15': {}
|
||||
|
||||
'@types/methods@1.1.4': {}
|
||||
@@ -5829,6 +6127,8 @@ snapshots:
|
||||
'@types/methods': 1.1.4
|
||||
'@types/superagent': 8.1.9
|
||||
|
||||
'@types/tough-cookie@4.0.5': {}
|
||||
|
||||
'@types/uuid@11.0.0':
|
||||
dependencies:
|
||||
uuid: 13.0.0
|
||||
@@ -6008,7 +6308,7 @@ snapshots:
|
||||
std-env: 3.10.0
|
||||
strip-literal: 2.1.1
|
||||
test-exclude: 6.0.0
|
||||
vitest: 1.6.1(@types/node@22.19.1)(@vitest/ui@1.6.1)(terser@5.44.1)
|
||||
vitest: 1.6.1(@types/node@22.19.1)(@vitest/ui@1.6.1)(jsdom@27.2.0)(terser@5.44.1)
|
||||
transitivePeerDependencies:
|
||||
- supports-color
|
||||
|
||||
@@ -6025,7 +6325,7 @@ snapshots:
|
||||
magicast: 0.5.1
|
||||
std-env: 3.10.0
|
||||
tinyrainbow: 3.0.3
|
||||
vitest: 4.0.13(@types/node@22.19.1)(@vitest/ui@4.0.13)(terser@5.44.1)(tsx@4.20.6)
|
||||
vitest: 4.0.13(@types/node@22.19.1)(@vitest/ui@4.0.13)(jsdom@27.2.0)(terser@5.44.1)(tsx@4.20.6)
|
||||
transitivePeerDependencies:
|
||||
- supports-color
|
||||
|
||||
@@ -6094,7 +6394,7 @@ snapshots:
|
||||
pathe: 1.1.2
|
||||
picocolors: 1.1.1
|
||||
sirv: 2.0.4
|
||||
vitest: 1.6.1(@types/node@22.19.1)(@vitest/ui@1.6.1)(terser@5.44.1)
|
||||
vitest: 1.6.1(@types/node@22.19.1)(@vitest/ui@1.6.1)(jsdom@27.2.0)(terser@5.44.1)
|
||||
|
||||
'@vitest/ui@4.0.13(vitest@4.0.13)':
|
||||
dependencies:
|
||||
@@ -6105,7 +6405,7 @@ snapshots:
|
||||
sirv: 3.0.2
|
||||
tinyglobby: 0.2.15
|
||||
tinyrainbow: 3.0.3
|
||||
vitest: 4.0.13(@types/node@22.19.1)(@vitest/ui@4.0.13)(terser@5.44.1)(tsx@4.20.6)
|
||||
vitest: 4.0.13(@types/node@22.19.1)(@vitest/ui@4.0.13)(jsdom@27.2.0)(terser@5.44.1)(tsx@4.20.6)
|
||||
|
||||
'@vitest/utils@1.6.1':
|
||||
dependencies:
|
||||
@@ -6213,6 +6513,8 @@ snapshots:
|
||||
|
||||
acorn@8.15.0: {}
|
||||
|
||||
agent-base@7.1.4: {}
|
||||
|
||||
ajv-formats@2.1.1(ajv@8.17.1):
|
||||
optionalDependencies:
|
||||
ajv: 8.17.1
|
||||
@@ -6285,6 +6587,10 @@ snapshots:
|
||||
|
||||
argparse@2.0.1: {}
|
||||
|
||||
aria-query@5.3.0:
|
||||
dependencies:
|
||||
dequal: 2.0.3
|
||||
|
||||
array-timsort@1.0.3: {}
|
||||
|
||||
asap@2.0.6: {}
|
||||
@@ -6363,6 +6669,10 @@ snapshots:
|
||||
|
||||
baseline-browser-mapping@2.8.31: {}
|
||||
|
||||
bidi-js@1.0.3:
|
||||
dependencies:
|
||||
require-from-string: 2.0.2
|
||||
|
||||
binary-extensions@2.3.0: {}
|
||||
|
||||
binaryextensions@6.11.0:
|
||||
@@ -6589,12 +6899,30 @@ snapshots:
|
||||
shebang-command: 2.0.0
|
||||
which: 2.0.2
|
||||
|
||||
css-tree@3.1.0:
|
||||
dependencies:
|
||||
mdn-data: 2.12.2
|
||||
source-map-js: 1.2.1
|
||||
|
||||
cssstyle@5.3.3:
|
||||
dependencies:
|
||||
'@asamuzakjp/css-color': 4.1.0
|
||||
'@csstools/css-syntax-patches-for-csstree': 1.0.20
|
||||
css-tree: 3.1.0
|
||||
|
||||
csstype@3.2.3: {}
|
||||
|
||||
data-urls@6.0.0:
|
||||
dependencies:
|
||||
whatwg-mimetype: 4.0.0
|
||||
whatwg-url: 15.1.0
|
||||
|
||||
debug@4.4.3:
|
||||
dependencies:
|
||||
ms: 2.1.3
|
||||
|
||||
decimal.js@10.6.0: {}
|
||||
|
||||
dedent@1.7.0: {}
|
||||
|
||||
deep-eql@4.1.4:
|
||||
@@ -6613,6 +6941,8 @@ snapshots:
|
||||
|
||||
denque@2.1.0: {}
|
||||
|
||||
dequal@2.0.3: {}
|
||||
|
||||
detect-newline@3.1.0: {}
|
||||
|
||||
dezalgo@1.0.4:
|
||||
@@ -6624,6 +6954,8 @@ snapshots:
|
||||
|
||||
diff@4.0.2: {}
|
||||
|
||||
dom-accessibility-api@0.5.16: {}
|
||||
|
||||
dunder-proto@1.0.1:
|
||||
dependencies:
|
||||
call-bind-apply-helpers: 1.0.2
|
||||
@@ -6649,6 +6981,8 @@ snapshots:
|
||||
graceful-fs: 4.2.11
|
||||
tapable: 2.3.0
|
||||
|
||||
entities@6.0.1: {}
|
||||
|
||||
environment@1.1.0: {}
|
||||
|
||||
error-ex@1.3.4:
|
||||
@@ -7130,12 +7464,34 @@ snapshots:
|
||||
dependencies:
|
||||
function-bind: 1.1.2
|
||||
|
||||
html-encoding-sniffer@4.0.0:
|
||||
dependencies:
|
||||
whatwg-encoding: 3.1.1
|
||||
|
||||
html-escaper@2.0.2: {}
|
||||
|
||||
http-proxy-agent@7.0.2:
|
||||
dependencies:
|
||||
agent-base: 7.1.4
|
||||
debug: 4.4.3
|
||||
transitivePeerDependencies:
|
||||
- supports-color
|
||||
|
||||
https-proxy-agent@7.0.6:
|
||||
dependencies:
|
||||
agent-base: 7.1.4
|
||||
debug: 4.4.3
|
||||
transitivePeerDependencies:
|
||||
- supports-color
|
||||
|
||||
human-signals@2.1.0: {}
|
||||
|
||||
human-signals@5.0.0: {}
|
||||
|
||||
iconv-lite@0.6.3:
|
||||
dependencies:
|
||||
safer-buffer: 2.1.2
|
||||
|
||||
iconv-lite@0.7.0:
|
||||
dependencies:
|
||||
safer-buffer: 2.1.2
|
||||
@@ -7254,6 +7610,8 @@ snapshots:
|
||||
|
||||
is-path-inside@4.0.0: {}
|
||||
|
||||
is-potential-custom-element-name@1.0.1: {}
|
||||
|
||||
is-stream@2.0.1: {}
|
||||
|
||||
is-stream@3.0.0: {}
|
||||
@@ -7648,6 +8006,33 @@ snapshots:
|
||||
dependencies:
|
||||
argparse: 2.0.1
|
||||
|
||||
jsdom@27.2.0:
|
||||
dependencies:
|
||||
'@acemir/cssom': 0.9.24
|
||||
'@asamuzakjp/dom-selector': 6.7.5
|
||||
cssstyle: 5.3.3
|
||||
data-urls: 6.0.0
|
||||
decimal.js: 10.6.0
|
||||
html-encoding-sniffer: 4.0.0
|
||||
http-proxy-agent: 7.0.2
|
||||
https-proxy-agent: 7.0.6
|
||||
is-potential-custom-element-name: 1.0.1
|
||||
parse5: 8.0.0
|
||||
saxes: 6.0.0
|
||||
symbol-tree: 3.2.4
|
||||
tough-cookie: 6.0.0
|
||||
w3c-xmlserializer: 5.0.0
|
||||
webidl-conversions: 8.0.0
|
||||
whatwg-encoding: 3.1.1
|
||||
whatwg-mimetype: 4.0.0
|
||||
whatwg-url: 15.1.0
|
||||
ws: 8.18.3
|
||||
xml-name-validator: 5.0.0
|
||||
transitivePeerDependencies:
|
||||
- bufferutil
|
||||
- supports-color
|
||||
- utf-8-validate
|
||||
|
||||
jsesc@3.1.0: {}
|
||||
|
||||
json-buffer@3.0.1: {}
|
||||
@@ -7737,6 +8122,8 @@ snapshots:
|
||||
dependencies:
|
||||
yallist: 3.1.1
|
||||
|
||||
lz-string@1.5.0: {}
|
||||
|
||||
magic-string@0.30.17:
|
||||
dependencies:
|
||||
'@jridgewell/sourcemap-codec': 1.5.5
|
||||
@@ -7769,6 +8156,8 @@ snapshots:
|
||||
|
||||
math-intrinsics@1.1.0: {}
|
||||
|
||||
mdn-data@2.12.2: {}
|
||||
|
||||
memfs@3.5.3:
|
||||
dependencies:
|
||||
fs-monkey: 1.1.0
|
||||
@@ -7941,6 +8330,14 @@ snapshots:
|
||||
json-parse-even-better-errors: 2.3.1
|
||||
lines-and-columns: 1.2.4
|
||||
|
||||
parse5@7.3.0:
|
||||
dependencies:
|
||||
entities: 6.0.1
|
||||
|
||||
parse5@8.0.0:
|
||||
dependencies:
|
||||
entities: 6.0.1
|
||||
|
||||
patch-console@2.0.0: {}
|
||||
|
||||
path-exists@4.0.0: {}
|
||||
@@ -8016,6 +8413,12 @@ snapshots:
|
||||
|
||||
prettier@3.6.2: {}
|
||||
|
||||
pretty-format@27.5.1:
|
||||
dependencies:
|
||||
ansi-regex: 5.0.1
|
||||
ansi-styles: 5.2.0
|
||||
react-is: 17.0.2
|
||||
|
||||
pretty-format@29.7.0:
|
||||
dependencies:
|
||||
'@jest/schemas': 29.6.3
|
||||
@@ -8051,6 +8454,14 @@ snapshots:
|
||||
transitivePeerDependencies:
|
||||
- supports-color
|
||||
|
||||
react-dom@18.3.1(react@18.3.1):
|
||||
dependencies:
|
||||
loose-envify: 1.4.0
|
||||
react: 18.3.1
|
||||
scheduler: 0.23.2
|
||||
|
||||
react-is@17.0.2: {}
|
||||
|
||||
react-is@18.3.1: {}
|
||||
|
||||
react-reconciler@0.29.2(react@18.3.1):
|
||||
@@ -8149,6 +8560,10 @@ snapshots:
|
||||
|
||||
safer-buffer@2.1.2: {}
|
||||
|
||||
saxes@6.0.0:
|
||||
dependencies:
|
||||
xmlchars: 2.2.0
|
||||
|
||||
scheduler@0.23.2:
|
||||
dependencies:
|
||||
loose-envify: 1.4.0
|
||||
@@ -8381,6 +8796,8 @@ snapshots:
|
||||
|
||||
symbol-observable@4.0.0: {}
|
||||
|
||||
symbol-tree@3.2.4: {}
|
||||
|
||||
synckit@0.11.11:
|
||||
dependencies:
|
||||
'@pkgr/core': 0.2.9
|
||||
@@ -8451,6 +8868,12 @@ snapshots:
|
||||
|
||||
tinyspy@2.2.1: {}
|
||||
|
||||
tldts-core@7.0.19: {}
|
||||
|
||||
tldts@7.0.19:
|
||||
dependencies:
|
||||
tldts-core: 7.0.19
|
||||
|
||||
tmpl@1.0.5: {}
|
||||
|
||||
to-regex-range@5.0.1:
|
||||
@@ -8465,6 +8888,14 @@ snapshots:
|
||||
|
||||
totalist@3.0.1: {}
|
||||
|
||||
tough-cookie@6.0.0:
|
||||
dependencies:
|
||||
tldts: 7.0.19
|
||||
|
||||
tr46@6.0.0:
|
||||
dependencies:
|
||||
punycode: 2.3.1
|
||||
|
||||
tree-kill@1.2.2: {}
|
||||
|
||||
tree-sitter-javascript@0.21.4(tree-sitter@0.21.1):
|
||||
@@ -8739,7 +9170,7 @@ snapshots:
|
||||
terser: 5.44.1
|
||||
tsx: 4.20.6
|
||||
|
||||
vitest@1.6.1(@types/node@22.19.1)(@vitest/ui@1.6.1)(terser@5.44.1):
|
||||
vitest@1.6.1(@types/node@22.19.1)(@vitest/ui@1.6.1)(jsdom@27.2.0)(terser@5.44.1):
|
||||
dependencies:
|
||||
'@vitest/expect': 1.6.1
|
||||
'@vitest/runner': 1.6.1
|
||||
@@ -8764,6 +9195,7 @@ snapshots:
|
||||
optionalDependencies:
|
||||
'@types/node': 22.19.1
|
||||
'@vitest/ui': 1.6.1(vitest@1.6.1)
|
||||
jsdom: 27.2.0
|
||||
transitivePeerDependencies:
|
||||
- less
|
||||
- lightningcss
|
||||
@@ -8774,7 +9206,7 @@ snapshots:
|
||||
- supports-color
|
||||
- terser
|
||||
|
||||
vitest@4.0.13(@types/node@22.19.1)(@vitest/ui@4.0.13)(terser@5.44.1)(tsx@4.20.6):
|
||||
vitest@4.0.13(@types/node@22.19.1)(@vitest/ui@4.0.13)(jsdom@27.2.0)(terser@5.44.1)(tsx@4.20.6):
|
||||
dependencies:
|
||||
'@vitest/expect': 4.0.13
|
||||
'@vitest/mocker': 4.0.13(vite@7.2.4(@types/node@22.19.1)(terser@5.44.1)(tsx@4.20.6))
|
||||
@@ -8799,6 +9231,7 @@ snapshots:
|
||||
optionalDependencies:
|
||||
'@types/node': 22.19.1
|
||||
'@vitest/ui': 4.0.13(vitest@4.0.13)
|
||||
jsdom: 27.2.0
|
||||
transitivePeerDependencies:
|
||||
- jiti
|
||||
- less
|
||||
@@ -8813,6 +9246,10 @@ snapshots:
|
||||
- tsx
|
||||
- yaml
|
||||
|
||||
w3c-xmlserializer@5.0.0:
|
||||
dependencies:
|
||||
xml-name-validator: 5.0.0
|
||||
|
||||
walker@1.0.8:
|
||||
dependencies:
|
||||
makeerror: 1.0.12
|
||||
@@ -8826,6 +9263,8 @@ snapshots:
|
||||
dependencies:
|
||||
defaults: 1.0.4
|
||||
|
||||
webidl-conversions@8.0.0: {}
|
||||
|
||||
webpack-node-externals@3.0.0: {}
|
||||
|
||||
webpack-sources@3.3.3: {}
|
||||
@@ -8862,8 +9301,19 @@ snapshots:
|
||||
- esbuild
|
||||
- uglify-js
|
||||
|
||||
whatwg-encoding@3.1.1:
|
||||
dependencies:
|
||||
iconv-lite: 0.6.3
|
||||
|
||||
whatwg-fetch@3.6.20: {}
|
||||
|
||||
whatwg-mimetype@4.0.0: {}
|
||||
|
||||
whatwg-url@15.1.0:
|
||||
dependencies:
|
||||
tr46: 6.0.0
|
||||
webidl-conversions: 8.0.0
|
||||
|
||||
which@2.0.2:
|
||||
dependencies:
|
||||
isexe: 2.0.0
|
||||
@@ -8908,6 +9358,10 @@ snapshots:
|
||||
|
||||
ws@8.18.3: {}
|
||||
|
||||
xml-name-validator@5.0.0: {}
|
||||
|
||||
xmlchars@2.2.0: {}
|
||||
|
||||
y18n@5.0.8: {}
|
||||
|
||||
yallist@3.1.1: {}
|
||||
|
||||
Reference in New Issue
Block a user