openclaw-productboard 1.0.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/CHANGELOG.md +35 -0
- package/LICENSE +21 -0
- package/README.md +230 -0
- package/dist/client/api-client.d.ts +64 -0
- package/dist/client/api-client.js +379 -0
- package/dist/client/errors.d.ts +51 -0
- package/dist/client/errors.js +128 -0
- package/dist/client/types.d.ts +262 -0
- package/dist/client/types.js +6 -0
- package/dist/index.d.ts +14 -0
- package/dist/index.js +62 -0
- package/dist/tools/features.d.ts +6 -0
- package/dist/tools/features.js +318 -0
- package/dist/tools/index.d.ts +7 -0
- package/dist/tools/index.js +15 -0
- package/dist/tools/notes.d.ts +6 -0
- package/dist/tools/notes.js +176 -0
- package/dist/tools/products.d.ts +6 -0
- package/dist/tools/products.js +148 -0
- package/dist/tools/search.d.ts +6 -0
- package/dist/tools/search.js +116 -0
- package/dist/utils/cache.d.ts +54 -0
- package/dist/utils/cache.js +123 -0
- package/dist/utils/rate-limiter.d.ts +58 -0
- package/dist/utils/rate-limiter.js +118 -0
- package/openclaw.plugin.json +57 -0
- package/package.json +53 -0
- package/skills/productboard-feedback/SKILL.md +105 -0
- package/skills/productboard-release/SKILL.md +146 -0
- package/skills/productboard-search/SKILL.md +62 -0
|
@@ -0,0 +1,318 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
/**
|
|
3
|
+
* ProductBoard Feature Management Tools
|
|
4
|
+
*/
|
|
5
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
6
|
+
exports.createFeatureTools = createFeatureTools;
|
|
7
|
+
function createFeatureTools(client) {
|
|
8
|
+
return [
|
|
9
|
+
// pb_feature_create
|
|
10
|
+
{
|
|
11
|
+
name: 'pb_feature_create',
|
|
12
|
+
description: 'Create a new feature in ProductBoard. Features represent product functionality, user stories, or items in your product backlog.',
|
|
13
|
+
parameters: {
|
|
14
|
+
type: 'object',
|
|
15
|
+
properties: {
|
|
16
|
+
name: {
|
|
17
|
+
type: 'string',
|
|
18
|
+
description: 'Name of the feature (required)',
|
|
19
|
+
},
|
|
20
|
+
description: {
|
|
21
|
+
type: 'string',
|
|
22
|
+
description: 'Detailed description of the feature (supports HTML)',
|
|
23
|
+
},
|
|
24
|
+
status: {
|
|
25
|
+
type: 'string',
|
|
26
|
+
description: 'Feature status',
|
|
27
|
+
enum: ['new', 'in-progress', 'shipped', 'archived', 'postponed', 'candidate'],
|
|
28
|
+
},
|
|
29
|
+
productId: {
|
|
30
|
+
type: 'string',
|
|
31
|
+
description: 'ID of the parent product to assign this feature to',
|
|
32
|
+
},
|
|
33
|
+
componentId: {
|
|
34
|
+
type: 'string',
|
|
35
|
+
description: 'ID of the parent component to assign this feature to',
|
|
36
|
+
},
|
|
37
|
+
parentFeatureId: {
|
|
38
|
+
type: 'string',
|
|
39
|
+
description: 'ID of the parent feature (for sub-features)',
|
|
40
|
+
},
|
|
41
|
+
ownerEmail: {
|
|
42
|
+
type: 'string',
|
|
43
|
+
description: 'Email of the feature owner',
|
|
44
|
+
},
|
|
45
|
+
startDate: {
|
|
46
|
+
type: 'string',
|
|
47
|
+
description: 'Planned start date (ISO 8601 format)',
|
|
48
|
+
},
|
|
49
|
+
endDate: {
|
|
50
|
+
type: 'string',
|
|
51
|
+
description: 'Planned end date (ISO 8601 format)',
|
|
52
|
+
},
|
|
53
|
+
},
|
|
54
|
+
required: ['name'],
|
|
55
|
+
},
|
|
56
|
+
handler: async (params) => {
|
|
57
|
+
const createParams = {
|
|
58
|
+
name: params.name,
|
|
59
|
+
description: params.description,
|
|
60
|
+
status: params.status,
|
|
61
|
+
};
|
|
62
|
+
// Build parent reference
|
|
63
|
+
if (params.productId || params.componentId || params.parentFeatureId) {
|
|
64
|
+
createParams.parent = {};
|
|
65
|
+
if (params.productId)
|
|
66
|
+
createParams.parent.product = { id: params.productId };
|
|
67
|
+
if (params.componentId)
|
|
68
|
+
createParams.parent.component = { id: params.componentId };
|
|
69
|
+
if (params.parentFeatureId)
|
|
70
|
+
createParams.parent.feature = { id: params.parentFeatureId };
|
|
71
|
+
}
|
|
72
|
+
// Set owner
|
|
73
|
+
if (params.ownerEmail) {
|
|
74
|
+
createParams.owner = { email: params.ownerEmail };
|
|
75
|
+
}
|
|
76
|
+
// Set timeframe
|
|
77
|
+
if (params.startDate || params.endDate) {
|
|
78
|
+
createParams.timeframe = {
|
|
79
|
+
startDate: params.startDate,
|
|
80
|
+
endDate: params.endDate,
|
|
81
|
+
};
|
|
82
|
+
}
|
|
83
|
+
const feature = await client.createFeature(createParams);
|
|
84
|
+
return {
|
|
85
|
+
success: true,
|
|
86
|
+
feature: {
|
|
87
|
+
id: feature.id,
|
|
88
|
+
name: feature.name,
|
|
89
|
+
status: feature.status,
|
|
90
|
+
url: feature.links?.html,
|
|
91
|
+
},
|
|
92
|
+
};
|
|
93
|
+
},
|
|
94
|
+
},
|
|
95
|
+
// pb_feature_list
|
|
96
|
+
{
|
|
97
|
+
name: 'pb_feature_list',
|
|
98
|
+
description: 'List features in ProductBoard with optional filters. Returns features from your product backlog.',
|
|
99
|
+
parameters: {
|
|
100
|
+
type: 'object',
|
|
101
|
+
properties: {
|
|
102
|
+
productId: {
|
|
103
|
+
type: 'string',
|
|
104
|
+
description: 'Filter by product ID',
|
|
105
|
+
},
|
|
106
|
+
componentId: {
|
|
107
|
+
type: 'string',
|
|
108
|
+
description: 'Filter by component ID',
|
|
109
|
+
},
|
|
110
|
+
status: {
|
|
111
|
+
type: 'string',
|
|
112
|
+
description: 'Filter by status',
|
|
113
|
+
enum: ['new', 'in-progress', 'shipped', 'archived', 'postponed', 'candidate'],
|
|
114
|
+
},
|
|
115
|
+
ownerId: {
|
|
116
|
+
type: 'string',
|
|
117
|
+
description: 'Filter by owner ID',
|
|
118
|
+
},
|
|
119
|
+
limit: {
|
|
120
|
+
type: 'number',
|
|
121
|
+
description: 'Maximum number of features to return (default: 50, max: 500)',
|
|
122
|
+
default: 50,
|
|
123
|
+
},
|
|
124
|
+
},
|
|
125
|
+
},
|
|
126
|
+
handler: async (params) => {
|
|
127
|
+
const listParams = {
|
|
128
|
+
productId: params.productId,
|
|
129
|
+
componentId: params.componentId,
|
|
130
|
+
status: params.status,
|
|
131
|
+
ownerId: params.ownerId,
|
|
132
|
+
limit: Math.min(params.limit || 50, 500),
|
|
133
|
+
};
|
|
134
|
+
const features = await client.listFeatures(listParams);
|
|
135
|
+
return {
|
|
136
|
+
count: features.length,
|
|
137
|
+
features: features.map((f) => ({
|
|
138
|
+
id: f.id,
|
|
139
|
+
name: f.name,
|
|
140
|
+
status: f.status,
|
|
141
|
+
description: f.description?.substring(0, 200),
|
|
142
|
+
owner: f.owner?.email,
|
|
143
|
+
url: f.links?.html,
|
|
144
|
+
})),
|
|
145
|
+
};
|
|
146
|
+
},
|
|
147
|
+
},
|
|
148
|
+
// pb_feature_get
|
|
149
|
+
{
|
|
150
|
+
name: 'pb_feature_get',
|
|
151
|
+
description: 'Get detailed information about a specific feature by ID.',
|
|
152
|
+
parameters: {
|
|
153
|
+
type: 'object',
|
|
154
|
+
properties: {
|
|
155
|
+
id: {
|
|
156
|
+
type: 'string',
|
|
157
|
+
description: 'Feature ID',
|
|
158
|
+
},
|
|
159
|
+
},
|
|
160
|
+
required: ['id'],
|
|
161
|
+
},
|
|
162
|
+
handler: async (params) => {
|
|
163
|
+
const feature = await client.getFeature(params.id);
|
|
164
|
+
return {
|
|
165
|
+
id: feature.id,
|
|
166
|
+
name: feature.name,
|
|
167
|
+
description: feature.description,
|
|
168
|
+
status: feature.status,
|
|
169
|
+
owner: feature.owner?.email,
|
|
170
|
+
parent: feature.parent,
|
|
171
|
+
timeframe: feature.timeframe,
|
|
172
|
+
createdAt: feature.createdAt,
|
|
173
|
+
updatedAt: feature.updatedAt,
|
|
174
|
+
url: feature.links?.html,
|
|
175
|
+
};
|
|
176
|
+
},
|
|
177
|
+
},
|
|
178
|
+
// pb_feature_update
|
|
179
|
+
{
|
|
180
|
+
name: 'pb_feature_update',
|
|
181
|
+
description: 'Update an existing feature in ProductBoard. Only provided fields will be updated.',
|
|
182
|
+
parameters: {
|
|
183
|
+
type: 'object',
|
|
184
|
+
properties: {
|
|
185
|
+
id: {
|
|
186
|
+
type: 'string',
|
|
187
|
+
description: 'Feature ID to update',
|
|
188
|
+
},
|
|
189
|
+
name: {
|
|
190
|
+
type: 'string',
|
|
191
|
+
description: 'New name for the feature',
|
|
192
|
+
},
|
|
193
|
+
description: {
|
|
194
|
+
type: 'string',
|
|
195
|
+
description: 'New description (supports HTML)',
|
|
196
|
+
},
|
|
197
|
+
status: {
|
|
198
|
+
type: 'string',
|
|
199
|
+
description: 'New status',
|
|
200
|
+
enum: ['new', 'in-progress', 'shipped', 'archived', 'postponed', 'candidate'],
|
|
201
|
+
},
|
|
202
|
+
productId: {
|
|
203
|
+
type: 'string',
|
|
204
|
+
description: 'Move to a different product',
|
|
205
|
+
},
|
|
206
|
+
componentId: {
|
|
207
|
+
type: 'string',
|
|
208
|
+
description: 'Move to a different component',
|
|
209
|
+
},
|
|
210
|
+
ownerEmail: {
|
|
211
|
+
type: 'string',
|
|
212
|
+
description: 'New owner email',
|
|
213
|
+
},
|
|
214
|
+
startDate: {
|
|
215
|
+
type: 'string',
|
|
216
|
+
description: 'New start date (ISO 8601)',
|
|
217
|
+
},
|
|
218
|
+
endDate: {
|
|
219
|
+
type: 'string',
|
|
220
|
+
description: 'New end date (ISO 8601)',
|
|
221
|
+
},
|
|
222
|
+
},
|
|
223
|
+
required: ['id'],
|
|
224
|
+
},
|
|
225
|
+
handler: async (params) => {
|
|
226
|
+
const updateParams = {};
|
|
227
|
+
if (params.name)
|
|
228
|
+
updateParams.name = params.name;
|
|
229
|
+
if (params.description)
|
|
230
|
+
updateParams.description = params.description;
|
|
231
|
+
if (params.status)
|
|
232
|
+
updateParams.status = params.status;
|
|
233
|
+
if (params.productId || params.componentId) {
|
|
234
|
+
updateParams.parent = {};
|
|
235
|
+
if (params.productId)
|
|
236
|
+
updateParams.parent.product = { id: params.productId };
|
|
237
|
+
if (params.componentId)
|
|
238
|
+
updateParams.parent.component = { id: params.componentId };
|
|
239
|
+
}
|
|
240
|
+
if (params.ownerEmail) {
|
|
241
|
+
updateParams.owner = { email: params.ownerEmail };
|
|
242
|
+
}
|
|
243
|
+
if (params.startDate || params.endDate) {
|
|
244
|
+
updateParams.timeframe = {
|
|
245
|
+
startDate: params.startDate,
|
|
246
|
+
endDate: params.endDate,
|
|
247
|
+
};
|
|
248
|
+
}
|
|
249
|
+
const feature = await client.updateFeature(params.id, updateParams);
|
|
250
|
+
return {
|
|
251
|
+
success: true,
|
|
252
|
+
feature: {
|
|
253
|
+
id: feature.id,
|
|
254
|
+
name: feature.name,
|
|
255
|
+
status: feature.status,
|
|
256
|
+
url: feature.links?.html,
|
|
257
|
+
},
|
|
258
|
+
};
|
|
259
|
+
},
|
|
260
|
+
},
|
|
261
|
+
// pb_feature_delete
|
|
262
|
+
{
|
|
263
|
+
name: 'pb_feature_delete',
|
|
264
|
+
description: 'Archive/delete a feature from ProductBoard. This action can be undone in ProductBoard.',
|
|
265
|
+
parameters: {
|
|
266
|
+
type: 'object',
|
|
267
|
+
properties: {
|
|
268
|
+
id: {
|
|
269
|
+
type: 'string',
|
|
270
|
+
description: 'Feature ID to delete',
|
|
271
|
+
},
|
|
272
|
+
},
|
|
273
|
+
required: ['id'],
|
|
274
|
+
},
|
|
275
|
+
handler: async (params) => {
|
|
276
|
+
await client.deleteFeature(params.id);
|
|
277
|
+
return {
|
|
278
|
+
success: true,
|
|
279
|
+
message: `Feature ${params.id} has been archived`,
|
|
280
|
+
};
|
|
281
|
+
},
|
|
282
|
+
},
|
|
283
|
+
// pb_feature_search
|
|
284
|
+
{
|
|
285
|
+
name: 'pb_feature_search',
|
|
286
|
+
description: 'Search for features by name or description. Returns matching features ordered by relevance.',
|
|
287
|
+
parameters: {
|
|
288
|
+
type: 'object',
|
|
289
|
+
properties: {
|
|
290
|
+
query: {
|
|
291
|
+
type: 'string',
|
|
292
|
+
description: 'Search query (searches name and description)',
|
|
293
|
+
},
|
|
294
|
+
limit: {
|
|
295
|
+
type: 'number',
|
|
296
|
+
description: 'Maximum results to return (default: 20)',
|
|
297
|
+
default: 20,
|
|
298
|
+
},
|
|
299
|
+
},
|
|
300
|
+
required: ['query'],
|
|
301
|
+
},
|
|
302
|
+
handler: async (params) => {
|
|
303
|
+
const features = await client.searchFeatures(params.query, params.limit || 20);
|
|
304
|
+
return {
|
|
305
|
+
count: features.length,
|
|
306
|
+
features: features.map((f) => ({
|
|
307
|
+
id: f.id,
|
|
308
|
+
name: f.name,
|
|
309
|
+
status: f.status,
|
|
310
|
+
description: f.description?.substring(0, 200),
|
|
311
|
+
url: f.links?.html,
|
|
312
|
+
})),
|
|
313
|
+
};
|
|
314
|
+
},
|
|
315
|
+
},
|
|
316
|
+
];
|
|
317
|
+
}
|
|
318
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"features.js","sourceRoot":"","sources":["../../src/tools/features.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * ProductBoard Feature Management Tools\n */\n\nimport { ProductBoardClient } from '../client/api-client';\nimport {\n  ToolDefinition,\n  CreateFeatureParams,\n  UpdateFeatureParams,\n  ListFeaturesParams,\n  FeatureStatus,\n} from '../client/types';\n\nexport function createFeatureTools(client: ProductBoardClient): ToolDefinition[] {\n  return [\n    // pb_feature_create\n    {\n      name: 'pb_feature_create',\n      description:\n        'Create a new feature in ProductBoard. Features represent product functionality, user stories, or items in your product backlog.',\n      parameters: {\n        type: 'object',\n        properties: {\n          name: {\n            type: 'string',\n            description: 'Name of the feature (required)',\n          },\n          description: {\n            type: 'string',\n            description: 'Detailed description of the feature (supports HTML)',\n          },\n          status: {\n            type: 'string',\n            description: 'Feature status',\n            enum: ['new', 'in-progress', 'shipped', 'archived', 'postponed', 'candidate'],\n          },\n          productId: {\n            type: 'string',\n            description: 'ID of the parent product to assign this feature to',\n          },\n          componentId: {\n            type: 'string',\n            description: 'ID of the parent component to assign this feature to',\n          },\n          parentFeatureId: {\n            type: 'string',\n            description: 'ID of the parent feature (for sub-features)',\n          },\n          ownerEmail: {\n            type: 'string',\n            description: 'Email of the feature owner',\n          },\n          startDate: {\n            type: 'string',\n            description: 'Planned start date (ISO 8601 format)',\n          },\n          endDate: {\n            type: 'string',\n            description: 'Planned end date (ISO 8601 format)',\n          },\n        },\n        required: ['name'],\n      },\n      handler: async (params) => {\n        const createParams: CreateFeatureParams = {\n          name: params.name as string,\n          description: params.description as string | undefined,\n          status: params.status as FeatureStatus | undefined,\n        };\n\n        // Build parent reference\n        if (params.productId || params.componentId || params.parentFeatureId) {\n          createParams.parent = {};\n          if (params.productId) createParams.parent.product = { id: params.productId as string };\n          if (params.componentId) createParams.parent.component = { id: params.componentId as string };\n          if (params.parentFeatureId) createParams.parent.feature = { id: params.parentFeatureId as string };\n        }\n\n        // Set owner\n        if (params.ownerEmail) {\n          createParams.owner = { email: params.ownerEmail as string };\n        }\n\n        // Set timeframe\n        if (params.startDate || params.endDate) {\n          createParams.timeframe = {\n            startDate: params.startDate as string | undefined,\n            endDate: params.endDate as string | undefined,\n          };\n        }\n\n        const feature = await client.createFeature(createParams);\n        return {\n          success: true,\n          feature: {\n            id: feature.id,\n            name: feature.name,\n            status: feature.status,\n            url: feature.links?.html,\n          },\n        };\n      },\n    },\n\n    // pb_feature_list\n    {\n      name: 'pb_feature_list',\n      description:\n        'List features in ProductBoard with optional filters. Returns features from your product backlog.',\n      parameters: {\n        type: 'object',\n        properties: {\n          productId: {\n            type: 'string',\n            description: 'Filter by product ID',\n          },\n          componentId: {\n            type: 'string',\n            description: 'Filter by component ID',\n          },\n          status: {\n            type: 'string',\n            description: 'Filter by status',\n            enum: ['new', 'in-progress', 'shipped', 'archived', 'postponed', 'candidate'],\n          },\n          ownerId: {\n            type: 'string',\n            description: 'Filter by owner ID',\n          },\n          limit: {\n            type: 'number',\n            description: 'Maximum number of features to return (default: 50, max: 500)',\n            default: 50,\n          },\n        },\n      },\n      handler: async (params) => {\n        const listParams: ListFeaturesParams = {\n          productId: params.productId as string | undefined,\n          componentId: params.componentId as string | undefined,\n          status: params.status as FeatureStatus | undefined,\n          ownerId: params.ownerId as string | undefined,\n          limit: Math.min(params.limit as number || 50, 500),\n        };\n\n        const features = await client.listFeatures(listParams);\n        return {\n          count: features.length,\n          features: features.map((f) => ({\n            id: f.id,\n            name: f.name,\n            status: f.status,\n            description: f.description?.substring(0, 200),\n            owner: f.owner?.email,\n            url: f.links?.html,\n          })),\n        };\n      },\n    },\n\n    // pb_feature_get\n    {\n      name: 'pb_feature_get',\n      description:\n        'Get detailed information about a specific feature by ID.',\n      parameters: {\n        type: 'object',\n        properties: {\n          id: {\n            type: 'string',\n            description: 'Feature ID',\n          },\n        },\n        required: ['id'],\n      },\n      handler: async (params) => {\n        const feature = await client.getFeature(params.id as string);\n        return {\n          id: feature.id,\n          name: feature.name,\n          description: feature.description,\n          status: feature.status,\n          owner: feature.owner?.email,\n          parent: feature.parent,\n          timeframe: feature.timeframe,\n          createdAt: feature.createdAt,\n          updatedAt: feature.updatedAt,\n          url: feature.links?.html,\n        };\n      },\n    },\n\n    // pb_feature_update\n    {\n      name: 'pb_feature_update',\n      description:\n        'Update an existing feature in ProductBoard. Only provided fields will be updated.',\n      parameters: {\n        type: 'object',\n        properties: {\n          id: {\n            type: 'string',\n            description: 'Feature ID to update',\n          },\n          name: {\n            type: 'string',\n            description: 'New name for the feature',\n          },\n          description: {\n            type: 'string',\n            description: 'New description (supports HTML)',\n          },\n          status: {\n            type: 'string',\n            description: 'New status',\n            enum: ['new', 'in-progress', 'shipped', 'archived', 'postponed', 'candidate'],\n          },\n          productId: {\n            type: 'string',\n            description: 'Move to a different product',\n          },\n          componentId: {\n            type: 'string',\n            description: 'Move to a different component',\n          },\n          ownerEmail: {\n            type: 'string',\n            description: 'New owner email',\n          },\n          startDate: {\n            type: 'string',\n            description: 'New start date (ISO 8601)',\n          },\n          endDate: {\n            type: 'string',\n            description: 'New end date (ISO 8601)',\n          },\n        },\n        required: ['id'],\n      },\n      handler: async (params) => {\n        const updateParams: UpdateFeatureParams = {};\n\n        if (params.name) updateParams.name = params.name as string;\n        if (params.description) updateParams.description = params.description as string;\n        if (params.status) updateParams.status = params.status as FeatureStatus;\n\n        if (params.productId || params.componentId) {\n          updateParams.parent = {};\n          if (params.productId) updateParams.parent.product = { id: params.productId as string };\n          if (params.componentId) updateParams.parent.component = { id: params.componentId as string };\n        }\n\n        if (params.ownerEmail) {\n          updateParams.owner = { email: params.ownerEmail as string };\n        }\n\n        if (params.startDate || params.endDate) {\n          updateParams.timeframe = {\n            startDate: params.startDate as string | undefined,\n            endDate: params.endDate as string | undefined,\n          };\n        }\n\n        const feature = await client.updateFeature(params.id as string, updateParams);\n        return {\n          success: true,\n          feature: {\n            id: feature.id,\n            name: feature.name,\n            status: feature.status,\n            url: feature.links?.html,\n          },\n        };\n      },\n    },\n\n    // pb_feature_delete\n    {\n      name: 'pb_feature_delete',\n      description:\n        'Archive/delete a feature from ProductBoard. This action can be undone in ProductBoard.',\n      parameters: {\n        type: 'object',\n        properties: {\n          id: {\n            type: 'string',\n            description: 'Feature ID to delete',\n          },\n        },\n        required: ['id'],\n      },\n      handler: async (params) => {\n        await client.deleteFeature(params.id as string);\n        return {\n          success: true,\n          message: `Feature ${params.id} has been archived`,\n        };\n      },\n    },\n\n    // pb_feature_search\n    {\n      name: 'pb_feature_search',\n      description:\n        'Search for features by name or description. Returns matching features ordered by relevance.',\n      parameters: {\n        type: 'object',\n        properties: {\n          query: {\n            type: 'string',\n            description: 'Search query (searches name and description)',\n          },\n          limit: {\n            type: 'number',\n            description: 'Maximum results to return (default: 20)',\n            default: 20,\n          },\n        },\n        required: ['query'],\n      },\n      handler: async (params) => {\n        const features = await client.searchFeatures(\n          params.query as string,\n          params.limit as number || 20\n        );\n        return {\n          count: features.length,\n          features: features.map((f) => ({\n            id: f.id,\n            name: f.name,\n            status: f.status,\n            description: f.description?.substring(0, 200),\n            url: f.links?.html,\n          })),\n        };\n      },\n    },\n  ];\n}\n"]}
|
|
@@ -0,0 +1,15 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
/**
|
|
3
|
+
* ProductBoard Tools - Export all tool creators
|
|
4
|
+
*/
|
|
5
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
6
|
+
exports.createSearchTools = exports.createNoteTools = exports.createProductTools = exports.createFeatureTools = void 0;
|
|
7
|
+
var features_1 = require("./features");
|
|
8
|
+
Object.defineProperty(exports, "createFeatureTools", { enumerable: true, get: function () { return features_1.createFeatureTools; } });
|
|
9
|
+
var products_1 = require("./products");
|
|
10
|
+
Object.defineProperty(exports, "createProductTools", { enumerable: true, get: function () { return products_1.createProductTools; } });
|
|
11
|
+
var notes_1 = require("./notes");
|
|
12
|
+
Object.defineProperty(exports, "createNoteTools", { enumerable: true, get: function () { return notes_1.createNoteTools; } });
|
|
13
|
+
var search_1 = require("./search");
|
|
14
|
+
Object.defineProperty(exports, "createSearchTools", { enumerable: true, get: function () { return search_1.createSearchTools; } });
|
|
15
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi9zcmMvdG9vbHMvaW5kZXgudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IjtBQUFBOztHQUVHOzs7QUFFSCx1Q0FBZ0Q7QUFBdkMsOEdBQUEsa0JBQWtCLE9BQUE7QUFDM0IsdUNBQWdEO0FBQXZDLDhHQUFBLGtCQUFrQixPQUFBO0FBQzNCLGlDQUEwQztBQUFqQyx3R0FBQSxlQUFlLE9BQUE7QUFDeEIsbUNBQTZDO0FBQXBDLDJHQUFBLGlCQUFpQixPQUFBIiwic291cmNlc0NvbnRlbnQiOlsiLyoqXG4gKiBQcm9kdWN0Qm9hcmQgVG9vbHMgLSBFeHBvcnQgYWxsIHRvb2wgY3JlYXRvcnNcbiAqL1xuXG5leHBvcnQgeyBjcmVhdGVGZWF0dXJlVG9vbHMgfSBmcm9tICcuL2ZlYXR1cmVzJztcbmV4cG9ydCB7IGNyZWF0ZVByb2R1Y3RUb29scyB9IGZyb20gJy4vcHJvZHVjdHMnO1xuZXhwb3J0IHsgY3JlYXRlTm90ZVRvb2xzIH0gZnJvbSAnLi9ub3Rlcyc7XG5leHBvcnQgeyBjcmVhdGVTZWFyY2hUb29scyB9IGZyb20gJy4vc2VhcmNoJztcbiJdfQ==
|
|
@@ -0,0 +1,176 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
/**
|
|
3
|
+
* ProductBoard Note/Feedback Management Tools
|
|
4
|
+
*/
|
|
5
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
6
|
+
exports.createNoteTools = createNoteTools;
|
|
7
|
+
function createNoteTools(client) {
|
|
8
|
+
return [
|
|
9
|
+
// pb_note_create
|
|
10
|
+
{
|
|
11
|
+
name: 'pb_note_create',
|
|
12
|
+
description: 'Create a customer feedback note in ProductBoard. Notes capture customer insights, feature requests, or user feedback that can be linked to features.',
|
|
13
|
+
parameters: {
|
|
14
|
+
type: 'object',
|
|
15
|
+
properties: {
|
|
16
|
+
content: {
|
|
17
|
+
type: 'string',
|
|
18
|
+
description: 'The note content/feedback text (required, supports HTML)',
|
|
19
|
+
},
|
|
20
|
+
title: {
|
|
21
|
+
type: 'string',
|
|
22
|
+
description: 'Optional title for the note',
|
|
23
|
+
},
|
|
24
|
+
displayUrl: {
|
|
25
|
+
type: 'string',
|
|
26
|
+
description: 'URL to the original source (e.g., support ticket, Slack message)',
|
|
27
|
+
},
|
|
28
|
+
sourceOrigin: {
|
|
29
|
+
type: 'string',
|
|
30
|
+
description: 'Origin system identifier (e.g., "zendesk", "intercom", "slack")',
|
|
31
|
+
},
|
|
32
|
+
sourceRecordId: {
|
|
33
|
+
type: 'string',
|
|
34
|
+
description: 'Record ID in the origin system',
|
|
35
|
+
},
|
|
36
|
+
userEmail: {
|
|
37
|
+
type: 'string',
|
|
38
|
+
description: 'Email of the user who provided the feedback',
|
|
39
|
+
},
|
|
40
|
+
userName: {
|
|
41
|
+
type: 'string',
|
|
42
|
+
description: 'Name of the user who provided the feedback',
|
|
43
|
+
},
|
|
44
|
+
companyName: {
|
|
45
|
+
type: 'string',
|
|
46
|
+
description: 'Company/organization name of the feedback source',
|
|
47
|
+
},
|
|
48
|
+
companyId: {
|
|
49
|
+
type: 'string',
|
|
50
|
+
description: 'External company ID',
|
|
51
|
+
},
|
|
52
|
+
tags: {
|
|
53
|
+
type: 'array',
|
|
54
|
+
description: 'Array of tags to categorize the note',
|
|
55
|
+
items: { type: 'string' },
|
|
56
|
+
},
|
|
57
|
+
},
|
|
58
|
+
required: ['content'],
|
|
59
|
+
},
|
|
60
|
+
handler: async (params) => {
|
|
61
|
+
const createParams = {
|
|
62
|
+
content: params.content,
|
|
63
|
+
title: params.title,
|
|
64
|
+
displayUrl: params.displayUrl,
|
|
65
|
+
tags: params.tags,
|
|
66
|
+
};
|
|
67
|
+
// Set source if provided
|
|
68
|
+
if (params.sourceOrigin || params.sourceRecordId) {
|
|
69
|
+
createParams.source = {
|
|
70
|
+
origin: params.sourceOrigin,
|
|
71
|
+
record_id: params.sourceRecordId,
|
|
72
|
+
};
|
|
73
|
+
}
|
|
74
|
+
// Set user if provided
|
|
75
|
+
if (params.userEmail) {
|
|
76
|
+
createParams.user = {
|
|
77
|
+
email: params.userEmail,
|
|
78
|
+
name: params.userName,
|
|
79
|
+
};
|
|
80
|
+
}
|
|
81
|
+
// Set company if provided
|
|
82
|
+
if (params.companyName || params.companyId) {
|
|
83
|
+
createParams.company = {
|
|
84
|
+
name: params.companyName,
|
|
85
|
+
id: params.companyId,
|
|
86
|
+
};
|
|
87
|
+
}
|
|
88
|
+
const note = await client.createNote(createParams);
|
|
89
|
+
return {
|
|
90
|
+
success: true,
|
|
91
|
+
note: {
|
|
92
|
+
id: note.id,
|
|
93
|
+
title: note.title,
|
|
94
|
+
content: note.content?.substring(0, 200),
|
|
95
|
+
user: note.user?.email,
|
|
96
|
+
company: note.company?.name,
|
|
97
|
+
tags: note.tags,
|
|
98
|
+
url: note.links?.html,
|
|
99
|
+
},
|
|
100
|
+
};
|
|
101
|
+
},
|
|
102
|
+
},
|
|
103
|
+
// pb_note_list
|
|
104
|
+
{
|
|
105
|
+
name: 'pb_note_list',
|
|
106
|
+
description: 'List customer feedback notes in ProductBoard. Can filter by date range.',
|
|
107
|
+
parameters: {
|
|
108
|
+
type: 'object',
|
|
109
|
+
properties: {
|
|
110
|
+
limit: {
|
|
111
|
+
type: 'number',
|
|
112
|
+
description: 'Maximum number of notes to return (default: 50)',
|
|
113
|
+
default: 50,
|
|
114
|
+
},
|
|
115
|
+
createdFrom: {
|
|
116
|
+
type: 'string',
|
|
117
|
+
description: 'Filter notes created on or after this date (ISO 8601)',
|
|
118
|
+
},
|
|
119
|
+
createdTo: {
|
|
120
|
+
type: 'string',
|
|
121
|
+
description: 'Filter notes created on or before this date (ISO 8601)',
|
|
122
|
+
},
|
|
123
|
+
},
|
|
124
|
+
},
|
|
125
|
+
handler: async (params) => {
|
|
126
|
+
const listParams = {
|
|
127
|
+
limit: params.limit || 50,
|
|
128
|
+
createdFrom: params.createdFrom,
|
|
129
|
+
createdTo: params.createdTo,
|
|
130
|
+
};
|
|
131
|
+
const notes = await client.listNotes(listParams);
|
|
132
|
+
return {
|
|
133
|
+
count: notes.length,
|
|
134
|
+
notes: notes.map((n) => ({
|
|
135
|
+
id: n.id,
|
|
136
|
+
title: n.title,
|
|
137
|
+
content: n.content?.substring(0, 200),
|
|
138
|
+
user: n.user?.email,
|
|
139
|
+
company: n.company?.name,
|
|
140
|
+
tags: n.tags,
|
|
141
|
+
featureCount: n.features?.length || 0,
|
|
142
|
+
createdAt: n.createdAt,
|
|
143
|
+
url: n.links?.html,
|
|
144
|
+
})),
|
|
145
|
+
};
|
|
146
|
+
},
|
|
147
|
+
},
|
|
148
|
+
// pb_note_attach
|
|
149
|
+
{
|
|
150
|
+
name: 'pb_note_attach',
|
|
151
|
+
description: 'Attach a note to a feature. This links customer feedback to a specific feature for insights aggregation.',
|
|
152
|
+
parameters: {
|
|
153
|
+
type: 'object',
|
|
154
|
+
properties: {
|
|
155
|
+
noteId: {
|
|
156
|
+
type: 'string',
|
|
157
|
+
description: 'ID of the note to attach',
|
|
158
|
+
},
|
|
159
|
+
featureId: {
|
|
160
|
+
type: 'string',
|
|
161
|
+
description: 'ID of the feature to attach the note to',
|
|
162
|
+
},
|
|
163
|
+
},
|
|
164
|
+
required: ['noteId', 'featureId'],
|
|
165
|
+
},
|
|
166
|
+
handler: async (params) => {
|
|
167
|
+
await client.attachNoteToFeature(params.noteId, params.featureId);
|
|
168
|
+
return {
|
|
169
|
+
success: true,
|
|
170
|
+
message: `Note ${params.noteId} has been attached to feature ${params.featureId}`,
|
|
171
|
+
};
|
|
172
|
+
},
|
|
173
|
+
},
|
|
174
|
+
];
|
|
175
|
+
}
|
|
176
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"notes.js","sourceRoot":"","sources":["../../src/tools/notes.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * ProductBoard Note/Feedback Management Tools\n */\n\nimport { ProductBoardClient } from '../client/api-client';\nimport { ToolDefinition, CreateNoteParams, ListNotesParams } from '../client/types';\n\nexport function createNoteTools(client: ProductBoardClient): ToolDefinition[] {\n  return [\n    // pb_note_create\n    {\n      name: 'pb_note_create',\n      description:\n        'Create a customer feedback note in ProductBoard. Notes capture customer insights, feature requests, or user feedback that can be linked to features.',\n      parameters: {\n        type: 'object',\n        properties: {\n          content: {\n            type: 'string',\n            description: 'The note content/feedback text (required, supports HTML)',\n          },\n          title: {\n            type: 'string',\n            description: 'Optional title for the note',\n          },\n          displayUrl: {\n            type: 'string',\n            description: 'URL to the original source (e.g., support ticket, Slack message)',\n          },\n          sourceOrigin: {\n            type: 'string',\n            description: 'Origin system identifier (e.g., \"zendesk\", \"intercom\", \"slack\")',\n          },\n          sourceRecordId: {\n            type: 'string',\n            description: 'Record ID in the origin system',\n          },\n          userEmail: {\n            type: 'string',\n            description: 'Email of the user who provided the feedback',\n          },\n          userName: {\n            type: 'string',\n            description: 'Name of the user who provided the feedback',\n          },\n          companyName: {\n            type: 'string',\n            description: 'Company/organization name of the feedback source',\n          },\n          companyId: {\n            type: 'string',\n            description: 'External company ID',\n          },\n          tags: {\n            type: 'array',\n            description: 'Array of tags to categorize the note',\n            items: { type: 'string' },\n          },\n        },\n        required: ['content'],\n      },\n      handler: async (params) => {\n        const createParams: CreateNoteParams = {\n          content: params.content as string,\n          title: params.title as string | undefined,\n          displayUrl: params.displayUrl as string | undefined,\n          tags: params.tags as string[] | undefined,\n        };\n\n        // Set source if provided\n        if (params.sourceOrigin || params.sourceRecordId) {\n          createParams.source = {\n            origin: params.sourceOrigin as string | undefined,\n            record_id: params.sourceRecordId as string | undefined,\n          };\n        }\n\n        // Set user if provided\n        if (params.userEmail) {\n          createParams.user = {\n            email: params.userEmail as string,\n            name: params.userName as string | undefined,\n          };\n        }\n\n        // Set company if provided\n        if (params.companyName || params.companyId) {\n          createParams.company = {\n            name: params.companyName as string | undefined,\n            id: params.companyId as string | undefined,\n          };\n        }\n\n        const note = await client.createNote(createParams);\n        return {\n          success: true,\n          note: {\n            id: note.id,\n            title: note.title,\n            content: note.content?.substring(0, 200),\n            user: note.user?.email,\n            company: note.company?.name,\n            tags: note.tags,\n            url: note.links?.html,\n          },\n        };\n      },\n    },\n\n    // pb_note_list\n    {\n      name: 'pb_note_list',\n      description:\n        'List customer feedback notes in ProductBoard. Can filter by date range.',\n      parameters: {\n        type: 'object',\n        properties: {\n          limit: {\n            type: 'number',\n            description: 'Maximum number of notes to return (default: 50)',\n            default: 50,\n          },\n          createdFrom: {\n            type: 'string',\n            description: 'Filter notes created on or after this date (ISO 8601)',\n          },\n          createdTo: {\n            type: 'string',\n            description: 'Filter notes created on or before this date (ISO 8601)',\n          },\n        },\n      },\n      handler: async (params) => {\n        const listParams: ListNotesParams = {\n          limit: params.limit as number || 50,\n          createdFrom: params.createdFrom as string | undefined,\n          createdTo: params.createdTo as string | undefined,\n        };\n\n        const notes = await client.listNotes(listParams);\n        return {\n          count: notes.length,\n          notes: notes.map((n) => ({\n            id: n.id,\n            title: n.title,\n            content: n.content?.substring(0, 200),\n            user: n.user?.email,\n            company: n.company?.name,\n            tags: n.tags,\n            featureCount: n.features?.length || 0,\n            createdAt: n.createdAt,\n            url: n.links?.html,\n          })),\n        };\n      },\n    },\n\n    // pb_note_attach\n    {\n      name: 'pb_note_attach',\n      description:\n        'Attach a note to a feature. This links customer feedback to a specific feature for insights aggregation.',\n      parameters: {\n        type: 'object',\n        properties: {\n          noteId: {\n            type: 'string',\n            description: 'ID of the note to attach',\n          },\n          featureId: {\n            type: 'string',\n            description: 'ID of the feature to attach the note to',\n          },\n        },\n        required: ['noteId', 'featureId'],\n      },\n      handler: async (params) => {\n        await client.attachNoteToFeature(\n          params.noteId as string,\n          params.featureId as string\n        );\n        return {\n          success: true,\n          message: `Note ${params.noteId} has been attached to feature ${params.featureId}`,\n        };\n      },\n    },\n  ];\n}\n"]}
|