@copilotz/admin 0.3.4
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/dist/index.d.ts +214 -0
- package/dist/index.js +669 -0
- package/dist/index.js.map +1 -0
- package/package.json +42 -0
package/dist/index.d.ts
ADDED
|
@@ -0,0 +1,214 @@
|
|
|
1
|
+
import React, { ReactNode } from 'react';
|
|
2
|
+
|
|
3
|
+
type RequestHeadersProvider = () => Record<string, string> | Promise<Record<string, string>>;
|
|
4
|
+
type AdminDatePreset = "24h" | "7d" | "30d";
|
|
5
|
+
type AdminActivityInterval = "hour" | "day";
|
|
6
|
+
interface AdminOverview {
|
|
7
|
+
threadTotals: {
|
|
8
|
+
total: number;
|
|
9
|
+
active: number;
|
|
10
|
+
archived: number;
|
|
11
|
+
};
|
|
12
|
+
queueTotals: {
|
|
13
|
+
total: number;
|
|
14
|
+
pending: number;
|
|
15
|
+
processing: number;
|
|
16
|
+
completed: number;
|
|
17
|
+
failed: number;
|
|
18
|
+
expired: number;
|
|
19
|
+
overwritten: number;
|
|
20
|
+
};
|
|
21
|
+
messageTotals: {
|
|
22
|
+
total: number;
|
|
23
|
+
toolCallMessages: number;
|
|
24
|
+
};
|
|
25
|
+
participantTotals: {
|
|
26
|
+
total: number;
|
|
27
|
+
humans: number;
|
|
28
|
+
agents: number;
|
|
29
|
+
};
|
|
30
|
+
llmTotals: {
|
|
31
|
+
totalCalls: number;
|
|
32
|
+
inputTokens: number;
|
|
33
|
+
outputTokens: number;
|
|
34
|
+
reasoningTokens: number;
|
|
35
|
+
cacheReadInputTokens: number;
|
|
36
|
+
cacheCreationInputTokens: number;
|
|
37
|
+
totalTokens: number;
|
|
38
|
+
};
|
|
39
|
+
}
|
|
40
|
+
interface AdminActivityPoint {
|
|
41
|
+
bucket: string;
|
|
42
|
+
messageCount: number;
|
|
43
|
+
llmCallCount: number;
|
|
44
|
+
toolCallMessageCount: number;
|
|
45
|
+
totalTokens: number;
|
|
46
|
+
}
|
|
47
|
+
interface AdminThreadSummary {
|
|
48
|
+
threadId: string;
|
|
49
|
+
name: string;
|
|
50
|
+
status: string;
|
|
51
|
+
summary: string | null;
|
|
52
|
+
participantIds: string[];
|
|
53
|
+
messageCount: number;
|
|
54
|
+
lastActivityAt: string | null;
|
|
55
|
+
lastMessagePreview: string | null;
|
|
56
|
+
createdAt: string | null;
|
|
57
|
+
updatedAt: string | null;
|
|
58
|
+
}
|
|
59
|
+
interface AdminParticipantSummary {
|
|
60
|
+
externalId: string;
|
|
61
|
+
displayName: string;
|
|
62
|
+
participantType: "human" | "agent";
|
|
63
|
+
namespace: string;
|
|
64
|
+
isGlobal: boolean;
|
|
65
|
+
messageCount: number;
|
|
66
|
+
threadCount: number;
|
|
67
|
+
lastActivityAt: string | null;
|
|
68
|
+
}
|
|
69
|
+
interface AdminAgentSummary {
|
|
70
|
+
agentId: string;
|
|
71
|
+
displayName: string;
|
|
72
|
+
description: string | null;
|
|
73
|
+
isConfigured: boolean;
|
|
74
|
+
namespace: string;
|
|
75
|
+
isGlobal: boolean;
|
|
76
|
+
messageCount: number;
|
|
77
|
+
llmCallCount: number;
|
|
78
|
+
toolCallMessageCount: number;
|
|
79
|
+
inputTokens: number;
|
|
80
|
+
outputTokens: number;
|
|
81
|
+
reasoningTokens: number;
|
|
82
|
+
cacheReadInputTokens: number;
|
|
83
|
+
cacheCreationInputTokens: number;
|
|
84
|
+
totalTokens: number;
|
|
85
|
+
lastActivityAt: string | null;
|
|
86
|
+
}
|
|
87
|
+
interface AdminFilters {
|
|
88
|
+
namespace?: string;
|
|
89
|
+
range?: AdminDatePreset;
|
|
90
|
+
interval?: AdminActivityInterval;
|
|
91
|
+
threadSearch?: string;
|
|
92
|
+
participantSearch?: string;
|
|
93
|
+
agentSearch?: string;
|
|
94
|
+
}
|
|
95
|
+
interface AdminConfig {
|
|
96
|
+
branding?: {
|
|
97
|
+
title?: string;
|
|
98
|
+
subtitle?: string;
|
|
99
|
+
logo?: ReactNode;
|
|
100
|
+
actions?: ReactNode;
|
|
101
|
+
};
|
|
102
|
+
labels?: {
|
|
103
|
+
overviewTitle?: string;
|
|
104
|
+
activityTitle?: string;
|
|
105
|
+
threadsTitle?: string;
|
|
106
|
+
participantsTitle?: string;
|
|
107
|
+
agentsTitle?: string;
|
|
108
|
+
range24h?: string;
|
|
109
|
+
range7d?: string;
|
|
110
|
+
range30d?: string;
|
|
111
|
+
intervalHour?: string;
|
|
112
|
+
intervalDay?: string;
|
|
113
|
+
refresh?: string;
|
|
114
|
+
retry?: string;
|
|
115
|
+
loading?: string;
|
|
116
|
+
emptyTitle?: string;
|
|
117
|
+
emptyDescription?: string;
|
|
118
|
+
threadSearchPlaceholder?: string;
|
|
119
|
+
participantSearchPlaceholder?: string;
|
|
120
|
+
agentSearchPlaceholder?: string;
|
|
121
|
+
messagesCard?: string;
|
|
122
|
+
activeThreadsCard?: string;
|
|
123
|
+
participantsCard?: string;
|
|
124
|
+
tokensCard?: string;
|
|
125
|
+
queueCard?: string;
|
|
126
|
+
statusActive?: string;
|
|
127
|
+
statusArchived?: string;
|
|
128
|
+
scopeGlobal?: string;
|
|
129
|
+
scopeScoped?: string;
|
|
130
|
+
configured?: string;
|
|
131
|
+
unconfigured?: string;
|
|
132
|
+
noResults?: string;
|
|
133
|
+
};
|
|
134
|
+
features?: {
|
|
135
|
+
showOverview?: boolean;
|
|
136
|
+
showActivity?: boolean;
|
|
137
|
+
showThreads?: boolean;
|
|
138
|
+
showParticipants?: boolean;
|
|
139
|
+
showAgents?: boolean;
|
|
140
|
+
};
|
|
141
|
+
ui?: {
|
|
142
|
+
compact?: boolean;
|
|
143
|
+
maxActivityBars?: number;
|
|
144
|
+
};
|
|
145
|
+
baseUrl?: string;
|
|
146
|
+
getRequestHeaders?: RequestHeadersProvider;
|
|
147
|
+
namespace?: string;
|
|
148
|
+
initialRange?: AdminDatePreset;
|
|
149
|
+
initialInterval?: AdminActivityInterval;
|
|
150
|
+
}
|
|
151
|
+
interface ResolvedAdminConfig {
|
|
152
|
+
branding: Required<NonNullable<AdminConfig["branding"]>>;
|
|
153
|
+
labels: Required<NonNullable<AdminConfig["labels"]>>;
|
|
154
|
+
features: Required<NonNullable<AdminConfig["features"]>>;
|
|
155
|
+
ui: Required<NonNullable<AdminConfig["ui"]>>;
|
|
156
|
+
baseUrl: string;
|
|
157
|
+
getRequestHeaders: RequestHeadersProvider;
|
|
158
|
+
namespace: string;
|
|
159
|
+
initialRange: AdminDatePreset;
|
|
160
|
+
initialInterval: AdminActivityInterval;
|
|
161
|
+
}
|
|
162
|
+
interface AdminSectionState {
|
|
163
|
+
overview: AdminOverview | null;
|
|
164
|
+
activity: AdminActivityPoint[];
|
|
165
|
+
threads: AdminThreadSummary[];
|
|
166
|
+
participants: AdminParticipantSummary[];
|
|
167
|
+
agents: AdminAgentSummary[];
|
|
168
|
+
}
|
|
169
|
+
interface UseCopilotzAdminOptions extends AdminFilters {
|
|
170
|
+
baseUrl?: string;
|
|
171
|
+
getRequestHeaders?: RequestHeadersProvider;
|
|
172
|
+
}
|
|
173
|
+
interface UseCopilotzAdminResult extends AdminSectionState {
|
|
174
|
+
filters: Required<Pick<AdminFilters, "range" | "interval">> & Omit<AdminFilters, "range" | "interval">;
|
|
175
|
+
isLoading: boolean;
|
|
176
|
+
error: Error | null;
|
|
177
|
+
refresh: () => Promise<void>;
|
|
178
|
+
setRange: (range: AdminDatePreset) => void;
|
|
179
|
+
setInterval: (interval: AdminActivityInterval) => void;
|
|
180
|
+
}
|
|
181
|
+
|
|
182
|
+
interface CopilotzAdminProps {
|
|
183
|
+
config?: Partial<AdminConfig>;
|
|
184
|
+
className?: string;
|
|
185
|
+
}
|
|
186
|
+
declare const CopilotzAdmin: React.FC<CopilotzAdminProps>;
|
|
187
|
+
|
|
188
|
+
declare function useCopilotzAdmin(options?: UseCopilotzAdminOptions): UseCopilotzAdminResult;
|
|
189
|
+
|
|
190
|
+
declare function fetchAdminOverview(range: AdminDatePreset, namespace?: string, options?: {
|
|
191
|
+
baseUrl?: string;
|
|
192
|
+
getRequestHeaders?: RequestHeadersProvider;
|
|
193
|
+
}): Promise<AdminOverview>;
|
|
194
|
+
declare function fetchAdminActivity(range: AdminDatePreset, interval: AdminActivityInterval, namespace?: string, options?: {
|
|
195
|
+
baseUrl?: string;
|
|
196
|
+
getRequestHeaders?: RequestHeadersProvider;
|
|
197
|
+
}): Promise<AdminActivityPoint[]>;
|
|
198
|
+
declare function fetchAdminThreads(search?: string, namespace?: string, options?: {
|
|
199
|
+
baseUrl?: string;
|
|
200
|
+
getRequestHeaders?: RequestHeadersProvider;
|
|
201
|
+
}): Promise<AdminThreadSummary[]>;
|
|
202
|
+
declare function fetchAdminParticipants(search?: string, namespace?: string, options?: {
|
|
203
|
+
baseUrl?: string;
|
|
204
|
+
getRequestHeaders?: RequestHeadersProvider;
|
|
205
|
+
}): Promise<AdminParticipantSummary[]>;
|
|
206
|
+
declare function fetchAdminAgents(search?: string, namespace?: string, options?: {
|
|
207
|
+
baseUrl?: string;
|
|
208
|
+
getRequestHeaders?: RequestHeadersProvider;
|
|
209
|
+
}): Promise<AdminAgentSummary[]>;
|
|
210
|
+
|
|
211
|
+
declare const defaultAdminConfig: ResolvedAdminConfig;
|
|
212
|
+
declare function mergeAdminConfig(_baseConfig: AdminConfig, userConfig?: Partial<AdminConfig>): ResolvedAdminConfig;
|
|
213
|
+
|
|
214
|
+
export { type AdminActivityInterval, type AdminActivityPoint, type AdminAgentSummary, type AdminConfig, type AdminDatePreset, type AdminFilters, type AdminOverview, type AdminParticipantSummary, type AdminSectionState, type AdminThreadSummary, CopilotzAdmin, type RequestHeadersProvider, type UseCopilotzAdminOptions, type UseCopilotzAdminResult, defaultAdminConfig, fetchAdminActivity, fetchAdminAgents, fetchAdminOverview, fetchAdminParticipants, fetchAdminThreads, mergeAdminConfig, useCopilotzAdmin };
|
package/dist/index.js
ADDED
|
@@ -0,0 +1,669 @@
|
|
|
1
|
+
// src/CopilotzAdmin.tsx
|
|
2
|
+
import {
|
|
3
|
+
useDeferredValue,
|
|
4
|
+
useMemo as useMemo2,
|
|
5
|
+
useState as useState2
|
|
6
|
+
} from "react";
|
|
7
|
+
|
|
8
|
+
// src/config.ts
|
|
9
|
+
var defaultAdminConfig = {
|
|
10
|
+
branding: {
|
|
11
|
+
title: "Copilotz Admin",
|
|
12
|
+
subtitle: "Read-only operational visibility for Copilotz clients",
|
|
13
|
+
logo: null,
|
|
14
|
+
actions: null
|
|
15
|
+
},
|
|
16
|
+
labels: {
|
|
17
|
+
overviewTitle: "Overview",
|
|
18
|
+
activityTitle: "Activity",
|
|
19
|
+
threadsTitle: "Threads",
|
|
20
|
+
participantsTitle: "Participants",
|
|
21
|
+
agentsTitle: "Agents",
|
|
22
|
+
range24h: "24h",
|
|
23
|
+
range7d: "7d",
|
|
24
|
+
range30d: "30d",
|
|
25
|
+
intervalHour: "Hourly",
|
|
26
|
+
intervalDay: "Daily",
|
|
27
|
+
refresh: "Refresh",
|
|
28
|
+
retry: "Try again",
|
|
29
|
+
loading: "Loading admin data...",
|
|
30
|
+
emptyTitle: "No activity yet",
|
|
31
|
+
emptyDescription: "Admin metrics will appear after threads, events, and graph data start flowing.",
|
|
32
|
+
threadSearchPlaceholder: "Search threads",
|
|
33
|
+
participantSearchPlaceholder: "Search participants",
|
|
34
|
+
agentSearchPlaceholder: "Search agents",
|
|
35
|
+
messagesCard: "Messages",
|
|
36
|
+
activeThreadsCard: "Active threads",
|
|
37
|
+
participantsCard: "Participants",
|
|
38
|
+
tokensCard: "LLM tokens",
|
|
39
|
+
queueCard: "Queued events",
|
|
40
|
+
statusActive: "Active",
|
|
41
|
+
statusArchived: "Archived",
|
|
42
|
+
scopeGlobal: "Global",
|
|
43
|
+
scopeScoped: "Scoped",
|
|
44
|
+
configured: "Configured",
|
|
45
|
+
unconfigured: "Observed only",
|
|
46
|
+
noResults: "No results"
|
|
47
|
+
},
|
|
48
|
+
features: {
|
|
49
|
+
showOverview: true,
|
|
50
|
+
showActivity: true,
|
|
51
|
+
showThreads: true,
|
|
52
|
+
showParticipants: true,
|
|
53
|
+
showAgents: true
|
|
54
|
+
},
|
|
55
|
+
ui: {
|
|
56
|
+
compact: false,
|
|
57
|
+
maxActivityBars: 18
|
|
58
|
+
},
|
|
59
|
+
baseUrl: "",
|
|
60
|
+
getRequestHeaders: async () => ({}),
|
|
61
|
+
namespace: "",
|
|
62
|
+
initialRange: "7d",
|
|
63
|
+
initialInterval: "day"
|
|
64
|
+
};
|
|
65
|
+
function mergeAdminConfig(_baseConfig, userConfig) {
|
|
66
|
+
if (!userConfig) return defaultAdminConfig;
|
|
67
|
+
return {
|
|
68
|
+
branding: {
|
|
69
|
+
...defaultAdminConfig.branding,
|
|
70
|
+
...userConfig.branding
|
|
71
|
+
},
|
|
72
|
+
labels: {
|
|
73
|
+
...defaultAdminConfig.labels,
|
|
74
|
+
...userConfig.labels
|
|
75
|
+
},
|
|
76
|
+
features: {
|
|
77
|
+
...defaultAdminConfig.features,
|
|
78
|
+
...userConfig.features
|
|
79
|
+
},
|
|
80
|
+
ui: {
|
|
81
|
+
...defaultAdminConfig.ui,
|
|
82
|
+
...userConfig.ui
|
|
83
|
+
},
|
|
84
|
+
baseUrl: userConfig.baseUrl ?? defaultAdminConfig.baseUrl,
|
|
85
|
+
getRequestHeaders: userConfig.getRequestHeaders ?? defaultAdminConfig.getRequestHeaders,
|
|
86
|
+
namespace: userConfig.namespace ?? defaultAdminConfig.namespace,
|
|
87
|
+
initialRange: userConfig.initialRange ?? defaultAdminConfig.initialRange,
|
|
88
|
+
initialInterval: userConfig.initialInterval ?? defaultAdminConfig.initialInterval
|
|
89
|
+
};
|
|
90
|
+
}
|
|
91
|
+
|
|
92
|
+
// src/useCopilotzAdmin.ts
|
|
93
|
+
import {
|
|
94
|
+
useCallback,
|
|
95
|
+
useEffect,
|
|
96
|
+
useMemo,
|
|
97
|
+
useState
|
|
98
|
+
} from "react";
|
|
99
|
+
|
|
100
|
+
// src/adminService.ts
|
|
101
|
+
var rawBaseValue = import.meta.env?.VITE_API_URL;
|
|
102
|
+
var rawBase = typeof rawBaseValue === "string" && rawBaseValue.length > 0 ? rawBaseValue : "/api";
|
|
103
|
+
var normalizedBase = rawBase.replace(/\/$/, "");
|
|
104
|
+
var runtimeProcess = typeof process !== "undefined" ? process : void 0;
|
|
105
|
+
var API_KEY = (() => {
|
|
106
|
+
const env = import.meta.env ?? {};
|
|
107
|
+
const candidates = [
|
|
108
|
+
env.VITE_API_KEY,
|
|
109
|
+
env.VITE_COPILOTZ_API_KEY,
|
|
110
|
+
runtimeProcess?.env?.COPILOTZ_API_KEY,
|
|
111
|
+
runtimeProcess?.env?.API_KEY
|
|
112
|
+
];
|
|
113
|
+
return candidates.find(
|
|
114
|
+
(value) => typeof value === "string" && value.length > 0
|
|
115
|
+
);
|
|
116
|
+
})();
|
|
117
|
+
var resolveBaseUrl = (baseUrl) => {
|
|
118
|
+
const candidate = (baseUrl && baseUrl.length > 0 ? baseUrl : normalizedBase).replace(/\/$/, "");
|
|
119
|
+
return candidate.startsWith("http") || candidate.startsWith("/") ? candidate : `/${candidate}`;
|
|
120
|
+
};
|
|
121
|
+
var withAuthHeaders = async (headers = {}, getRequestHeaders) => {
|
|
122
|
+
const providedHeaders = getRequestHeaders ? await getRequestHeaders() : void 0;
|
|
123
|
+
if (providedHeaders && Object.keys(providedHeaders).length > 0) {
|
|
124
|
+
return { ...headers, ...providedHeaders };
|
|
125
|
+
}
|
|
126
|
+
if (API_KEY) {
|
|
127
|
+
return { ...headers, Authorization: `Bearer ${API_KEY}` };
|
|
128
|
+
}
|
|
129
|
+
return headers;
|
|
130
|
+
};
|
|
131
|
+
var getRangeWindow = (range) => {
|
|
132
|
+
const to = /* @__PURE__ */ new Date();
|
|
133
|
+
const from = new Date(to);
|
|
134
|
+
if (range === "24h") {
|
|
135
|
+
from.setHours(from.getHours() - 24);
|
|
136
|
+
} else if (range === "30d") {
|
|
137
|
+
from.setDate(from.getDate() - 30);
|
|
138
|
+
} else {
|
|
139
|
+
from.setDate(from.getDate() - 7);
|
|
140
|
+
}
|
|
141
|
+
return {
|
|
142
|
+
from: from.toISOString(),
|
|
143
|
+
to: to.toISOString()
|
|
144
|
+
};
|
|
145
|
+
};
|
|
146
|
+
async function fetchAdminJson(path, params, options) {
|
|
147
|
+
const url = new URL(`${resolveBaseUrl(options?.baseUrl)}${path}`, window.location.origin);
|
|
148
|
+
for (const [key, value] of Object.entries(params)) {
|
|
149
|
+
if (typeof value === "string" && value.length > 0) {
|
|
150
|
+
url.searchParams.set(key, value);
|
|
151
|
+
}
|
|
152
|
+
}
|
|
153
|
+
const response = await fetch(url.toString(), {
|
|
154
|
+
headers: await withAuthHeaders({}, options?.getRequestHeaders)
|
|
155
|
+
});
|
|
156
|
+
if (!response.ok) {
|
|
157
|
+
throw new Error(`Admin request failed (${response.status})`);
|
|
158
|
+
}
|
|
159
|
+
const payload = await response.json();
|
|
160
|
+
return payload.data;
|
|
161
|
+
}
|
|
162
|
+
async function fetchAdminOverview(range, namespace, options) {
|
|
163
|
+
const windowRange = getRangeWindow(range);
|
|
164
|
+
return await fetchAdminJson("/v1/admin/overview", {
|
|
165
|
+
namespace,
|
|
166
|
+
from: windowRange.from,
|
|
167
|
+
to: windowRange.to
|
|
168
|
+
}, options);
|
|
169
|
+
}
|
|
170
|
+
async function fetchAdminActivity(range, interval, namespace, options) {
|
|
171
|
+
const windowRange = getRangeWindow(range);
|
|
172
|
+
return await fetchAdminJson("/v1/admin/activity", {
|
|
173
|
+
namespace,
|
|
174
|
+
interval,
|
|
175
|
+
from: windowRange.from,
|
|
176
|
+
to: windowRange.to
|
|
177
|
+
}, options);
|
|
178
|
+
}
|
|
179
|
+
async function fetchAdminThreads(search, namespace, options) {
|
|
180
|
+
return await fetchAdminJson("/v1/admin/threads", {
|
|
181
|
+
search,
|
|
182
|
+
namespace,
|
|
183
|
+
limit: "8"
|
|
184
|
+
}, options);
|
|
185
|
+
}
|
|
186
|
+
async function fetchAdminParticipants(search, namespace, options) {
|
|
187
|
+
return await fetchAdminJson("/v1/admin/participants", {
|
|
188
|
+
search,
|
|
189
|
+
namespace,
|
|
190
|
+
limit: "8"
|
|
191
|
+
}, options);
|
|
192
|
+
}
|
|
193
|
+
async function fetchAdminAgents(search, namespace, options) {
|
|
194
|
+
return await fetchAdminJson("/v1/admin/agents", {
|
|
195
|
+
search,
|
|
196
|
+
namespace,
|
|
197
|
+
limit: "8"
|
|
198
|
+
}, options);
|
|
199
|
+
}
|
|
200
|
+
|
|
201
|
+
// src/useCopilotzAdmin.ts
|
|
202
|
+
function useCopilotzAdmin(options = {}) {
|
|
203
|
+
const [range, setRange] = useState(options.range ?? "7d");
|
|
204
|
+
const [interval, setInterval] = useState(
|
|
205
|
+
options.interval ?? "day"
|
|
206
|
+
);
|
|
207
|
+
const [overview, setOverview] = useState(
|
|
208
|
+
null
|
|
209
|
+
);
|
|
210
|
+
const [activity, setActivity] = useState(
|
|
211
|
+
[]
|
|
212
|
+
);
|
|
213
|
+
const [threads, setThreads] = useState([]);
|
|
214
|
+
const [participants, setParticipants] = useState([]);
|
|
215
|
+
const [agents, setAgents] = useState([]);
|
|
216
|
+
const [isLoading, setIsLoading] = useState(true);
|
|
217
|
+
const [error, setError] = useState(null);
|
|
218
|
+
const fetchAll = useCallback(async () => {
|
|
219
|
+
setIsLoading(true);
|
|
220
|
+
setError(null);
|
|
221
|
+
try {
|
|
222
|
+
const shared = {
|
|
223
|
+
baseUrl: options.baseUrl,
|
|
224
|
+
getRequestHeaders: options.getRequestHeaders
|
|
225
|
+
};
|
|
226
|
+
const [
|
|
227
|
+
nextOverview,
|
|
228
|
+
nextActivity,
|
|
229
|
+
nextThreads,
|
|
230
|
+
nextParticipants,
|
|
231
|
+
nextAgents
|
|
232
|
+
] = await Promise.all([
|
|
233
|
+
fetchAdminOverview(range, options.namespace, shared),
|
|
234
|
+
fetchAdminActivity(range, interval, options.namespace, shared),
|
|
235
|
+
fetchAdminThreads(options.threadSearch, options.namespace, shared),
|
|
236
|
+
fetchAdminParticipants(
|
|
237
|
+
options.participantSearch,
|
|
238
|
+
options.namespace,
|
|
239
|
+
shared
|
|
240
|
+
),
|
|
241
|
+
fetchAdminAgents(options.agentSearch, options.namespace, shared)
|
|
242
|
+
]);
|
|
243
|
+
setOverview(nextOverview);
|
|
244
|
+
setActivity(nextActivity);
|
|
245
|
+
setThreads(nextThreads);
|
|
246
|
+
setParticipants(nextParticipants);
|
|
247
|
+
setAgents(nextAgents);
|
|
248
|
+
} catch (nextError) {
|
|
249
|
+
setError(nextError instanceof Error ? nextError : new Error("Failed to load admin data"));
|
|
250
|
+
} finally {
|
|
251
|
+
setIsLoading(false);
|
|
252
|
+
}
|
|
253
|
+
}, [
|
|
254
|
+
interval,
|
|
255
|
+
options.agentSearch,
|
|
256
|
+
options.baseUrl,
|
|
257
|
+
options.getRequestHeaders,
|
|
258
|
+
options.namespace,
|
|
259
|
+
options.participantSearch,
|
|
260
|
+
options.threadSearch,
|
|
261
|
+
range
|
|
262
|
+
]);
|
|
263
|
+
useEffect(() => {
|
|
264
|
+
void fetchAll();
|
|
265
|
+
}, [fetchAll]);
|
|
266
|
+
return useMemo(() => ({
|
|
267
|
+
overview,
|
|
268
|
+
activity,
|
|
269
|
+
threads,
|
|
270
|
+
participants,
|
|
271
|
+
agents,
|
|
272
|
+
filters: {
|
|
273
|
+
namespace: options.namespace,
|
|
274
|
+
threadSearch: options.threadSearch,
|
|
275
|
+
participantSearch: options.participantSearch,
|
|
276
|
+
agentSearch: options.agentSearch,
|
|
277
|
+
range,
|
|
278
|
+
interval
|
|
279
|
+
},
|
|
280
|
+
isLoading,
|
|
281
|
+
error,
|
|
282
|
+
refresh: fetchAll,
|
|
283
|
+
setRange,
|
|
284
|
+
setInterval
|
|
285
|
+
}), [
|
|
286
|
+
activity,
|
|
287
|
+
agents,
|
|
288
|
+
error,
|
|
289
|
+
fetchAll,
|
|
290
|
+
interval,
|
|
291
|
+
isLoading,
|
|
292
|
+
options.agentSearch,
|
|
293
|
+
options.namespace,
|
|
294
|
+
options.participantSearch,
|
|
295
|
+
options.threadSearch,
|
|
296
|
+
overview,
|
|
297
|
+
participants,
|
|
298
|
+
range,
|
|
299
|
+
threads
|
|
300
|
+
]);
|
|
301
|
+
}
|
|
302
|
+
|
|
303
|
+
// src/CopilotzAdmin.tsx
|
|
304
|
+
import { jsx, jsxs } from "react/jsx-runtime";
|
|
305
|
+
var CopilotzAdmin = ({
|
|
306
|
+
config: userConfig,
|
|
307
|
+
className
|
|
308
|
+
}) => {
|
|
309
|
+
const config = useMemo2(
|
|
310
|
+
() => mergeAdminConfig(defaultAdminConfig, userConfig),
|
|
311
|
+
[userConfig]
|
|
312
|
+
);
|
|
313
|
+
const [threadSearch, setThreadSearch] = useState2("");
|
|
314
|
+
const [participantSearch, setParticipantSearch] = useState2("");
|
|
315
|
+
const [agentSearch, setAgentSearch] = useState2("");
|
|
316
|
+
const deferredThreadSearch = useDeferredValue(threadSearch);
|
|
317
|
+
const deferredParticipantSearch = useDeferredValue(participantSearch);
|
|
318
|
+
const deferredAgentSearch = useDeferredValue(agentSearch);
|
|
319
|
+
const admin = useCopilotzAdmin({
|
|
320
|
+
baseUrl: config.baseUrl,
|
|
321
|
+
getRequestHeaders: config.getRequestHeaders,
|
|
322
|
+
namespace: config.namespace,
|
|
323
|
+
range: config.initialRange,
|
|
324
|
+
interval: config.initialInterval,
|
|
325
|
+
threadSearch: deferredThreadSearch,
|
|
326
|
+
participantSearch: deferredParticipantSearch,
|
|
327
|
+
agentSearch: deferredAgentSearch
|
|
328
|
+
});
|
|
329
|
+
const cards = [
|
|
330
|
+
{
|
|
331
|
+
label: config.labels.messagesCard,
|
|
332
|
+
value: admin.overview?.messageTotals.total ?? 0,
|
|
333
|
+
detail: `${admin.overview?.messageTotals.toolCallMessages ?? 0} tool-call messages`
|
|
334
|
+
},
|
|
335
|
+
{
|
|
336
|
+
label: config.labels.activeThreadsCard,
|
|
337
|
+
value: admin.overview?.threadTotals.active ?? 0,
|
|
338
|
+
detail: `${admin.overview?.threadTotals.total ?? 0} total threads`
|
|
339
|
+
},
|
|
340
|
+
{
|
|
341
|
+
label: config.labels.participantsCard,
|
|
342
|
+
value: admin.overview?.participantTotals.total ?? 0,
|
|
343
|
+
detail: `${admin.overview?.participantTotals.agents ?? 0} agents`
|
|
344
|
+
},
|
|
345
|
+
{
|
|
346
|
+
label: config.labels.tokensCard,
|
|
347
|
+
value: admin.overview?.llmTotals.totalTokens ?? 0,
|
|
348
|
+
detail: `${admin.overview?.llmTotals.totalCalls ?? 0} calls`
|
|
349
|
+
},
|
|
350
|
+
{
|
|
351
|
+
label: config.labels.queueCard,
|
|
352
|
+
value: admin.overview?.queueTotals.pending ?? 0,
|
|
353
|
+
detail: `${admin.overview?.queueTotals.failed ?? 0} failed`
|
|
354
|
+
}
|
|
355
|
+
];
|
|
356
|
+
if (admin.isLoading && !admin.overview) {
|
|
357
|
+
return /* @__PURE__ */ jsx("div", { className: joinClassNames("rounded-3xl border border-slate-200 bg-white p-8 text-slate-700 shadow-sm", className), children: config.labels.loading });
|
|
358
|
+
}
|
|
359
|
+
if (admin.error && !admin.overview) {
|
|
360
|
+
return /* @__PURE__ */ jsxs("div", { className: joinClassNames("rounded-3xl border border-rose-200 bg-rose-50 p-8 text-rose-700 shadow-sm", className), children: [
|
|
361
|
+
/* @__PURE__ */ jsx("p", { className: "text-base font-semibold", children: admin.error.message }),
|
|
362
|
+
/* @__PURE__ */ jsx(
|
|
363
|
+
"button",
|
|
364
|
+
{
|
|
365
|
+
className: "mt-4 rounded-xl bg-rose-600 px-4 py-2 text-sm font-medium text-white",
|
|
366
|
+
onClick: () => void admin.refresh(),
|
|
367
|
+
type: "button",
|
|
368
|
+
children: config.labels.retry
|
|
369
|
+
}
|
|
370
|
+
)
|
|
371
|
+
] });
|
|
372
|
+
}
|
|
373
|
+
const isEmpty = cards.every((card) => card.value === 0) && admin.activity.length === 0 && admin.threads.length === 0 && admin.participants.length === 0 && admin.agents.length === 0;
|
|
374
|
+
return /* @__PURE__ */ jsxs("div", { className: joinClassNames("space-y-6 rounded-[28px] border border-slate-200 bg-[linear-gradient(180deg,#ffffff_0%,#f8fafc_100%)] p-6 text-slate-900 shadow-sm", className), children: [
|
|
375
|
+
/* @__PURE__ */ jsxs("header", { className: "flex flex-col gap-4 lg:flex-row lg:items-center lg:justify-between", children: [
|
|
376
|
+
/* @__PURE__ */ jsxs("div", { className: "space-y-2", children: [
|
|
377
|
+
/* @__PURE__ */ jsxs("div", { className: "flex items-center gap-3", children: [
|
|
378
|
+
config.branding.logo ? /* @__PURE__ */ jsx("div", { className: "flex h-10 w-10 items-center justify-center rounded-2xl border border-slate-200 bg-white shadow-sm", children: config.branding.logo }) : null,
|
|
379
|
+
/* @__PURE__ */ jsxs("div", { children: [
|
|
380
|
+
/* @__PURE__ */ jsx("h2", { className: "text-2xl font-semibold tracking-tight", children: config.branding.title }),
|
|
381
|
+
/* @__PURE__ */ jsx("p", { className: "text-sm text-slate-500", children: config.branding.subtitle })
|
|
382
|
+
] })
|
|
383
|
+
] }),
|
|
384
|
+
config.namespace ? /* @__PURE__ */ jsxs("p", { className: "text-xs font-medium uppercase tracking-[0.18em] text-slate-400", children: [
|
|
385
|
+
"Namespace: ",
|
|
386
|
+
config.namespace
|
|
387
|
+
] }) : null
|
|
388
|
+
] }),
|
|
389
|
+
/* @__PURE__ */ jsxs("div", { className: "flex flex-wrap items-center gap-2", children: [
|
|
390
|
+
/* @__PURE__ */ jsx(
|
|
391
|
+
RangeButton,
|
|
392
|
+
{
|
|
393
|
+
active: admin.filters.range === "24h",
|
|
394
|
+
label: config.labels.range24h,
|
|
395
|
+
onClick: () => admin.setRange("24h")
|
|
396
|
+
}
|
|
397
|
+
),
|
|
398
|
+
/* @__PURE__ */ jsx(
|
|
399
|
+
RangeButton,
|
|
400
|
+
{
|
|
401
|
+
active: admin.filters.range === "7d",
|
|
402
|
+
label: config.labels.range7d,
|
|
403
|
+
onClick: () => admin.setRange("7d")
|
|
404
|
+
}
|
|
405
|
+
),
|
|
406
|
+
/* @__PURE__ */ jsx(
|
|
407
|
+
RangeButton,
|
|
408
|
+
{
|
|
409
|
+
active: admin.filters.range === "30d",
|
|
410
|
+
label: config.labels.range30d,
|
|
411
|
+
onClick: () => admin.setRange("30d")
|
|
412
|
+
}
|
|
413
|
+
),
|
|
414
|
+
/* @__PURE__ */ jsxs(
|
|
415
|
+
"select",
|
|
416
|
+
{
|
|
417
|
+
className: "rounded-xl border border-slate-200 bg-white px-3 py-2 text-sm",
|
|
418
|
+
onChange: (event) => admin.setInterval(event.target.value),
|
|
419
|
+
value: admin.filters.interval,
|
|
420
|
+
children: [
|
|
421
|
+
/* @__PURE__ */ jsx("option", { value: "hour", children: config.labels.intervalHour }),
|
|
422
|
+
/* @__PURE__ */ jsx("option", { value: "day", children: config.labels.intervalDay })
|
|
423
|
+
]
|
|
424
|
+
}
|
|
425
|
+
),
|
|
426
|
+
/* @__PURE__ */ jsx(
|
|
427
|
+
"button",
|
|
428
|
+
{
|
|
429
|
+
className: "rounded-xl border border-slate-200 bg-white px-3 py-2 text-sm font-medium text-slate-700",
|
|
430
|
+
onClick: () => void admin.refresh(),
|
|
431
|
+
type: "button",
|
|
432
|
+
children: config.labels.refresh
|
|
433
|
+
}
|
|
434
|
+
),
|
|
435
|
+
config.branding.actions
|
|
436
|
+
] })
|
|
437
|
+
] }),
|
|
438
|
+
isEmpty ? /* @__PURE__ */ jsxs("section", { className: "rounded-3xl border border-dashed border-slate-300 bg-white/70 p-10 text-center", children: [
|
|
439
|
+
/* @__PURE__ */ jsx("h3", { className: "text-lg font-semibold", children: config.labels.emptyTitle }),
|
|
440
|
+
/* @__PURE__ */ jsx("p", { className: "mt-2 text-sm text-slate-500", children: config.labels.emptyDescription })
|
|
441
|
+
] }) : null,
|
|
442
|
+
config.features.showOverview ? /* @__PURE__ */ jsxs("section", { className: "space-y-4", children: [
|
|
443
|
+
/* @__PURE__ */ jsx(SectionHeading, { title: config.labels.overviewTitle }),
|
|
444
|
+
/* @__PURE__ */ jsx("div", { className: "grid gap-4 md:grid-cols-2 xl:grid-cols-5", children: cards.map((card) => /* @__PURE__ */ jsxs(
|
|
445
|
+
"div",
|
|
446
|
+
{
|
|
447
|
+
className: "rounded-3xl border border-slate-200 bg-white p-5 shadow-sm",
|
|
448
|
+
children: [
|
|
449
|
+
/* @__PURE__ */ jsx("p", { className: "text-sm font-medium text-slate-500", children: card.label }),
|
|
450
|
+
/* @__PURE__ */ jsx("p", { className: "mt-3 text-3xl font-semibold tracking-tight", children: formatNumber(card.value) }),
|
|
451
|
+
/* @__PURE__ */ jsx("p", { className: "mt-2 text-xs text-slate-400", children: card.detail })
|
|
452
|
+
]
|
|
453
|
+
},
|
|
454
|
+
card.label
|
|
455
|
+
)) })
|
|
456
|
+
] }) : null,
|
|
457
|
+
config.features.showActivity ? /* @__PURE__ */ jsxs("section", { className: "space-y-4", children: [
|
|
458
|
+
/* @__PURE__ */ jsx(SectionHeading, { title: config.labels.activityTitle }),
|
|
459
|
+
/* @__PURE__ */ jsx(
|
|
460
|
+
ActivityChart,
|
|
461
|
+
{
|
|
462
|
+
compact: config.ui.compact,
|
|
463
|
+
interval: admin.filters.interval,
|
|
464
|
+
labels: config.labels,
|
|
465
|
+
maxBars: config.ui.maxActivityBars,
|
|
466
|
+
points: admin.activity
|
|
467
|
+
}
|
|
468
|
+
)
|
|
469
|
+
] }) : null,
|
|
470
|
+
/* @__PURE__ */ jsxs("div", { className: "grid gap-6 xl:grid-cols-3", children: [
|
|
471
|
+
config.features.showThreads ? /* @__PURE__ */ jsx(
|
|
472
|
+
DataTable,
|
|
473
|
+
{
|
|
474
|
+
rows: admin.threads,
|
|
475
|
+
searchPlaceholder: config.labels.threadSearchPlaceholder,
|
|
476
|
+
searchValue: threadSearch,
|
|
477
|
+
setSearchValue: setThreadSearch,
|
|
478
|
+
title: config.labels.threadsTitle,
|
|
479
|
+
children: /* @__PURE__ */ jsx(ThreadsTable, { rows: admin.threads, labels: config.labels })
|
|
480
|
+
}
|
|
481
|
+
) : null,
|
|
482
|
+
config.features.showParticipants ? /* @__PURE__ */ jsx(
|
|
483
|
+
DataTable,
|
|
484
|
+
{
|
|
485
|
+
rows: admin.participants,
|
|
486
|
+
searchPlaceholder: config.labels.participantSearchPlaceholder,
|
|
487
|
+
searchValue: participantSearch,
|
|
488
|
+
setSearchValue: setParticipantSearch,
|
|
489
|
+
title: config.labels.participantsTitle,
|
|
490
|
+
children: /* @__PURE__ */ jsx(ParticipantsTable, { rows: admin.participants, labels: config.labels })
|
|
491
|
+
}
|
|
492
|
+
) : null,
|
|
493
|
+
config.features.showAgents ? /* @__PURE__ */ jsx(
|
|
494
|
+
DataTable,
|
|
495
|
+
{
|
|
496
|
+
rows: admin.agents,
|
|
497
|
+
searchPlaceholder: config.labels.agentSearchPlaceholder,
|
|
498
|
+
searchValue: agentSearch,
|
|
499
|
+
setSearchValue: setAgentSearch,
|
|
500
|
+
title: config.labels.agentsTitle,
|
|
501
|
+
children: /* @__PURE__ */ jsx(AgentsTable, { rows: admin.agents, labels: config.labels })
|
|
502
|
+
}
|
|
503
|
+
) : null
|
|
504
|
+
] })
|
|
505
|
+
] });
|
|
506
|
+
};
|
|
507
|
+
function SectionHeading({ title }) {
|
|
508
|
+
return /* @__PURE__ */ jsx("h3", { className: "text-lg font-semibold tracking-tight", children: title });
|
|
509
|
+
}
|
|
510
|
+
function RangeButton({ active, label, onClick }) {
|
|
511
|
+
return /* @__PURE__ */ jsx(
|
|
512
|
+
"button",
|
|
513
|
+
{
|
|
514
|
+
className: joinClassNames(
|
|
515
|
+
"rounded-xl px-3 py-2 text-sm font-medium transition",
|
|
516
|
+
active ? "bg-slate-900 text-white" : "border border-slate-200 bg-white text-slate-600"
|
|
517
|
+
),
|
|
518
|
+
onClick,
|
|
519
|
+
type: "button",
|
|
520
|
+
children: label
|
|
521
|
+
}
|
|
522
|
+
);
|
|
523
|
+
}
|
|
524
|
+
function ActivityChart(props) {
|
|
525
|
+
const trimmedPoints = props.points.slice(-props.maxBars);
|
|
526
|
+
const maxMessages = Math.max(...trimmedPoints.map((point) => point.messageCount), 1);
|
|
527
|
+
return /* @__PURE__ */ jsx("div", { className: "rounded-3xl border border-slate-200 bg-white p-5 shadow-sm", children: trimmedPoints.length === 0 ? /* @__PURE__ */ jsx("p", { className: "text-sm text-slate-500", children: props.labels.noResults }) : /* @__PURE__ */ jsx("div", { className: "flex min-h-56 items-end gap-2", children: trimmedPoints.map((point) => /* @__PURE__ */ jsxs("div", { className: "flex min-w-0 flex-1 flex-col items-center gap-2", children: [
|
|
528
|
+
/* @__PURE__ */ jsx("div", { className: "flex h-40 w-full items-end rounded-2xl bg-slate-100/80 px-1 pb-1", children: /* @__PURE__ */ jsx(
|
|
529
|
+
"div",
|
|
530
|
+
{
|
|
531
|
+
className: "w-full rounded-xl bg-slate-900",
|
|
532
|
+
style: {
|
|
533
|
+
height: `${Math.max(point.messageCount / maxMessages * 100, 8)}%`
|
|
534
|
+
}
|
|
535
|
+
}
|
|
536
|
+
) }),
|
|
537
|
+
/* @__PURE__ */ jsxs("div", { className: "text-center", children: [
|
|
538
|
+
/* @__PURE__ */ jsx("p", { className: "text-xs font-medium text-slate-700", children: formatBucket(point.bucket, props.interval) }),
|
|
539
|
+
/* @__PURE__ */ jsxs("p", { className: "text-[11px] text-slate-400", children: [
|
|
540
|
+
formatNumber(point.messageCount),
|
|
541
|
+
" msg"
|
|
542
|
+
] })
|
|
543
|
+
] })
|
|
544
|
+
] }, point.bucket)) }) });
|
|
545
|
+
}
|
|
546
|
+
function DataTable(props) {
|
|
547
|
+
return /* @__PURE__ */ jsxs("section", { className: "rounded-3xl border border-slate-200 bg-white p-5 shadow-sm", children: [
|
|
548
|
+
/* @__PURE__ */ jsxs("div", { className: "mb-4 flex items-center justify-between gap-3", children: [
|
|
549
|
+
/* @__PURE__ */ jsx(SectionHeading, { title: props.title }),
|
|
550
|
+
/* @__PURE__ */ jsx(
|
|
551
|
+
"input",
|
|
552
|
+
{
|
|
553
|
+
className: "w-full max-w-44 rounded-xl border border-slate-200 bg-slate-50 px-3 py-2 text-sm outline-none ring-0",
|
|
554
|
+
onChange: (event) => props.setSearchValue(event.target.value),
|
|
555
|
+
placeholder: props.searchPlaceholder,
|
|
556
|
+
value: props.searchValue
|
|
557
|
+
}
|
|
558
|
+
)
|
|
559
|
+
] }),
|
|
560
|
+
props.rows.length === 0 ? /* @__PURE__ */ jsx("p", { className: "text-sm text-slate-500", children: "No results" }) : props.children
|
|
561
|
+
] });
|
|
562
|
+
}
|
|
563
|
+
function ThreadsTable({ rows, labels }) {
|
|
564
|
+
return /* @__PURE__ */ jsx("div", { className: "space-y-3", children: rows.map((thread) => /* @__PURE__ */ jsxs("div", { className: "rounded-2xl border border-slate-200 bg-slate-50/70 p-4", children: [
|
|
565
|
+
/* @__PURE__ */ jsxs("div", { className: "flex items-start justify-between gap-3", children: [
|
|
566
|
+
/* @__PURE__ */ jsxs("div", { children: [
|
|
567
|
+
/* @__PURE__ */ jsx("p", { className: "font-medium text-slate-900", children: thread.name }),
|
|
568
|
+
/* @__PURE__ */ jsx("p", { className: "mt-1 text-xs text-slate-500", children: thread.summary ?? thread.lastMessagePreview ?? "No summary yet" })
|
|
569
|
+
] }),
|
|
570
|
+
/* @__PURE__ */ jsx("span", { className: "rounded-full bg-slate-900 px-2.5 py-1 text-[11px] font-semibold text-white", children: thread.status === "archived" ? labels.statusArchived : labels.statusActive })
|
|
571
|
+
] }),
|
|
572
|
+
/* @__PURE__ */ jsxs("div", { className: "mt-3 flex flex-wrap gap-3 text-xs text-slate-500", children: [
|
|
573
|
+
/* @__PURE__ */ jsxs("span", { children: [
|
|
574
|
+
formatNumber(thread.messageCount),
|
|
575
|
+
" messages"
|
|
576
|
+
] }),
|
|
577
|
+
/* @__PURE__ */ jsxs("span", { children: [
|
|
578
|
+
thread.participantIds.length,
|
|
579
|
+
" participants"
|
|
580
|
+
] }),
|
|
581
|
+
/* @__PURE__ */ jsx("span", { children: formatDate(thread.lastActivityAt) })
|
|
582
|
+
] })
|
|
583
|
+
] }, thread.threadId)) });
|
|
584
|
+
}
|
|
585
|
+
function ParticipantsTable({ rows, labels }) {
|
|
586
|
+
return /* @__PURE__ */ jsx("div", { className: "space-y-3", children: rows.map((participant) => /* @__PURE__ */ jsxs("div", { className: "rounded-2xl border border-slate-200 bg-slate-50/70 p-4", children: [
|
|
587
|
+
/* @__PURE__ */ jsxs("div", { className: "flex items-start justify-between gap-3", children: [
|
|
588
|
+
/* @__PURE__ */ jsxs("div", { children: [
|
|
589
|
+
/* @__PURE__ */ jsx("p", { className: "font-medium text-slate-900", children: participant.displayName }),
|
|
590
|
+
/* @__PURE__ */ jsx("p", { className: "mt-1 text-xs uppercase tracking-[0.18em] text-slate-400", children: participant.participantType })
|
|
591
|
+
] }),
|
|
592
|
+
/* @__PURE__ */ jsx("span", { className: "rounded-full border border-slate-200 bg-white px-2.5 py-1 text-[11px] font-semibold text-slate-600", children: participant.isGlobal ? labels.scopeGlobal : labels.scopeScoped })
|
|
593
|
+
] }),
|
|
594
|
+
/* @__PURE__ */ jsxs("div", { className: "mt-3 flex flex-wrap gap-3 text-xs text-slate-500", children: [
|
|
595
|
+
/* @__PURE__ */ jsxs("span", { children: [
|
|
596
|
+
formatNumber(participant.messageCount),
|
|
597
|
+
" messages"
|
|
598
|
+
] }),
|
|
599
|
+
/* @__PURE__ */ jsxs("span", { children: [
|
|
600
|
+
formatNumber(participant.threadCount),
|
|
601
|
+
" threads"
|
|
602
|
+
] }),
|
|
603
|
+
/* @__PURE__ */ jsx("span", { children: formatDate(participant.lastActivityAt) })
|
|
604
|
+
] })
|
|
605
|
+
] }, `${participant.namespace}:${participant.externalId}`)) });
|
|
606
|
+
}
|
|
607
|
+
function AgentsTable({ rows, labels }) {
|
|
608
|
+
return /* @__PURE__ */ jsx("div", { className: "space-y-3", children: rows.map((agent) => /* @__PURE__ */ jsxs("div", { className: "rounded-2xl border border-slate-200 bg-slate-50/70 p-4", children: [
|
|
609
|
+
/* @__PURE__ */ jsxs("div", { className: "flex items-start justify-between gap-3", children: [
|
|
610
|
+
/* @__PURE__ */ jsxs("div", { children: [
|
|
611
|
+
/* @__PURE__ */ jsx("p", { className: "font-medium text-slate-900", children: agent.displayName }),
|
|
612
|
+
/* @__PURE__ */ jsx("p", { className: "mt-1 text-xs text-slate-500", children: agent.description ?? agent.agentId })
|
|
613
|
+
] }),
|
|
614
|
+
/* @__PURE__ */ jsx("span", { className: "rounded-full border border-slate-200 bg-white px-2.5 py-1 text-[11px] font-semibold text-slate-600", children: agent.isConfigured ? labels.configured : labels.unconfigured })
|
|
615
|
+
] }),
|
|
616
|
+
/* @__PURE__ */ jsxs("div", { className: "mt-3 grid grid-cols-2 gap-2 text-xs text-slate-500", children: [
|
|
617
|
+
/* @__PURE__ */ jsxs("span", { children: [
|
|
618
|
+
formatNumber(agent.messageCount),
|
|
619
|
+
" messages"
|
|
620
|
+
] }),
|
|
621
|
+
/* @__PURE__ */ jsxs("span", { children: [
|
|
622
|
+
formatNumber(agent.llmCallCount),
|
|
623
|
+
" LLM calls"
|
|
624
|
+
] }),
|
|
625
|
+
/* @__PURE__ */ jsxs("span", { children: [
|
|
626
|
+
formatNumber(agent.toolCallMessageCount),
|
|
627
|
+
" tool calls"
|
|
628
|
+
] }),
|
|
629
|
+
/* @__PURE__ */ jsxs("span", { children: [
|
|
630
|
+
formatNumber(agent.totalTokens),
|
|
631
|
+
" tokens"
|
|
632
|
+
] })
|
|
633
|
+
] })
|
|
634
|
+
] }, `${agent.namespace}:${agent.agentId}`)) });
|
|
635
|
+
}
|
|
636
|
+
function formatBucket(bucket, interval) {
|
|
637
|
+
const date = new Date(bucket);
|
|
638
|
+
if (Number.isNaN(date.getTime())) return bucket;
|
|
639
|
+
return interval === "hour" ? date.toLocaleString(void 0, { hour: "numeric", month: "short", day: "numeric" }) : date.toLocaleDateString(void 0, { month: "short", day: "numeric" });
|
|
640
|
+
}
|
|
641
|
+
function formatDate(value) {
|
|
642
|
+
if (!value) return "No activity";
|
|
643
|
+
const date = new Date(value);
|
|
644
|
+
if (Number.isNaN(date.getTime())) return value;
|
|
645
|
+
return date.toLocaleString(void 0, {
|
|
646
|
+
month: "short",
|
|
647
|
+
day: "numeric",
|
|
648
|
+
hour: "numeric",
|
|
649
|
+
minute: "2-digit"
|
|
650
|
+
});
|
|
651
|
+
}
|
|
652
|
+
function formatNumber(value) {
|
|
653
|
+
return new Intl.NumberFormat().format(value);
|
|
654
|
+
}
|
|
655
|
+
function joinClassNames(...values) {
|
|
656
|
+
return values.filter(Boolean).join(" ");
|
|
657
|
+
}
|
|
658
|
+
export {
|
|
659
|
+
CopilotzAdmin,
|
|
660
|
+
defaultAdminConfig,
|
|
661
|
+
fetchAdminActivity,
|
|
662
|
+
fetchAdminAgents,
|
|
663
|
+
fetchAdminOverview,
|
|
664
|
+
fetchAdminParticipants,
|
|
665
|
+
fetchAdminThreads,
|
|
666
|
+
mergeAdminConfig,
|
|
667
|
+
useCopilotzAdmin
|
|
668
|
+
};
|
|
669
|
+
//# sourceMappingURL=index.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"sources":["../src/CopilotzAdmin.tsx","../src/config.ts","../src/useCopilotzAdmin.ts","../src/adminService.ts"],"sourcesContent":["import React, {\n useDeferredValue,\n useMemo,\n useState,\n} from \"react\";\nimport { defaultAdminConfig, mergeAdminConfig } from \"./config\";\nimport { useCopilotzAdmin } from \"./useCopilotzAdmin\";\nimport type {\n AdminActivityPoint,\n AdminAgentSummary,\n AdminConfig,\n AdminParticipantSummary,\n ResolvedAdminConfig,\n AdminThreadSummary,\n} from \"./types\";\n\nexport interface CopilotzAdminProps {\n config?: Partial<AdminConfig>;\n className?: string;\n}\n\nexport const CopilotzAdmin: React.FC<CopilotzAdminProps> = ({\n config: userConfig,\n className,\n}) => {\n const config = useMemo(\n () => mergeAdminConfig(defaultAdminConfig, userConfig),\n [userConfig],\n );\n const [threadSearch, setThreadSearch] = useState(\"\");\n const [participantSearch, setParticipantSearch] = useState(\"\");\n const [agentSearch, setAgentSearch] = useState(\"\");\n const deferredThreadSearch = useDeferredValue(threadSearch);\n const deferredParticipantSearch = useDeferredValue(participantSearch);\n const deferredAgentSearch = useDeferredValue(agentSearch);\n\n const admin = useCopilotzAdmin({\n baseUrl: config.baseUrl,\n getRequestHeaders: config.getRequestHeaders,\n namespace: config.namespace,\n range: config.initialRange,\n interval: config.initialInterval,\n threadSearch: deferredThreadSearch,\n participantSearch: deferredParticipantSearch,\n agentSearch: deferredAgentSearch,\n });\n\n const cards = [\n {\n label: config.labels.messagesCard,\n value: admin.overview?.messageTotals.total ?? 0,\n detail: `${admin.overview?.messageTotals.toolCallMessages ?? 0} tool-call messages`,\n },\n {\n label: config.labels.activeThreadsCard,\n value: admin.overview?.threadTotals.active ?? 0,\n detail: `${admin.overview?.threadTotals.total ?? 0} total threads`,\n },\n {\n label: config.labels.participantsCard,\n value: admin.overview?.participantTotals.total ?? 0,\n detail: `${admin.overview?.participantTotals.agents ?? 0} agents`,\n },\n {\n label: config.labels.tokensCard,\n value: admin.overview?.llmTotals.totalTokens ?? 0,\n detail: `${admin.overview?.llmTotals.totalCalls ?? 0} calls`,\n },\n {\n label: config.labels.queueCard,\n value: admin.overview?.queueTotals.pending ?? 0,\n detail: `${admin.overview?.queueTotals.failed ?? 0} failed`,\n },\n ];\n\n if (admin.isLoading && !admin.overview) {\n return (\n <div className={joinClassNames(\"rounded-3xl border border-slate-200 bg-white p-8 text-slate-700 shadow-sm\", className)}>\n {config.labels.loading}\n </div>\n );\n }\n\n if (admin.error && !admin.overview) {\n return (\n <div className={joinClassNames(\"rounded-3xl border border-rose-200 bg-rose-50 p-8 text-rose-700 shadow-sm\", className)}>\n <p className=\"text-base font-semibold\">{admin.error.message}</p>\n <button\n className=\"mt-4 rounded-xl bg-rose-600 px-4 py-2 text-sm font-medium text-white\"\n onClick={() => void admin.refresh()}\n type=\"button\"\n >\n {config.labels.retry}\n </button>\n </div>\n );\n }\n\n const isEmpty = cards.every((card) => card.value === 0) &&\n admin.activity.length === 0 &&\n admin.threads.length === 0 &&\n admin.participants.length === 0 &&\n admin.agents.length === 0;\n\n return (\n <div className={joinClassNames(\"space-y-6 rounded-[28px] border border-slate-200 bg-[linear-gradient(180deg,#ffffff_0%,#f8fafc_100%)] p-6 text-slate-900 shadow-sm\", className)}>\n <header className=\"flex flex-col gap-4 lg:flex-row lg:items-center lg:justify-between\">\n <div className=\"space-y-2\">\n <div className=\"flex items-center gap-3\">\n {config.branding.logo ? (\n <div className=\"flex h-10 w-10 items-center justify-center rounded-2xl border border-slate-200 bg-white shadow-sm\">\n {config.branding.logo}\n </div>\n ) : null}\n <div>\n <h2 className=\"text-2xl font-semibold tracking-tight\">\n {config.branding.title}\n </h2>\n <p className=\"text-sm text-slate-500\">{config.branding.subtitle}</p>\n </div>\n </div>\n {config.namespace ? (\n <p className=\"text-xs font-medium uppercase tracking-[0.18em] text-slate-400\">\n Namespace: {config.namespace}\n </p>\n ) : null}\n </div>\n <div className=\"flex flex-wrap items-center gap-2\">\n <RangeButton\n active={admin.filters.range === \"24h\"}\n label={config.labels.range24h}\n onClick={() => admin.setRange(\"24h\")}\n />\n <RangeButton\n active={admin.filters.range === \"7d\"}\n label={config.labels.range7d}\n onClick={() => admin.setRange(\"7d\")}\n />\n <RangeButton\n active={admin.filters.range === \"30d\"}\n label={config.labels.range30d}\n onClick={() => admin.setRange(\"30d\")}\n />\n <select\n className=\"rounded-xl border border-slate-200 bg-white px-3 py-2 text-sm\"\n onChange={(event) => admin.setInterval(event.target.value as \"hour\" | \"day\")}\n value={admin.filters.interval}\n >\n <option value=\"hour\">{config.labels.intervalHour}</option>\n <option value=\"day\">{config.labels.intervalDay}</option>\n </select>\n <button\n className=\"rounded-xl border border-slate-200 bg-white px-3 py-2 text-sm font-medium text-slate-700\"\n onClick={() => void admin.refresh()}\n type=\"button\"\n >\n {config.labels.refresh}\n </button>\n {config.branding.actions}\n </div>\n </header>\n\n {isEmpty ? (\n <section className=\"rounded-3xl border border-dashed border-slate-300 bg-white/70 p-10 text-center\">\n <h3 className=\"text-lg font-semibold\">{config.labels.emptyTitle}</h3>\n <p className=\"mt-2 text-sm text-slate-500\">{config.labels.emptyDescription}</p>\n </section>\n ) : null}\n\n {config.features.showOverview ? (\n <section className=\"space-y-4\">\n <SectionHeading title={config.labels.overviewTitle} />\n <div className=\"grid gap-4 md:grid-cols-2 xl:grid-cols-5\">\n {cards.map((card) => (\n <div\n className=\"rounded-3xl border border-slate-200 bg-white p-5 shadow-sm\"\n key={card.label}\n >\n <p className=\"text-sm font-medium text-slate-500\">{card.label}</p>\n <p className=\"mt-3 text-3xl font-semibold tracking-tight\">\n {formatNumber(card.value)}\n </p>\n <p className=\"mt-2 text-xs text-slate-400\">{card.detail}</p>\n </div>\n ))}\n </div>\n </section>\n ) : null}\n\n {config.features.showActivity ? (\n <section className=\"space-y-4\">\n <SectionHeading title={config.labels.activityTitle} />\n <ActivityChart\n compact={config.ui.compact}\n interval={admin.filters.interval}\n labels={config.labels}\n maxBars={config.ui.maxActivityBars}\n points={admin.activity}\n />\n </section>\n ) : null}\n\n <div className=\"grid gap-6 xl:grid-cols-3\">\n {config.features.showThreads ? (\n <DataTable\n rows={admin.threads}\n searchPlaceholder={config.labels.threadSearchPlaceholder}\n searchValue={threadSearch}\n setSearchValue={setThreadSearch}\n title={config.labels.threadsTitle}\n >\n <ThreadsTable rows={admin.threads} labels={config.labels} />\n </DataTable>\n ) : null}\n {config.features.showParticipants ? (\n <DataTable\n rows={admin.participants}\n searchPlaceholder={config.labels.participantSearchPlaceholder}\n searchValue={participantSearch}\n setSearchValue={setParticipantSearch}\n title={config.labels.participantsTitle}\n >\n <ParticipantsTable rows={admin.participants} labels={config.labels} />\n </DataTable>\n ) : null}\n {config.features.showAgents ? (\n <DataTable\n rows={admin.agents}\n searchPlaceholder={config.labels.agentSearchPlaceholder}\n searchValue={agentSearch}\n setSearchValue={setAgentSearch}\n title={config.labels.agentsTitle}\n >\n <AgentsTable rows={admin.agents} labels={config.labels} />\n </DataTable>\n ) : null}\n </div>\n </div>\n );\n};\n\nfunction SectionHeading({ title }: { title: string }) {\n return <h3 className=\"text-lg font-semibold tracking-tight\">{title}</h3>;\n}\n\nfunction RangeButton(\n { active, label, onClick }: { active: boolean; label: string; onClick: () => void },\n) {\n return (\n <button\n className={joinClassNames(\n \"rounded-xl px-3 py-2 text-sm font-medium transition\",\n active\n ? \"bg-slate-900 text-white\"\n : \"border border-slate-200 bg-white text-slate-600\",\n )}\n onClick={onClick}\n type=\"button\"\n >\n {label}\n </button>\n );\n}\n\nfunction ActivityChart(\n props: {\n points: AdminActivityPoint[];\n interval: \"hour\" | \"day\";\n maxBars: number;\n compact: boolean;\n labels: ResolvedAdminConfig[\"labels\"];\n },\n) {\n const trimmedPoints = props.points.slice(-props.maxBars);\n const maxMessages = Math.max(...trimmedPoints.map((point) => point.messageCount), 1);\n\n return (\n <div className=\"rounded-3xl border border-slate-200 bg-white p-5 shadow-sm\">\n {trimmedPoints.length === 0 ? (\n <p className=\"text-sm text-slate-500\">{props.labels.noResults}</p>\n ) : (\n <div className=\"flex min-h-56 items-end gap-2\">\n {trimmedPoints.map((point) => (\n <div className=\"flex min-w-0 flex-1 flex-col items-center gap-2\" key={point.bucket}>\n <div className=\"flex h-40 w-full items-end rounded-2xl bg-slate-100/80 px-1 pb-1\">\n <div\n className=\"w-full rounded-xl bg-slate-900\"\n style={{\n height: `${Math.max((point.messageCount / maxMessages) * 100, 8)}%`,\n }}\n />\n </div>\n <div className=\"text-center\">\n <p className=\"text-xs font-medium text-slate-700\">\n {formatBucket(point.bucket, props.interval)}\n </p>\n <p className=\"text-[11px] text-slate-400\">\n {formatNumber(point.messageCount)} msg\n </p>\n </div>\n </div>\n ))}\n </div>\n )}\n </div>\n );\n}\n\nfunction DataTable<T>(\n props: {\n title: string;\n searchPlaceholder: string;\n searchValue: string;\n setSearchValue: (value: string) => void;\n rows: T[];\n children: React.ReactNode;\n },\n) {\n return (\n <section className=\"rounded-3xl border border-slate-200 bg-white p-5 shadow-sm\">\n <div className=\"mb-4 flex items-center justify-between gap-3\">\n <SectionHeading title={props.title} />\n <input\n className=\"w-full max-w-44 rounded-xl border border-slate-200 bg-slate-50 px-3 py-2 text-sm outline-none ring-0\"\n onChange={(event) => props.setSearchValue(event.target.value)}\n placeholder={props.searchPlaceholder}\n value={props.searchValue}\n />\n </div>\n {props.rows.length === 0 ? (\n <p className=\"text-sm text-slate-500\">No results</p>\n ) : props.children}\n </section>\n );\n}\n\nfunction ThreadsTable(\n { rows, labels }: { rows: AdminThreadSummary[]; labels: ResolvedAdminConfig[\"labels\"] },\n) {\n return (\n <div className=\"space-y-3\">\n {rows.map((thread) => (\n <div className=\"rounded-2xl border border-slate-200 bg-slate-50/70 p-4\" key={thread.threadId}>\n <div className=\"flex items-start justify-between gap-3\">\n <div>\n <p className=\"font-medium text-slate-900\">{thread.name}</p>\n <p className=\"mt-1 text-xs text-slate-500\">{thread.summary ?? thread.lastMessagePreview ?? \"No summary yet\"}</p>\n </div>\n <span className=\"rounded-full bg-slate-900 px-2.5 py-1 text-[11px] font-semibold text-white\">\n {thread.status === \"archived\" ? labels.statusArchived : labels.statusActive}\n </span>\n </div>\n <div className=\"mt-3 flex flex-wrap gap-3 text-xs text-slate-500\">\n <span>{formatNumber(thread.messageCount)} messages</span>\n <span>{thread.participantIds.length} participants</span>\n <span>{formatDate(thread.lastActivityAt)}</span>\n </div>\n </div>\n ))}\n </div>\n );\n}\n\nfunction ParticipantsTable(\n { rows, labels }: { rows: AdminParticipantSummary[]; labels: ResolvedAdminConfig[\"labels\"] },\n) {\n return (\n <div className=\"space-y-3\">\n {rows.map((participant) => (\n <div className=\"rounded-2xl border border-slate-200 bg-slate-50/70 p-4\" key={`${participant.namespace}:${participant.externalId}`}>\n <div className=\"flex items-start justify-between gap-3\">\n <div>\n <p className=\"font-medium text-slate-900\">{participant.displayName}</p>\n <p className=\"mt-1 text-xs uppercase tracking-[0.18em] text-slate-400\">\n {participant.participantType}\n </p>\n </div>\n <span className=\"rounded-full border border-slate-200 bg-white px-2.5 py-1 text-[11px] font-semibold text-slate-600\">\n {participant.isGlobal ? labels.scopeGlobal : labels.scopeScoped}\n </span>\n </div>\n <div className=\"mt-3 flex flex-wrap gap-3 text-xs text-slate-500\">\n <span>{formatNumber(participant.messageCount)} messages</span>\n <span>{formatNumber(participant.threadCount)} threads</span>\n <span>{formatDate(participant.lastActivityAt)}</span>\n </div>\n </div>\n ))}\n </div>\n );\n}\n\nfunction AgentsTable(\n { rows, labels }: { rows: AdminAgentSummary[]; labels: ResolvedAdminConfig[\"labels\"] },\n) {\n return (\n <div className=\"space-y-3\">\n {rows.map((agent) => (\n <div className=\"rounded-2xl border border-slate-200 bg-slate-50/70 p-4\" key={`${agent.namespace}:${agent.agentId}`}>\n <div className=\"flex items-start justify-between gap-3\">\n <div>\n <p className=\"font-medium text-slate-900\">{agent.displayName}</p>\n <p className=\"mt-1 text-xs text-slate-500\">\n {agent.description ?? agent.agentId}\n </p>\n </div>\n <span className=\"rounded-full border border-slate-200 bg-white px-2.5 py-1 text-[11px] font-semibold text-slate-600\">\n {agent.isConfigured ? labels.configured : labels.unconfigured}\n </span>\n </div>\n <div className=\"mt-3 grid grid-cols-2 gap-2 text-xs text-slate-500\">\n <span>{formatNumber(agent.messageCount)} messages</span>\n <span>{formatNumber(agent.llmCallCount)} LLM calls</span>\n <span>{formatNumber(agent.toolCallMessageCount)} tool calls</span>\n <span>{formatNumber(agent.totalTokens)} tokens</span>\n </div>\n </div>\n ))}\n </div>\n );\n}\n\nfunction formatBucket(bucket: string, interval: \"hour\" | \"day\") {\n const date = new Date(bucket);\n if (Number.isNaN(date.getTime())) return bucket;\n return interval === \"hour\"\n ? date.toLocaleString(undefined, { hour: \"numeric\", month: \"short\", day: \"numeric\" })\n : date.toLocaleDateString(undefined, { month: \"short\", day: \"numeric\" });\n}\n\nfunction formatDate(value: string | null) {\n if (!value) return \"No activity\";\n const date = new Date(value);\n if (Number.isNaN(date.getTime())) return value;\n return date.toLocaleString(undefined, {\n month: \"short\",\n day: \"numeric\",\n hour: \"numeric\",\n minute: \"2-digit\",\n });\n}\n\nfunction formatNumber(value: number) {\n return new Intl.NumberFormat().format(value);\n}\n\nfunction joinClassNames(...values: Array<string | undefined | null | false>) {\n return values.filter(Boolean).join(\" \");\n}\n","import type { AdminConfig, ResolvedAdminConfig } from \"./types\";\n\nexport const defaultAdminConfig: ResolvedAdminConfig = {\n branding: {\n title: \"Copilotz Admin\",\n subtitle: \"Read-only operational visibility for Copilotz clients\",\n logo: null,\n actions: null,\n },\n labels: {\n overviewTitle: \"Overview\",\n activityTitle: \"Activity\",\n threadsTitle: \"Threads\",\n participantsTitle: \"Participants\",\n agentsTitle: \"Agents\",\n range24h: \"24h\",\n range7d: \"7d\",\n range30d: \"30d\",\n intervalHour: \"Hourly\",\n intervalDay: \"Daily\",\n refresh: \"Refresh\",\n retry: \"Try again\",\n loading: \"Loading admin data...\",\n emptyTitle: \"No activity yet\",\n emptyDescription: \"Admin metrics will appear after threads, events, and graph data start flowing.\",\n threadSearchPlaceholder: \"Search threads\",\n participantSearchPlaceholder: \"Search participants\",\n agentSearchPlaceholder: \"Search agents\",\n messagesCard: \"Messages\",\n activeThreadsCard: \"Active threads\",\n participantsCard: \"Participants\",\n tokensCard: \"LLM tokens\",\n queueCard: \"Queued events\",\n statusActive: \"Active\",\n statusArchived: \"Archived\",\n scopeGlobal: \"Global\",\n scopeScoped: \"Scoped\",\n configured: \"Configured\",\n unconfigured: \"Observed only\",\n noResults: \"No results\",\n },\n features: {\n showOverview: true,\n showActivity: true,\n showThreads: true,\n showParticipants: true,\n showAgents: true,\n },\n ui: {\n compact: false,\n maxActivityBars: 18,\n },\n baseUrl: \"\",\n getRequestHeaders: async () => ({}),\n namespace: \"\",\n initialRange: \"7d\",\n initialInterval: \"day\",\n};\n\nexport function mergeAdminConfig(\n _baseConfig: AdminConfig,\n userConfig?: Partial<AdminConfig>,\n): ResolvedAdminConfig {\n if (!userConfig) return defaultAdminConfig;\n\n return {\n branding: {\n ...defaultAdminConfig.branding,\n ...userConfig.branding,\n },\n labels: {\n ...defaultAdminConfig.labels,\n ...userConfig.labels,\n },\n features: {\n ...defaultAdminConfig.features,\n ...userConfig.features,\n },\n ui: {\n ...defaultAdminConfig.ui,\n ...userConfig.ui,\n },\n baseUrl: userConfig.baseUrl ?? defaultAdminConfig.baseUrl,\n getRequestHeaders: userConfig.getRequestHeaders ??\n defaultAdminConfig.getRequestHeaders,\n namespace: userConfig.namespace ?? defaultAdminConfig.namespace,\n initialRange: userConfig.initialRange ?? defaultAdminConfig.initialRange,\n initialInterval: userConfig.initialInterval ??\n defaultAdminConfig.initialInterval,\n };\n}\n","import {\n useCallback,\n useEffect,\n useMemo,\n useState,\n} from \"react\";\nimport {\n fetchAdminActivity,\n fetchAdminAgents,\n fetchAdminOverview,\n fetchAdminParticipants,\n fetchAdminThreads,\n} from \"./adminService\";\nimport type {\n AdminActivityInterval,\n AdminDatePreset,\n UseCopilotzAdminOptions,\n UseCopilotzAdminResult,\n} from \"./types\";\n\nexport function useCopilotzAdmin(\n options: UseCopilotzAdminOptions = {},\n): UseCopilotzAdminResult {\n const [range, setRange] = useState<AdminDatePreset>(options.range ?? \"7d\");\n const [interval, setInterval] = useState<AdminActivityInterval>(\n options.interval ?? \"day\",\n );\n const [overview, setOverview] = useState<UseCopilotzAdminResult[\"overview\"]>(\n null,\n );\n const [activity, setActivity] = useState<UseCopilotzAdminResult[\"activity\"]>(\n [],\n );\n const [threads, setThreads] = useState<UseCopilotzAdminResult[\"threads\"]>([]);\n const [participants, setParticipants] = useState<\n UseCopilotzAdminResult[\"participants\"]\n >([]);\n const [agents, setAgents] = useState<UseCopilotzAdminResult[\"agents\"]>([]);\n const [isLoading, setIsLoading] = useState(true);\n const [error, setError] = useState<Error | null>(null);\n\n const fetchAll = useCallback(async () => {\n setIsLoading(true);\n setError(null);\n try {\n const shared = {\n baseUrl: options.baseUrl,\n getRequestHeaders: options.getRequestHeaders,\n };\n const [\n nextOverview,\n nextActivity,\n nextThreads,\n nextParticipants,\n nextAgents,\n ] = await Promise.all([\n fetchAdminOverview(range, options.namespace, shared),\n fetchAdminActivity(range, interval, options.namespace, shared),\n fetchAdminThreads(options.threadSearch, options.namespace, shared),\n fetchAdminParticipants(\n options.participantSearch,\n options.namespace,\n shared,\n ),\n fetchAdminAgents(options.agentSearch, options.namespace, shared),\n ]);\n\n setOverview(nextOverview);\n setActivity(nextActivity);\n setThreads(nextThreads);\n setParticipants(nextParticipants);\n setAgents(nextAgents);\n } catch (nextError) {\n setError(nextError instanceof Error\n ? nextError\n : new Error(\"Failed to load admin data\"));\n } finally {\n setIsLoading(false);\n }\n }, [\n interval,\n options.agentSearch,\n options.baseUrl,\n options.getRequestHeaders,\n options.namespace,\n options.participantSearch,\n options.threadSearch,\n range,\n ]);\n\n useEffect(() => {\n void fetchAll();\n }, [fetchAll]);\n\n return useMemo(() => ({\n overview,\n activity,\n threads,\n participants,\n agents,\n filters: {\n namespace: options.namespace,\n threadSearch: options.threadSearch,\n participantSearch: options.participantSearch,\n agentSearch: options.agentSearch,\n range,\n interval,\n },\n isLoading,\n error,\n refresh: fetchAll,\n setRange,\n setInterval,\n }), [\n activity,\n agents,\n error,\n fetchAll,\n interval,\n isLoading,\n options.agentSearch,\n options.namespace,\n options.participantSearch,\n options.threadSearch,\n overview,\n participants,\n range,\n threads,\n ]);\n}\n","import type {\n AdminActivityInterval,\n AdminActivityPoint,\n AdminAgentSummary,\n AdminDatePreset,\n AdminOverview,\n AdminParticipantSummary,\n AdminThreadSummary,\n RequestHeadersProvider,\n} from \"./types\";\n\nconst rawBaseValue =\n (import.meta as { env?: Record<string, string | undefined> }).env\n ?.VITE_API_URL;\nconst rawBase = typeof rawBaseValue === \"string\" && rawBaseValue.length > 0\n ? rawBaseValue\n : \"/api\";\nconst normalizedBase = rawBase.replace(/\\/$/, \"\");\n\nconst runtimeProcess: typeof process | undefined =\n typeof process !== \"undefined\" ? process : undefined;\n\nconst API_KEY = (() => {\n const env =\n (import.meta as { env?: Record<string, string | undefined> }).env ?? {};\n const candidates = [\n env.VITE_API_KEY,\n env.VITE_COPILOTZ_API_KEY,\n runtimeProcess?.env?.COPILOTZ_API_KEY,\n runtimeProcess?.env?.API_KEY,\n ];\n return candidates.find((value) =>\n typeof value === \"string\" && value.length > 0\n );\n})();\n\nconst resolveBaseUrl = (baseUrl?: string) => {\n const candidate = (baseUrl && baseUrl.length > 0 ? baseUrl : normalizedBase)\n .replace(/\\/$/, \"\");\n return candidate.startsWith(\"http\") || candidate.startsWith(\"/\")\n ? candidate\n : `/${candidate}`;\n};\n\nconst withAuthHeaders = async (\n headers: Record<string, string> = {},\n getRequestHeaders?: RequestHeadersProvider,\n): Promise<Record<string, string>> => {\n const providedHeaders = getRequestHeaders\n ? await getRequestHeaders()\n : undefined;\n if (providedHeaders && Object.keys(providedHeaders).length > 0) {\n return { ...headers, ...providedHeaders };\n }\n if (API_KEY) {\n return { ...headers, Authorization: `Bearer ${API_KEY}` };\n }\n return headers;\n};\n\nconst getRangeWindow = (range: AdminDatePreset) => {\n const to = new Date();\n const from = new Date(to);\n if (range === \"24h\") {\n from.setHours(from.getHours() - 24);\n } else if (range === \"30d\") {\n from.setDate(from.getDate() - 30);\n } else {\n from.setDate(from.getDate() - 7);\n }\n return {\n from: from.toISOString(),\n to: to.toISOString(),\n };\n};\n\nasync function fetchAdminJson<T>(\n path: string,\n params: Record<string, string | undefined>,\n options?: {\n baseUrl?: string;\n getRequestHeaders?: RequestHeadersProvider;\n },\n): Promise<T> {\n const url = new URL(`${resolveBaseUrl(options?.baseUrl)}${path}`, window.location.origin);\n for (const [key, value] of Object.entries(params)) {\n if (typeof value === \"string\" && value.length > 0) {\n url.searchParams.set(key, value);\n }\n }\n\n const response = await fetch(url.toString(), {\n headers: await withAuthHeaders({}, options?.getRequestHeaders),\n });\n\n if (!response.ok) {\n throw new Error(`Admin request failed (${response.status})`);\n }\n\n const payload = await response.json() as { data?: T };\n return payload.data as T;\n}\n\nexport async function fetchAdminOverview(\n range: AdminDatePreset,\n namespace?: string,\n options?: {\n baseUrl?: string;\n getRequestHeaders?: RequestHeadersProvider;\n },\n): Promise<AdminOverview> {\n const windowRange = getRangeWindow(range);\n return await fetchAdminJson<AdminOverview>(\"/v1/admin/overview\", {\n namespace,\n from: windowRange.from,\n to: windowRange.to,\n }, options);\n}\n\nexport async function fetchAdminActivity(\n range: AdminDatePreset,\n interval: AdminActivityInterval,\n namespace?: string,\n options?: {\n baseUrl?: string;\n getRequestHeaders?: RequestHeadersProvider;\n },\n): Promise<AdminActivityPoint[]> {\n const windowRange = getRangeWindow(range);\n return await fetchAdminJson<AdminActivityPoint[]>(\"/v1/admin/activity\", {\n namespace,\n interval,\n from: windowRange.from,\n to: windowRange.to,\n }, options);\n}\n\nexport async function fetchAdminThreads(\n search?: string,\n namespace?: string,\n options?: {\n baseUrl?: string;\n getRequestHeaders?: RequestHeadersProvider;\n },\n): Promise<AdminThreadSummary[]> {\n return await fetchAdminJson<AdminThreadSummary[]>(\"/v1/admin/threads\", {\n search,\n namespace,\n limit: \"8\",\n }, options);\n}\n\nexport async function fetchAdminParticipants(\n search?: string,\n namespace?: string,\n options?: {\n baseUrl?: string;\n getRequestHeaders?: RequestHeadersProvider;\n },\n): Promise<AdminParticipantSummary[]> {\n return await fetchAdminJson<AdminParticipantSummary[]>(\"/v1/admin/participants\", {\n search,\n namespace,\n limit: \"8\",\n }, options);\n}\n\nexport async function fetchAdminAgents(\n search?: string,\n namespace?: string,\n options?: {\n baseUrl?: string;\n getRequestHeaders?: RequestHeadersProvider;\n },\n): Promise<AdminAgentSummary[]> {\n return await fetchAdminJson<AdminAgentSummary[]>(\"/v1/admin/agents\", {\n search,\n namespace,\n limit: \"8\",\n }, options);\n}\n"],"mappings":"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","names":["useMemo","useState","useMemo","useState"]}
|
package/package.json
ADDED
|
@@ -0,0 +1,42 @@
|
|
|
1
|
+
{
|
|
2
|
+
"name": "@copilotz/admin",
|
|
3
|
+
"version": "0.3.4",
|
|
4
|
+
"description": "Copilotz admin dashboard package",
|
|
5
|
+
"type": "module",
|
|
6
|
+
"module": "./dist/index.js",
|
|
7
|
+
"types": "./dist/index.d.ts",
|
|
8
|
+
"exports": {
|
|
9
|
+
".": {
|
|
10
|
+
"types": "./dist/index.d.ts",
|
|
11
|
+
"import": "./dist/index.js"
|
|
12
|
+
}
|
|
13
|
+
},
|
|
14
|
+
"files": [
|
|
15
|
+
"dist"
|
|
16
|
+
],
|
|
17
|
+
"scripts": {
|
|
18
|
+
"build": "tsup",
|
|
19
|
+
"build:watch": "tsup --watch",
|
|
20
|
+
"dev": "npm run build:watch",
|
|
21
|
+
"prepack": "npm run build"
|
|
22
|
+
},
|
|
23
|
+
"peerDependencies": {
|
|
24
|
+
"react": ">=18",
|
|
25
|
+
"react-dom": ">=18"
|
|
26
|
+
},
|
|
27
|
+
"devDependencies": {
|
|
28
|
+
"@types/react": "^19.1.10",
|
|
29
|
+
"@types/react-dom": "^19.1.7",
|
|
30
|
+
"tsup": "^8.3.5",
|
|
31
|
+
"typescript": "~5.8.2"
|
|
32
|
+
},
|
|
33
|
+
"license": "MIT",
|
|
34
|
+
"repository": {
|
|
35
|
+
"type": "git",
|
|
36
|
+
"url": "git+ssh://git@github.com/copilotzhq/packages.git"
|
|
37
|
+
},
|
|
38
|
+
"homepage": "https://github.com/copilotzhq/packages#readme",
|
|
39
|
+
"publishConfig": {
|
|
40
|
+
"access": "public"
|
|
41
|
+
}
|
|
42
|
+
}
|