@questionbase/deskfree 0.1.0
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/README.md +129 -0
- package/dist/channel.d.ts +3 -0
- package/dist/channel.d.ts.map +1 -0
- package/dist/channel.js +503 -0
- package/dist/channel.js.map +1 -0
- package/dist/client.d.ts +148 -0
- package/dist/client.d.ts.map +1 -0
- package/dist/client.js +255 -0
- package/dist/client.js.map +1 -0
- package/dist/deliver.d.ts +22 -0
- package/dist/deliver.d.ts.map +1 -0
- package/dist/deliver.js +350 -0
- package/dist/deliver.js.map +1 -0
- package/dist/gateway.d.ts +13 -0
- package/dist/gateway.d.ts.map +1 -0
- package/dist/gateway.js +687 -0
- package/dist/gateway.js.map +1 -0
- package/dist/index.d.ts +11 -0
- package/dist/index.d.ts.map +1 -0
- package/dist/index.js +19 -0
- package/dist/index.js.map +1 -0
- package/dist/llm-definitions.d.ts +116 -0
- package/dist/llm-definitions.d.ts.map +1 -0
- package/dist/llm-definitions.js +148 -0
- package/dist/llm-definitions.js.map +1 -0
- package/dist/offline-queue.d.ts +45 -0
- package/dist/offline-queue.d.ts.map +1 -0
- package/dist/offline-queue.js +109 -0
- package/dist/offline-queue.js.map +1 -0
- package/dist/paths.d.ts +10 -0
- package/dist/paths.d.ts.map +1 -0
- package/dist/paths.js +29 -0
- package/dist/paths.js.map +1 -0
- package/dist/runtime.d.ts +17 -0
- package/dist/runtime.d.ts.map +1 -0
- package/dist/runtime.js +24 -0
- package/dist/runtime.js.map +1 -0
- package/dist/tools.d.ts +35 -0
- package/dist/tools.d.ts.map +1 -0
- package/dist/tools.js +527 -0
- package/dist/tools.js.map +1 -0
- package/dist/types.d.ts +389 -0
- package/dist/types.d.ts.map +1 -0
- package/dist/types.js +2 -0
- package/dist/types.js.map +1 -0
- package/dist/workspace.d.ts +18 -0
- package/dist/workspace.d.ts.map +1 -0
- package/dist/workspace.js +83 -0
- package/dist/workspace.js.map +1 -0
- package/openclaw.plugin.json +8 -0
- package/package.json +63 -0
- package/skills/deskfree/SKILL.md +271 -0
package/README.md
ADDED
|
@@ -0,0 +1,129 @@
|
|
|
1
|
+
# @questionbase/deskfree
|
|
2
|
+
|
|
3
|
+
OpenClaw plugin that connects AI agents to [DeskFree](https://github.com/deskfree) — a task management platform built for human-AI collaboration.
|
|
4
|
+
|
|
5
|
+
Agents can pick up tasks, post progress updates, mark work as done, and exchange messages with humans — all through a structured workflow with human oversight gates.
|
|
6
|
+
|
|
7
|
+
## How it works
|
|
8
|
+
|
|
9
|
+
```
|
|
10
|
+
┌──────────────────────────────────────────────────────────────────┐
|
|
11
|
+
│ OpenClaw Agent │
|
|
12
|
+
│ │
|
|
13
|
+
│ Tools (HTTP) Channel (WebSocket + polling) │
|
|
14
|
+
│ ───────────── ────────────────────────────── │
|
|
15
|
+
│ get_next_task Inbound messages from humans │
|
|
16
|
+
│ comment / done / giveup Outbound replies from agent │
|
|
17
|
+
│ list_tasks / create_task Real-time notifications │
|
|
18
|
+
└──────────┬───────────────────────────┬───────────────────────────┘
|
|
19
|
+
│ │
|
|
20
|
+
▼ ▼
|
|
21
|
+
┌──────────────────────────────────────────────────────────────────┐
|
|
22
|
+
│ DeskFree Backend │
|
|
23
|
+
│ │
|
|
24
|
+
│ Tasks Messages WebSocket Gateway │
|
|
25
|
+
│ ────── ──────── ───────────────── │
|
|
26
|
+
│ State machine 1:1 bot↔user DynamoDB-backed │
|
|
27
|
+
│ RLS-scoped Threaded by Thin notifications │
|
|
28
|
+
│ Atomic claims task + polling fallback │
|
|
29
|
+
└──────────────────────────────────────────────────────────────────┘
|
|
30
|
+
```
|
|
31
|
+
|
|
32
|
+
The plugin registers three things with OpenClaw:
|
|
33
|
+
|
|
34
|
+
1. **Channel** — bidirectional messaging between DeskFree users and the agent
|
|
35
|
+
2. **Tools** — 8 task management tools the agent can call
|
|
36
|
+
3. **Skill** — workflow knowledge that teaches the agent how to use the tools correctly
|
|
37
|
+
|
|
38
|
+
## Install
|
|
39
|
+
|
|
40
|
+
```bash
|
|
41
|
+
openclaw plugins install @questionbase/deskfree
|
|
42
|
+
```
|
|
43
|
+
|
|
44
|
+
Requires OpenClaw >= 0.1.0.
|
|
45
|
+
|
|
46
|
+
This installs the plugin to `~/.openclaw/extensions/deskfree/` and enables it automatically.
|
|
47
|
+
|
|
48
|
+
### Verify
|
|
49
|
+
|
|
50
|
+
```bash
|
|
51
|
+
openclaw plugins list
|
|
52
|
+
openclaw plugins info deskfree
|
|
53
|
+
```
|
|
54
|
+
|
|
55
|
+
## Configuration
|
|
56
|
+
|
|
57
|
+
After installation, configure the DeskFree channel. You can either:
|
|
58
|
+
|
|
59
|
+
### Option A: Interactive setup
|
|
60
|
+
|
|
61
|
+
Run the OpenClaw onboarding wizard — it will detect the unconfigured DeskFree channel and prompt for credentials.
|
|
62
|
+
|
|
63
|
+
### Option B: Manual config
|
|
64
|
+
|
|
65
|
+
Edit your OpenClaw config file (`~/.openclaw/config.json5`):
|
|
66
|
+
|
|
67
|
+
**Single account (most common):**
|
|
68
|
+
```json5
|
|
69
|
+
{
|
|
70
|
+
channels: {
|
|
71
|
+
deskfree: {
|
|
72
|
+
botToken: "bot_...",
|
|
73
|
+
apiUrl: "https://api.example.com/v1/bot",
|
|
74
|
+
wsUrl: "wss://ws.example.com"
|
|
75
|
+
}
|
|
76
|
+
}
|
|
77
|
+
}
|
|
78
|
+
```
|
|
79
|
+
|
|
80
|
+
**Multiple accounts:**
|
|
81
|
+
```json5
|
|
82
|
+
{
|
|
83
|
+
channels: {
|
|
84
|
+
deskfree: {
|
|
85
|
+
accounts: {
|
|
86
|
+
"my-bot": {
|
|
87
|
+
botToken: "bot_...",
|
|
88
|
+
apiUrl: "https://api.example.com/v1/bot",
|
|
89
|
+
wsUrl: "wss://ws.example.com"
|
|
90
|
+
}
|
|
91
|
+
}
|
|
92
|
+
}
|
|
93
|
+
}
|
|
94
|
+
}
|
|
95
|
+
```
|
|
96
|
+
|
|
97
|
+
| Field | Description |
|
|
98
|
+
|-------|-------------|
|
|
99
|
+
| `botToken` | Bot authentication token (starts with `bot_`) |
|
|
100
|
+
| `apiUrl` | DeskFree bot API base URL (the `/v1/bot` tRPC endpoint) |
|
|
101
|
+
| `wsUrl` | WebSocket gateway URL for real-time notifications |
|
|
102
|
+
|
|
103
|
+
### Getting credentials
|
|
104
|
+
|
|
105
|
+
1. Create a bot in your DeskFree instance
|
|
106
|
+
2. Copy the bot token from the bot settings page
|
|
107
|
+
3. Your DeskFree admin can provide the API and WebSocket URLs
|
|
108
|
+
|
|
109
|
+
## Task workflow
|
|
110
|
+
|
|
111
|
+
Tasks follow a strict state machine. The agent claims a task, works on it, and either completes or gives it up — always returning control to a human.
|
|
112
|
+
|
|
113
|
+
```
|
|
114
|
+
ready_for_bot ──[get_next_task]──> working_on_it ──[done]──> waiting_for_human
|
|
115
|
+
│
|
|
116
|
+
└───[giveup]──> waiting_for_human
|
|
117
|
+
```
|
|
118
|
+
|
|
119
|
+
See [docs/tools.md](docs/tools.md) for the full tool reference and [docs/architecture.md](docs/architecture.md) for how the system works under the hood.
|
|
120
|
+
|
|
121
|
+
## Documentation
|
|
122
|
+
|
|
123
|
+
- **[Architecture](docs/architecture.md)** — system design, message flow, WebSocket gateway
|
|
124
|
+
- **[Tools reference](docs/tools.md)** — all 8 agent tools with parameters and behavior
|
|
125
|
+
- **[Publishing](PUBLISH.md)** — how to release new versions
|
|
126
|
+
|
|
127
|
+
## License
|
|
128
|
+
|
|
129
|
+
MIT
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"channel.d.ts","sourceRoot":"","sources":["../src/channel.ts"],"names":[],"mappings":"AAQA,OAAO,KAAK,EAGV,aAAa,EAId,MAAM,SAAS,CAAC;AAMjB,eAAO,MAAM,cAAc,EAAE,aA8iB5B,CAAC"}
|
package/dist/channel.js
ADDED
|
@@ -0,0 +1,503 @@
|
|
|
1
|
+
import { DeskFreeClient } from './client';
|
|
2
|
+
import { getActiveTaskId, startDeskFreeConnection } from './gateway';
|
|
3
|
+
import { CHANNEL_HINTS, CHANNEL_META, STATUS_MESSAGES, } from './llm-definitions';
|
|
4
|
+
import { getDeskFreeRuntime } from './runtime';
|
|
5
|
+
function getChannelConfig(cfg) {
|
|
6
|
+
return cfg?.channels?.deskfree ?? null;
|
|
7
|
+
}
|
|
8
|
+
export const deskFreePlugin = {
|
|
9
|
+
id: 'deskfree',
|
|
10
|
+
meta: CHANNEL_META,
|
|
11
|
+
capabilities: {
|
|
12
|
+
text: true,
|
|
13
|
+
media: false,
|
|
14
|
+
reactions: false,
|
|
15
|
+
threads: true,
|
|
16
|
+
editing: false,
|
|
17
|
+
},
|
|
18
|
+
config: {
|
|
19
|
+
listAccountIds(cfg) {
|
|
20
|
+
const ch = getChannelConfig(cfg);
|
|
21
|
+
if (!ch)
|
|
22
|
+
return [];
|
|
23
|
+
if (ch.accounts && Object.keys(ch.accounts).length > 0) {
|
|
24
|
+
return Object.keys(ch.accounts);
|
|
25
|
+
}
|
|
26
|
+
return ['default'];
|
|
27
|
+
},
|
|
28
|
+
resolveAccount(cfg, accountId) {
|
|
29
|
+
const ch = getChannelConfig(cfg);
|
|
30
|
+
if (!ch) {
|
|
31
|
+
return {
|
|
32
|
+
accountId: accountId ?? 'default',
|
|
33
|
+
botToken: '',
|
|
34
|
+
apiUrl: '',
|
|
35
|
+
wsUrl: '',
|
|
36
|
+
userId: '',
|
|
37
|
+
enabled: false,
|
|
38
|
+
};
|
|
39
|
+
}
|
|
40
|
+
const id = accountId ?? 'default';
|
|
41
|
+
const acct = id !== 'default' && ch.accounts?.[id] ? ch.accounts[id] : ch;
|
|
42
|
+
return {
|
|
43
|
+
accountId: id,
|
|
44
|
+
botToken: acct.botToken ?? '',
|
|
45
|
+
apiUrl: acct.apiUrl ?? '',
|
|
46
|
+
wsUrl: acct.wsUrl ?? '',
|
|
47
|
+
userId: acct.userId ?? '',
|
|
48
|
+
enabled: acct.enabled !== false,
|
|
49
|
+
};
|
|
50
|
+
},
|
|
51
|
+
isConfigured(account) {
|
|
52
|
+
return Boolean(account.botToken && account.apiUrl);
|
|
53
|
+
},
|
|
54
|
+
isEnabled(account) {
|
|
55
|
+
return account.enabled;
|
|
56
|
+
},
|
|
57
|
+
describeAccount(account) {
|
|
58
|
+
const isConfigured = Boolean(account.botToken && account.apiUrl);
|
|
59
|
+
return {
|
|
60
|
+
accountId: account.accountId,
|
|
61
|
+
enabled: account.enabled,
|
|
62
|
+
configured: isConfigured,
|
|
63
|
+
};
|
|
64
|
+
},
|
|
65
|
+
},
|
|
66
|
+
outbound: {
|
|
67
|
+
deliveryMode: 'direct',
|
|
68
|
+
chunkerMode: 'markdown',
|
|
69
|
+
textChunkLimit: 2000,
|
|
70
|
+
resolveTarget(_target, { cfg, accountId }) {
|
|
71
|
+
// Resolve to the userId configured for this account
|
|
72
|
+
const acct = deskFreePlugin.config.resolveAccount(cfg, accountId);
|
|
73
|
+
if (!acct.userId)
|
|
74
|
+
return null;
|
|
75
|
+
return { to: acct.userId };
|
|
76
|
+
},
|
|
77
|
+
async sendText(ctx) {
|
|
78
|
+
const runtime = getDeskFreeRuntime();
|
|
79
|
+
const cfg = runtime.config.loadConfig();
|
|
80
|
+
const ch = getChannelConfig(cfg);
|
|
81
|
+
if (!ch) {
|
|
82
|
+
return { channel: 'deskfree', success: false };
|
|
83
|
+
}
|
|
84
|
+
const acct = deskFreePlugin.config.resolveAccount(cfg, ctx.accountId);
|
|
85
|
+
const client = new DeskFreeClient(acct.botToken, acct.apiUrl);
|
|
86
|
+
const log = runtime.logging.createLogger('deskfree');
|
|
87
|
+
try {
|
|
88
|
+
// Use explicit threadId if present, otherwise auto-thread
|
|
89
|
+
// into the active task
|
|
90
|
+
const taskId = ctx.threadId
|
|
91
|
+
? String(ctx.threadId)
|
|
92
|
+
: (getActiveTaskId() ?? undefined);
|
|
93
|
+
await client.sendMessage({
|
|
94
|
+
content: ctx.text,
|
|
95
|
+
taskId,
|
|
96
|
+
});
|
|
97
|
+
return {
|
|
98
|
+
channel: 'deskfree',
|
|
99
|
+
success: true,
|
|
100
|
+
threadId: taskId,
|
|
101
|
+
};
|
|
102
|
+
}
|
|
103
|
+
catch (err) {
|
|
104
|
+
const message = err instanceof Error ? err.message : String(err);
|
|
105
|
+
log.error(`Failed to send message: ${message}`);
|
|
106
|
+
return {
|
|
107
|
+
channel: 'deskfree',
|
|
108
|
+
success: false,
|
|
109
|
+
error: message,
|
|
110
|
+
};
|
|
111
|
+
}
|
|
112
|
+
},
|
|
113
|
+
},
|
|
114
|
+
gateway: {
|
|
115
|
+
async startAccount(ctx) {
|
|
116
|
+
return startDeskFreeConnection(ctx);
|
|
117
|
+
},
|
|
118
|
+
async stopAccount(ctx) {
|
|
119
|
+
const log = ctx.log ?? getDeskFreeRuntime().logging.createLogger('deskfree');
|
|
120
|
+
log.info(`Stopping DeskFree account ${ctx.accountId}`);
|
|
121
|
+
ctx.setStatus({ running: false, lastStopAt: Date.now() });
|
|
122
|
+
},
|
|
123
|
+
async logoutAccount(ctx) {
|
|
124
|
+
const log = ctx.log ?? getDeskFreeRuntime().logging.createLogger('deskfree');
|
|
125
|
+
const runtime = ctx.runtime;
|
|
126
|
+
// Clone config immutably
|
|
127
|
+
const config = JSON.parse(JSON.stringify(ctx.cfg));
|
|
128
|
+
const channels = (config.channels ?? {});
|
|
129
|
+
const deskfree = (channels.deskfree ?? {});
|
|
130
|
+
if (ctx.accountId === 'default') {
|
|
131
|
+
delete deskfree.botToken;
|
|
132
|
+
}
|
|
133
|
+
else {
|
|
134
|
+
const accounts = (deskfree.accounts ?? {});
|
|
135
|
+
if (accounts[ctx.accountId]) {
|
|
136
|
+
delete accounts[ctx.accountId].botToken;
|
|
137
|
+
}
|
|
138
|
+
}
|
|
139
|
+
channels.deskfree = deskfree;
|
|
140
|
+
config.channels = channels;
|
|
141
|
+
await runtime.config.writeConfigFile(config);
|
|
142
|
+
log.info(`Cleared bot token for account ${ctx.accountId}`);
|
|
143
|
+
return { cleared: true, loggedOut: true };
|
|
144
|
+
},
|
|
145
|
+
},
|
|
146
|
+
setup: {
|
|
147
|
+
applyAccountConfig(params) {
|
|
148
|
+
const { cfg, accountId, input } = params;
|
|
149
|
+
const config = cfg;
|
|
150
|
+
const channels = (config.channels ?? {});
|
|
151
|
+
const deskfree = (channels.deskfree ?? {});
|
|
152
|
+
if (accountId === 'default') {
|
|
153
|
+
deskfree.botToken = input.botToken;
|
|
154
|
+
deskfree.apiUrl = input.apiUrl;
|
|
155
|
+
deskfree.wsUrl = input.wsUrl;
|
|
156
|
+
deskfree.userId = input.userId;
|
|
157
|
+
}
|
|
158
|
+
else {
|
|
159
|
+
const accounts = (deskfree.accounts ?? {});
|
|
160
|
+
accounts[accountId] = {
|
|
161
|
+
botToken: input.botToken,
|
|
162
|
+
apiUrl: input.apiUrl,
|
|
163
|
+
wsUrl: input.wsUrl,
|
|
164
|
+
userId: input.userId,
|
|
165
|
+
enabled: true,
|
|
166
|
+
};
|
|
167
|
+
deskfree.accounts = accounts;
|
|
168
|
+
}
|
|
169
|
+
// Always set top-level channel enabled
|
|
170
|
+
deskfree.enabled = true;
|
|
171
|
+
channels.deskfree = deskfree;
|
|
172
|
+
config.channels = channels;
|
|
173
|
+
// Register plugin entry with channel ID to prevent auto-enable
|
|
174
|
+
// from creating a phantom disabled entry under the NPM package ID
|
|
175
|
+
const plugins = (config.plugins ?? {});
|
|
176
|
+
const entries = (plugins.entries ?? {});
|
|
177
|
+
const existing = (entries.deskfree ?? {});
|
|
178
|
+
entries.deskfree = { ...existing, enabled: true };
|
|
179
|
+
plugins.entries = entries;
|
|
180
|
+
config.plugins = plugins;
|
|
181
|
+
return config;
|
|
182
|
+
},
|
|
183
|
+
validateInput(params) {
|
|
184
|
+
const { input } = params;
|
|
185
|
+
// Validate bot token
|
|
186
|
+
if (!input.botToken) {
|
|
187
|
+
return 'Bot token is required';
|
|
188
|
+
}
|
|
189
|
+
if (typeof input.botToken !== 'string') {
|
|
190
|
+
return 'Bot token must be a string';
|
|
191
|
+
}
|
|
192
|
+
const botTokenTrimmed = input.botToken.trim();
|
|
193
|
+
if (botTokenTrimmed !== input.botToken) {
|
|
194
|
+
return 'Bot token must not have leading or trailing whitespace';
|
|
195
|
+
}
|
|
196
|
+
if (!botTokenTrimmed.startsWith('bot_')) {
|
|
197
|
+
return 'Bot token must start with "bot_"';
|
|
198
|
+
}
|
|
199
|
+
if (botTokenTrimmed.length < 10) {
|
|
200
|
+
return 'Bot token appears to be too short (minimum 10 characters)';
|
|
201
|
+
}
|
|
202
|
+
if (botTokenTrimmed.length > 100) {
|
|
203
|
+
return 'Bot token appears to be too long (maximum 100 characters)';
|
|
204
|
+
}
|
|
205
|
+
if (!/^bot_[a-zA-Z0-9_-]+$/.test(botTokenTrimmed)) {
|
|
206
|
+
return 'Bot token contains invalid characters (only alphanumeric, underscore, and dash allowed after "bot_")';
|
|
207
|
+
}
|
|
208
|
+
// Validate API URL
|
|
209
|
+
if (!input.apiUrl) {
|
|
210
|
+
return 'API URL is required';
|
|
211
|
+
}
|
|
212
|
+
if (typeof input.apiUrl !== 'string') {
|
|
213
|
+
return 'API URL must be a string';
|
|
214
|
+
}
|
|
215
|
+
const apiUrlTrimmed = input.apiUrl.trim();
|
|
216
|
+
if (apiUrlTrimmed !== input.apiUrl) {
|
|
217
|
+
return 'API URL must not have leading or trailing whitespace';
|
|
218
|
+
}
|
|
219
|
+
try {
|
|
220
|
+
const apiUrl = new URL(apiUrlTrimmed);
|
|
221
|
+
if (apiUrl.protocol !== 'https:') {
|
|
222
|
+
return 'API URL must use HTTPS protocol for security';
|
|
223
|
+
}
|
|
224
|
+
if (!apiUrl.hostname) {
|
|
225
|
+
return 'API URL must have a valid hostname';
|
|
226
|
+
}
|
|
227
|
+
if (apiUrl.hostname === 'localhost' ||
|
|
228
|
+
apiUrl.hostname === '127.0.0.1') {
|
|
229
|
+
return 'API URL cannot use localhost (use a publicly accessible URL)';
|
|
230
|
+
}
|
|
231
|
+
if (apiUrl.pathname && !apiUrl.pathname.endsWith('/')) {
|
|
232
|
+
// Warn if path doesn't end with slash but don't fail validation
|
|
233
|
+
// since the client will normalize this
|
|
234
|
+
}
|
|
235
|
+
}
|
|
236
|
+
catch (err) {
|
|
237
|
+
const message = err instanceof Error ? err.message : 'Invalid URL format';
|
|
238
|
+
return `API URL must be a valid URL: ${message}`;
|
|
239
|
+
}
|
|
240
|
+
// Validate WebSocket URL
|
|
241
|
+
if (!input.wsUrl) {
|
|
242
|
+
return 'WebSocket URL is required';
|
|
243
|
+
}
|
|
244
|
+
if (typeof input.wsUrl !== 'string') {
|
|
245
|
+
return 'WebSocket URL must be a string';
|
|
246
|
+
}
|
|
247
|
+
const wsUrlTrimmed = input.wsUrl.trim();
|
|
248
|
+
if (wsUrlTrimmed !== input.wsUrl) {
|
|
249
|
+
return 'WebSocket URL must not have leading or trailing whitespace';
|
|
250
|
+
}
|
|
251
|
+
try {
|
|
252
|
+
const wsUrl = new URL(wsUrlTrimmed);
|
|
253
|
+
if (!['ws:', 'wss:'].includes(wsUrl.protocol)) {
|
|
254
|
+
return 'WebSocket URL must use ws:// or wss:// protocol';
|
|
255
|
+
}
|
|
256
|
+
if (!wsUrl.hostname) {
|
|
257
|
+
return 'WebSocket URL must have a valid hostname';
|
|
258
|
+
}
|
|
259
|
+
if (wsUrl.hostname === 'localhost' || wsUrl.hostname === '127.0.0.1') {
|
|
260
|
+
return 'WebSocket URL cannot use localhost (use a publicly accessible URL)';
|
|
261
|
+
}
|
|
262
|
+
// Recommend wss:// for security if using ws://
|
|
263
|
+
if (wsUrl.protocol === 'ws:' && wsUrl.hostname !== 'localhost') {
|
|
264
|
+
// This is just a recommendation, not a hard failure
|
|
265
|
+
// Most production deployments should use wss://
|
|
266
|
+
}
|
|
267
|
+
}
|
|
268
|
+
catch (err) {
|
|
269
|
+
const message = err instanceof Error ? err.message : 'Invalid URL format';
|
|
270
|
+
return `WebSocket URL must be a valid URL: ${message}`;
|
|
271
|
+
}
|
|
272
|
+
// Validate User ID
|
|
273
|
+
if (!input.userId) {
|
|
274
|
+
return 'User ID is required';
|
|
275
|
+
}
|
|
276
|
+
if (typeof input.userId !== 'string') {
|
|
277
|
+
return 'User ID must be a string';
|
|
278
|
+
}
|
|
279
|
+
const userIdTrimmed = input.userId.trim().toUpperCase();
|
|
280
|
+
if (userIdTrimmed !== input.userId.toUpperCase()) {
|
|
281
|
+
return 'User ID must not have leading or trailing whitespace and should be uppercase';
|
|
282
|
+
}
|
|
283
|
+
if (!/^[UBPT][A-Z0-9]{10}$/.test(userIdTrimmed)) {
|
|
284
|
+
const prefix = userIdTrimmed.charAt(0);
|
|
285
|
+
if (!'UBPT'.includes(prefix)) {
|
|
286
|
+
return 'User ID must start with U, B, P, or T (got: ' + prefix + ')';
|
|
287
|
+
}
|
|
288
|
+
if (userIdTrimmed.length !== 11) {
|
|
289
|
+
return `User ID must be exactly 11 characters long (got: ${userIdTrimmed.length})`;
|
|
290
|
+
}
|
|
291
|
+
return 'User ID must be a valid DeskFree ID format: one letter (U/B/P/T) + 10 alphanumeric characters (e.g. U9QF3C6X1A)';
|
|
292
|
+
}
|
|
293
|
+
return null;
|
|
294
|
+
},
|
|
295
|
+
},
|
|
296
|
+
messaging: {
|
|
297
|
+
normalizeTarget(raw) {
|
|
298
|
+
const trimmed = raw.trim();
|
|
299
|
+
if (/^[UBPT][A-Z0-9]{10}$/i.test(trimmed))
|
|
300
|
+
return `user:${trimmed}`;
|
|
301
|
+
return undefined;
|
|
302
|
+
},
|
|
303
|
+
targetResolver: {
|
|
304
|
+
hint: CHANNEL_HINTS.targetResolver,
|
|
305
|
+
looksLikeId(raw) {
|
|
306
|
+
const t = raw.trim();
|
|
307
|
+
return /^[UBPT][A-Z0-9]{10}$/i.test(t) || /^user:/i.test(t);
|
|
308
|
+
},
|
|
309
|
+
},
|
|
310
|
+
},
|
|
311
|
+
security: {
|
|
312
|
+
resolveDmPolicy({ accountId }) {
|
|
313
|
+
const basePath = accountId && accountId !== 'default'
|
|
314
|
+
? `channels.deskfree.accounts.${accountId}.dm.`
|
|
315
|
+
: 'channels.deskfree.dm.';
|
|
316
|
+
return {
|
|
317
|
+
policy: 'open',
|
|
318
|
+
allowFrom: ['*'],
|
|
319
|
+
allowFromPath: basePath,
|
|
320
|
+
approveHint: 'DeskFree uses bot token auth — no pairing needed.',
|
|
321
|
+
};
|
|
322
|
+
},
|
|
323
|
+
},
|
|
324
|
+
onboarding: {
|
|
325
|
+
channel: 'deskfree',
|
|
326
|
+
async getStatus(ctx) {
|
|
327
|
+
const ch = getChannelConfig(ctx.cfg);
|
|
328
|
+
const hasBotToken = Boolean(ch?.botToken);
|
|
329
|
+
const hasApiUrl = Boolean(ch?.apiUrl);
|
|
330
|
+
const hasUserId = Boolean(ch?.userId);
|
|
331
|
+
const configured = hasBotToken && hasApiUrl && hasUserId;
|
|
332
|
+
const statusLines = [];
|
|
333
|
+
if (configured) {
|
|
334
|
+
statusLines.push('Bot token: configured');
|
|
335
|
+
statusLines.push(`API URL: ${ch.apiUrl}`);
|
|
336
|
+
if (ch.wsUrl)
|
|
337
|
+
statusLines.push(`WS URL: ${ch.wsUrl}`);
|
|
338
|
+
}
|
|
339
|
+
else {
|
|
340
|
+
statusLines.push('Not configured');
|
|
341
|
+
}
|
|
342
|
+
return {
|
|
343
|
+
channel: 'deskfree',
|
|
344
|
+
configured,
|
|
345
|
+
statusLines,
|
|
346
|
+
selectionHint: CHANNEL_HINTS.onboardingSelection,
|
|
347
|
+
quickstartScore: configured ? 0 : 50,
|
|
348
|
+
};
|
|
349
|
+
},
|
|
350
|
+
async configure(ctx) {
|
|
351
|
+
const ch = getChannelConfig(ctx.cfg);
|
|
352
|
+
const accountId = ctx.accountOverrides.accountId ?? 'default';
|
|
353
|
+
const botToken = await ctx.prompter.text({
|
|
354
|
+
message: 'Bot token',
|
|
355
|
+
initialValue: ch?.botToken ?? '',
|
|
356
|
+
placeholder: 'bot_xxxxxxxxxxxxxxxx',
|
|
357
|
+
validate: (v) => {
|
|
358
|
+
if (!v)
|
|
359
|
+
return 'Bot token is required';
|
|
360
|
+
if (!v.startsWith('bot_'))
|
|
361
|
+
return 'Must start with "bot_"';
|
|
362
|
+
if (v.length < 10)
|
|
363
|
+
return 'Bot token appears to be too short';
|
|
364
|
+
return undefined;
|
|
365
|
+
},
|
|
366
|
+
});
|
|
367
|
+
const apiUrl = await ctx.prompter.text({
|
|
368
|
+
message: 'API URL',
|
|
369
|
+
initialValue: ch?.apiUrl ?? '',
|
|
370
|
+
placeholder: 'https://app.deskfree.ai/v1/bot',
|
|
371
|
+
validate: (v) => {
|
|
372
|
+
if (!v)
|
|
373
|
+
return 'API URL is required';
|
|
374
|
+
try {
|
|
375
|
+
const url = new URL(v);
|
|
376
|
+
if (url.protocol !== 'https:')
|
|
377
|
+
return 'Must use HTTPS protocol for security';
|
|
378
|
+
return undefined;
|
|
379
|
+
}
|
|
380
|
+
catch {
|
|
381
|
+
return 'Must be a valid HTTPS URL';
|
|
382
|
+
}
|
|
383
|
+
},
|
|
384
|
+
});
|
|
385
|
+
const wsUrl = await ctx.prompter.text({
|
|
386
|
+
message: 'WebSocket URL',
|
|
387
|
+
initialValue: ch?.wsUrl ?? '',
|
|
388
|
+
placeholder: 'wss://ws.deskfree.ai',
|
|
389
|
+
validate: (v) => {
|
|
390
|
+
if (!v)
|
|
391
|
+
return 'WebSocket URL is required';
|
|
392
|
+
try {
|
|
393
|
+
const url = new URL(v);
|
|
394
|
+
if (!['ws:', 'wss:'].includes(url.protocol)) {
|
|
395
|
+
return 'Must use ws:// or wss:// protocol';
|
|
396
|
+
}
|
|
397
|
+
return undefined;
|
|
398
|
+
}
|
|
399
|
+
catch {
|
|
400
|
+
return 'Must be a valid WebSocket URL';
|
|
401
|
+
}
|
|
402
|
+
},
|
|
403
|
+
});
|
|
404
|
+
const userId = await ctx.prompter.text({
|
|
405
|
+
message: 'Your DeskFree User ID (shown in install instructions)',
|
|
406
|
+
initialValue: ch?.userId ?? '',
|
|
407
|
+
placeholder: 'U9QF3C6X1A',
|
|
408
|
+
validate: (v) => {
|
|
409
|
+
if (!v)
|
|
410
|
+
return 'User ID is required';
|
|
411
|
+
if (!/^[UBPT][A-Z0-9]{10}$/i.test(v)) {
|
|
412
|
+
return 'Must be a valid DeskFree ID (e.g. U9QF3C6X1A)';
|
|
413
|
+
}
|
|
414
|
+
return undefined;
|
|
415
|
+
},
|
|
416
|
+
});
|
|
417
|
+
const updatedCfg = deskFreePlugin.setup.applyAccountConfig({
|
|
418
|
+
cfg: ctx.cfg,
|
|
419
|
+
accountId,
|
|
420
|
+
input: { botToken, apiUrl, wsUrl, userId },
|
|
421
|
+
});
|
|
422
|
+
await ctx.prompter.note('DeskFree channel configured.', 'Done');
|
|
423
|
+
return { cfg: updatedCfg, accountId };
|
|
424
|
+
},
|
|
425
|
+
disable(cfg) {
|
|
426
|
+
const config = JSON.parse(JSON.stringify(cfg));
|
|
427
|
+
const channels = (config.channels ?? {});
|
|
428
|
+
const deskfree = (channels.deskfree ?? {});
|
|
429
|
+
deskfree.enabled = false;
|
|
430
|
+
channels.deskfree = deskfree;
|
|
431
|
+
config.channels = channels;
|
|
432
|
+
return config;
|
|
433
|
+
},
|
|
434
|
+
},
|
|
435
|
+
status: {
|
|
436
|
+
async probeAccount(params) {
|
|
437
|
+
const isConfigured = Boolean(params.account.botToken && params.account.apiUrl);
|
|
438
|
+
if (!isConfigured) {
|
|
439
|
+
return { ok: false, error: STATUS_MESSAGES.probeNotConfigured };
|
|
440
|
+
}
|
|
441
|
+
const client = new DeskFreeClient(params.account.botToken, params.account.apiUrl);
|
|
442
|
+
return client.probe(params.timeoutMs);
|
|
443
|
+
},
|
|
444
|
+
buildAccountSnapshot(params) {
|
|
445
|
+
const isConfigured = Boolean(params.account.botToken && params.account.apiUrl);
|
|
446
|
+
return {
|
|
447
|
+
accountId: params.account.accountId,
|
|
448
|
+
enabled: params.account.enabled,
|
|
449
|
+
configured: isConfigured,
|
|
450
|
+
running: params.runtime?.running,
|
|
451
|
+
lastStartAt: params.runtime?.lastStartAt,
|
|
452
|
+
lastStopAt: params.runtime?.lastStopAt,
|
|
453
|
+
lastError: params.runtime?.lastError,
|
|
454
|
+
probe: params.probe,
|
|
455
|
+
};
|
|
456
|
+
},
|
|
457
|
+
collectStatusIssues(accounts) {
|
|
458
|
+
const issues = [];
|
|
459
|
+
for (const snap of accounts) {
|
|
460
|
+
const id = snap.accountId ?? 'unknown';
|
|
461
|
+
if (!snap.configured) {
|
|
462
|
+
issues.push({
|
|
463
|
+
channel: 'deskfree',
|
|
464
|
+
accountId: id,
|
|
465
|
+
kind: 'config',
|
|
466
|
+
message: STATUS_MESSAGES.notConfigured.message,
|
|
467
|
+
fix: STATUS_MESSAGES.notConfigured.fix,
|
|
468
|
+
});
|
|
469
|
+
continue;
|
|
470
|
+
}
|
|
471
|
+
if (!snap.enabled) {
|
|
472
|
+
issues.push({
|
|
473
|
+
channel: 'deskfree',
|
|
474
|
+
accountId: id,
|
|
475
|
+
kind: 'config',
|
|
476
|
+
message: STATUS_MESSAGES.disabled.message,
|
|
477
|
+
fix: STATUS_MESSAGES.disabled.fix,
|
|
478
|
+
});
|
|
479
|
+
continue;
|
|
480
|
+
}
|
|
481
|
+
if (!snap.running) {
|
|
482
|
+
issues.push({
|
|
483
|
+
channel: 'deskfree',
|
|
484
|
+
accountId: id,
|
|
485
|
+
kind: 'runtime',
|
|
486
|
+
message: STATUS_MESSAGES.notRunning.message,
|
|
487
|
+
});
|
|
488
|
+
}
|
|
489
|
+
if (snap.probe && !snap.probe.ok) {
|
|
490
|
+
issues.push({
|
|
491
|
+
channel: 'deskfree',
|
|
492
|
+
accountId: id,
|
|
493
|
+
kind: 'auth',
|
|
494
|
+
message: `Probe failed: ${snap.probe.error ?? 'unknown error'}`,
|
|
495
|
+
fix: STATUS_MESSAGES.probeFailed.fix,
|
|
496
|
+
});
|
|
497
|
+
}
|
|
498
|
+
}
|
|
499
|
+
return issues;
|
|
500
|
+
},
|
|
501
|
+
},
|
|
502
|
+
};
|
|
503
|
+
//# sourceMappingURL=channel.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"channel.js","sourceRoot":"","sources":["../src/channel.ts"],"names":[],"mappings":"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"}
|