openclaw-productboard 1.0.1 → 1.0.3

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.
@@ -1,7 +1,7 @@
1
1
  /**
2
2
  * ProductBoard API Client
3
3
  */
4
- import { Feature, Product, Component, Note, User, CurrentUser, SearchResult, CreateFeatureParams, UpdateFeatureParams, ListFeaturesParams, ListProductsParams, CreateNoteParams, ListNotesParams, ListUsersParams, SearchParams, PluginConfig, ProductHierarchy } from './types';
4
+ import { Feature, Product, Component, Note, User, CurrentUser, SearchResult, CreateFeatureParams, UpdateFeatureParams, ListFeaturesParams, ListProductsParams, CreateNoteParams, ListNotesParams, ListUsersParams, SearchParams, PluginConfig, ProductHierarchy } from './types.js';
5
5
  export declare class ProductBoardClient {
6
6
  private client;
7
7
  private cache;
@@ -1,26 +1,20 @@
1
- "use strict";
2
1
  /**
3
2
  * ProductBoard API Client
4
3
  */
5
- var __importDefault = (this && this.__importDefault) || function (mod) {
6
- return (mod && mod.__esModule) ? mod : { "default": mod };
7
- };
8
- Object.defineProperty(exports, "__esModule", { value: true });
9
- exports.ProductBoardClient = void 0;
10
- const axios_1 = __importDefault(require("axios"));
11
- const errors_1 = require("./errors");
12
- const cache_1 = require("../utils/cache");
13
- const rate_limiter_1 = require("../utils/rate-limiter");
4
+ import axios from 'axios';
5
+ import { parseApiError, isRetryableError, getRetryDelay, ProductBoardError, } from './errors.js';
6
+ import { ApiCache, getCache } from '../utils/cache.js';
7
+ import { getRateLimiter } from '../utils/rate-limiter.js';
14
8
  const DEFAULT_BASE_URL = 'https://api.productboard.com';
15
9
  const MAX_RETRIES = 3;
16
- class ProductBoardClient {
10
+ export class ProductBoardClient {
17
11
  client;
18
12
  cache;
19
13
  rateLimiter;
20
14
  baseUrl;
21
15
  constructor(config) {
22
16
  this.baseUrl = config.apiBaseUrl || DEFAULT_BASE_URL;
23
- this.client = axios_1.default.create({
17
+ this.client = axios.create({
24
18
  baseURL: this.baseUrl,
25
19
  headers: {
26
20
  'Authorization': `Bearer ${config.apiToken}`,
@@ -29,10 +23,10 @@ class ProductBoardClient {
29
23
  },
30
24
  timeout: 30000,
31
25
  });
32
- this.cache = (0, cache_1.getCache)({
26
+ this.cache = getCache({
33
27
  ttl: (config.cacheTtlSeconds || 300) * 1000,
34
28
  });
35
- this.rateLimiter = (0, rate_limiter_1.getRateLimiter)({
29
+ this.rateLimiter = getRateLimiter({
36
30
  maxTokens: config.rateLimitPerMinute || 100,
37
31
  });
38
32
  this.setupInterceptors();
@@ -42,9 +36,9 @@ class ProductBoardClient {
42
36
  this.client.interceptors.response.use((response) => response, (error) => {
43
37
  if (error.response) {
44
38
  const { status, data, headers } = error.response;
45
- throw (0, errors_1.parseApiError)(status, data, headers['retry-after']);
39
+ throw parseApiError(status, data, headers['retry-after']);
46
40
  }
47
- throw new errors_1.ProductBoardError(error.message || 'Network error', 'NETWORK_ERROR', 0);
41
+ throw new ProductBoardError(error.message || 'Network error', 'NETWORK_ERROR', 0);
48
42
  });
49
43
  }
50
44
  /**
@@ -57,8 +51,8 @@ class ProductBoardClient {
57
51
  return response.data;
58
52
  }
59
53
  catch (error) {
60
- if ((0, errors_1.isRetryableError)(error) && retryCount < MAX_RETRIES) {
61
- const delay = (0, errors_1.getRetryDelay)(error, retryCount);
54
+ if (isRetryableError(error) && retryCount < MAX_RETRIES) {
55
+ const delay = getRetryDelay(error, retryCount);
62
56
  await this.sleep(delay);
63
57
  return this.request(config, retryCount + 1);
64
58
  }
@@ -106,7 +100,7 @@ class ProductBoardClient {
106
100
  return result.data;
107
101
  }
108
102
  async listFeatures(params = {}) {
109
- const cacheKey = cache_1.ApiCache.generateKey('pb_feature_list', params);
103
+ const cacheKey = ApiCache.generateKey('pb_feature_list', params);
110
104
  return this.cache.wrap(cacheKey, async () => {
111
105
  const queryParams = {};
112
106
  if (params.productId)
@@ -121,7 +115,7 @@ class ProductBoardClient {
121
115
  });
122
116
  }
123
117
  async getFeature(id) {
124
- const cacheKey = cache_1.ApiCache.generateKey('pb_feature_get', { id });
118
+ const cacheKey = ApiCache.generateKey('pb_feature_get', { id });
125
119
  return this.cache.wrap(cacheKey, async () => {
126
120
  const result = await this.request({
127
121
  method: 'GET',
@@ -137,7 +131,7 @@ class ProductBoardClient {
137
131
  data: { data: params },
138
132
  });
139
133
  // Invalidate caches
140
- this.cache.delete(cache_1.ApiCache.generateKey('pb_feature_get', { id }));
134
+ this.cache.delete(ApiCache.generateKey('pb_feature_get', { id }));
141
135
  this.cache.invalidatePattern('pb_feature_list:');
142
136
  this.cache.invalidatePattern('pb_feature_search:');
143
137
  return result.data;
@@ -148,12 +142,12 @@ class ProductBoardClient {
148
142
  url: `/features/${id}`,
149
143
  });
150
144
  // Invalidate caches
151
- this.cache.delete(cache_1.ApiCache.generateKey('pb_feature_get', { id }));
145
+ this.cache.delete(ApiCache.generateKey('pb_feature_get', { id }));
152
146
  this.cache.invalidatePattern('pb_feature_list:');
153
147
  this.cache.invalidatePattern('pb_feature_search:');
154
148
  }
155
149
  async searchFeatures(query, limit = 50) {
156
- const cacheKey = cache_1.ApiCache.generateKey('pb_feature_search', { query, limit });
150
+ const cacheKey = ApiCache.generateKey('pb_feature_search', { query, limit });
157
151
  return this.cache.wrap(cacheKey, async () => {
158
152
  // ProductBoard doesn't have a dedicated search endpoint for features
159
153
  // We'll list all and filter client-side
@@ -172,13 +166,13 @@ class ProductBoardClient {
172
166
  // Product Methods
173
167
  // ============================================
174
168
  async listProducts(params = {}) {
175
- const cacheKey = cache_1.ApiCache.generateKey('pb_product_list', params);
169
+ const cacheKey = ApiCache.generateKey('pb_product_list', params);
176
170
  return this.cache.wrap(cacheKey, async () => {
177
171
  return this.paginate('/products', {}, params.limit);
178
172
  });
179
173
  }
180
174
  async getProduct(id) {
181
- const cacheKey = cache_1.ApiCache.generateKey('pb_product_get', { id });
175
+ const cacheKey = ApiCache.generateKey('pb_product_get', { id });
182
176
  return this.cache.wrap(cacheKey, async () => {
183
177
  const result = await this.request({
184
178
  method: 'GET',
@@ -188,7 +182,7 @@ class ProductBoardClient {
188
182
  });
189
183
  }
190
184
  async listComponents(params = {}) {
191
- const cacheKey = cache_1.ApiCache.generateKey('pb_component_list', params);
185
+ const cacheKey = ApiCache.generateKey('pb_component_list', params);
192
186
  return this.cache.wrap(cacheKey, async () => {
193
187
  const queryParams = {};
194
188
  if (params.productId)
@@ -197,7 +191,7 @@ class ProductBoardClient {
197
191
  });
198
192
  }
199
193
  async getProductHierarchy() {
200
- const cacheKey = cache_1.ApiCache.generateKey('pb_product_hierarchy', {});
194
+ const cacheKey = ApiCache.generateKey('pb_product_hierarchy', {});
201
195
  return this.cache.wrap(cacheKey, async () => {
202
196
  const [products, components] = await Promise.all([
203
197
  this.listProducts({ limit: 500 }),
@@ -220,7 +214,7 @@ class ProductBoardClient {
220
214
  return result.data;
221
215
  }
222
216
  async listNotes(params = {}) {
223
- const cacheKey = cache_1.ApiCache.generateKey('pb_note_list', params);
217
+ const cacheKey = ApiCache.generateKey('pb_note_list', params);
224
218
  return this.cache.wrap(cacheKey, async () => {
225
219
  const queryParams = {};
226
220
  if (params.createdFrom)
@@ -231,7 +225,7 @@ class ProductBoardClient {
231
225
  });
232
226
  }
233
227
  async getNote(id) {
234
- const cacheKey = cache_1.ApiCache.generateKey('pb_note_get', { id });
228
+ const cacheKey = ApiCache.generateKey('pb_note_get', { id });
235
229
  return this.cache.wrap(cacheKey, async () => {
236
230
  const result = await this.request({
237
231
  method: 'GET',
@@ -251,14 +245,14 @@ class ProductBoardClient {
251
245
  },
252
246
  });
253
247
  // Invalidate caches
254
- this.cache.delete(cache_1.ApiCache.generateKey('pb_note_get', { id: noteId }));
255
- this.cache.delete(cache_1.ApiCache.generateKey('pb_feature_get', { id: featureId }));
248
+ this.cache.delete(ApiCache.generateKey('pb_note_get', { id: noteId }));
249
+ this.cache.delete(ApiCache.generateKey('pb_feature_get', { id: featureId }));
256
250
  }
257
251
  // ============================================
258
252
  // User Methods
259
253
  // ============================================
260
254
  async getCurrentUser() {
261
- const cacheKey = cache_1.ApiCache.generateKey('pb_user_current', {});
255
+ const cacheKey = ApiCache.generateKey('pb_user_current', {});
262
256
  return this.cache.wrap(cacheKey, async () => {
263
257
  const result = await this.request({
264
258
  method: 'GET',
@@ -268,7 +262,7 @@ class ProductBoardClient {
268
262
  });
269
263
  }
270
264
  async listUsers(params = {}) {
271
- const cacheKey = cache_1.ApiCache.generateKey('pb_user_list', params);
265
+ const cacheKey = ApiCache.generateKey('pb_user_list', params);
272
266
  return this.cache.wrap(cacheKey, async () => {
273
267
  return this.paginate('/users', {}, params.limit);
274
268
  });
@@ -277,7 +271,7 @@ class ProductBoardClient {
277
271
  // Search Methods
278
272
  // ============================================
279
273
  async search(params) {
280
- const cacheKey = cache_1.ApiCache.generateKey('pb_search', { query: params.query, type: params.type, limit: params.limit });
274
+ const cacheKey = ApiCache.generateKey('pb_search', { query: params.query, type: params.type, limit: params.limit });
281
275
  return this.cache.wrap(cacheKey, async () => {
282
276
  const results = [];
283
277
  const queryLower = params.query.toLowerCase();
@@ -375,5 +369,4 @@ class ProductBoardClient {
375
369
  return this.rateLimiter.stats();
376
370
  }
377
371
  }
378
- exports.ProductBoardClient = ProductBoardClient;
379
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"api-client.js","sourceRoot":"","sources":["../../src/client/api-client.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * ProductBoard API Client\n */\n\nimport axios, { AxiosInstance, AxiosError, AxiosRequestConfig } from 'axios';\nimport {\n  Feature,\n  Product,\n  Component,\n  Note,\n  User,\n  CurrentUser,\n  SearchResult,\n  PaginatedResponse,\n  CreateFeatureParams,\n  UpdateFeatureParams,\n  ListFeaturesParams,\n  ListProductsParams,\n  CreateNoteParams,\n  ListNotesParams,\n  ListUsersParams,\n  SearchParams,\n  PluginConfig,\n  ProductHierarchy,\n} from './types';\nimport {\n  parseApiError,\n  isRetryableError,\n  getRetryDelay,\n  ProductBoardError,\n} from './errors';\nimport { ApiCache, getCache } from '../utils/cache';\nimport { RateLimiter, getRateLimiter } from '../utils/rate-limiter';\n\nconst DEFAULT_BASE_URL = 'https://api.productboard.com';\nconst MAX_RETRIES = 3;\n\nexport class ProductBoardClient {\n  private client: AxiosInstance;\n  private cache: ApiCache;\n  private rateLimiter: RateLimiter;\n  private baseUrl: string;\n\n  constructor(config: PluginConfig) {\n    this.baseUrl = config.apiBaseUrl || DEFAULT_BASE_URL;\n\n    this.client = axios.create({\n      baseURL: this.baseUrl,\n      headers: {\n        'Authorization': `Bearer ${config.apiToken}`,\n        'Content-Type': 'application/json',\n        'X-Version': '1',\n      },\n      timeout: 30000,\n    });\n\n    this.cache = getCache({\n      ttl: (config.cacheTtlSeconds || 300) * 1000,\n    });\n\n    this.rateLimiter = getRateLimiter({\n      maxTokens: config.rateLimitPerMinute || 100,\n    });\n\n    this.setupInterceptors();\n  }\n\n  private setupInterceptors(): void {\n    // Response interceptor for error handling\n    this.client.interceptors.response.use(\n      (response) => response,\n      (error: AxiosError) => {\n        if (error.response) {\n          const { status, data, headers } = error.response;\n          throw parseApiError(\n            status,\n            data as { code?: string; message?: string; details?: Record<string, unknown> },\n            headers['retry-after'] as string | undefined\n          );\n        }\n        throw new ProductBoardError(\n          error.message || 'Network error',\n          'NETWORK_ERROR',\n          0\n        );\n      }\n    );\n  }\n\n  /**\n   * Execute a request with retry logic and rate limiting\n   */\n  private async request<T>(\n    config: AxiosRequestConfig,\n    retryCount = 0\n  ): Promise<T> {\n    await this.rateLimiter.acquire();\n\n    try {\n      const response = await this.client.request<T>(config);\n      return response.data;\n    } catch (error) {\n      if (isRetryableError(error) && retryCount < MAX_RETRIES) {\n        const delay = getRetryDelay(error, retryCount);\n        await this.sleep(delay);\n        return this.request<T>(config, retryCount + 1);\n      }\n      throw error;\n    }\n  }\n\n  private sleep(ms: number): Promise<void> {\n    return new Promise((resolve) => setTimeout(resolve, ms));\n  }\n\n  /**\n   * Paginate through all results\n   */\n  private async paginate<T>(\n    endpoint: string,\n    params: Record<string, unknown> = {},\n    maxItems?: number\n  ): Promise<T[]> {\n    const results: T[] = [];\n    let cursor: string | undefined;\n    const limit = Math.min(params.limit as number || 100, 100);\n\n    do {\n      const response = await this.request<PaginatedResponse<T>>({\n        method: 'GET',\n        url: endpoint,\n        params: { ...params, limit, pageCursor: cursor },\n      });\n\n      results.push(...response.data);\n\n      if (maxItems && results.length >= maxItems) {\n        return results.slice(0, maxItems);\n      }\n\n      cursor = response.links?.next\n        ? new URL(response.links.next).searchParams.get('pageCursor') || undefined\n        : undefined;\n    } while (cursor);\n\n    return results;\n  }\n\n  // ============================================\n  // Feature Methods\n  // ============================================\n\n  async createFeature(params: CreateFeatureParams): Promise<Feature> {\n    const result = await this.request<{ data: Feature }>({\n      method: 'POST',\n      url: '/features',\n      data: { data: params },\n    });\n\n    // Invalidate feature list caches\n    this.cache.invalidatePattern('pb_feature_list:');\n    this.cache.invalidatePattern('pb_feature_search:');\n\n    return result.data;\n  }\n\n  async listFeatures(params: ListFeaturesParams = {}): Promise<Feature[]> {\n    const cacheKey = ApiCache.generateKey('pb_feature_list', params as Record<string, unknown>);\n\n    return this.cache.wrap(cacheKey, async () => {\n      const queryParams: Record<string, unknown> = {};\n\n      if (params.productId) queryParams['product.id'] = params.productId;\n      if (params.componentId) queryParams['component.id'] = params.componentId;\n      if (params.status) queryParams.status = params.status;\n      if (params.ownerId) queryParams['owner.id'] = params.ownerId;\n\n      return this.paginate<Feature>('/features', queryParams, params.limit);\n    });\n  }\n\n  async getFeature(id: string): Promise<Feature> {\n    const cacheKey = ApiCache.generateKey('pb_feature_get', { id });\n\n    return this.cache.wrap(cacheKey, async () => {\n      const result = await this.request<{ data: Feature }>({\n        method: 'GET',\n        url: `/features/${id}`,\n      });\n      return result.data;\n    });\n  }\n\n  async updateFeature(id: string, params: UpdateFeatureParams): Promise<Feature> {\n    const result = await this.request<{ data: Feature }>({\n      method: 'PATCH',\n      url: `/features/${id}`,\n      data: { data: params },\n    });\n\n    // Invalidate caches\n    this.cache.delete(ApiCache.generateKey('pb_feature_get', { id }));\n    this.cache.invalidatePattern('pb_feature_list:');\n    this.cache.invalidatePattern('pb_feature_search:');\n\n    return result.data;\n  }\n\n  async deleteFeature(id: string): Promise<void> {\n    await this.request<void>({\n      method: 'DELETE',\n      url: `/features/${id}`,\n    });\n\n    // Invalidate caches\n    this.cache.delete(ApiCache.generateKey('pb_feature_get', { id }));\n    this.cache.invalidatePattern('pb_feature_list:');\n    this.cache.invalidatePattern('pb_feature_search:');\n  }\n\n  async searchFeatures(query: string, limit = 50): Promise<Feature[]> {\n    const cacheKey = ApiCache.generateKey('pb_feature_search', { query, limit });\n\n    return this.cache.wrap(cacheKey, async () => {\n      // ProductBoard doesn't have a dedicated search endpoint for features\n      // We'll list all and filter client-side\n      const features = await this.listFeatures({ limit: 500 });\n      const queryLower = query.toLowerCase();\n\n      return features\n        .filter((f) => {\n          const nameMatch = f.name?.toLowerCase().includes(queryLower);\n          const descMatch = f.description?.toLowerCase().includes(queryLower);\n          return nameMatch || descMatch;\n        })\n        .slice(0, limit);\n    });\n  }\n\n  // ============================================\n  // Product Methods\n  // ============================================\n\n  async listProducts(params: ListProductsParams = {}): Promise<Product[]> {\n    const cacheKey = ApiCache.generateKey('pb_product_list', params as Record<string, unknown>);\n\n    return this.cache.wrap(cacheKey, async () => {\n      return this.paginate<Product>('/products', {}, params.limit);\n    });\n  }\n\n  async getProduct(id: string): Promise<Product> {\n    const cacheKey = ApiCache.generateKey('pb_product_get', { id });\n\n    return this.cache.wrap(cacheKey, async () => {\n      const result = await this.request<{ data: Product }>({\n        method: 'GET',\n        url: `/products/${id}`,\n      });\n      return result.data;\n    });\n  }\n\n  async listComponents(params: { productId?: string; limit?: number } = {}): Promise<Component[]> {\n    const cacheKey = ApiCache.generateKey('pb_component_list', params as Record<string, unknown>);\n\n    return this.cache.wrap(cacheKey, async () => {\n      const queryParams: Record<string, unknown> = {};\n      if (params.productId) queryParams['product.id'] = params.productId;\n\n      return this.paginate<Component>('/components', queryParams, params.limit);\n    });\n  }\n\n  async getProductHierarchy(): Promise<ProductHierarchy> {\n    const cacheKey = ApiCache.generateKey('pb_product_hierarchy', {});\n\n    return this.cache.wrap(cacheKey, async () => {\n      const [products, components] = await Promise.all([\n        this.listProducts({ limit: 500 }),\n        this.listComponents({ limit: 500 }),\n      ]);\n\n      return { products, components };\n    });\n  }\n\n  // ============================================\n  // Note Methods\n  // ============================================\n\n  async createNote(params: CreateNoteParams): Promise<Note> {\n    const result = await this.request<{ data: Note }>({\n      method: 'POST',\n      url: '/notes',\n      data: { data: params },\n    });\n\n    // Invalidate note list caches\n    this.cache.invalidatePattern('pb_note_list:');\n\n    return result.data;\n  }\n\n  async listNotes(params: ListNotesParams = {}): Promise<Note[]> {\n    const cacheKey = ApiCache.generateKey('pb_note_list', params as Record<string, unknown>);\n\n    return this.cache.wrap(cacheKey, async () => {\n      const queryParams: Record<string, unknown> = {};\n      if (params.createdFrom) queryParams.createdFrom = params.createdFrom;\n      if (params.createdTo) queryParams.createdTo = params.createdTo;\n\n      return this.paginate<Note>('/notes', queryParams, params.limit);\n    });\n  }\n\n  async getNote(id: string): Promise<Note> {\n    const cacheKey = ApiCache.generateKey('pb_note_get', { id });\n\n    return this.cache.wrap(cacheKey, async () => {\n      const result = await this.request<{ data: Note }>({\n        method: 'GET',\n        url: `/notes/${id}`,\n      });\n      return result.data;\n    });\n  }\n\n  async attachNoteToFeature(noteId: string, featureId: string): Promise<void> {\n    await this.request<void>({\n      method: 'POST',\n      url: `/notes/${noteId}/connections`,\n      data: {\n        data: {\n          feature: { id: featureId },\n        },\n      },\n    });\n\n    // Invalidate caches\n    this.cache.delete(ApiCache.generateKey('pb_note_get', { id: noteId }));\n    this.cache.delete(ApiCache.generateKey('pb_feature_get', { id: featureId }));\n  }\n\n  // ============================================\n  // User Methods\n  // ============================================\n\n  async getCurrentUser(): Promise<CurrentUser> {\n    const cacheKey = ApiCache.generateKey('pb_user_current', {});\n\n    return this.cache.wrap(cacheKey, async () => {\n      const result = await this.request<{ data: CurrentUser }>({\n        method: 'GET',\n        url: '/users/me',\n      });\n      return result.data;\n    });\n  }\n\n  async listUsers(params: ListUsersParams = {}): Promise<User[]> {\n    const cacheKey = ApiCache.generateKey('pb_user_list', params as Record<string, unknown>);\n\n    return this.cache.wrap(cacheKey, async () => {\n      return this.paginate<User>('/users', {}, params.limit);\n    });\n  }\n\n  // ============================================\n  // Search Methods\n  // ============================================\n\n  async search(params: SearchParams): Promise<SearchResult[]> {\n    const cacheKey = ApiCache.generateKey('pb_search', { query: params.query, type: params.type, limit: params.limit });\n\n    return this.cache.wrap(cacheKey, async () => {\n      const results: SearchResult[] = [];\n      const queryLower = params.query.toLowerCase();\n      const limit = params.limit || 50;\n\n      // Search features\n      if (!params.type || params.type === 'feature') {\n        const features = await this.searchFeatures(queryLower, limit);\n        results.push(\n          ...features.map((f) => ({\n            type: 'feature' as const,\n            id: f.id,\n            name: f.name,\n            description: f.description,\n            links: f.links,\n          }))\n        );\n      }\n\n      // Search products\n      if (!params.type || params.type === 'product') {\n        const products = await this.listProducts({ limit: 100 });\n        const matchingProducts = products\n          .filter(\n            (p) =>\n              p.name?.toLowerCase().includes(queryLower) ||\n              p.description?.toLowerCase().includes(queryLower)\n          )\n          .slice(0, limit);\n\n        results.push(\n          ...matchingProducts.map((p) => ({\n            type: 'product' as const,\n            id: p.id,\n            name: p.name,\n            description: p.description,\n            links: p.links,\n          }))\n        );\n      }\n\n      // Search components\n      if (!params.type || params.type === 'component') {\n        const components = await this.listComponents({ limit: 100 });\n        const matchingComponents = components\n          .filter(\n            (c) =>\n              c.name?.toLowerCase().includes(queryLower) ||\n              c.description?.toLowerCase().includes(queryLower)\n          )\n          .slice(0, limit);\n\n        results.push(\n          ...matchingComponents.map((c) => ({\n            type: 'component' as const,\n            id: c.id,\n            name: c.name,\n            description: c.description,\n            links: c.links,\n          }))\n        );\n      }\n\n      // Search notes\n      if (!params.type || params.type === 'note') {\n        const notes = await this.listNotes({ limit: 100 });\n        const matchingNotes = notes\n          .filter(\n            (n) =>\n              n.title?.toLowerCase().includes(queryLower) ||\n              n.content?.toLowerCase().includes(queryLower)\n          )\n          .slice(0, limit);\n\n        results.push(\n          ...matchingNotes.map((n) => ({\n            type: 'note' as const,\n            id: n.id,\n            title: n.title,\n            content: n.content?.substring(0, 200),\n            links: n.links,\n          }))\n        );\n      }\n\n      return results.slice(0, limit);\n    });\n  }\n\n  // ============================================\n  // Utility Methods\n  // ============================================\n\n  /**\n   * Validate the API token by making a test request\n   */\n  async validateToken(): Promise<boolean> {\n    try {\n      await this.getCurrentUser();\n      return true;\n    } catch {\n      return false;\n    }\n  }\n\n  /**\n   * Clear all cached data\n   */\n  clearCache(): void {\n    this.cache.clear();\n  }\n\n  /**\n   * Get cache statistics\n   */\n  getCacheStats(): { size: number; max: number } {\n    return this.cache.stats();\n  }\n\n  /**\n   * Get rate limiter statistics\n   */\n  getRateLimiterStats(): { tokens: number; maxTokens: number; waitTime: number } {\n    return this.rateLimiter.stats();\n  }\n}\n"]}
372
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"api-client.js","sourceRoot":"","sources":["../../src/client/api-client.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * ProductBoard API Client\n */\n\nimport axios, { AxiosInstance, AxiosError, AxiosRequestConfig } from 'axios';\nimport {\n  Feature,\n  Product,\n  Component,\n  Note,\n  User,\n  CurrentUser,\n  SearchResult,\n  PaginatedResponse,\n  CreateFeatureParams,\n  UpdateFeatureParams,\n  ListFeaturesParams,\n  ListProductsParams,\n  CreateNoteParams,\n  ListNotesParams,\n  ListUsersParams,\n  SearchParams,\n  PluginConfig,\n  ProductHierarchy,\n} from './types.js';\nimport {\n  parseApiError,\n  isRetryableError,\n  getRetryDelay,\n  ProductBoardError,\n} from './errors.js';\nimport { ApiCache, getCache } from '../utils/cache.js';\nimport { RateLimiter, getRateLimiter } from '../utils/rate-limiter.js';\n\nconst DEFAULT_BASE_URL = 'https://api.productboard.com';\nconst MAX_RETRIES = 3;\n\nexport class ProductBoardClient {\n  private client: AxiosInstance;\n  private cache: ApiCache;\n  private rateLimiter: RateLimiter;\n  private baseUrl: string;\n\n  constructor(config: PluginConfig) {\n    this.baseUrl = config.apiBaseUrl || DEFAULT_BASE_URL;\n\n    this.client = axios.create({\n      baseURL: this.baseUrl,\n      headers: {\n        'Authorization': `Bearer ${config.apiToken}`,\n        'Content-Type': 'application/json',\n        'X-Version': '1',\n      },\n      timeout: 30000,\n    });\n\n    this.cache = getCache({\n      ttl: (config.cacheTtlSeconds || 300) * 1000,\n    });\n\n    this.rateLimiter = getRateLimiter({\n      maxTokens: config.rateLimitPerMinute || 100,\n    });\n\n    this.setupInterceptors();\n  }\n\n  private setupInterceptors(): void {\n    // Response interceptor for error handling\n    this.client.interceptors.response.use(\n      (response) => response,\n      (error: AxiosError) => {\n        if (error.response) {\n          const { status, data, headers } = error.response;\n          throw parseApiError(\n            status,\n            data as { code?: string; message?: string; details?: Record<string, unknown> },\n            headers['retry-after'] as string | undefined\n          );\n        }\n        throw new ProductBoardError(\n          error.message || 'Network error',\n          'NETWORK_ERROR',\n          0\n        );\n      }\n    );\n  }\n\n  /**\n   * Execute a request with retry logic and rate limiting\n   */\n  private async request<T>(\n    config: AxiosRequestConfig,\n    retryCount = 0\n  ): Promise<T> {\n    await this.rateLimiter.acquire();\n\n    try {\n      const response = await this.client.request<T>(config);\n      return response.data;\n    } catch (error) {\n      if (isRetryableError(error) && retryCount < MAX_RETRIES) {\n        const delay = getRetryDelay(error, retryCount);\n        await this.sleep(delay);\n        return this.request<T>(config, retryCount + 1);\n      }\n      throw error;\n    }\n  }\n\n  private sleep(ms: number): Promise<void> {\n    return new Promise((resolve) => setTimeout(resolve, ms));\n  }\n\n  /**\n   * Paginate through all results\n   */\n  private async paginate<T>(\n    endpoint: string,\n    params: Record<string, unknown> = {},\n    maxItems?: number\n  ): Promise<T[]> {\n    const results: T[] = [];\n    let cursor: string | undefined;\n    const limit = Math.min(params.limit as number || 100, 100);\n\n    do {\n      const response = await this.request<PaginatedResponse<T>>({\n        method: 'GET',\n        url: endpoint,\n        params: { ...params, limit, pageCursor: cursor },\n      });\n\n      results.push(...response.data);\n\n      if (maxItems && results.length >= maxItems) {\n        return results.slice(0, maxItems);\n      }\n\n      cursor = response.links?.next\n        ? new URL(response.links.next).searchParams.get('pageCursor') || undefined\n        : undefined;\n    } while (cursor);\n\n    return results;\n  }\n\n  // ============================================\n  // Feature Methods\n  // ============================================\n\n  async createFeature(params: CreateFeatureParams): Promise<Feature> {\n    const result = await this.request<{ data: Feature }>({\n      method: 'POST',\n      url: '/features',\n      data: { data: params },\n    });\n\n    // Invalidate feature list caches\n    this.cache.invalidatePattern('pb_feature_list:');\n    this.cache.invalidatePattern('pb_feature_search:');\n\n    return result.data;\n  }\n\n  async listFeatures(params: ListFeaturesParams = {}): Promise<Feature[]> {\n    const cacheKey = ApiCache.generateKey('pb_feature_list', params as Record<string, unknown>);\n\n    return this.cache.wrap(cacheKey, async () => {\n      const queryParams: Record<string, unknown> = {};\n\n      if (params.productId) queryParams['product.id'] = params.productId;\n      if (params.componentId) queryParams['component.id'] = params.componentId;\n      if (params.status) queryParams.status = params.status;\n      if (params.ownerId) queryParams['owner.id'] = params.ownerId;\n\n      return this.paginate<Feature>('/features', queryParams, params.limit);\n    });\n  }\n\n  async getFeature(id: string): Promise<Feature> {\n    const cacheKey = ApiCache.generateKey('pb_feature_get', { id });\n\n    return this.cache.wrap(cacheKey, async () => {\n      const result = await this.request<{ data: Feature }>({\n        method: 'GET',\n        url: `/features/${id}`,\n      });\n      return result.data;\n    });\n  }\n\n  async updateFeature(id: string, params: UpdateFeatureParams): Promise<Feature> {\n    const result = await this.request<{ data: Feature }>({\n      method: 'PATCH',\n      url: `/features/${id}`,\n      data: { data: params },\n    });\n\n    // Invalidate caches\n    this.cache.delete(ApiCache.generateKey('pb_feature_get', { id }));\n    this.cache.invalidatePattern('pb_feature_list:');\n    this.cache.invalidatePattern('pb_feature_search:');\n\n    return result.data;\n  }\n\n  async deleteFeature(id: string): Promise<void> {\n    await this.request<void>({\n      method: 'DELETE',\n      url: `/features/${id}`,\n    });\n\n    // Invalidate caches\n    this.cache.delete(ApiCache.generateKey('pb_feature_get', { id }));\n    this.cache.invalidatePattern('pb_feature_list:');\n    this.cache.invalidatePattern('pb_feature_search:');\n  }\n\n  async searchFeatures(query: string, limit = 50): Promise<Feature[]> {\n    const cacheKey = ApiCache.generateKey('pb_feature_search', { query, limit });\n\n    return this.cache.wrap(cacheKey, async () => {\n      // ProductBoard doesn't have a dedicated search endpoint for features\n      // We'll list all and filter client-side\n      const features = await this.listFeatures({ limit: 500 });\n      const queryLower = query.toLowerCase();\n\n      return features\n        .filter((f) => {\n          const nameMatch = f.name?.toLowerCase().includes(queryLower);\n          const descMatch = f.description?.toLowerCase().includes(queryLower);\n          return nameMatch || descMatch;\n        })\n        .slice(0, limit);\n    });\n  }\n\n  // ============================================\n  // Product Methods\n  // ============================================\n\n  async listProducts(params: ListProductsParams = {}): Promise<Product[]> {\n    const cacheKey = ApiCache.generateKey('pb_product_list', params as Record<string, unknown>);\n\n    return this.cache.wrap(cacheKey, async () => {\n      return this.paginate<Product>('/products', {}, params.limit);\n    });\n  }\n\n  async getProduct(id: string): Promise<Product> {\n    const cacheKey = ApiCache.generateKey('pb_product_get', { id });\n\n    return this.cache.wrap(cacheKey, async () => {\n      const result = await this.request<{ data: Product }>({\n        method: 'GET',\n        url: `/products/${id}`,\n      });\n      return result.data;\n    });\n  }\n\n  async listComponents(params: { productId?: string; limit?: number } = {}): Promise<Component[]> {\n    const cacheKey = ApiCache.generateKey('pb_component_list', params as Record<string, unknown>);\n\n    return this.cache.wrap(cacheKey, async () => {\n      const queryParams: Record<string, unknown> = {};\n      if (params.productId) queryParams['product.id'] = params.productId;\n\n      return this.paginate<Component>('/components', queryParams, params.limit);\n    });\n  }\n\n  async getProductHierarchy(): Promise<ProductHierarchy> {\n    const cacheKey = ApiCache.generateKey('pb_product_hierarchy', {});\n\n    return this.cache.wrap(cacheKey, async () => {\n      const [products, components] = await Promise.all([\n        this.listProducts({ limit: 500 }),\n        this.listComponents({ limit: 500 }),\n      ]);\n\n      return { products, components };\n    });\n  }\n\n  // ============================================\n  // Note Methods\n  // ============================================\n\n  async createNote(params: CreateNoteParams): Promise<Note> {\n    const result = await this.request<{ data: Note }>({\n      method: 'POST',\n      url: '/notes',\n      data: { data: params },\n    });\n\n    // Invalidate note list caches\n    this.cache.invalidatePattern('pb_note_list:');\n\n    return result.data;\n  }\n\n  async listNotes(params: ListNotesParams = {}): Promise<Note[]> {\n    const cacheKey = ApiCache.generateKey('pb_note_list', params as Record<string, unknown>);\n\n    return this.cache.wrap(cacheKey, async () => {\n      const queryParams: Record<string, unknown> = {};\n      if (params.createdFrom) queryParams.createdFrom = params.createdFrom;\n      if (params.createdTo) queryParams.createdTo = params.createdTo;\n\n      return this.paginate<Note>('/notes', queryParams, params.limit);\n    });\n  }\n\n  async getNote(id: string): Promise<Note> {\n    const cacheKey = ApiCache.generateKey('pb_note_get', { id });\n\n    return this.cache.wrap(cacheKey, async () => {\n      const result = await this.request<{ data: Note }>({\n        method: 'GET',\n        url: `/notes/${id}`,\n      });\n      return result.data;\n    });\n  }\n\n  async attachNoteToFeature(noteId: string, featureId: string): Promise<void> {\n    await this.request<void>({\n      method: 'POST',\n      url: `/notes/${noteId}/connections`,\n      data: {\n        data: {\n          feature: { id: featureId },\n        },\n      },\n    });\n\n    // Invalidate caches\n    this.cache.delete(ApiCache.generateKey('pb_note_get', { id: noteId }));\n    this.cache.delete(ApiCache.generateKey('pb_feature_get', { id: featureId }));\n  }\n\n  // ============================================\n  // User Methods\n  // ============================================\n\n  async getCurrentUser(): Promise<CurrentUser> {\n    const cacheKey = ApiCache.generateKey('pb_user_current', {});\n\n    return this.cache.wrap(cacheKey, async () => {\n      const result = await this.request<{ data: CurrentUser }>({\n        method: 'GET',\n        url: '/users/me',\n      });\n      return result.data;\n    });\n  }\n\n  async listUsers(params: ListUsersParams = {}): Promise<User[]> {\n    const cacheKey = ApiCache.generateKey('pb_user_list', params as Record<string, unknown>);\n\n    return this.cache.wrap(cacheKey, async () => {\n      return this.paginate<User>('/users', {}, params.limit);\n    });\n  }\n\n  // ============================================\n  // Search Methods\n  // ============================================\n\n  async search(params: SearchParams): Promise<SearchResult[]> {\n    const cacheKey = ApiCache.generateKey('pb_search', { query: params.query, type: params.type, limit: params.limit });\n\n    return this.cache.wrap(cacheKey, async () => {\n      const results: SearchResult[] = [];\n      const queryLower = params.query.toLowerCase();\n      const limit = params.limit || 50;\n\n      // Search features\n      if (!params.type || params.type === 'feature') {\n        const features = await this.searchFeatures(queryLower, limit);\n        results.push(\n          ...features.map((f) => ({\n            type: 'feature' as const,\n            id: f.id,\n            name: f.name,\n            description: f.description,\n            links: f.links,\n          }))\n        );\n      }\n\n      // Search products\n      if (!params.type || params.type === 'product') {\n        const products = await this.listProducts({ limit: 100 });\n        const matchingProducts = products\n          .filter(\n            (p) =>\n              p.name?.toLowerCase().includes(queryLower) ||\n              p.description?.toLowerCase().includes(queryLower)\n          )\n          .slice(0, limit);\n\n        results.push(\n          ...matchingProducts.map((p) => ({\n            type: 'product' as const,\n            id: p.id,\n            name: p.name,\n            description: p.description,\n            links: p.links,\n          }))\n        );\n      }\n\n      // Search components\n      if (!params.type || params.type === 'component') {\n        const components = await this.listComponents({ limit: 100 });\n        const matchingComponents = components\n          .filter(\n            (c) =>\n              c.name?.toLowerCase().includes(queryLower) ||\n              c.description?.toLowerCase().includes(queryLower)\n          )\n          .slice(0, limit);\n\n        results.push(\n          ...matchingComponents.map((c) => ({\n            type: 'component' as const,\n            id: c.id,\n            name: c.name,\n            description: c.description,\n            links: c.links,\n          }))\n        );\n      }\n\n      // Search notes\n      if (!params.type || params.type === 'note') {\n        const notes = await this.listNotes({ limit: 100 });\n        const matchingNotes = notes\n          .filter(\n            (n) =>\n              n.title?.toLowerCase().includes(queryLower) ||\n              n.content?.toLowerCase().includes(queryLower)\n          )\n          .slice(0, limit);\n\n        results.push(\n          ...matchingNotes.map((n) => ({\n            type: 'note' as const,\n            id: n.id,\n            title: n.title,\n            content: n.content?.substring(0, 200),\n            links: n.links,\n          }))\n        );\n      }\n\n      return results.slice(0, limit);\n    });\n  }\n\n  // ============================================\n  // Utility Methods\n  // ============================================\n\n  /**\n   * Validate the API token by making a test request\n   */\n  async validateToken(): Promise<boolean> {\n    try {\n      await this.getCurrentUser();\n      return true;\n    } catch {\n      return false;\n    }\n  }\n\n  /**\n   * Clear all cached data\n   */\n  clearCache(): void {\n    this.cache.clear();\n  }\n\n  /**\n   * Get cache statistics\n   */\n  getCacheStats(): { size: number; max: number } {\n    return this.cache.stats();\n  }\n\n  /**\n   * Get rate limiter statistics\n   */\n  getRateLimiterStats(): { tokens: number; maxTokens: number; waitTime: number } {\n    return this.rateLimiter.stats();\n  }\n}\n"]}
@@ -1,13 +1,7 @@
1
- "use strict";
2
1
  /**
3
2
  * ProductBoard API Error Handling
4
3
  */
5
- Object.defineProperty(exports, "__esModule", { value: true });
6
- exports.ServerError = exports.ValidationError = exports.RateLimitError = exports.NotFoundError = exports.AuthorizationError = exports.AuthenticationError = exports.ProductBoardError = void 0;
7
- exports.isRetryableError = isRetryableError;
8
- exports.getRetryDelay = getRetryDelay;
9
- exports.parseApiError = parseApiError;
10
- class ProductBoardError extends Error {
4
+ export class ProductBoardError extends Error {
11
5
  code;
12
6
  statusCode;
13
7
  details;
@@ -28,29 +22,25 @@ class ProductBoardError extends Error {
28
22
  };
29
23
  }
30
24
  }
31
- exports.ProductBoardError = ProductBoardError;
32
- class AuthenticationError extends ProductBoardError {
25
+ export class AuthenticationError extends ProductBoardError {
33
26
  constructor(message = 'Invalid or expired API token') {
34
27
  super(message, 'AUTHENTICATION_ERROR', 401);
35
28
  this.name = 'AuthenticationError';
36
29
  }
37
30
  }
38
- exports.AuthenticationError = AuthenticationError;
39
- class AuthorizationError extends ProductBoardError {
31
+ export class AuthorizationError extends ProductBoardError {
40
32
  constructor(message = 'Insufficient permissions for this operation') {
41
33
  super(message, 'AUTHORIZATION_ERROR', 403);
42
34
  this.name = 'AuthorizationError';
43
35
  }
44
36
  }
45
- exports.AuthorizationError = AuthorizationError;
46
- class NotFoundError extends ProductBoardError {
37
+ export class NotFoundError extends ProductBoardError {
47
38
  constructor(resource, id) {
48
39
  super(`${resource} with id '${id}' not found`, 'NOT_FOUND', 404);
49
40
  this.name = 'NotFoundError';
50
41
  }
51
42
  }
52
- exports.NotFoundError = NotFoundError;
53
- class RateLimitError extends ProductBoardError {
43
+ export class RateLimitError extends ProductBoardError {
54
44
  retryAfter;
55
45
  constructor(retryAfter = 60) {
56
46
  super(`Rate limit exceeded. Retry after ${retryAfter} seconds`, 'RATE_LIMIT_EXCEEDED', 429);
@@ -58,25 +48,22 @@ class RateLimitError extends ProductBoardError {
58
48
  this.retryAfter = retryAfter;
59
49
  }
60
50
  }
61
- exports.RateLimitError = RateLimitError;
62
- class ValidationError extends ProductBoardError {
51
+ export class ValidationError extends ProductBoardError {
63
52
  constructor(message, details) {
64
53
  super(message, 'VALIDATION_ERROR', 400, details);
65
54
  this.name = 'ValidationError';
66
55
  }
67
56
  }
68
- exports.ValidationError = ValidationError;
69
- class ServerError extends ProductBoardError {
57
+ export class ServerError extends ProductBoardError {
70
58
  constructor(message = 'ProductBoard server error') {
71
59
  super(message, 'SERVER_ERROR', 500);
72
60
  this.name = 'ServerError';
73
61
  }
74
62
  }
75
- exports.ServerError = ServerError;
76
63
  /**
77
64
  * Check if an error is retryable
78
65
  */
79
- function isRetryableError(error) {
66
+ export function isRetryableError(error) {
80
67
  if (error instanceof RateLimitError) {
81
68
  return true;
82
69
  }
@@ -91,7 +78,7 @@ function isRetryableError(error) {
91
78
  /**
92
79
  * Get retry delay for an error
93
80
  */
94
- function getRetryDelay(error, attempt) {
81
+ export function getRetryDelay(error, attempt) {
95
82
  if (error instanceof RateLimitError) {
96
83
  return error.retryAfter * 1000;
97
84
  }
@@ -101,7 +88,7 @@ function getRetryDelay(error, attempt) {
101
88
  /**
102
89
  * Parse API error response into appropriate error class
103
90
  */
104
- function parseApiError(statusCode, responseData, retryAfterHeader) {
91
+ export function parseApiError(statusCode, responseData, retryAfterHeader) {
105
92
  const message = responseData?.message || 'Unknown error';
106
93
  const details = responseData?.details;
107
94
  switch (statusCode) {
@@ -125,4 +112,4 @@ function parseApiError(statusCode, responseData, retryAfterHeader) {
125
112
  return new ProductBoardError(message, responseData?.code || 'UNKNOWN_ERROR', statusCode, details);
126
113
  }
127
114
  }
128
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"errors.js","sourceRoot":"","sources":["../../src/client/errors.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * ProductBoard API Error Handling\n */\n\nexport class ProductBoardError extends Error {\n  public readonly code: string;\n  public readonly statusCode: number;\n  public readonly details?: Record<string, unknown>;\n\n  constructor(\n    message: string,\n    code: string,\n    statusCode: number,\n    details?: Record<string, unknown>\n  ) {\n    super(message);\n    this.name = 'ProductBoardError';\n    this.code = code;\n    this.statusCode = statusCode;\n    this.details = details;\n  }\n\n  toJSON() {\n    return {\n      name: this.name,\n      message: this.message,\n      code: this.code,\n      statusCode: this.statusCode,\n      details: this.details,\n    };\n  }\n}\n\nexport class AuthenticationError extends ProductBoardError {\n  constructor(message = 'Invalid or expired API token') {\n    super(message, 'AUTHENTICATION_ERROR', 401);\n    this.name = 'AuthenticationError';\n  }\n}\n\nexport class AuthorizationError extends ProductBoardError {\n  constructor(message = 'Insufficient permissions for this operation') {\n    super(message, 'AUTHORIZATION_ERROR', 403);\n    this.name = 'AuthorizationError';\n  }\n}\n\nexport class NotFoundError extends ProductBoardError {\n  constructor(resource: string, id: string) {\n    super(`${resource} with id '${id}' not found`, 'NOT_FOUND', 404);\n    this.name = 'NotFoundError';\n  }\n}\n\nexport class RateLimitError extends ProductBoardError {\n  public readonly retryAfter: number;\n\n  constructor(retryAfter = 60) {\n    super(\n      `Rate limit exceeded. Retry after ${retryAfter} seconds`,\n      'RATE_LIMIT_EXCEEDED',\n      429\n    );\n    this.name = 'RateLimitError';\n    this.retryAfter = retryAfter;\n  }\n}\n\nexport class ValidationError extends ProductBoardError {\n  constructor(message: string, details?: Record<string, unknown>) {\n    super(message, 'VALIDATION_ERROR', 400, details);\n    this.name = 'ValidationError';\n  }\n}\n\nexport class ServerError extends ProductBoardError {\n  constructor(message = 'ProductBoard server error') {\n    super(message, 'SERVER_ERROR', 500);\n    this.name = 'ServerError';\n  }\n}\n\n/**\n * Check if an error is retryable\n */\nexport function isRetryableError(error: unknown): boolean {\n  if (error instanceof RateLimitError) {\n    return true;\n  }\n  if (error instanceof ServerError) {\n    return true;\n  }\n  if (error instanceof ProductBoardError) {\n    return error.statusCode >= 500;\n  }\n  return false;\n}\n\n/**\n * Get retry delay for an error\n */\nexport function getRetryDelay(error: unknown, attempt: number): number {\n  if (error instanceof RateLimitError) {\n    return error.retryAfter * 1000;\n  }\n  // Exponential backoff: 1s, 2s, 4s, 8s...\n  return Math.min(1000 * Math.pow(2, attempt), 30000);\n}\n\n/**\n * Parse API error response into appropriate error class\n */\nexport function parseApiError(\n  statusCode: number,\n  responseData?: { code?: string; message?: string; details?: Record<string, unknown> },\n  retryAfterHeader?: string\n): ProductBoardError {\n  const message = responseData?.message || 'Unknown error';\n  const details = responseData?.details;\n\n  switch (statusCode) {\n    case 400:\n      return new ValidationError(message, details);\n    case 401:\n      return new AuthenticationError(message);\n    case 403:\n      return new AuthorizationError(message);\n    case 404:\n      return new ProductBoardError(message, 'NOT_FOUND', 404, details);\n    case 429:\n      const retryAfter = retryAfterHeader ? parseInt(retryAfterHeader, 10) : 60;\n      return new RateLimitError(retryAfter);\n    case 500:\n    case 502:\n    case 503:\n    case 504:\n      return new ServerError(message);\n    default:\n      return new ProductBoardError(message, responseData?.code || 'UNKNOWN_ERROR', statusCode, details);\n  }\n}\n"]}
115
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"errors.js","sourceRoot":"","sources":["../../src/client/errors.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * ProductBoard API Error Handling\n */\n\nexport class ProductBoardError extends Error {\n  public readonly code: string;\n  public readonly statusCode: number;\n  public readonly details?: Record<string, unknown>;\n\n  constructor(\n    message: string,\n    code: string,\n    statusCode: number,\n    details?: Record<string, unknown>\n  ) {\n    super(message);\n    this.name = 'ProductBoardError';\n    this.code = code;\n    this.statusCode = statusCode;\n    this.details = details;\n  }\n\n  toJSON() {\n    return {\n      name: this.name,\n      message: this.message,\n      code: this.code,\n      statusCode: this.statusCode,\n      details: this.details,\n    };\n  }\n}\n\nexport class AuthenticationError extends ProductBoardError {\n  constructor(message = 'Invalid or expired API token') {\n    super(message, 'AUTHENTICATION_ERROR', 401);\n    this.name = 'AuthenticationError';\n  }\n}\n\nexport class AuthorizationError extends ProductBoardError {\n  constructor(message = 'Insufficient permissions for this operation') {\n    super(message, 'AUTHORIZATION_ERROR', 403);\n    this.name = 'AuthorizationError';\n  }\n}\n\nexport class NotFoundError extends ProductBoardError {\n  constructor(resource: string, id: string) {\n    super(`${resource} with id '${id}' not found`, 'NOT_FOUND', 404);\n    this.name = 'NotFoundError';\n  }\n}\n\nexport class RateLimitError extends ProductBoardError {\n  public readonly retryAfter: number;\n\n  constructor(retryAfter = 60) {\n    super(\n      `Rate limit exceeded. Retry after ${retryAfter} seconds`,\n      'RATE_LIMIT_EXCEEDED',\n      429\n    );\n    this.name = 'RateLimitError';\n    this.retryAfter = retryAfter;\n  }\n}\n\nexport class ValidationError extends ProductBoardError {\n  constructor(message: string, details?: Record<string, unknown>) {\n    super(message, 'VALIDATION_ERROR', 400, details);\n    this.name = 'ValidationError';\n  }\n}\n\nexport class ServerError extends ProductBoardError {\n  constructor(message = 'ProductBoard server error') {\n    super(message, 'SERVER_ERROR', 500);\n    this.name = 'ServerError';\n  }\n}\n\n/**\n * Check if an error is retryable\n */\nexport function isRetryableError(error: unknown): boolean {\n  if (error instanceof RateLimitError) {\n    return true;\n  }\n  if (error instanceof ServerError) {\n    return true;\n  }\n  if (error instanceof ProductBoardError) {\n    return error.statusCode >= 500;\n  }\n  return false;\n}\n\n/**\n * Get retry delay for an error\n */\nexport function getRetryDelay(error: unknown, attempt: number): number {\n  if (error instanceof RateLimitError) {\n    return error.retryAfter * 1000;\n  }\n  // Exponential backoff: 1s, 2s, 4s, 8s...\n  return Math.min(1000 * Math.pow(2, attempt), 30000);\n}\n\n/**\n * Parse API error response into appropriate error class\n */\nexport function parseApiError(\n  statusCode: number,\n  responseData?: { code?: string; message?: string; details?: Record<string, unknown> },\n  retryAfterHeader?: string\n): ProductBoardError {\n  const message = responseData?.message || 'Unknown error';\n  const details = responseData?.details;\n\n  switch (statusCode) {\n    case 400:\n      return new ValidationError(message, details);\n    case 401:\n      return new AuthenticationError(message);\n    case 403:\n      return new AuthorizationError(message);\n    case 404:\n      return new ProductBoardError(message, 'NOT_FOUND', 404, details);\n    case 429:\n      const retryAfter = retryAfterHeader ? parseInt(retryAfterHeader, 10) : 60;\n      return new RateLimitError(retryAfter);\n    case 500:\n    case 502:\n    case 503:\n    case 504:\n      return new ServerError(message);\n    default:\n      return new ProductBoardError(message, responseData?.code || 'UNKNOWN_ERROR', statusCode, details);\n  }\n}\n"]}
@@ -1,6 +1,5 @@
1
- "use strict";
2
1
  /**
3
2
  * ProductBoard API Type Definitions
4
3
  */
5
- Object.defineProperty(exports, "__esModule", { value: true });
6
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidHlwZXMuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi9zcmMvY2xpZW50L3R5cGVzLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiI7QUFBQTs7R0FFRyIsInNvdXJjZXNDb250ZW50IjpbIi8qKlxuICogUHJvZHVjdEJvYXJkIEFQSSBUeXBlIERlZmluaXRpb25zXG4gKi9cblxuLy8gQmFzZSB0eXBlc1xuZXhwb3J0IGludGVyZmFjZSBQYWdpbmF0ZWRSZXNwb25zZTxUPiB7XG4gIGRhdGE6IFRbXTtcbiAgbGlua3M/OiB7XG4gICAgbmV4dD86IHN0cmluZztcbiAgICBwcmV2Pzogc3RyaW5nO1xuICB9O1xuICB0b3RhbFJlc3VsdHM/OiBudW1iZXI7XG59XG5cbmV4cG9ydCBpbnRlcmZhY2UgQXBpRXJyb3Ige1xuICBjb2RlOiBzdHJpbmc7XG4gIG1lc3NhZ2U6IHN0cmluZztcbiAgZGV0YWlscz86IFJlY29yZDxzdHJpbmcsIHVua25vd24+O1xufVxuXG4vLyBGZWF0dXJlIHR5cGVzXG5leHBvcnQgaW50ZXJmYWNlIEZlYXR1cmUge1xuICBpZDogc3RyaW5nO1xuICBuYW1lOiBzdHJpbmc7XG4gIGRlc2NyaXB0aW9uPzogc3RyaW5nO1xuICBzdGF0dXM/OiBGZWF0dXJlU3RhdHVzO1xuICBwYXJlbnQ/OiB7XG4gICAgZmVhdHVyZT86IHsgaWQ6IHN0cmluZyB9O1xuICAgIHByb2R1Y3Q/OiB7IGlkOiBzdHJpbmcgfTtcbiAgICBjb21wb25lbnQ/OiB7IGlkOiBzdHJpbmcgfTtcbiAgfTtcbiAgb3duZXI/OiB7IGVtYWlsOiBzdHJpbmcgfTtcbiAgdGltZWZyYW1lPzogeyBzdGFydERhdGU/OiBzdHJpbmc7IGVuZERhdGU/OiBzdHJpbmcgfTtcbiAgY3JlYXRlZEF0OiBzdHJpbmc7XG4gIHVwZGF0ZWRBdDogc3RyaW5nO1xuICBsaW5rcz86IHtcbiAgICBzZWxmOiBzdHJpbmc7XG4gICAgaHRtbDogc3RyaW5nO1xuICB9O1xufVxuXG5leHBvcnQgdHlwZSBGZWF0dXJlU3RhdHVzID1cbiAgfCAnbmV3J1xuICB8ICdpbi1wcm9ncmVzcydcbiAgfCAnc2hpcHBlZCdcbiAgfCAnYXJjaGl2ZWQnXG4gIHwgJ3Bvc3Rwb25lZCdcbiAgfCAnY2FuZGlkYXRlJztcblxuZXhwb3J0IGludGVyZmFjZSBDcmVhdGVGZWF0dXJlUGFyYW1zIHtcbiAgbmFtZTogc3RyaW5nO1xuICBkZXNjcmlwdGlvbj86IHN0cmluZztcbiAgc3RhdHVzPzogRmVhdHVyZVN0YXR1cztcbiAgcGFyZW50Pzoge1xuICAgIHByb2R1Y3Q/OiB7IGlkOiBzdHJpbmcgfTtcbiAgICBjb21wb25lbnQ/OiB7IGlkOiBzdHJpbmcgfTtcbiAgICBmZWF0dXJlPzogeyBpZDogc3RyaW5nIH07XG4gIH07XG4gIG93bmVyPzogeyBlbWFpbDogc3RyaW5nIH07XG4gIHRpbWVmcmFtZT86IHsgc3RhcnREYXRlPzogc3RyaW5nOyBlbmREYXRlPzogc3RyaW5nIH07XG59XG5cbmV4cG9ydCBpbnRlcmZhY2UgVXBkYXRlRmVhdHVyZVBhcmFtcyB7XG4gIG5hbWU/OiBzdHJpbmc7XG4gIGRlc2NyaXB0aW9uPzogc3RyaW5nO1xuICBzdGF0dXM/OiBGZWF0dXJlU3RhdHVzO1xuICBwYXJlbnQ/OiB7XG4gICAgcHJvZHVjdD86IHsgaWQ6IHN0cmluZyB9O1xuICAgIGNvbXBvbmVudD86IHsgaWQ6IHN0cmluZyB9O1xuICAgIGZlYXR1cmU/OiB7IGlkOiBzdHJpbmcgfTtcbiAgfTtcbiAgb3duZXI/OiB7IGVtYWlsOiBzdHJpbmcgfTtcbiAgdGltZWZyYW1lPzogeyBzdGFydERhdGU/OiBzdHJpbmc7IGVuZERhdGU/OiBzdHJpbmcgfTtcbn1cblxuZXhwb3J0IGludGVyZmFjZSBMaXN0RmVhdHVyZXNQYXJhbXMge1xuICBwcm9kdWN0SWQ/OiBzdHJpbmc7XG4gIGNvbXBvbmVudElkPzogc3RyaW5nO1xuICBzdGF0dXM/OiBGZWF0dXJlU3RhdHVzO1xuICBvd25lcklkPzogc3RyaW5nO1xuICBsaW1pdD86IG51bWJlcjtcbiAgY3Vyc29yPzogc3RyaW5nO1xufVxuXG4vLyBQcm9kdWN0IHR5cGVzXG5leHBvcnQgaW50ZXJmYWNlIFByb2R1Y3Qge1xuICBpZDogc3RyaW5nO1xuICBuYW1lOiBzdHJpbmc7XG4gIGRlc2NyaXB0aW9uPzogc3RyaW5nO1xuICBjcmVhdGVkQXQ6IHN0cmluZztcbiAgdXBkYXRlZEF0OiBzdHJpbmc7XG4gIGxpbmtzPzoge1xuICAgIHNlbGY6IHN0cmluZztcbiAgICBodG1sOiBzdHJpbmc7XG4gIH07XG59XG5cbmV4cG9ydCBpbnRlcmZhY2UgQ29tcG9uZW50IHtcbiAgaWQ6IHN0cmluZztcbiAgbmFtZTogc3RyaW5nO1xuICBkZXNjcmlwdGlvbj86IHN0cmluZztcbiAgcGFyZW50Pzoge1xuICAgIHByb2R1Y3Q/OiB7IGlkOiBzdHJpbmcgfTtcbiAgICBjb21wb25lbnQ/OiB7IGlkOiBzdHJpbmcgfTtcbiAgfTtcbiAgY3JlYXRlZEF0OiBzdHJpbmc7XG4gIHVwZGF0ZWRBdDogc3RyaW5nO1xuICBsaW5rcz86IHtcbiAgICBzZWxmOiBzdHJpbmc7XG4gICAgaHRtbDogc3RyaW5nO1xuICB9O1xufVxuXG5leHBvcnQgaW50ZXJmYWNlIFByb2R1Y3RIaWVyYXJjaHkge1xuICBwcm9kdWN0czogUHJvZHVjdFtdO1xuICBjb21wb25lbnRzOiBDb21wb25lbnRbXTtcbn1cblxuZXhwb3J0IGludGVyZmFjZSBMaXN0UHJvZHVjdHNQYXJhbXMge1xuICBsaW1pdD86IG51bWJlcjtcbiAgY3Vyc29yPzogc3RyaW5nO1xufVxuXG4vLyBOb3RlIHR5cGVzXG5leHBvcnQgaW50ZXJmYWNlIE5vdGUge1xuICBpZDogc3RyaW5nO1xuICB0aXRsZT86IHN0cmluZztcbiAgY29udGVudDogc3RyaW5nO1xuICBkaXNwbGF5VXJsPzogc3RyaW5nO1xuICBzb3VyY2U/OiB7XG4gICAgb3JpZ2luPzogc3RyaW5nO1xuICAgIHJlY29yZF9pZD86IHN0cmluZztcbiAgfTtcbiAgdXNlcj86IHtcbiAgICBlbWFpbDogc3RyaW5nO1xuICAgIG5hbWU/OiBzdHJpbmc7XG4gIH07XG4gIGNvbXBhbnk/OiB7XG4gICAgaWQ/OiBzdHJpbmc7XG4gICAgbmFtZT86IHN0cmluZztcbiAgfTtcbiAgdGFncz86IHN0cmluZ1tdO1xuICBmZWF0dXJlcz86IEFycmF5PHsgaWQ6IHN0cmluZyB9PjtcbiAgY3JlYXRlZEF0OiBzdHJpbmc7XG4gIHVwZGF0ZWRBdDogc3RyaW5nO1xuICBsaW5rcz86IHtcbiAgICBzZWxmOiBzdHJpbmc7XG4gICAgaHRtbDogc3RyaW5nO1xuICB9O1xufVxuXG5leHBvcnQgaW50ZXJmYWNlIENyZWF0ZU5vdGVQYXJhbXMge1xuICB0aXRsZT86IHN0cmluZztcbiAgY29udGVudDogc3RyaW5nO1xuICBkaXNwbGF5VXJsPzogc3RyaW5nO1xuICBzb3VyY2U/OiB7XG4gICAgb3JpZ2luPzogc3RyaW5nO1xuICAgIHJlY29yZF9pZD86IHN0cmluZztcbiAgfTtcbiAgdXNlcj86IHtcbiAgICBlbWFpbDogc3RyaW5nO1xuICAgIG5hbWU/OiBzdHJpbmc7XG4gIH07XG4gIGNvbXBhbnk/OiB7XG4gICAgaWQ/OiBzdHJpbmc7XG4gICAgbmFtZT86IHN0cmluZztcbiAgfTtcbiAgdGFncz86IHN0cmluZ1tdO1xufVxuXG5leHBvcnQgaW50ZXJmYWNlIExpc3ROb3Rlc1BhcmFtcyB7XG4gIGxpbWl0PzogbnVtYmVyO1xuICBjdXJzb3I/OiBzdHJpbmc7XG4gIGNyZWF0ZWRGcm9tPzogc3RyaW5nO1xuICBjcmVhdGVkVG8/OiBzdHJpbmc7XG59XG5cbmV4cG9ydCBpbnRlcmZhY2UgQXR0YWNoTm90ZVBhcmFtcyB7XG4gIG5vdGVJZDogc3RyaW5nO1xuICBmZWF0dXJlSWQ6IHN0cmluZztcbn1cblxuLy8gVXNlciB0eXBlc1xuZXhwb3J0IGludGVyZmFjZSBVc2VyIHtcbiAgaWQ6IHN0cmluZztcbiAgZW1haWw6IHN0cmluZztcbiAgbmFtZT86IHN0cmluZztcbiAgcm9sZT86IHN0cmluZztcbiAgY3JlYXRlZEF0Pzogc3RyaW5nO1xufVxuXG5leHBvcnQgaW50ZXJmYWNlIEN1cnJlbnRVc2VyIGV4dGVuZHMgVXNlciB7XG4gIHdvcmtzcGFjZUlkPzogc3RyaW5nO1xuICB3b3Jrc3BhY2VOYW1lPzogc3RyaW5nO1xufVxuXG5leHBvcnQgaW50ZXJmYWNlIExpc3RVc2Vyc1BhcmFtcyB7XG4gIGxpbWl0PzogbnVtYmVyO1xuICBjdXJzb3I/OiBzdHJpbmc7XG59XG5cbi8vIFNlYXJjaCB0eXBlc1xuZXhwb3J0IGludGVyZmFjZSBTZWFyY2hQYXJhbXMge1xuICBxdWVyeTogc3RyaW5nO1xuICB0eXBlPzogJ2ZlYXR1cmUnIHwgJ25vdGUnIHwgJ3Byb2R1Y3QnIHwgJ2NvbXBvbmVudCc7XG4gIGxpbWl0PzogbnVtYmVyO1xufVxuXG5leHBvcnQgaW50ZXJmYWNlIFNlYXJjaFJlc3VsdCB7XG4gIHR5cGU6ICdmZWF0dXJlJyB8ICdub3RlJyB8ICdwcm9kdWN0JyB8ICdjb21wb25lbnQnO1xuICBpZDogc3RyaW5nO1xuICBuYW1lPzogc3RyaW5nO1xuICB0aXRsZT86IHN0cmluZztcbiAgY29udGVudD86IHN0cmluZztcbiAgZGVzY3JpcHRpb24/OiBzdHJpbmc7XG4gIHNjb3JlPzogbnVtYmVyO1xuICBsaW5rcz86IHtcbiAgICBzZWxmOiBzdHJpbmc7XG4gICAgaHRtbDogc3RyaW5nO1xuICB9O1xufVxuXG4vLyBQbHVnaW4gY29uZmlndXJhdGlvblxuZXhwb3J0IGludGVyZmFjZSBQbHVnaW5Db25maWcge1xuICBhcGlUb2tlbjogc3RyaW5nO1xuICBhcGlCYXNlVXJsPzogc3RyaW5nO1xuICBjYWNoZVR0bFNlY29uZHM/OiBudW1iZXI7XG4gIHJhdGVMaW1pdFBlck1pbnV0ZT86IG51bWJlcjtcbn1cblxuLy8gT3BlbkNsYXcgUGx1Z2luIEFQSSB0eXBlc1xuZXhwb3J0IGludGVyZmFjZSBUb29sRGVmaW5pdGlvbiB7XG4gIG5hbWU6IHN0cmluZztcbiAgZGVzY3JpcHRpb246IHN0cmluZztcbiAgcGFyYW1ldGVyczoge1xuICAgIHR5cGU6ICdvYmplY3QnO1xuICAgIHByb3BlcnRpZXM6IFJlY29yZDxzdHJpbmcsIFBhcmFtZXRlclNjaGVtYT47XG4gICAgcmVxdWlyZWQ/OiBzdHJpbmdbXTtcbiAgfTtcbiAgaGFuZGxlcjogKHBhcmFtczogUmVjb3JkPHN0cmluZywgdW5rbm93bj4pID0+IFByb21pc2U8dW5rbm93bj47XG59XG5cbmV4cG9ydCBpbnRlcmZhY2UgUGFyYW1ldGVyU2NoZW1hIHtcbiAgdHlwZTogJ3N0cmluZycgfCAnbnVtYmVyJyB8ICdib29sZWFuJyB8ICdhcnJheScgfCAnb2JqZWN0JztcbiAgZGVzY3JpcHRpb24/OiBzdHJpbmc7XG4gIGVudW0/OiBzdHJpbmdbXTtcbiAgZGVmYXVsdD86IHVua25vd247XG4gIGl0ZW1zPzogUGFyYW1ldGVyU2NoZW1hO1xuICBwcm9wZXJ0aWVzPzogUmVjb3JkPHN0cmluZywgUGFyYW1ldGVyU2NoZW1hPjtcbn1cblxuZXhwb3J0IGludGVyZmFjZSBQbHVnaW5BUEkge1xuICBjb25maWc6IFBsdWdpbkNvbmZpZztcbiAgcmVnaXN0ZXJUb29sOiAodG9vbDogVG9vbERlZmluaXRpb24pID0+IHZvaWQ7XG4gIGxvZzoge1xuICAgIGluZm86IChtZXNzYWdlOiBzdHJpbmcsIGRhdGE/OiB1bmtub3duKSA9PiB2b2lkO1xuICAgIHdhcm46IChtZXNzYWdlOiBzdHJpbmcsIGRhdGE/OiB1bmtub3duKSA9PiB2b2lkO1xuICAgIGVycm9yOiAobWVzc2FnZTogc3RyaW5nLCBkYXRhPzogdW5rbm93bikgPT4gdm9pZDtcbiAgICBkZWJ1ZzogKG1lc3NhZ2U6IHN0cmluZywgZGF0YT86IHVua25vd24pID0+IHZvaWQ7XG4gIH07XG59XG4iXX0=
4
+ export {};
5
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidHlwZXMuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi9zcmMvY2xpZW50L3R5cGVzLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBOztHQUVHIiwic291cmNlc0NvbnRlbnQiOlsiLyoqXG4gKiBQcm9kdWN0Qm9hcmQgQVBJIFR5cGUgRGVmaW5pdGlvbnNcbiAqL1xuXG4vLyBCYXNlIHR5cGVzXG5leHBvcnQgaW50ZXJmYWNlIFBhZ2luYXRlZFJlc3BvbnNlPFQ+IHtcbiAgZGF0YTogVFtdO1xuICBsaW5rcz86IHtcbiAgICBuZXh0Pzogc3RyaW5nO1xuICAgIHByZXY/OiBzdHJpbmc7XG4gIH07XG4gIHRvdGFsUmVzdWx0cz86IG51bWJlcjtcbn1cblxuZXhwb3J0IGludGVyZmFjZSBBcGlFcnJvciB7XG4gIGNvZGU6IHN0cmluZztcbiAgbWVzc2FnZTogc3RyaW5nO1xuICBkZXRhaWxzPzogUmVjb3JkPHN0cmluZywgdW5rbm93bj47XG59XG5cbi8vIEZlYXR1cmUgdHlwZXNcbmV4cG9ydCBpbnRlcmZhY2UgRmVhdHVyZSB7XG4gIGlkOiBzdHJpbmc7XG4gIG5hbWU6IHN0cmluZztcbiAgZGVzY3JpcHRpb24/OiBzdHJpbmc7XG4gIHN0YXR1cz86IEZlYXR1cmVTdGF0dXM7XG4gIHBhcmVudD86IHtcbiAgICBmZWF0dXJlPzogeyBpZDogc3RyaW5nIH07XG4gICAgcHJvZHVjdD86IHsgaWQ6IHN0cmluZyB9O1xuICAgIGNvbXBvbmVudD86IHsgaWQ6IHN0cmluZyB9O1xuICB9O1xuICBvd25lcj86IHsgZW1haWw6IHN0cmluZyB9O1xuICB0aW1lZnJhbWU/OiB7IHN0YXJ0RGF0ZT86IHN0cmluZzsgZW5kRGF0ZT86IHN0cmluZyB9O1xuICBjcmVhdGVkQXQ6IHN0cmluZztcbiAgdXBkYXRlZEF0OiBzdHJpbmc7XG4gIGxpbmtzPzoge1xuICAgIHNlbGY6IHN0cmluZztcbiAgICBodG1sOiBzdHJpbmc7XG4gIH07XG59XG5cbmV4cG9ydCB0eXBlIEZlYXR1cmVTdGF0dXMgPVxuICB8ICduZXcnXG4gIHwgJ2luLXByb2dyZXNzJ1xuICB8ICdzaGlwcGVkJ1xuICB8ICdhcmNoaXZlZCdcbiAgfCAncG9zdHBvbmVkJ1xuICB8ICdjYW5kaWRhdGUnO1xuXG5leHBvcnQgaW50ZXJmYWNlIENyZWF0ZUZlYXR1cmVQYXJhbXMge1xuICBuYW1lOiBzdHJpbmc7XG4gIGRlc2NyaXB0aW9uPzogc3RyaW5nO1xuICBzdGF0dXM/OiBGZWF0dXJlU3RhdHVzO1xuICBwYXJlbnQ/OiB7XG4gICAgcHJvZHVjdD86IHsgaWQ6IHN0cmluZyB9O1xuICAgIGNvbXBvbmVudD86IHsgaWQ6IHN0cmluZyB9O1xuICAgIGZlYXR1cmU/OiB7IGlkOiBzdHJpbmcgfTtcbiAgfTtcbiAgb3duZXI/OiB7IGVtYWlsOiBzdHJpbmcgfTtcbiAgdGltZWZyYW1lPzogeyBzdGFydERhdGU/OiBzdHJpbmc7IGVuZERhdGU/OiBzdHJpbmcgfTtcbn1cblxuZXhwb3J0IGludGVyZmFjZSBVcGRhdGVGZWF0dXJlUGFyYW1zIHtcbiAgbmFtZT86IHN0cmluZztcbiAgZGVzY3JpcHRpb24/OiBzdHJpbmc7XG4gIHN0YXR1cz86IEZlYXR1cmVTdGF0dXM7XG4gIHBhcmVudD86IHtcbiAgICBwcm9kdWN0PzogeyBpZDogc3RyaW5nIH07XG4gICAgY29tcG9uZW50PzogeyBpZDogc3RyaW5nIH07XG4gICAgZmVhdHVyZT86IHsgaWQ6IHN0cmluZyB9O1xuICB9O1xuICBvd25lcj86IHsgZW1haWw6IHN0cmluZyB9O1xuICB0aW1lZnJhbWU/OiB7IHN0YXJ0RGF0ZT86IHN0cmluZzsgZW5kRGF0ZT86IHN0cmluZyB9O1xufVxuXG5leHBvcnQgaW50ZXJmYWNlIExpc3RGZWF0dXJlc1BhcmFtcyB7XG4gIHByb2R1Y3RJZD86IHN0cmluZztcbiAgY29tcG9uZW50SWQ/OiBzdHJpbmc7XG4gIHN0YXR1cz86IEZlYXR1cmVTdGF0dXM7XG4gIG93bmVySWQ/OiBzdHJpbmc7XG4gIGxpbWl0PzogbnVtYmVyO1xuICBjdXJzb3I/OiBzdHJpbmc7XG59XG5cbi8vIFByb2R1Y3QgdHlwZXNcbmV4cG9ydCBpbnRlcmZhY2UgUHJvZHVjdCB7XG4gIGlkOiBzdHJpbmc7XG4gIG5hbWU6IHN0cmluZztcbiAgZGVzY3JpcHRpb24/OiBzdHJpbmc7XG4gIGNyZWF0ZWRBdDogc3RyaW5nO1xuICB1cGRhdGVkQXQ6IHN0cmluZztcbiAgbGlua3M/OiB7XG4gICAgc2VsZjogc3RyaW5nO1xuICAgIGh0bWw6IHN0cmluZztcbiAgfTtcbn1cblxuZXhwb3J0IGludGVyZmFjZSBDb21wb25lbnQge1xuICBpZDogc3RyaW5nO1xuICBuYW1lOiBzdHJpbmc7XG4gIGRlc2NyaXB0aW9uPzogc3RyaW5nO1xuICBwYXJlbnQ/OiB7XG4gICAgcHJvZHVjdD86IHsgaWQ6IHN0cmluZyB9O1xuICAgIGNvbXBvbmVudD86IHsgaWQ6IHN0cmluZyB9O1xuICB9O1xuICBjcmVhdGVkQXQ6IHN0cmluZztcbiAgdXBkYXRlZEF0OiBzdHJpbmc7XG4gIGxpbmtzPzoge1xuICAgIHNlbGY6IHN0cmluZztcbiAgICBodG1sOiBzdHJpbmc7XG4gIH07XG59XG5cbmV4cG9ydCBpbnRlcmZhY2UgUHJvZHVjdEhpZXJhcmNoeSB7XG4gIHByb2R1Y3RzOiBQcm9kdWN0W107XG4gIGNvbXBvbmVudHM6IENvbXBvbmVudFtdO1xufVxuXG5leHBvcnQgaW50ZXJmYWNlIExpc3RQcm9kdWN0c1BhcmFtcyB7XG4gIGxpbWl0PzogbnVtYmVyO1xuICBjdXJzb3I/OiBzdHJpbmc7XG59XG5cbi8vIE5vdGUgdHlwZXNcbmV4cG9ydCBpbnRlcmZhY2UgTm90ZSB7XG4gIGlkOiBzdHJpbmc7XG4gIHRpdGxlPzogc3RyaW5nO1xuICBjb250ZW50OiBzdHJpbmc7XG4gIGRpc3BsYXlVcmw/OiBzdHJpbmc7XG4gIHNvdXJjZT86IHtcbiAgICBvcmlnaW4/OiBzdHJpbmc7XG4gICAgcmVjb3JkX2lkPzogc3RyaW5nO1xuICB9O1xuICB1c2VyPzoge1xuICAgIGVtYWlsOiBzdHJpbmc7XG4gICAgbmFtZT86IHN0cmluZztcbiAgfTtcbiAgY29tcGFueT86IHtcbiAgICBpZD86IHN0cmluZztcbiAgICBuYW1lPzogc3RyaW5nO1xuICB9O1xuICB0YWdzPzogc3RyaW5nW107XG4gIGZlYXR1cmVzPzogQXJyYXk8eyBpZDogc3RyaW5nIH0+O1xuICBjcmVhdGVkQXQ6IHN0cmluZztcbiAgdXBkYXRlZEF0OiBzdHJpbmc7XG4gIGxpbmtzPzoge1xuICAgIHNlbGY6IHN0cmluZztcbiAgICBodG1sOiBzdHJpbmc7XG4gIH07XG59XG5cbmV4cG9ydCBpbnRlcmZhY2UgQ3JlYXRlTm90ZVBhcmFtcyB7XG4gIHRpdGxlPzogc3RyaW5nO1xuICBjb250ZW50OiBzdHJpbmc7XG4gIGRpc3BsYXlVcmw/OiBzdHJpbmc7XG4gIHNvdXJjZT86IHtcbiAgICBvcmlnaW4/OiBzdHJpbmc7XG4gICAgcmVjb3JkX2lkPzogc3RyaW5nO1xuICB9O1xuICB1c2VyPzoge1xuICAgIGVtYWlsOiBzdHJpbmc7XG4gICAgbmFtZT86IHN0cmluZztcbiAgfTtcbiAgY29tcGFueT86IHtcbiAgICBpZD86IHN0cmluZztcbiAgICBuYW1lPzogc3RyaW5nO1xuICB9O1xuICB0YWdzPzogc3RyaW5nW107XG59XG5cbmV4cG9ydCBpbnRlcmZhY2UgTGlzdE5vdGVzUGFyYW1zIHtcbiAgbGltaXQ/OiBudW1iZXI7XG4gIGN1cnNvcj86IHN0cmluZztcbiAgY3JlYXRlZEZyb20/OiBzdHJpbmc7XG4gIGNyZWF0ZWRUbz86IHN0cmluZztcbn1cblxuZXhwb3J0IGludGVyZmFjZSBBdHRhY2hOb3RlUGFyYW1zIHtcbiAgbm90ZUlkOiBzdHJpbmc7XG4gIGZlYXR1cmVJZDogc3RyaW5nO1xufVxuXG4vLyBVc2VyIHR5cGVzXG5leHBvcnQgaW50ZXJmYWNlIFVzZXIge1xuICBpZDogc3RyaW5nO1xuICBlbWFpbDogc3RyaW5nO1xuICBuYW1lPzogc3RyaW5nO1xuICByb2xlPzogc3RyaW5nO1xuICBjcmVhdGVkQXQ/OiBzdHJpbmc7XG59XG5cbmV4cG9ydCBpbnRlcmZhY2UgQ3VycmVudFVzZXIgZXh0ZW5kcyBVc2VyIHtcbiAgd29ya3NwYWNlSWQ/OiBzdHJpbmc7XG4gIHdvcmtzcGFjZU5hbWU/OiBzdHJpbmc7XG59XG5cbmV4cG9ydCBpbnRlcmZhY2UgTGlzdFVzZXJzUGFyYW1zIHtcbiAgbGltaXQ/OiBudW1iZXI7XG4gIGN1cnNvcj86IHN0cmluZztcbn1cblxuLy8gU2VhcmNoIHR5cGVzXG5leHBvcnQgaW50ZXJmYWNlIFNlYXJjaFBhcmFtcyB7XG4gIHF1ZXJ5OiBzdHJpbmc7XG4gIHR5cGU/OiAnZmVhdHVyZScgfCAnbm90ZScgfCAncHJvZHVjdCcgfCAnY29tcG9uZW50JztcbiAgbGltaXQ/OiBudW1iZXI7XG59XG5cbmV4cG9ydCBpbnRlcmZhY2UgU2VhcmNoUmVzdWx0IHtcbiAgdHlwZTogJ2ZlYXR1cmUnIHwgJ25vdGUnIHwgJ3Byb2R1Y3QnIHwgJ2NvbXBvbmVudCc7XG4gIGlkOiBzdHJpbmc7XG4gIG5hbWU/OiBzdHJpbmc7XG4gIHRpdGxlPzogc3RyaW5nO1xuICBjb250ZW50Pzogc3RyaW5nO1xuICBkZXNjcmlwdGlvbj86IHN0cmluZztcbiAgc2NvcmU/OiBudW1iZXI7XG4gIGxpbmtzPzoge1xuICAgIHNlbGY6IHN0cmluZztcbiAgICBodG1sOiBzdHJpbmc7XG4gIH07XG59XG5cbi8vIFBsdWdpbiBjb25maWd1cmF0aW9uXG5leHBvcnQgaW50ZXJmYWNlIFBsdWdpbkNvbmZpZyB7XG4gIGFwaVRva2VuOiBzdHJpbmc7XG4gIGFwaUJhc2VVcmw/OiBzdHJpbmc7XG4gIGNhY2hlVHRsU2Vjb25kcz86IG51bWJlcjtcbiAgcmF0ZUxpbWl0UGVyTWludXRlPzogbnVtYmVyO1xufVxuXG4vLyBPcGVuQ2xhdyBQbHVnaW4gQVBJIHR5cGVzXG5leHBvcnQgaW50ZXJmYWNlIFRvb2xEZWZpbml0aW9uIHtcbiAgbmFtZTogc3RyaW5nO1xuICBkZXNjcmlwdGlvbjogc3RyaW5nO1xuICBwYXJhbWV0ZXJzOiB7XG4gICAgdHlwZTogJ29iamVjdCc7XG4gICAgcHJvcGVydGllczogUmVjb3JkPHN0cmluZywgUGFyYW1ldGVyU2NoZW1hPjtcbiAgICByZXF1aXJlZD86IHN0cmluZ1tdO1xuICB9O1xuICBoYW5kbGVyOiAocGFyYW1zOiBSZWNvcmQ8c3RyaW5nLCB1bmtub3duPikgPT4gUHJvbWlzZTx1bmtub3duPjtcbn1cblxuZXhwb3J0IGludGVyZmFjZSBQYXJhbWV0ZXJTY2hlbWEge1xuICB0eXBlOiAnc3RyaW5nJyB8ICdudW1iZXInIHwgJ2Jvb2xlYW4nIHwgJ2FycmF5JyB8ICdvYmplY3QnO1xuICBkZXNjcmlwdGlvbj86IHN0cmluZztcbiAgZW51bT86IHN0cmluZ1tdO1xuICBkZWZhdWx0PzogdW5rbm93bjtcbiAgaXRlbXM/OiBQYXJhbWV0ZXJTY2hlbWE7XG4gIHByb3BlcnRpZXM/OiBSZWNvcmQ8c3RyaW5nLCBQYXJhbWV0ZXJTY2hlbWE+O1xufVxuXG5leHBvcnQgaW50ZXJmYWNlIFBsdWdpbkFQSSB7XG4gIGNvbmZpZzogUGx1Z2luQ29uZmlnO1xuICByZWdpc3RlclRvb2w6ICh0b29sOiBUb29sRGVmaW5pdGlvbikgPT4gdm9pZDtcbiAgbG9nOiB7XG4gICAgaW5mbzogKG1lc3NhZ2U6IHN0cmluZywgZGF0YT86IHVua25vd24pID0+IHZvaWQ7XG4gICAgd2FybjogKG1lc3NhZ2U6IHN0cmluZywgZGF0YT86IHVua25vd24pID0+IHZvaWQ7XG4gICAgZXJyb3I6IChtZXNzYWdlOiBzdHJpbmcsIGRhdGE/OiB1bmtub3duKSA9PiB2b2lkO1xuICAgIGRlYnVnOiAobWVzc2FnZTogc3RyaW5nLCBkYXRhPzogdW5rbm93bikgPT4gdm9pZDtcbiAgfTtcbn1cbiJdfQ==
package/dist/index.d.ts CHANGED
@@ -4,11 +4,11 @@
4
4
  * Provides integration with ProductBoard for managing features, products,
5
5
  * customer feedback notes, and workspace users.
6
6
  */
7
- import { PluginAPI } from './client/types';
7
+ import { PluginAPI } from './client/types.js';
8
8
  /**
9
9
  * Plugin registration function called by OpenClaw
10
10
  */
11
11
  export default function register(api: PluginAPI): void;
12
- export type { PluginAPI, PluginConfig } from './client/types';
13
- export { ProductBoardClient } from './client/api-client';
14
- export { ProductBoardError } from './client/errors';
12
+ export type { PluginAPI, PluginConfig } from './client/types.js';
13
+ export { ProductBoardClient } from './client/api-client.js';
14
+ export { ProductBoardError } from './client/errors.js';