Compare commits
4 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
| 5d6ca3d8cc | |||
| db7948d279 | |||
| 7cee3b3a97 | |||
| 01f8974314 |
@@ -450,7 +450,7 @@ function SidebarContent({
|
||||
{!sidebarCollapsed && (
|
||||
<div className="overflow-hidden">
|
||||
<h1 className="font-display text-xl font-semibold text-gray-900 dark:text-gray-50">
|
||||
Nex<span className="text-brand-600">us</span>
|
||||
Capa<span className="text-brand-600">Kraken</span>
|
||||
</h1>
|
||||
<p className="text-xs uppercase tracking-[0.18em] text-gray-500 dark:text-gray-400">
|
||||
Resource & Capacity Planning
|
||||
@@ -984,7 +984,7 @@ export function AppShell({
|
||||
<HamburgerIcon />
|
||||
</button>
|
||||
<span className="ml-3 font-display text-sm font-semibold text-gray-900 dark:text-gray-50">
|
||||
Nex<span className="text-brand-600">us</span>
|
||||
Capa<span className="text-brand-600">Kraken</span>
|
||||
</span>
|
||||
</div>
|
||||
<PageTransition>{children}</PageTransition>
|
||||
|
||||
@@ -1,5 +1,5 @@
|
||||
# Nexus nginx Security Hardening
|
||||
# Apply to the server block for nexus.hartmut-noerenberg.com
|
||||
# CapaKraken nginx Security Hardening
|
||||
# Apply to the server block for capakraken.hartmut-noerenberg.com
|
||||
#
|
||||
# References:
|
||||
# - EAPPS 3.3.1.3.04 (Server Header entfernen)
|
||||
@@ -113,5 +113,5 @@ log_format security '$remote_addr - $remote_user [$time_local] '
|
||||
'"$http_referer" "$http_user_agent" '
|
||||
'$request_time $upstream_response_time';
|
||||
|
||||
access_log /var/log/nginx/nexus_access.log security;
|
||||
error_log /var/log/nginx/nexus_error.log warn;
|
||||
access_log /var/log/nginx/capakraken_access.log security;
|
||||
error_log /var/log/nginx/capakraken_error.log warn;
|
||||
|
||||
@@ -7,14 +7,12 @@ vi.mock("../lib/audit.js", () => ({
|
||||
vi.mock("../router/assistant-approvals.js", () => ({
|
||||
clearPendingAssistantApproval: vi.fn().mockResolvedValue(undefined),
|
||||
consumePendingAssistantApproval: vi.fn(),
|
||||
toApprovalPayload: vi.fn(
|
||||
(approval: { id: string; toolName: string; summary: string }, status: string) => ({
|
||||
id: approval.id,
|
||||
toolName: approval.toolName,
|
||||
summary: approval.summary,
|
||||
status,
|
||||
}),
|
||||
),
|
||||
toApprovalPayload: vi.fn((approval: { id: string; toolName: string; summary: string }, status: string) => ({
|
||||
id: approval.id,
|
||||
toolName: approval.toolName,
|
||||
summary: approval.summary,
|
||||
status,
|
||||
})),
|
||||
}));
|
||||
|
||||
vi.mock("../router/assistant-confirmation.js", () => ({
|
||||
@@ -41,10 +39,16 @@ import {
|
||||
clearPendingAssistantApproval,
|
||||
consumePendingAssistantApproval,
|
||||
} from "../router/assistant-approvals.js";
|
||||
import { canExecuteMutationTool, isCancellationReply } from "../router/assistant-confirmation.js";
|
||||
import {
|
||||
canExecuteMutationTool,
|
||||
isCancellationReply,
|
||||
} from "../router/assistant-confirmation.js";
|
||||
import { buildAssistantInsight } from "../router/assistant-insights.js";
|
||||
import { handlePendingAssistantApproval } from "../router/assistant-chat-response.js";
|
||||
import { readToolError, readToolSuccessMessage } from "../router/assistant-tool-results.js";
|
||||
import {
|
||||
readToolError,
|
||||
readToolSuccessMessage,
|
||||
} from "../router/assistant-tool-results.js";
|
||||
import { executeTool } from "../router/assistant-tools.js";
|
||||
|
||||
function createPendingApproval() {
|
||||
@@ -53,16 +57,14 @@ function createPendingApproval() {
|
||||
userId: "user_1",
|
||||
conversationId: "conv_1",
|
||||
toolName: "create_project",
|
||||
toolArguments: '{"name":"Apollo"}',
|
||||
toolArguments: "{\"name\":\"Apollo\"}",
|
||||
summary: "create project (name=Apollo)",
|
||||
createdAt: Date.now(),
|
||||
expiresAt: Date.now() + 60_000,
|
||||
};
|
||||
}
|
||||
|
||||
function createHandleInput(
|
||||
overrides: Partial<Parameters<typeof handlePendingAssistantApproval>[0]> = {},
|
||||
) {
|
||||
function createHandleInput(overrides: Partial<Parameters<typeof handlePendingAssistantApproval>[0]> = {}) {
|
||||
return {
|
||||
db: {} as never,
|
||||
dbUserId: "user_1",
|
||||
@@ -79,10 +81,7 @@ function createHandleInput(
|
||||
pendingApproval: createPendingApproval(),
|
||||
lastUserMessage: { role: "user" as const, content: "ja" },
|
||||
messages: [
|
||||
{
|
||||
role: "assistant" as const,
|
||||
content: "__NEXUS_CONFIRM__ create project (name=Apollo). Bitte bestätigen.",
|
||||
},
|
||||
{ role: "assistant" as const, content: "__CAPAKRAKEN_CONFIRM__ create project (name=Apollo). Bitte bestätigen." },
|
||||
{ role: "user" as const, content: "ja" },
|
||||
],
|
||||
collectedActions: [],
|
||||
@@ -104,11 +103,9 @@ describe("assistant pending approval handling", () => {
|
||||
it("cancels pending approvals when the user aborts", async () => {
|
||||
vi.mocked(isCancellationReply).mockReturnValue(true);
|
||||
|
||||
const result = await handlePendingAssistantApproval(
|
||||
createHandleInput({
|
||||
lastUserMessage: { role: "user", content: "nein, abbrechen" },
|
||||
}),
|
||||
);
|
||||
const result = await handlePendingAssistantApproval(createHandleInput({
|
||||
lastUserMessage: { role: "user", content: "nein, abbrechen" },
|
||||
}));
|
||||
|
||||
expect(result).toMatchObject({
|
||||
response: {
|
||||
@@ -130,7 +127,7 @@ describe("assistant pending approval handling", () => {
|
||||
summary: "create project (name=Apollo, status=DRAFT)",
|
||||
} as never);
|
||||
vi.mocked(executeTool).mockResolvedValue({
|
||||
content: '{"message":"Projekt Apollo angelegt"}',
|
||||
content: "{\"message\":\"Projekt Apollo angelegt\"}",
|
||||
data: { message: "Projekt Apollo angelegt" },
|
||||
action: { type: "refresh" },
|
||||
} as never);
|
||||
@@ -151,35 +148,29 @@ describe("assistant pending approval handling", () => {
|
||||
status: "approved",
|
||||
},
|
||||
actions: [{ type: "refresh" }],
|
||||
insights: [
|
||||
{
|
||||
kind: "holiday_region",
|
||||
title: "Berlin",
|
||||
},
|
||||
],
|
||||
insights: [{
|
||||
kind: "holiday_region",
|
||||
title: "Berlin",
|
||||
}],
|
||||
},
|
||||
});
|
||||
expect(executeTool).toHaveBeenCalledWith(
|
||||
"create_project",
|
||||
'{"name":"Apollo"}',
|
||||
"{\"name\":\"Apollo\"}",
|
||||
expect.objectContaining({ userId: "user_1" }),
|
||||
);
|
||||
expect(createAuditEntry).toHaveBeenCalledWith(
|
||||
expect.objectContaining({
|
||||
entityName: "create_project",
|
||||
summary: "AI executed previously approved tool: create_project",
|
||||
}),
|
||||
);
|
||||
expect(createAuditEntry).toHaveBeenCalledWith(expect.objectContaining({
|
||||
entityName: "create_project",
|
||||
summary: "AI executed previously approved tool: create_project",
|
||||
}));
|
||||
});
|
||||
|
||||
it("does nothing when the user reply is not a valid confirmation", async () => {
|
||||
vi.mocked(canExecuteMutationTool).mockReturnValue(false);
|
||||
|
||||
const result = await handlePendingAssistantApproval(
|
||||
createHandleInput({
|
||||
lastUserMessage: { role: "user", content: "vielleicht" },
|
||||
}),
|
||||
);
|
||||
const result = await handlePendingAssistantApproval(createHandleInput({
|
||||
lastUserMessage: { role: "user", content: "vielleicht" },
|
||||
}));
|
||||
|
||||
expect(result).toBeNull();
|
||||
expect(consumePendingAssistantApproval).not.toHaveBeenCalled();
|
||||
|
||||
@@ -1,4 +1,4 @@
|
||||
// Nexus — Prisma Schema
|
||||
// CapaKraken — Prisma Schema
|
||||
// All monetary values stored as integer cents to avoid float precision issues.
|
||||
|
||||
generator client {
|
||||
|
||||
+1
-1
@@ -1,5 +1,5 @@
|
||||
#!/usr/bin/env bash
|
||||
# restart.sh — Rebuild the Nexus app container from scratch.
|
||||
# restart.sh — Rebuild the CapaKraken app container from scratch.
|
||||
#
|
||||
# When to use:
|
||||
# - After changing pnpm-lock.yaml (new/removed dependencies)
|
||||
|
||||
+1
-1
@@ -2,7 +2,7 @@
|
||||
set -euo pipefail
|
||||
cd "$(dirname "$0")/.."
|
||||
|
||||
echo "Restarting Nexus..."
|
||||
echo "Restarting CapaKraken..."
|
||||
echo ""
|
||||
|
||||
# Stop
|
||||
|
||||
+2
-2
@@ -5,7 +5,7 @@ cd "$(dirname "$0")/.."
|
||||
APP_PORT="${APP_PORT:-3100}"
|
||||
APP_CONTAINER="${APP_CONTAINER:-$(docker compose --profile full ps -q app 2>/dev/null | head -1)}"
|
||||
|
||||
echo "Starting Nexus..."
|
||||
echo "Starting CapaKraken..."
|
||||
|
||||
# 1. Start Docker services
|
||||
echo " Starting PostgreSQL + Redis..."
|
||||
@@ -34,7 +34,7 @@ echo " Waiting for server (up to 90s)..."
|
||||
for i in {1..90}; do
|
||||
if curl -sf "http://localhost:${APP_PORT}/api/health" > /dev/null 2>&1; then
|
||||
echo ""
|
||||
echo "Nexus is running!"
|
||||
echo "CapaKraken is running!"
|
||||
curl -s "http://localhost:${APP_PORT}/api/ready" | python3 -m json.tool 2>/dev/null || curl -s "http://localhost:${APP_PORT}/api/ready"
|
||||
echo ""
|
||||
echo " URL: http://localhost:${APP_PORT}"
|
||||
|
||||
+2
-2
@@ -2,7 +2,7 @@
|
||||
set -euo pipefail
|
||||
cd "$(dirname "$0")/.."
|
||||
|
||||
echo "Stopping Nexus..."
|
||||
echo "Stopping CapaKraken..."
|
||||
|
||||
# 1. Stop any legacy local dev server
|
||||
if [ -f /tmp/nexus-dev.pid ]; then
|
||||
@@ -28,4 +28,4 @@ echo " Stopping app, PostgreSQL and Redis..."
|
||||
docker compose --profile full stop app postgres redis 2>/dev/null || true
|
||||
|
||||
echo ""
|
||||
echo "Nexus stopped."
|
||||
echo "CapaKraken stopped."
|
||||
|
||||
@@ -63,7 +63,7 @@ docker compose -p "$OLD_PROJECT" -f "$COMPOSE_FILE" stop app 2>/dev/null || true
|
||||
echo "[2/7] Capturing pre-rename row counts..."
|
||||
PRE_COUNTS=$(docker compose -p "$OLD_PROJECT" -f "$COMPOSE_FILE" exec -T postgres \
|
||||
psql -U capakraken -d capakraken -t -c \
|
||||
"SELECT relname, n_live_tup FROM pg_stat_user_tables ORDER BY relname;")
|
||||
"SELECT table_name, n_live_tup FROM pg_stat_user_tables ORDER BY table_name;")
|
||||
echo "$PRE_COUNTS" | head -20
|
||||
echo "..."
|
||||
|
||||
@@ -149,7 +149,7 @@ sleep 15
|
||||
echo "=== Verification ==="
|
||||
POST_COUNTS=$(docker compose -p "$NEW_PROJECT" -f "$COMPOSE_FILE" exec -T postgres \
|
||||
psql -U nexus -d nexus -t -c \
|
||||
"SELECT relname, n_live_tup FROM pg_stat_user_tables ORDER BY relname;")
|
||||
"SELECT table_name, n_live_tup FROM pg_stat_user_tables ORDER BY table_name;")
|
||||
echo "Post-rename row counts (sample):"
|
||||
echo "$POST_COUNTS" | head -20
|
||||
|
||||
|
||||
Reference in New Issue
Block a user