mirror of
https://github.com/AutoMaker-Org/automaker.git
synced 2026-02-03 21:03:08 +00:00
feat: add delete session functionality with confirmation dialog
- Introduced a new DeleteSessionDialog component for confirming session deletions. - Integrated the delete session dialog into the SessionManager component, allowing users to delete sessions with a confirmation prompt. - Updated the UI to handle session deletion more intuitively, enhancing user experience. - Refactored existing delete confirmation logic to utilize the new DeleteConfirmDialog component for consistency across the application.
This commit is contained in:
52
apps/app/src/components/delete-session-dialog.tsx
Normal file
52
apps/app/src/components/delete-session-dialog.tsx
Normal file
@@ -0,0 +1,52 @@
|
|||||||
|
import { MessageSquare } from "lucide-react";
|
||||||
|
import { DeleteConfirmDialog } from "@/components/ui/delete-confirm-dialog";
|
||||||
|
import type { SessionListItem } from "@/types/electron";
|
||||||
|
|
||||||
|
interface DeleteSessionDialogProps {
|
||||||
|
open: boolean;
|
||||||
|
onOpenChange: (open: boolean) => void;
|
||||||
|
session: SessionListItem | null;
|
||||||
|
onConfirm: (sessionId: string) => void;
|
||||||
|
}
|
||||||
|
|
||||||
|
export function DeleteSessionDialog({
|
||||||
|
open,
|
||||||
|
onOpenChange,
|
||||||
|
session,
|
||||||
|
onConfirm,
|
||||||
|
}: DeleteSessionDialogProps) {
|
||||||
|
const handleConfirm = () => {
|
||||||
|
if (session) {
|
||||||
|
onConfirm(session.id);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
return (
|
||||||
|
<DeleteConfirmDialog
|
||||||
|
open={open}
|
||||||
|
onOpenChange={onOpenChange}
|
||||||
|
onConfirm={handleConfirm}
|
||||||
|
title="Delete Session"
|
||||||
|
description="Are you sure you want to delete this session? This action cannot be undone."
|
||||||
|
confirmText="Delete Session"
|
||||||
|
testId="delete-session-dialog"
|
||||||
|
confirmTestId="confirm-delete-session"
|
||||||
|
>
|
||||||
|
{session && (
|
||||||
|
<div className="flex items-center gap-3 p-4 rounded-lg bg-sidebar-accent/10 border border-sidebar-border">
|
||||||
|
<div className="w-10 h-10 rounded-lg bg-sidebar-accent/20 border border-sidebar-border flex items-center justify-center shrink-0">
|
||||||
|
<MessageSquare className="w-5 h-5 text-brand-500" />
|
||||||
|
</div>
|
||||||
|
<div className="min-w-0">
|
||||||
|
<p className="font-medium text-foreground truncate">
|
||||||
|
{session.name}
|
||||||
|
</p>
|
||||||
|
<p className="text-xs text-muted-foreground">
|
||||||
|
{session.messageCount} messages
|
||||||
|
</p>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
)}
|
||||||
|
</DeleteConfirmDialog>
|
||||||
|
);
|
||||||
|
}
|
||||||
@@ -26,6 +26,7 @@ import { cn } from "@/lib/utils";
|
|||||||
import type { SessionListItem } from "@/types/electron";
|
import type { SessionListItem } from "@/types/electron";
|
||||||
import { useKeyboardShortcutsConfig } from "@/hooks/use-keyboard-shortcuts";
|
import { useKeyboardShortcutsConfig } from "@/hooks/use-keyboard-shortcuts";
|
||||||
import { getElectronAPI } from "@/lib/electron";
|
import { getElectronAPI } from "@/lib/electron";
|
||||||
|
import { DeleteSessionDialog } from "@/components/delete-session-dialog";
|
||||||
|
|
||||||
// Random session name generator
|
// Random session name generator
|
||||||
const adjectives = [
|
const adjectives = [
|
||||||
@@ -113,6 +114,8 @@ export function SessionManager({
|
|||||||
const [runningSessions, setRunningSessions] = useState<Set<string>>(
|
const [runningSessions, setRunningSessions] = useState<Set<string>>(
|
||||||
new Set()
|
new Set()
|
||||||
);
|
);
|
||||||
|
const [isDeleteDialogOpen, setIsDeleteDialogOpen] = useState(false);
|
||||||
|
const [sessionToDelete, setSessionToDelete] = useState<SessionListItem | null>(null);
|
||||||
|
|
||||||
// Check running state for all sessions
|
// Check running state for all sessions
|
||||||
const checkRunningSessions = async (sessionList: SessionListItem[]) => {
|
const checkRunningSessions = async (sessionList: SessionListItem[]) => {
|
||||||
@@ -286,11 +289,16 @@ export function SessionManager({
|
|||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
// Delete session
|
// Open delete session dialog
|
||||||
const handleDeleteSession = async (sessionId: string) => {
|
const handleDeleteSession = (session: SessionListItem) => {
|
||||||
|
setSessionToDelete(session);
|
||||||
|
setIsDeleteDialogOpen(true);
|
||||||
|
};
|
||||||
|
|
||||||
|
// Confirm delete session
|
||||||
|
const confirmDeleteSession = async (sessionId: string) => {
|
||||||
const api = getElectronAPI();
|
const api = getElectronAPI();
|
||||||
if (!api?.sessions) return;
|
if (!api?.sessions) return;
|
||||||
if (!confirm("Are you sure you want to delete this session?")) return;
|
|
||||||
|
|
||||||
const result = await api.sessions.delete(sessionId);
|
const result = await api.sessions.delete(sessionId);
|
||||||
if (result.success) {
|
if (result.success) {
|
||||||
@@ -303,6 +311,7 @@ export function SessionManager({
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
setSessionToDelete(null);
|
||||||
};
|
};
|
||||||
|
|
||||||
const activeSessions = sessions.filter((s) => !s.isArchived);
|
const activeSessions = sessions.filter((s) => !s.isArchived);
|
||||||
@@ -315,20 +324,24 @@ export function SessionManager({
|
|||||||
<CardHeader className="pb-3">
|
<CardHeader className="pb-3">
|
||||||
<div className="flex items-center justify-between mb-4">
|
<div className="flex items-center justify-between mb-4">
|
||||||
<CardTitle>Agent Sessions</CardTitle>
|
<CardTitle>Agent Sessions</CardTitle>
|
||||||
{activeTab === "active" && (
|
<HotkeyButton
|
||||||
<HotkeyButton
|
variant="default"
|
||||||
variant="default"
|
size="sm"
|
||||||
size="sm"
|
onClick={() => {
|
||||||
onClick={handleQuickCreateSession}
|
// Switch to active tab if on archived tab
|
||||||
hotkey={shortcuts.newSession}
|
if (activeTab === "archived") {
|
||||||
hotkeyActive={false}
|
setActiveTab("active");
|
||||||
data-testid="new-session-button"
|
}
|
||||||
title={`New Session (${shortcuts.newSession})`}
|
handleQuickCreateSession();
|
||||||
>
|
}}
|
||||||
<Plus className="w-4 h-4 mr-1" />
|
hotkey={shortcuts.newSession}
|
||||||
New
|
hotkeyActive={false}
|
||||||
</HotkeyButton>
|
data-testid="new-session-button"
|
||||||
)}
|
title={`New Session (${shortcuts.newSession})`}
|
||||||
|
>
|
||||||
|
<Plus className="w-4 h-4 mr-1" />
|
||||||
|
New
|
||||||
|
</HotkeyButton>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<Tabs
|
<Tabs
|
||||||
@@ -525,8 +538,9 @@ export function SessionManager({
|
|||||||
<Button
|
<Button
|
||||||
size="sm"
|
size="sm"
|
||||||
variant="ghost"
|
variant="ghost"
|
||||||
onClick={() => handleDeleteSession(session.id)}
|
onClick={() => handleDeleteSession(session)}
|
||||||
className="h-7 w-7 p-0 text-destructive"
|
className="h-7 w-7 p-0 text-destructive"
|
||||||
|
data-testid={`delete-session-${session.id}`}
|
||||||
>
|
>
|
||||||
<Trash2 className="w-3 h-3" />
|
<Trash2 className="w-3 h-3" />
|
||||||
</Button>
|
</Button>
|
||||||
@@ -552,6 +566,14 @@ export function SessionManager({
|
|||||||
</div>
|
</div>
|
||||||
)}
|
)}
|
||||||
</CardContent>
|
</CardContent>
|
||||||
|
|
||||||
|
{/* Delete Session Confirmation Dialog */}
|
||||||
|
<DeleteSessionDialog
|
||||||
|
open={isDeleteDialogOpen}
|
||||||
|
onOpenChange={setIsDeleteDialogOpen}
|
||||||
|
session={sessionToDelete}
|
||||||
|
onConfirm={confirmDeleteSession}
|
||||||
|
/>
|
||||||
</Card>
|
</Card>
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|||||||
88
apps/app/src/components/ui/delete-confirm-dialog.tsx
Normal file
88
apps/app/src/components/ui/delete-confirm-dialog.tsx
Normal file
@@ -0,0 +1,88 @@
|
|||||||
|
import { Trash2 } from "lucide-react";
|
||||||
|
import {
|
||||||
|
Dialog,
|
||||||
|
DialogContent,
|
||||||
|
DialogDescription,
|
||||||
|
DialogFooter,
|
||||||
|
DialogHeader,
|
||||||
|
DialogTitle,
|
||||||
|
} from "@/components/ui/dialog";
|
||||||
|
import { Button } from "@/components/ui/button";
|
||||||
|
import { HotkeyButton } from "@/components/ui/hotkey-button";
|
||||||
|
import type { ReactNode } from "react";
|
||||||
|
|
||||||
|
interface DeleteConfirmDialogProps {
|
||||||
|
open: boolean;
|
||||||
|
onOpenChange: (open: boolean) => void;
|
||||||
|
onConfirm: () => void;
|
||||||
|
title: string;
|
||||||
|
description: string;
|
||||||
|
/** Optional content to show between description and buttons (e.g., item preview card) */
|
||||||
|
children?: ReactNode;
|
||||||
|
/** Text for the confirm button. Defaults to "Delete" */
|
||||||
|
confirmText?: string;
|
||||||
|
/** Test ID for the dialog */
|
||||||
|
testId?: string;
|
||||||
|
/** Test ID for the confirm button */
|
||||||
|
confirmTestId?: string;
|
||||||
|
}
|
||||||
|
|
||||||
|
export function DeleteConfirmDialog({
|
||||||
|
open,
|
||||||
|
onOpenChange,
|
||||||
|
onConfirm,
|
||||||
|
title,
|
||||||
|
description,
|
||||||
|
children,
|
||||||
|
confirmText = "Delete",
|
||||||
|
testId = "delete-confirm-dialog",
|
||||||
|
confirmTestId = "confirm-delete-button",
|
||||||
|
}: DeleteConfirmDialogProps) {
|
||||||
|
const handleConfirm = () => {
|
||||||
|
onConfirm();
|
||||||
|
onOpenChange(false);
|
||||||
|
};
|
||||||
|
|
||||||
|
return (
|
||||||
|
<Dialog open={open} onOpenChange={onOpenChange}>
|
||||||
|
<DialogContent
|
||||||
|
className="bg-popover border-border max-w-md"
|
||||||
|
data-testid={testId}
|
||||||
|
>
|
||||||
|
<DialogHeader>
|
||||||
|
<DialogTitle className="flex items-center gap-2">
|
||||||
|
<Trash2 className="w-5 h-5 text-destructive" />
|
||||||
|
{title}
|
||||||
|
</DialogTitle>
|
||||||
|
<DialogDescription className="text-muted-foreground">
|
||||||
|
{description}
|
||||||
|
</DialogDescription>
|
||||||
|
</DialogHeader>
|
||||||
|
|
||||||
|
{children}
|
||||||
|
|
||||||
|
<DialogFooter className="gap-2 sm:gap-2 pt-4">
|
||||||
|
<Button
|
||||||
|
variant="ghost"
|
||||||
|
onClick={() => onOpenChange(false)}
|
||||||
|
className="px-4"
|
||||||
|
data-testid="cancel-delete-button"
|
||||||
|
>
|
||||||
|
Cancel
|
||||||
|
</Button>
|
||||||
|
<HotkeyButton
|
||||||
|
variant="destructive"
|
||||||
|
onClick={handleConfirm}
|
||||||
|
data-testid={confirmTestId}
|
||||||
|
hotkey={{ key: "Enter", cmdCtrl: true }}
|
||||||
|
hotkeyActive={open}
|
||||||
|
className="px-4"
|
||||||
|
>
|
||||||
|
<Trash2 className="w-4 h-4 mr-2" />
|
||||||
|
{confirmText}
|
||||||
|
</HotkeyButton>
|
||||||
|
</DialogFooter>
|
||||||
|
</DialogContent>
|
||||||
|
</Dialog>
|
||||||
|
);
|
||||||
|
}
|
||||||
@@ -21,6 +21,7 @@ import {
|
|||||||
DialogHeader,
|
DialogHeader,
|
||||||
DialogTitle,
|
DialogTitle,
|
||||||
} from "@/components/ui/dialog";
|
} from "@/components/ui/dialog";
|
||||||
|
import { DeleteConfirmDialog } from "@/components/ui/delete-confirm-dialog";
|
||||||
import {
|
import {
|
||||||
DropdownMenu,
|
DropdownMenu,
|
||||||
DropdownMenuContent,
|
DropdownMenuContent,
|
||||||
@@ -195,14 +196,9 @@ export const KanbanCard = memo(function KanbanCard({
|
|||||||
};
|
};
|
||||||
|
|
||||||
const handleConfirmDelete = () => {
|
const handleConfirmDelete = () => {
|
||||||
setIsDeleteDialogOpen(false);
|
|
||||||
onDelete();
|
onDelete();
|
||||||
};
|
};
|
||||||
|
|
||||||
const handleCancelDelete = () => {
|
|
||||||
setIsDeleteDialogOpen(false);
|
|
||||||
};
|
|
||||||
|
|
||||||
// Dragging logic:
|
// Dragging logic:
|
||||||
// - Backlog items can always be dragged
|
// - Backlog items can always be dragged
|
||||||
// - skipTests items can be dragged even when in_progress or verified (unless currently running)
|
// - skipTests items can be dragged even when in_progress or verified (unless currently running)
|
||||||
@@ -805,35 +801,15 @@ export const KanbanCard = memo(function KanbanCard({
|
|||||||
</CardContent>
|
</CardContent>
|
||||||
|
|
||||||
{/* Delete Confirmation Dialog */}
|
{/* Delete Confirmation Dialog */}
|
||||||
<Dialog open={isDeleteDialogOpen} onOpenChange={setIsDeleteDialogOpen}>
|
<DeleteConfirmDialog
|
||||||
<DialogContent data-testid="delete-confirmation-dialog">
|
open={isDeleteDialogOpen}
|
||||||
<DialogHeader>
|
onOpenChange={setIsDeleteDialogOpen}
|
||||||
<DialogTitle>Delete Feature</DialogTitle>
|
onConfirm={handleConfirmDelete}
|
||||||
<DialogDescription>
|
title="Delete Feature"
|
||||||
Are you sure you want to delete this feature? This action cannot
|
description="Are you sure you want to delete this feature? This action cannot be undone."
|
||||||
be undone.
|
testId="delete-confirmation-dialog"
|
||||||
</DialogDescription>
|
confirmTestId="confirm-delete-button"
|
||||||
</DialogHeader>
|
/>
|
||||||
<DialogFooter className="mt-6">
|
|
||||||
<Button
|
|
||||||
variant="ghost"
|
|
||||||
onClick={handleCancelDelete}
|
|
||||||
data-testid="cancel-delete-button"
|
|
||||||
>
|
|
||||||
Cancel
|
|
||||||
</Button>
|
|
||||||
<HotkeyButton
|
|
||||||
variant="destructive"
|
|
||||||
onClick={handleConfirmDelete}
|
|
||||||
data-testid="confirm-delete-button"
|
|
||||||
hotkey={{ key: "Enter", cmdCtrl: true }}
|
|
||||||
hotkeyActive={isDeleteDialogOpen}
|
|
||||||
>
|
|
||||||
Delete
|
|
||||||
</HotkeyButton>
|
|
||||||
</DialogFooter>
|
|
||||||
</DialogContent>
|
|
||||||
</Dialog>
|
|
||||||
|
|
||||||
{/* Summary Modal */}
|
{/* Summary Modal */}
|
||||||
<Dialog open={isSummaryDialogOpen} onOpenChange={setIsSummaryDialogOpen}>
|
<Dialog open={isSummaryDialogOpen} onOpenChange={setIsSummaryDialogOpen}>
|
||||||
|
|||||||
@@ -1,13 +1,5 @@
|
|||||||
import { Trash2, Folder } from "lucide-react";
|
import { Folder, Trash2 } from "lucide-react";
|
||||||
import {
|
import { DeleteConfirmDialog } from "@/components/ui/delete-confirm-dialog";
|
||||||
Dialog,
|
|
||||||
DialogContent,
|
|
||||||
DialogDescription,
|
|
||||||
DialogFooter,
|
|
||||||
DialogHeader,
|
|
||||||
DialogTitle,
|
|
||||||
} from "@/components/ui/dialog";
|
|
||||||
import { Button } from "@/components/ui/button";
|
|
||||||
import type { Project } from "@/lib/electron";
|
import type { Project } from "@/lib/electron";
|
||||||
|
|
||||||
interface DeleteProjectDialogProps {
|
interface DeleteProjectDialogProps {
|
||||||
@@ -26,24 +18,22 @@ export function DeleteProjectDialog({
|
|||||||
const handleConfirm = () => {
|
const handleConfirm = () => {
|
||||||
if (project) {
|
if (project) {
|
||||||
onConfirm(project.id);
|
onConfirm(project.id);
|
||||||
onOpenChange(false);
|
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<Dialog open={open} onOpenChange={onOpenChange}>
|
<DeleteConfirmDialog
|
||||||
<DialogContent className="bg-popover border-border max-w-md">
|
open={open}
|
||||||
<DialogHeader>
|
onOpenChange={onOpenChange}
|
||||||
<DialogTitle className="flex items-center gap-2">
|
onConfirm={handleConfirm}
|
||||||
<Trash2 className="w-5 h-5 text-destructive" />
|
title="Delete Project"
|
||||||
Delete Project
|
description="Are you sure you want to move this project to Trash?"
|
||||||
</DialogTitle>
|
confirmText="Move to Trash"
|
||||||
<DialogDescription className="text-muted-foreground">
|
testId="delete-project-dialog"
|
||||||
Are you sure you want to move this project to Trash?
|
confirmTestId="confirm-delete-project"
|
||||||
</DialogDescription>
|
>
|
||||||
</DialogHeader>
|
{project && (
|
||||||
|
<>
|
||||||
{project && (
|
|
||||||
<div className="flex items-center gap-3 p-4 rounded-lg bg-sidebar-accent/10 border border-sidebar-border">
|
<div className="flex items-center gap-3 p-4 rounded-lg bg-sidebar-accent/10 border border-sidebar-border">
|
||||||
<div className="w-10 h-10 rounded-lg bg-sidebar-accent/20 border border-sidebar-border flex items-center justify-center shrink-0">
|
<div className="w-10 h-10 rounded-lg bg-sidebar-accent/20 border border-sidebar-border flex items-center justify-center shrink-0">
|
||||||
<Folder className="w-5 h-5 text-brand-500" />
|
<Folder className="w-5 h-5 text-brand-500" />
|
||||||
@@ -57,27 +47,13 @@ export function DeleteProjectDialog({
|
|||||||
</p>
|
</p>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
)}
|
|
||||||
|
|
||||||
<p className="text-sm text-muted-foreground">
|
<p className="text-sm text-muted-foreground">
|
||||||
The folder will remain on disk until you permanently delete it from
|
The folder will remain on disk until you permanently delete it from
|
||||||
Trash.
|
Trash.
|
||||||
</p>
|
</p>
|
||||||
|
</>
|
||||||
<DialogFooter className="gap-2 sm:gap-0">
|
)}
|
||||||
<Button variant="ghost" onClick={() => onOpenChange(false)}>
|
</DeleteConfirmDialog>
|
||||||
Cancel
|
|
||||||
</Button>
|
|
||||||
<Button
|
|
||||||
variant="destructive"
|
|
||||||
onClick={handleConfirm}
|
|
||||||
data-testid="confirm-delete-project"
|
|
||||||
>
|
|
||||||
<Trash2 className="w-4 h-4 mr-2" />
|
|
||||||
Move to Trash
|
|
||||||
</Button>
|
|
||||||
</DialogFooter>
|
|
||||||
</DialogContent>
|
|
||||||
</Dialog>
|
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -147,7 +147,7 @@ export const DEFAULT_KEYBOARD_SHORTCUTS: KeyboardShortcuts = {
|
|||||||
// Note: Some shortcuts share the same key (e.g., "N" for addFeature, newSession, addProfile)
|
// Note: Some shortcuts share the same key (e.g., "N" for addFeature, newSession, addProfile)
|
||||||
// This is intentional as they are context-specific and only active in their respective views
|
// This is intentional as they are context-specific and only active in their respective views
|
||||||
addFeature: "N", // Only active in board view
|
addFeature: "N", // Only active in board view
|
||||||
addContextFile: "F", // Only active in context view
|
addContextFile: "N", // Only active in context view
|
||||||
startNext: "G", // Only active in board view
|
startNext: "G", // Only active in board view
|
||||||
newSession: "N", // Only active in agent view
|
newSession: "N", // Only active in agent view
|
||||||
openProject: "O", // Global shortcut
|
openProject: "O", // Global shortcut
|
||||||
@@ -1136,6 +1136,20 @@ export const useAppStore = create<AppState & AppActions>()(
|
|||||||
}),
|
}),
|
||||||
{
|
{
|
||||||
name: "automaker-storage",
|
name: "automaker-storage",
|
||||||
|
version: 1, // Increment when making breaking changes to persisted state
|
||||||
|
migrate: (persistedState: unknown, version: number) => {
|
||||||
|
const state = persistedState as Partial<AppState>;
|
||||||
|
|
||||||
|
// Migration from version 0 (no version) to version 1:
|
||||||
|
// - Change addContextFile shortcut from "F" to "N"
|
||||||
|
if (version === 0) {
|
||||||
|
if (state.keyboardShortcuts?.addContextFile === "F") {
|
||||||
|
state.keyboardShortcuts.addContextFile = "N";
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return state as AppState;
|
||||||
|
},
|
||||||
partialize: (state) => ({
|
partialize: (state) => ({
|
||||||
// Project management
|
// Project management
|
||||||
projects: state.projects,
|
projects: state.projects,
|
||||||
|
|||||||
@@ -1,38 +0,0 @@
|
|||||||
[
|
|
||||||
{
|
|
||||||
"id": "msg_1765523527444_d8xyk8rh9",
|
|
||||||
"role": "user",
|
|
||||||
"content": "this is a test",
|
|
||||||
"timestamp": "2025-12-12T07:12:07.444Z"
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"id": "msg_1765523534075_351i0rcft",
|
|
||||||
"role": "assistant",
|
|
||||||
"content": "Hello! This is a test response. I'm working correctly and ready to help you with your software development projects.\n\nIs there something specific you'd like to work on today? I can help you with:\n\n- **Planning a new project** - Define requirements, architecture, and technical approach\n- **Writing code** - Create new files, implement features, or build components\n- **Editing existing code** - Modify, refactor, or fix bugs in your codebase\n- **Running commands** - Execute tests, builds, or other terminal commands\n- **Exploring your codebase** - Search for patterns, understand existing code structure\n\nJust let me know what you need!",
|
|
||||||
"timestamp": "2025-12-12T07:12:14.075Z"
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"id": "msg_1765524006202_3d2cuya6b",
|
|
||||||
"role": "user",
|
|
||||||
"content": "test",
|
|
||||||
"timestamp": "2025-12-12T07:20:06.202Z"
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"id": "msg_1765524011389_sdfbuai3k",
|
|
||||||
"role": "assistant",
|
|
||||||
"content": "Hello! I'm here to help you build software. How can I assist you today?\n\nI can help you with:\n\n- **Planning** - Define requirements, discuss architecture, and design your project\n- **Coding** - Write, edit, and modify code files\n- **Debugging** - Analyze issues and fix bugs\n- **Testing** - Run tests and verify functionality\n- **Exploring** - Search and understand existing codebases\n\nWhat would you like to work on?",
|
|
||||||
"timestamp": "2025-12-12T07:20:11.389Z"
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"id": "msg_1765524491273_7ocghnbb5",
|
|
||||||
"role": "user",
|
|
||||||
"content": "test",
|
|
||||||
"timestamp": "2025-12-12T07:28:11.273Z"
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"id": "msg_1765524498237_x8b4148gc",
|
|
||||||
"role": "assistant",
|
|
||||||
"content": "Hello! I'm here to help you build software. How can I assist you today?\n\nI can help you with:\n- **Planning** - Define requirements, architecture, and technical approaches\n- **Coding** - Write, edit, and modify code files\n- **Debugging** - Find and fix issues in your codebase\n- **Testing** - Run tests and help improve test coverage\n- **Exploring** - Search and analyze your existing codebase\n\nWhat would you like to work on?",
|
|
||||||
"timestamp": "2025-12-12T07:28:18.237Z"
|
|
||||||
}
|
|
||||||
]
|
|
||||||
File diff suppressed because one or more lines are too long
@@ -1,18 +0,0 @@
|
|||||||
{
|
|
||||||
"msg_1765523524581_xhk6u45v2": {
|
|
||||||
"id": "msg_1765523524581_xhk6u45v2",
|
|
||||||
"name": "Bright Agent 2",
|
|
||||||
"projectPath": "/Users/webdevcody/Workspace/automaker",
|
|
||||||
"workingDirectory": "/Users/webdevcody/Workspace/automaker",
|
|
||||||
"createdAt": "2025-12-12T07:12:04.582Z",
|
|
||||||
"updatedAt": "2025-12-12T07:28:18.571Z"
|
|
||||||
},
|
|
||||||
"msg_1765525491205_xeuqv7i9v": {
|
|
||||||
"id": "msg_1765525491205_xeuqv7i9v",
|
|
||||||
"name": "Optimal Helper 52",
|
|
||||||
"projectPath": "/Users/webdevcody/Workspace/automaker",
|
|
||||||
"workingDirectory": "/Users/webdevcody/Workspace/automaker",
|
|
||||||
"createdAt": "2025-12-12T07:44:51.205Z",
|
|
||||||
"updatedAt": "2025-12-12T07:46:03.339Z"
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -313,6 +313,46 @@ export class AutoModeService {
|
|||||||
// No worktree, use project path
|
// No worktree, use project path
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Load feature info for context
|
||||||
|
const feature = await this.loadFeature(projectPath, featureId);
|
||||||
|
|
||||||
|
// Load previous agent output if it exists
|
||||||
|
const contextPath = path.join(
|
||||||
|
projectPath,
|
||||||
|
".automaker",
|
||||||
|
"features",
|
||||||
|
featureId,
|
||||||
|
"agent-output.md"
|
||||||
|
);
|
||||||
|
let previousContext = "";
|
||||||
|
try {
|
||||||
|
previousContext = await fs.readFile(contextPath, "utf-8");
|
||||||
|
} catch {
|
||||||
|
// No previous context
|
||||||
|
}
|
||||||
|
|
||||||
|
// Build complete prompt with feature info, previous context, and follow-up instructions
|
||||||
|
let fullPrompt = `## Follow-up on Feature Implementation
|
||||||
|
|
||||||
|
${feature ? this.buildFeaturePrompt(feature) : `**Feature ID:** ${featureId}`}
|
||||||
|
`;
|
||||||
|
|
||||||
|
if (previousContext) {
|
||||||
|
fullPrompt += `
|
||||||
|
## Previous Agent Work
|
||||||
|
The following is the output from the previous implementation attempt:
|
||||||
|
|
||||||
|
${previousContext}
|
||||||
|
`;
|
||||||
|
}
|
||||||
|
|
||||||
|
fullPrompt += `
|
||||||
|
## Follow-up Instructions
|
||||||
|
${prompt}
|
||||||
|
|
||||||
|
## Task
|
||||||
|
Address the follow-up instructions above. Review the previous work and make the requested changes or fixes.`;
|
||||||
|
|
||||||
this.runningFeatures.set(featureId, {
|
this.runningFeatures.set(featureId, {
|
||||||
featureId,
|
featureId,
|
||||||
projectPath,
|
projectPath,
|
||||||
@@ -326,11 +366,14 @@ export class AutoModeService {
|
|||||||
this.emitAutoModeEvent("auto_mode_feature_start", {
|
this.emitAutoModeEvent("auto_mode_feature_start", {
|
||||||
featureId,
|
featureId,
|
||||||
projectPath,
|
projectPath,
|
||||||
feature: { id: featureId, title: "Follow-up", description: prompt.substring(0, 100) },
|
feature: feature || { id: featureId, title: "Follow-up", description: prompt.substring(0, 100) },
|
||||||
});
|
});
|
||||||
|
|
||||||
try {
|
try {
|
||||||
await this.runAgent(workDir, featureId, prompt, abortController, imagePaths);
|
await this.runAgent(workDir, featureId, fullPrompt, abortController, imagePaths);
|
||||||
|
|
||||||
|
// Mark as waiting_approval for user review
|
||||||
|
await this.updateFeatureStatus(projectPath, featureId, "waiting_approval");
|
||||||
|
|
||||||
this.emitAutoModeEvent("auto_mode_feature_complete", {
|
this.emitAutoModeEvent("auto_mode_feature_complete", {
|
||||||
featureId,
|
featureId,
|
||||||
|
|||||||
182
package-lock.json
generated
182
package-lock.json
generated
@@ -17,11 +17,9 @@
|
|||||||
"version": "0.1.0",
|
"version": "0.1.0",
|
||||||
"license": "Unlicense",
|
"license": "Unlicense",
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
"@anthropic-ai/claude-agent-sdk": "^0.1.61",
|
|
||||||
"@dnd-kit/core": "^6.3.1",
|
"@dnd-kit/core": "^6.3.1",
|
||||||
"@dnd-kit/sortable": "^10.0.0",
|
"@dnd-kit/sortable": "^10.0.0",
|
||||||
"@dnd-kit/utilities": "^3.2.2",
|
"@dnd-kit/utilities": "^3.2.2",
|
||||||
"@homebridge/node-pty-prebuilt-multiarch": "^0.13.1",
|
|
||||||
"@radix-ui/react-checkbox": "^1.3.3",
|
"@radix-ui/react-checkbox": "^1.3.3",
|
||||||
"@radix-ui/react-dialog": "^1.1.15",
|
"@radix-ui/react-dialog": "^1.1.15",
|
||||||
"@radix-ui/react-dropdown-menu": "^2.1.16",
|
"@radix-ui/react-dropdown-menu": "^2.1.16",
|
||||||
@@ -1216,22 +1214,6 @@
|
|||||||
"@hapi/hoek": "^11.0.2"
|
"@hapi/hoek": "^11.0.2"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"apps/app/node_modules/@homebridge/node-pty-prebuilt-multiarch": {
|
|
||||||
"version": "0.13.1",
|
|
||||||
"hasInstallScript": true,
|
|
||||||
"license": "MIT",
|
|
||||||
"dependencies": {
|
|
||||||
"node-addon-api": "^7.1.0",
|
|
||||||
"prebuild-install": "^7.1.2"
|
|
||||||
},
|
|
||||||
"engines": {
|
|
||||||
"node": ">=18.0.0 <25.0.0"
|
|
||||||
}
|
|
||||||
},
|
|
||||||
"apps/app/node_modules/@homebridge/node-pty-prebuilt-multiarch/node_modules/node-addon-api": {
|
|
||||||
"version": "7.1.1",
|
|
||||||
"license": "MIT"
|
|
||||||
},
|
|
||||||
"apps/app/node_modules/@humanfs/core": {
|
"apps/app/node_modules/@humanfs/core": {
|
||||||
"version": "0.19.1",
|
"version": "0.19.1",
|
||||||
"dev": true,
|
"dev": true,
|
||||||
@@ -3554,6 +3536,7 @@
|
|||||||
},
|
},
|
||||||
"apps/app/node_modules/base64-js": {
|
"apps/app/node_modules/base64-js": {
|
||||||
"version": "1.5.1",
|
"version": "1.5.1",
|
||||||
|
"dev": true,
|
||||||
"funding": [
|
"funding": [
|
||||||
{
|
{
|
||||||
"type": "github",
|
"type": "github",
|
||||||
@@ -3580,6 +3563,7 @@
|
|||||||
},
|
},
|
||||||
"apps/app/node_modules/bl": {
|
"apps/app/node_modules/bl": {
|
||||||
"version": "4.1.0",
|
"version": "4.1.0",
|
||||||
|
"dev": true,
|
||||||
"license": "MIT",
|
"license": "MIT",
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
"buffer": "^5.5.0",
|
"buffer": "^5.5.0",
|
||||||
@@ -3638,6 +3622,7 @@
|
|||||||
},
|
},
|
||||||
"apps/app/node_modules/buffer": {
|
"apps/app/node_modules/buffer": {
|
||||||
"version": "5.7.1",
|
"version": "5.7.1",
|
||||||
|
"dev": true,
|
||||||
"funding": [
|
"funding": [
|
||||||
{
|
{
|
||||||
"type": "github",
|
"type": "github",
|
||||||
@@ -4351,6 +4336,7 @@
|
|||||||
},
|
},
|
||||||
"apps/app/node_modules/decompress-response": {
|
"apps/app/node_modules/decompress-response": {
|
||||||
"version": "6.0.0",
|
"version": "6.0.0",
|
||||||
|
"dev": true,
|
||||||
"license": "MIT",
|
"license": "MIT",
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
"mimic-response": "^3.1.0"
|
"mimic-response": "^3.1.0"
|
||||||
@@ -4364,6 +4350,7 @@
|
|||||||
},
|
},
|
||||||
"apps/app/node_modules/decompress-response/node_modules/mimic-response": {
|
"apps/app/node_modules/decompress-response/node_modules/mimic-response": {
|
||||||
"version": "3.1.0",
|
"version": "3.1.0",
|
||||||
|
"dev": true,
|
||||||
"license": "MIT",
|
"license": "MIT",
|
||||||
"engines": {
|
"engines": {
|
||||||
"node": ">=10"
|
"node": ">=10"
|
||||||
@@ -4785,6 +4772,7 @@
|
|||||||
},
|
},
|
||||||
"apps/app/node_modules/end-of-stream": {
|
"apps/app/node_modules/end-of-stream": {
|
||||||
"version": "1.4.5",
|
"version": "1.4.5",
|
||||||
|
"dev": true,
|
||||||
"license": "MIT",
|
"license": "MIT",
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
"once": "^1.4.0"
|
"once": "^1.4.0"
|
||||||
@@ -5364,13 +5352,6 @@
|
|||||||
"node": ">=0.10.0"
|
"node": ">=0.10.0"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"apps/app/node_modules/expand-template": {
|
|
||||||
"version": "2.0.3",
|
|
||||||
"license": "(MIT OR WTFPL)",
|
|
||||||
"engines": {
|
|
||||||
"node": ">=6"
|
|
||||||
}
|
|
||||||
},
|
|
||||||
"apps/app/node_modules/exponential-backoff": {
|
"apps/app/node_modules/exponential-backoff": {
|
||||||
"version": "3.1.3",
|
"version": "3.1.3",
|
||||||
"dev": true,
|
"dev": true,
|
||||||
@@ -5615,10 +5596,6 @@
|
|||||||
"node": ">= 6"
|
"node": ">= 6"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"apps/app/node_modules/fs-constants": {
|
|
||||||
"version": "1.0.0",
|
|
||||||
"license": "MIT"
|
|
||||||
},
|
|
||||||
"apps/app/node_modules/fs-extra": {
|
"apps/app/node_modules/fs-extra": {
|
||||||
"version": "8.1.0",
|
"version": "8.1.0",
|
||||||
"dev": true,
|
"dev": true,
|
||||||
@@ -5748,10 +5725,6 @@
|
|||||||
"url": "https://github.com/sponsors/ljharb"
|
"url": "https://github.com/sponsors/ljharb"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"apps/app/node_modules/github-from-package": {
|
|
||||||
"version": "0.0.0",
|
|
||||||
"license": "MIT"
|
|
||||||
},
|
|
||||||
"apps/app/node_modules/glob": {
|
"apps/app/node_modules/glob": {
|
||||||
"version": "7.2.3",
|
"version": "7.2.3",
|
||||||
"dev": true,
|
"dev": true,
|
||||||
@@ -6083,6 +6056,7 @@
|
|||||||
},
|
},
|
||||||
"apps/app/node_modules/ieee754": {
|
"apps/app/node_modules/ieee754": {
|
||||||
"version": "1.2.1",
|
"version": "1.2.1",
|
||||||
|
"dev": true,
|
||||||
"funding": [
|
"funding": [
|
||||||
{
|
{
|
||||||
"type": "github",
|
"type": "github",
|
||||||
@@ -7730,18 +7704,10 @@
|
|||||||
"node": ">=10"
|
"node": ">=10"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"apps/app/node_modules/mkdirp-classic": {
|
|
||||||
"version": "0.5.3",
|
|
||||||
"license": "MIT"
|
|
||||||
},
|
|
||||||
"apps/app/node_modules/ms": {
|
"apps/app/node_modules/ms": {
|
||||||
"version": "2.1.3",
|
"version": "2.1.3",
|
||||||
"license": "MIT"
|
"license": "MIT"
|
||||||
},
|
},
|
||||||
"apps/app/node_modules/napi-build-utils": {
|
|
||||||
"version": "2.0.0",
|
|
||||||
"license": "MIT"
|
|
||||||
},
|
|
||||||
"apps/app/node_modules/napi-postinstall": {
|
"apps/app/node_modules/napi-postinstall": {
|
||||||
"version": "0.3.4",
|
"version": "0.3.4",
|
||||||
"dev": true,
|
"dev": true,
|
||||||
@@ -8358,50 +8324,6 @@
|
|||||||
"node": "^12.20.0 || >=14"
|
"node": "^12.20.0 || >=14"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"apps/app/node_modules/prebuild-install": {
|
|
||||||
"version": "7.1.3",
|
|
||||||
"license": "MIT",
|
|
||||||
"dependencies": {
|
|
||||||
"detect-libc": "^2.0.0",
|
|
||||||
"expand-template": "^2.0.3",
|
|
||||||
"github-from-package": "0.0.0",
|
|
||||||
"minimist": "^1.2.3",
|
|
||||||
"mkdirp-classic": "^0.5.3",
|
|
||||||
"napi-build-utils": "^2.0.0",
|
|
||||||
"node-abi": "^3.3.0",
|
|
||||||
"pump": "^3.0.0",
|
|
||||||
"rc": "^1.2.7",
|
|
||||||
"simple-get": "^4.0.0",
|
|
||||||
"tar-fs": "^2.0.0",
|
|
||||||
"tunnel-agent": "^0.6.0"
|
|
||||||
},
|
|
||||||
"bin": {
|
|
||||||
"prebuild-install": "bin.js"
|
|
||||||
},
|
|
||||||
"engines": {
|
|
||||||
"node": ">=10"
|
|
||||||
}
|
|
||||||
},
|
|
||||||
"apps/app/node_modules/prebuild-install/node_modules/node-abi": {
|
|
||||||
"version": "3.85.0",
|
|
||||||
"license": "MIT",
|
|
||||||
"dependencies": {
|
|
||||||
"semver": "^7.3.5"
|
|
||||||
},
|
|
||||||
"engines": {
|
|
||||||
"node": ">=10"
|
|
||||||
}
|
|
||||||
},
|
|
||||||
"apps/app/node_modules/prebuild-install/node_modules/semver": {
|
|
||||||
"version": "7.7.3",
|
|
||||||
"license": "ISC",
|
|
||||||
"bin": {
|
|
||||||
"semver": "bin/semver.js"
|
|
||||||
},
|
|
||||||
"engines": {
|
|
||||||
"node": ">=10"
|
|
||||||
}
|
|
||||||
},
|
|
||||||
"apps/app/node_modules/prelude-ls": {
|
"apps/app/node_modules/prelude-ls": {
|
||||||
"version": "1.2.1",
|
"version": "1.2.1",
|
||||||
"dev": true,
|
"dev": true,
|
||||||
@@ -8468,6 +8390,7 @@
|
|||||||
},
|
},
|
||||||
"apps/app/node_modules/pump": {
|
"apps/app/node_modules/pump": {
|
||||||
"version": "3.0.3",
|
"version": "3.0.3",
|
||||||
|
"dev": true,
|
||||||
"license": "MIT",
|
"license": "MIT",
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
"end-of-stream": "^1.1.0",
|
"end-of-stream": "^1.1.0",
|
||||||
@@ -8610,6 +8533,7 @@
|
|||||||
},
|
},
|
||||||
"apps/app/node_modules/readable-stream": {
|
"apps/app/node_modules/readable-stream": {
|
||||||
"version": "3.6.2",
|
"version": "3.6.2",
|
||||||
|
"dev": true,
|
||||||
"license": "MIT",
|
"license": "MIT",
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
"inherits": "^2.0.3",
|
"inherits": "^2.0.3",
|
||||||
@@ -8991,47 +8915,6 @@
|
|||||||
"url": "https://github.com/sponsors/ljharb"
|
"url": "https://github.com/sponsors/ljharb"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"apps/app/node_modules/simple-concat": {
|
|
||||||
"version": "1.0.1",
|
|
||||||
"funding": [
|
|
||||||
{
|
|
||||||
"type": "github",
|
|
||||||
"url": "https://github.com/sponsors/feross"
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"type": "patreon",
|
|
||||||
"url": "https://www.patreon.com/feross"
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"type": "consulting",
|
|
||||||
"url": "https://feross.org/support"
|
|
||||||
}
|
|
||||||
],
|
|
||||||
"license": "MIT"
|
|
||||||
},
|
|
||||||
"apps/app/node_modules/simple-get": {
|
|
||||||
"version": "4.0.1",
|
|
||||||
"funding": [
|
|
||||||
{
|
|
||||||
"type": "github",
|
|
||||||
"url": "https://github.com/sponsors/feross"
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"type": "patreon",
|
|
||||||
"url": "https://www.patreon.com/feross"
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"type": "consulting",
|
|
||||||
"url": "https://feross.org/support"
|
|
||||||
}
|
|
||||||
],
|
|
||||||
"license": "MIT",
|
|
||||||
"dependencies": {
|
|
||||||
"decompress-response": "^6.0.0",
|
|
||||||
"once": "^1.3.1",
|
|
||||||
"simple-concat": "^1.0.0"
|
|
||||||
}
|
|
||||||
},
|
|
||||||
"apps/app/node_modules/simple-update-notifier": {
|
"apps/app/node_modules/simple-update-notifier": {
|
||||||
"version": "2.0.0",
|
"version": "2.0.0",
|
||||||
"dev": true,
|
"dev": true,
|
||||||
@@ -9188,6 +9071,7 @@
|
|||||||
},
|
},
|
||||||
"apps/app/node_modules/string_decoder": {
|
"apps/app/node_modules/string_decoder": {
|
||||||
"version": "1.3.0",
|
"version": "1.3.0",
|
||||||
|
"dev": true,
|
||||||
"license": "MIT",
|
"license": "MIT",
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
"safe-buffer": "~5.2.0"
|
"safe-buffer": "~5.2.0"
|
||||||
@@ -9462,34 +9346,6 @@
|
|||||||
"node": ">=10"
|
"node": ">=10"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"apps/app/node_modules/tar-fs": {
|
|
||||||
"version": "2.1.4",
|
|
||||||
"license": "MIT",
|
|
||||||
"dependencies": {
|
|
||||||
"chownr": "^1.1.1",
|
|
||||||
"mkdirp-classic": "^0.5.2",
|
|
||||||
"pump": "^3.0.0",
|
|
||||||
"tar-stream": "^2.1.4"
|
|
||||||
}
|
|
||||||
},
|
|
||||||
"apps/app/node_modules/tar-fs/node_modules/chownr": {
|
|
||||||
"version": "1.1.4",
|
|
||||||
"license": "ISC"
|
|
||||||
},
|
|
||||||
"apps/app/node_modules/tar-stream": {
|
|
||||||
"version": "2.2.0",
|
|
||||||
"license": "MIT",
|
|
||||||
"dependencies": {
|
|
||||||
"bl": "^4.0.3",
|
|
||||||
"end-of-stream": "^1.4.1",
|
|
||||||
"fs-constants": "^1.0.0",
|
|
||||||
"inherits": "^2.0.3",
|
|
||||||
"readable-stream": "^3.1.1"
|
|
||||||
},
|
|
||||||
"engines": {
|
|
||||||
"node": ">=6"
|
|
||||||
}
|
|
||||||
},
|
|
||||||
"apps/app/node_modules/tar/node_modules/minipass": {
|
"apps/app/node_modules/tar/node_modules/minipass": {
|
||||||
"version": "5.0.0",
|
"version": "5.0.0",
|
||||||
"dev": true,
|
"dev": true,
|
||||||
@@ -9731,16 +9587,6 @@
|
|||||||
"json5": "lib/cli.js"
|
"json5": "lib/cli.js"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"apps/app/node_modules/tunnel-agent": {
|
|
||||||
"version": "0.6.0",
|
|
||||||
"license": "Apache-2.0",
|
|
||||||
"dependencies": {
|
|
||||||
"safe-buffer": "^5.0.1"
|
|
||||||
},
|
|
||||||
"engines": {
|
|
||||||
"node": "*"
|
|
||||||
}
|
|
||||||
},
|
|
||||||
"apps/app/node_modules/tw-animate-css": {
|
"apps/app/node_modules/tw-animate-css": {
|
||||||
"version": "1.4.0",
|
"version": "1.4.0",
|
||||||
"dev": true,
|
"dev": true,
|
||||||
@@ -10094,6 +9940,7 @@
|
|||||||
},
|
},
|
||||||
"apps/app/node_modules/util-deprecate": {
|
"apps/app/node_modules/util-deprecate": {
|
||||||
"version": "1.0.2",
|
"version": "1.0.2",
|
||||||
|
"dev": true,
|
||||||
"license": "MIT"
|
"license": "MIT"
|
||||||
},
|
},
|
||||||
"apps/app/node_modules/verror": {
|
"apps/app/node_modules/verror": {
|
||||||
@@ -12668,6 +12515,7 @@
|
|||||||
"version": "0.6.0",
|
"version": "0.6.0",
|
||||||
"resolved": "https://registry.npmjs.org/deep-extend/-/deep-extend-0.6.0.tgz",
|
"resolved": "https://registry.npmjs.org/deep-extend/-/deep-extend-0.6.0.tgz",
|
||||||
"integrity": "sha512-LOHxIOaPYdHlJRtCQfDIVZtfw/ufM8+rVj649RIHzcm/vGwQRXFt6OPqIFWsm2XEMrNIEtWR64sY1LEKD2vAOA==",
|
"integrity": "sha512-LOHxIOaPYdHlJRtCQfDIVZtfw/ufM8+rVj649RIHzcm/vGwQRXFt6OPqIFWsm2XEMrNIEtWR64sY1LEKD2vAOA==",
|
||||||
|
"dev": true,
|
||||||
"license": "MIT",
|
"license": "MIT",
|
||||||
"engines": {
|
"engines": {
|
||||||
"node": ">=4.0.0"
|
"node": ">=4.0.0"
|
||||||
@@ -12686,6 +12534,7 @@
|
|||||||
"version": "2.1.2",
|
"version": "2.1.2",
|
||||||
"resolved": "https://registry.npmjs.org/detect-libc/-/detect-libc-2.1.2.tgz",
|
"resolved": "https://registry.npmjs.org/detect-libc/-/detect-libc-2.1.2.tgz",
|
||||||
"integrity": "sha512-Btj2BOOO83o3WyH59e8MgXsxEQVcarkUOpEYrubB0urwnN10yQ364rsiByU11nZlqWYZm05i/of7io4mzihBtQ==",
|
"integrity": "sha512-Btj2BOOO83o3WyH59e8MgXsxEQVcarkUOpEYrubB0urwnN10yQ364rsiByU11nZlqWYZm05i/of7io4mzihBtQ==",
|
||||||
|
"devOptional": true,
|
||||||
"license": "Apache-2.0",
|
"license": "Apache-2.0",
|
||||||
"engines": {
|
"engines": {
|
||||||
"node": ">=8"
|
"node": ">=8"
|
||||||
@@ -13228,6 +13077,7 @@
|
|||||||
"version": "1.3.8",
|
"version": "1.3.8",
|
||||||
"resolved": "https://registry.npmjs.org/ini/-/ini-1.3.8.tgz",
|
"resolved": "https://registry.npmjs.org/ini/-/ini-1.3.8.tgz",
|
||||||
"integrity": "sha512-JV/yugV2uzW5iMRSiZAyDtQd+nxtUnjeLt0acNdw98kKLrvuRVyB80tsREOE7yvGVgalhZ6RNXCmEHkUKBKxew==",
|
"integrity": "sha512-JV/yugV2uzW5iMRSiZAyDtQd+nxtUnjeLt0acNdw98kKLrvuRVyB80tsREOE7yvGVgalhZ6RNXCmEHkUKBKxew==",
|
||||||
|
"dev": true,
|
||||||
"license": "ISC"
|
"license": "ISC"
|
||||||
},
|
},
|
||||||
"node_modules/ipaddr.js": {
|
"node_modules/ipaddr.js": {
|
||||||
@@ -13573,6 +13423,7 @@
|
|||||||
"version": "1.2.8",
|
"version": "1.2.8",
|
||||||
"resolved": "https://registry.npmjs.org/minimist/-/minimist-1.2.8.tgz",
|
"resolved": "https://registry.npmjs.org/minimist/-/minimist-1.2.8.tgz",
|
||||||
"integrity": "sha512-2yyAR8qBkN3YuheJanUpWC5U3bb5osDywNB8RzDVlDwDHbocAJveqqj1u8+SVD7jkWT4yvsHCpWqqWqAxb0zCA==",
|
"integrity": "sha512-2yyAR8qBkN3YuheJanUpWC5U3bb5osDywNB8RzDVlDwDHbocAJveqqj1u8+SVD7jkWT4yvsHCpWqqWqAxb0zCA==",
|
||||||
|
"dev": true,
|
||||||
"license": "MIT",
|
"license": "MIT",
|
||||||
"funding": {
|
"funding": {
|
||||||
"url": "https://github.com/sponsors/ljharb"
|
"url": "https://github.com/sponsors/ljharb"
|
||||||
@@ -13881,6 +13732,7 @@
|
|||||||
"version": "1.2.8",
|
"version": "1.2.8",
|
||||||
"resolved": "https://registry.npmjs.org/rc/-/rc-1.2.8.tgz",
|
"resolved": "https://registry.npmjs.org/rc/-/rc-1.2.8.tgz",
|
||||||
"integrity": "sha512-y3bGgqKj3QBdxLbLkomlohkvsA8gdAiUQlSBJnBhfn+BPxg4bc62d8TcBW15wavDfgexCgccckhcZvywyQYPOw==",
|
"integrity": "sha512-y3bGgqKj3QBdxLbLkomlohkvsA8gdAiUQlSBJnBhfn+BPxg4bc62d8TcBW15wavDfgexCgccckhcZvywyQYPOw==",
|
||||||
|
"dev": true,
|
||||||
"license": "(BSD-2-Clause OR MIT OR Apache-2.0)",
|
"license": "(BSD-2-Clause OR MIT OR Apache-2.0)",
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
"deep-extend": "^0.6.0",
|
"deep-extend": "^0.6.0",
|
||||||
@@ -14010,6 +13862,7 @@
|
|||||||
"version": "5.2.1",
|
"version": "5.2.1",
|
||||||
"resolved": "https://registry.npmjs.org/safe-buffer/-/safe-buffer-5.2.1.tgz",
|
"resolved": "https://registry.npmjs.org/safe-buffer/-/safe-buffer-5.2.1.tgz",
|
||||||
"integrity": "sha512-rp3So07KcdmmKbGvgaNxQSJr7bGVSVk5S9Eq1F+ppbRo70+YeaDxkw5Dd8NPN+GD6bjnYm2VuPuCXmpuYvmCXQ==",
|
"integrity": "sha512-rp3So07KcdmmKbGvgaNxQSJr7bGVSVk5S9Eq1F+ppbRo70+YeaDxkw5Dd8NPN+GD6bjnYm2VuPuCXmpuYvmCXQ==",
|
||||||
|
"dev": true,
|
||||||
"funding": [
|
"funding": [
|
||||||
{
|
{
|
||||||
"type": "github",
|
"type": "github",
|
||||||
@@ -14407,6 +14260,7 @@
|
|||||||
"version": "2.0.1",
|
"version": "2.0.1",
|
||||||
"resolved": "https://registry.npmjs.org/strip-json-comments/-/strip-json-comments-2.0.1.tgz",
|
"resolved": "https://registry.npmjs.org/strip-json-comments/-/strip-json-comments-2.0.1.tgz",
|
||||||
"integrity": "sha512-4gB8na07fecVVkOI6Rs4e7T6NOTki5EmL7TUduTs6bu3EdnSycntVJ4re8kgZA+wx9IueI2Y11bfbgwtzuE0KQ==",
|
"integrity": "sha512-4gB8na07fecVVkOI6Rs4e7T6NOTki5EmL7TUduTs6bu3EdnSycntVJ4re8kgZA+wx9IueI2Y11bfbgwtzuE0KQ==",
|
||||||
|
"dev": true,
|
||||||
"license": "MIT",
|
"license": "MIT",
|
||||||
"engines": {
|
"engines": {
|
||||||
"node": ">=0.10.0"
|
"node": ">=0.10.0"
|
||||||
|
|||||||
Reference in New Issue
Block a user