openclaw-productboard 1.0.13 → 1.0.15
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.
|
@@ -147,6 +147,9 @@ export class ProductBoardClient {
|
|
|
147
147
|
this.cache.invalidatePattern('pb_feature_search:');
|
|
148
148
|
}
|
|
149
149
|
async searchFeatures(query, limit = 50) {
|
|
150
|
+
if (!query) {
|
|
151
|
+
return [];
|
|
152
|
+
}
|
|
150
153
|
const cacheKey = ApiCache.generateKey('pb_feature_search', { query, limit });
|
|
151
154
|
return this.cache.wrap(cacheKey, async () => {
|
|
152
155
|
// ProductBoard doesn't have a dedicated search endpoint for features
|
|
@@ -271,6 +274,9 @@ export class ProductBoardClient {
|
|
|
271
274
|
// Search Methods
|
|
272
275
|
// ============================================
|
|
273
276
|
async search(params) {
|
|
277
|
+
if (!params.query) {
|
|
278
|
+
return [];
|
|
279
|
+
}
|
|
274
280
|
const cacheKey = ApiCache.generateKey('pb_search', { query: params.query, type: params.type, limit: params.limit });
|
|
275
281
|
return this.cache.wrap(cacheKey, async () => {
|
|
276
282
|
const results = [];
|
|
@@ -369,4 +375,4 @@ export class ProductBoardClient {
|
|
|
369
375
|
return this.rateLimiter.stats();
|
|
370
376
|
}
|
|
371
377
|
}
|
|
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"]}
|
|
378
|
+
//# 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    if (!query) {\n      return [];\n    }\n\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    if (!params.query) {\n      return [];\n    }\n\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"]}
|
package/dist/client/types.d.ts
CHANGED
|
@@ -240,7 +240,7 @@ export interface ToolDefinition {
|
|
|
240
240
|
properties: Record<string, ParameterSchema>;
|
|
241
241
|
required?: string[];
|
|
242
242
|
};
|
|
243
|
-
|
|
243
|
+
execute: (params: Record<string, unknown>) => Promise<unknown>;
|
|
244
244
|
}
|
|
245
245
|
export interface ParameterSchema {
|
|
246
246
|
type: 'string' | 'number' | 'boolean' | 'array' | 'object';
|
package/dist/client/types.js
CHANGED
|
@@ -2,4 +2,4 @@
|
|
|
2
2
|
* ProductBoard API Type Definitions
|
|
3
3
|
*/
|
|
4
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/
|
|
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/OiBzdHJpbmc7XG4gIGNhY2hlVHRsU2Vjb25kcz86IG51bWJlcjtcbiAgcmF0ZUxpbWl0UGVyTWludXRlPzogbnVtYmVyO1xufVxuXG4vLyBPcGVuQ2xhdyBQbHVnaW4gQVBJIHR5cGVzXG5leHBvcnQgaW50ZXJmYWNlIFRvb2xEZWZpbml0aW9uIHtcbiAgbmFtZTogc3RyaW5nO1xuICBkZXNjcmlwdGlvbjogc3RyaW5nO1xuICBwYXJhbWV0ZXJzOiB7XG4gICAgdHlwZTogJ29iamVjdCc7XG4gICAgcHJvcGVydGllczogUmVjb3JkPHN0cmluZywgUGFyYW1ldGVyU2NoZW1hPjtcbiAgICByZXF1aXJlZD86IHN0cmluZ1tdO1xuICB9O1xuICBleGVjdXRlOiAocGFyYW1zOiBSZWNvcmQ8c3RyaW5nLCB1bmtub3duPikgPT4gUHJvbWlzZTx1bmtub3duPjtcbn1cblxuZXhwb3J0IGludGVyZmFjZSBQYXJhbWV0ZXJTY2hlbWEge1xuICB0eXBlOiAnc3RyaW5nJyB8ICdudW1iZXInIHwgJ2Jvb2xlYW4nIHwgJ2FycmF5JyB8ICdvYmplY3QnO1xuICBkZXNjcmlwdGlvbj86IHN0cmluZztcbiAgZW51bT86IHN0cmluZ1tdO1xuICBkZWZhdWx0PzogdW5rbm93bjtcbiAgaXRlbXM/OiBQYXJhbWV0ZXJTY2hlbWE7XG4gIHByb3BlcnRpZXM/OiBSZWNvcmQ8c3RyaW5nLCBQYXJhbWV0ZXJTY2hlbWE+O1xufVxuXG5leHBvcnQgaW50ZXJmYWNlIFBsdWdpbkFQSSB7XG4gIGNvbmZpZzogUmVjb3JkPHN0cmluZywgdW5rbm93bj47ICAvLyBGdWxsIE9wZW5DbGF3IGNvbmZpZ1xuICBwbHVnaW5Db25maWc/OiBQbHVnaW5Db25maWc7ICAgICAgLy8gUGx1Z2luLXNwZWNpZmljIGNvbmZpZyAocHJlZmVycmVkKVxuICByZWdpc3RlclRvb2w6ICh0b29sOiBUb29sRGVmaW5pdGlvbikgPT4gdm9pZDtcbiAgbG9nPzoge1xuICAgIGluZm86IChtZXNzYWdlOiBzdHJpbmcsIGRhdGE/OiB1bmtub3duKSA9PiB2b2lkO1xuICAgIHdhcm46IChtZXNzYWdlOiBzdHJpbmcsIGRhdGE/OiB1bmtub3duKSA9PiB2b2lkO1xuICAgIGVycm9yOiAobWVzc2FnZTogc3RyaW5nLCBkYXRhPzogdW5rbm93bikgPT4gdm9pZDtcbiAgICBkZWJ1ZzogKG1lc3NhZ2U6IHN0cmluZywgZGF0YT86IHVua25vd24pID0+IHZvaWQ7XG4gIH07XG59XG4iXX0=
|
package/dist/tools/features.js
CHANGED
|
@@ -51,7 +51,7 @@ export function createFeatureTools(client) {
|
|
|
51
51
|
},
|
|
52
52
|
required: ['name'],
|
|
53
53
|
},
|
|
54
|
-
|
|
54
|
+
execute: async (params) => {
|
|
55
55
|
const createParams = {
|
|
56
56
|
name: params.name,
|
|
57
57
|
description: params.description,
|
|
@@ -121,7 +121,7 @@ export function createFeatureTools(client) {
|
|
|
121
121
|
},
|
|
122
122
|
},
|
|
123
123
|
},
|
|
124
|
-
|
|
124
|
+
execute: async (params) => {
|
|
125
125
|
const listParams = {
|
|
126
126
|
productId: params.productId,
|
|
127
127
|
componentId: params.componentId,
|
|
@@ -157,7 +157,7 @@ export function createFeatureTools(client) {
|
|
|
157
157
|
},
|
|
158
158
|
required: ['id'],
|
|
159
159
|
},
|
|
160
|
-
|
|
160
|
+
execute: async (params) => {
|
|
161
161
|
const feature = await client.getFeature(params.id);
|
|
162
162
|
return {
|
|
163
163
|
id: feature.id,
|
|
@@ -220,7 +220,7 @@ export function createFeatureTools(client) {
|
|
|
220
220
|
},
|
|
221
221
|
required: ['id'],
|
|
222
222
|
},
|
|
223
|
-
|
|
223
|
+
execute: async (params) => {
|
|
224
224
|
const updateParams = {};
|
|
225
225
|
if (params.name)
|
|
226
226
|
updateParams.name = params.name;
|
|
@@ -270,7 +270,7 @@ export function createFeatureTools(client) {
|
|
|
270
270
|
},
|
|
271
271
|
required: ['id'],
|
|
272
272
|
},
|
|
273
|
-
|
|
273
|
+
execute: async (params) => {
|
|
274
274
|
await client.deleteFeature(params.id);
|
|
275
275
|
return {
|
|
276
276
|
success: true,
|
|
@@ -297,7 +297,7 @@ export function createFeatureTools(client) {
|
|
|
297
297
|
},
|
|
298
298
|
required: ['query'],
|
|
299
299
|
},
|
|
300
|
-
|
|
300
|
+
execute: async (params) => {
|
|
301
301
|
const features = await client.searchFeatures(params.query, params.limit || 20);
|
|
302
302
|
return {
|
|
303
303
|
count: features.length,
|
|
@@ -313,4 +313,4 @@ export function createFeatureTools(client) {
|
|
|
313
313
|
},
|
|
314
314
|
];
|
|
315
315
|
}
|
|
316
|
-
//# 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.js';\nimport {\n  ToolDefinition,\n  CreateFeatureParams,\n  UpdateFeatureParams,\n  ListFeaturesParams,\n  FeatureStatus,\n} from '../client/types.js';\nimport { cleanText } from '../utils/sanitize.js';\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: cleanText(f.description, 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: cleanText(f.description, 200),\n            url: f.links?.html,\n          })),\n        };\n      },\n    },\n  ];\n}\n"]}
|
|
316
|
+
//# 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.js';\nimport {\n  ToolDefinition,\n  CreateFeatureParams,\n  UpdateFeatureParams,\n  ListFeaturesParams,\n  FeatureStatus,\n} from '../client/types.js';\nimport { cleanText } from '../utils/sanitize.js';\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      execute: 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      execute: 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: cleanText(f.description, 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      execute: 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      execute: 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      execute: 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      execute: 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: cleanText(f.description, 200),\n            url: f.links?.html,\n          })),\n        };\n      },\n    },\n  ];\n}\n"]}
|
package/dist/tools/notes.js
CHANGED
|
@@ -55,7 +55,7 @@ export function createNoteTools(client) {
|
|
|
55
55
|
},
|
|
56
56
|
required: ['content'],
|
|
57
57
|
},
|
|
58
|
-
|
|
58
|
+
execute: async (params) => {
|
|
59
59
|
const createParams = {
|
|
60
60
|
content: params.content,
|
|
61
61
|
title: params.title,
|
|
@@ -120,7 +120,7 @@ export function createNoteTools(client) {
|
|
|
120
120
|
},
|
|
121
121
|
},
|
|
122
122
|
},
|
|
123
|
-
|
|
123
|
+
execute: async (params) => {
|
|
124
124
|
const listParams = {
|
|
125
125
|
limit: params.limit || 50,
|
|
126
126
|
createdFrom: params.createdFrom,
|
|
@@ -161,7 +161,7 @@ export function createNoteTools(client) {
|
|
|
161
161
|
},
|
|
162
162
|
required: ['noteId', 'featureId'],
|
|
163
163
|
},
|
|
164
|
-
|
|
164
|
+
execute: async (params) => {
|
|
165
165
|
await client.attachNoteToFeature(params.noteId, params.featureId);
|
|
166
166
|
return {
|
|
167
167
|
success: true,
|
|
@@ -171,4 +171,4 @@ export function createNoteTools(client) {
|
|
|
171
171
|
},
|
|
172
172
|
];
|
|
173
173
|
}
|
|
174
|
-
//# 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.js';\nimport { ToolDefinition, CreateNoteParams, ListNotesParams } from '../client/types.js';\nimport { cleanText } from '../utils/sanitize.js';\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: cleanText(note.content, 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: cleanText(n.content, 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"]}
|
|
174
|
+
//# 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.js';\nimport { ToolDefinition, CreateNoteParams, ListNotesParams } from '../client/types.js';\nimport { cleanText } from '../utils/sanitize.js';\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      execute: 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: cleanText(note.content, 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      execute: 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: cleanText(n.content, 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      execute: 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"]}
|
package/dist/tools/products.js
CHANGED
|
@@ -18,7 +18,7 @@ export function createProductTools(client) {
|
|
|
18
18
|
},
|
|
19
19
|
},
|
|
20
20
|
},
|
|
21
|
-
|
|
21
|
+
execute: async (params) => {
|
|
22
22
|
const products = await client.listProducts({
|
|
23
23
|
limit: params.limit || 50,
|
|
24
24
|
});
|
|
@@ -53,7 +53,7 @@ export function createProductTools(client) {
|
|
|
53
53
|
},
|
|
54
54
|
required: ['id'],
|
|
55
55
|
},
|
|
56
|
-
|
|
56
|
+
execute: async (params) => {
|
|
57
57
|
const product = await client.getProduct(params.id);
|
|
58
58
|
const result = {
|
|
59
59
|
id: product.id,
|
|
@@ -87,7 +87,7 @@ export function createProductTools(client) {
|
|
|
87
87
|
type: 'object',
|
|
88
88
|
properties: {},
|
|
89
89
|
},
|
|
90
|
-
|
|
90
|
+
execute: async () => {
|
|
91
91
|
const hierarchy = await client.getProductHierarchy();
|
|
92
92
|
// Build tree structure
|
|
93
93
|
const productMap = new Map();
|
|
@@ -143,4 +143,4 @@ export function createProductTools(client) {
|
|
|
143
143
|
},
|
|
144
144
|
];
|
|
145
145
|
}
|
|
146
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"products.js","sourceRoot":"","sources":["../../src/tools/products.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * ProductBoard Product Management Tools\n */\n\nimport { ProductBoardClient } from '../client/api-client.js';\nimport { ToolDefinition } from '../client/types.js';\nimport { cleanText } from '../utils/sanitize.js';\n\nexport function createProductTools(client: ProductBoardClient): ToolDefinition[] {\n  return [\n    // pb_product_list\n    {\n      name: 'pb_product_list',\n      description:\n        'List all products in the ProductBoard workspace. Products are top-level containers for organizing features.',\n      parameters: {\n        type: 'object',\n        properties: {\n          limit: {\n            type: 'number',\n            description: 'Maximum number of products to return (default: 50)',\n            default: 50,\n          },\n        },\n      },\n      handler: async (params) => {\n        const products = await client.listProducts({\n          limit: params.limit as number || 50,\n        });\n        return {\n          count: products.length,\n          products: products.map((p) => ({\n            id: p.id,\n            name: p.name,\n            description: cleanText(p.description, 200),\n            createdAt: p.createdAt,\n            url: p.links?.html,\n          })),\n        };\n      },\n    },\n\n    // pb_product_get\n    {\n      name: 'pb_product_get',\n      description:\n        'Get detailed information about a specific product by ID, including its components.',\n      parameters: {\n        type: 'object',\n        properties: {\n          id: {\n            type: 'string',\n            description: 'Product ID',\n          },\n          includeComponents: {\n            type: 'boolean',\n            description: 'Include the list of components under this product',\n            default: true,\n          },\n        },\n        required: ['id'],\n      },\n      handler: async (params) => {\n        const product = await client.getProduct(params.id as string);\n\n        const result: Record<string, unknown> = {\n          id: product.id,\n          name: product.name,\n          description: product.description,\n          createdAt: product.createdAt,\n          updatedAt: product.updatedAt,\n          url: product.links?.html,\n        };\n\n        // Optionally include components\n        if (params.includeComponents !== false) {\n          const components = await client.listComponents({\n            productId: product.id,\n            limit: 100,\n          });\n          result.components = components.map((c) => ({\n            id: c.id,\n            name: c.name,\n            description: cleanText(c.description, 200),\n          }));\n          result.componentCount = components.length;\n        }\n\n        return result;\n      },\n    },\n\n    // pb_product_hierarchy\n    {\n      name: 'pb_product_hierarchy',\n      description:\n        'Get the complete product hierarchy including all products and their components. Useful for understanding the workspace structure.',\n      parameters: {\n        type: 'object',\n        properties: {},\n      },\n      handler: async () => {\n        const hierarchy = await client.getProductHierarchy();\n\n        // Build tree structure\n        const productMap = new Map<string, {\n          id: string;\n          name: string;\n          description?: string;\n          components: Array<{\n            id: string;\n            name: string;\n            description?: string;\n            subcomponents: Array<{ id: string; name: string }>;\n          }>;\n        }>();\n\n        // Initialize products\n        for (const product of hierarchy.products) {\n          productMap.set(product.id, {\n            id: product.id,\n            name: product.name,\n            description: cleanText(product.description, 200),\n            components: [],\n          });\n        }\n\n        // Build component tree (components can be nested)\n        const componentMap = new Map<string, typeof hierarchy.components[0] & { subcomponents: Array<{ id: string; name: string }> }>();\n\n        for (const component of hierarchy.components) {\n          componentMap.set(component.id, {\n            ...component,\n            subcomponents: [],\n          });\n        }\n\n        // Assign components to parents\n        for (const component of hierarchy.components) {\n          const comp = componentMap.get(component.id)!;\n\n          if (component.parent?.product?.id) {\n            // Top-level component under a product\n            const product = productMap.get(component.parent.product.id);\n            if (product) {\n              product.components.push({\n                id: comp.id,\n                name: comp.name,\n                description: cleanText(comp.description, 200),\n                subcomponents: comp.subcomponents,\n              });\n            }\n          } else if (component.parent?.component?.id) {\n            // Nested component\n            const parent = componentMap.get(component.parent.component.id);\n            if (parent) {\n              parent.subcomponents.push({\n                id: comp.id,\n                name: comp.name,\n              });\n            }\n          }\n        }\n\n        return {\n          productCount: hierarchy.products.length,\n          componentCount: hierarchy.components.length,\n          hierarchy: Array.from(productMap.values()),\n        };\n      },\n    },\n  ];\n}\n"]}
|
|
146
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"products.js","sourceRoot":"","sources":["../../src/tools/products.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * ProductBoard Product Management Tools\n */\n\nimport { ProductBoardClient } from '../client/api-client.js';\nimport { ToolDefinition } from '../client/types.js';\nimport { cleanText } from '../utils/sanitize.js';\n\nexport function createProductTools(client: ProductBoardClient): ToolDefinition[] {\n  return [\n    // pb_product_list\n    {\n      name: 'pb_product_list',\n      description:\n        'List all products in the ProductBoard workspace. Products are top-level containers for organizing features.',\n      parameters: {\n        type: 'object',\n        properties: {\n          limit: {\n            type: 'number',\n            description: 'Maximum number of products to return (default: 50)',\n            default: 50,\n          },\n        },\n      },\n      execute: async (params) => {\n        const products = await client.listProducts({\n          limit: params.limit as number || 50,\n        });\n        return {\n          count: products.length,\n          products: products.map((p) => ({\n            id: p.id,\n            name: p.name,\n            description: cleanText(p.description, 200),\n            createdAt: p.createdAt,\n            url: p.links?.html,\n          })),\n        };\n      },\n    },\n\n    // pb_product_get\n    {\n      name: 'pb_product_get',\n      description:\n        'Get detailed information about a specific product by ID, including its components.',\n      parameters: {\n        type: 'object',\n        properties: {\n          id: {\n            type: 'string',\n            description: 'Product ID',\n          },\n          includeComponents: {\n            type: 'boolean',\n            description: 'Include the list of components under this product',\n            default: true,\n          },\n        },\n        required: ['id'],\n      },\n      execute: async (params) => {\n        const product = await client.getProduct(params.id as string);\n\n        const result: Record<string, unknown> = {\n          id: product.id,\n          name: product.name,\n          description: product.description,\n          createdAt: product.createdAt,\n          updatedAt: product.updatedAt,\n          url: product.links?.html,\n        };\n\n        // Optionally include components\n        if (params.includeComponents !== false) {\n          const components = await client.listComponents({\n            productId: product.id,\n            limit: 100,\n          });\n          result.components = components.map((c) => ({\n            id: c.id,\n            name: c.name,\n            description: cleanText(c.description, 200),\n          }));\n          result.componentCount = components.length;\n        }\n\n        return result;\n      },\n    },\n\n    // pb_product_hierarchy\n    {\n      name: 'pb_product_hierarchy',\n      description:\n        'Get the complete product hierarchy including all products and their components. Useful for understanding the workspace structure.',\n      parameters: {\n        type: 'object',\n        properties: {},\n      },\n      execute: async () => {\n        const hierarchy = await client.getProductHierarchy();\n\n        // Build tree structure\n        const productMap = new Map<string, {\n          id: string;\n          name: string;\n          description?: string;\n          components: Array<{\n            id: string;\n            name: string;\n            description?: string;\n            subcomponents: Array<{ id: string; name: string }>;\n          }>;\n        }>();\n\n        // Initialize products\n        for (const product of hierarchy.products) {\n          productMap.set(product.id, {\n            id: product.id,\n            name: product.name,\n            description: cleanText(product.description, 200),\n            components: [],\n          });\n        }\n\n        // Build component tree (components can be nested)\n        const componentMap = new Map<string, typeof hierarchy.components[0] & { subcomponents: Array<{ id: string; name: string }> }>();\n\n        for (const component of hierarchy.components) {\n          componentMap.set(component.id, {\n            ...component,\n            subcomponents: [],\n          });\n        }\n\n        // Assign components to parents\n        for (const component of hierarchy.components) {\n          const comp = componentMap.get(component.id)!;\n\n          if (component.parent?.product?.id) {\n            // Top-level component under a product\n            const product = productMap.get(component.parent.product.id);\n            if (product) {\n              product.components.push({\n                id: comp.id,\n                name: comp.name,\n                description: cleanText(comp.description, 200),\n                subcomponents: comp.subcomponents,\n              });\n            }\n          } else if (component.parent?.component?.id) {\n            // Nested component\n            const parent = componentMap.get(component.parent.component.id);\n            if (parent) {\n              parent.subcomponents.push({\n                id: comp.id,\n                name: comp.name,\n              });\n            }\n          }\n        }\n\n        return {\n          productCount: hierarchy.products.length,\n          componentCount: hierarchy.components.length,\n          hierarchy: Array.from(productMap.values()),\n        };\n      },\n    },\n  ];\n}\n"]}
|
package/dist/tools/search.js
CHANGED
|
@@ -27,7 +27,7 @@ export function createSearchTools(client) {
|
|
|
27
27
|
},
|
|
28
28
|
required: ['query'],
|
|
29
29
|
},
|
|
30
|
-
|
|
30
|
+
execute: async (params) => {
|
|
31
31
|
const searchParams = {
|
|
32
32
|
query: params.query,
|
|
33
33
|
type: params.type,
|
|
@@ -67,7 +67,7 @@ export function createSearchTools(client) {
|
|
|
67
67
|
type: 'object',
|
|
68
68
|
properties: {},
|
|
69
69
|
},
|
|
70
|
-
|
|
70
|
+
execute: async () => {
|
|
71
71
|
const user = await client.getCurrentUser();
|
|
72
72
|
return {
|
|
73
73
|
id: user.id,
|
|
@@ -93,7 +93,7 @@ export function createSearchTools(client) {
|
|
|
93
93
|
},
|
|
94
94
|
},
|
|
95
95
|
},
|
|
96
|
-
|
|
96
|
+
execute: async (params) => {
|
|
97
97
|
const users = await client.listUsers({
|
|
98
98
|
limit: params.limit || 100,
|
|
99
99
|
});
|
|
@@ -110,4 +110,4 @@ export function createSearchTools(client) {
|
|
|
110
110
|
},
|
|
111
111
|
];
|
|
112
112
|
}
|
|
113
|
-
//# sourceMappingURL=data:application/json;base64,
|
|
113
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"search.js","sourceRoot":"","sources":["../../src/tools/search.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * ProductBoard Search and User Tools\n */\n\nimport { ProductBoardClient } from '../client/api-client.js';\nimport { ToolDefinition, SearchParams } from '../client/types.js';\n\nexport function createSearchTools(client: ProductBoardClient): ToolDefinition[] {\n  return [\n    // pb_search\n    {\n      name: 'pb_search',\n      description:\n        'Global search across ProductBoard. Searches features, products, components, and notes by name, title, or content.',\n      parameters: {\n        type: 'object',\n        properties: {\n          query: {\n            type: 'string',\n            description: 'Search query text',\n          },\n          type: {\n            type: 'string',\n            description: 'Limit search to specific type',\n            enum: ['feature', 'product', 'component', 'note'],\n          },\n          limit: {\n            type: 'number',\n            description: 'Maximum results to return (default: 25)',\n            default: 25,\n          },\n        },\n        required: ['query'],\n      },\n      execute: async (params) => {\n        const searchParams: SearchParams = {\n          query: params.query as string,\n          type: params.type as 'feature' | 'note' | 'product' | 'component' | undefined,\n          limit: params.limit as number || 25,\n        };\n\n        const results = await client.search(searchParams);\n\n        // Group results by type\n        const grouped: Record<string, unknown[]> = {\n          features: [],\n          products: [],\n          components: [],\n          notes: [],\n        };\n\n        for (const result of results) {\n          const key = result.type + 's';\n          if (grouped[key]) {\n            grouped[key].push({\n              id: result.id,\n              name: result.name || result.title,\n              description: result.description || result.content,\n              url: result.links?.html,\n            });\n          }\n        }\n\n        return {\n          totalCount: results.length,\n          query: params.query,\n          results: grouped,\n        };\n      },\n    },\n\n    // pb_user_current\n    {\n      name: 'pb_user_current',\n      description:\n        'Get information about the currently authenticated user, including workspace details.',\n      parameters: {\n        type: 'object',\n        properties: {},\n      },\n      execute: async () => {\n        const user = await client.getCurrentUser();\n        return {\n          id: user.id,\n          email: user.email,\n          name: user.name,\n          role: user.role,\n          workspaceId: user.workspaceId,\n          workspaceName: user.workspaceName,\n        };\n      },\n    },\n\n    // pb_user_list\n    {\n      name: 'pb_user_list',\n      description:\n        'List all users in the ProductBoard workspace. Useful for finding user IDs for assigning features.',\n      parameters: {\n        type: 'object',\n        properties: {\n          limit: {\n            type: 'number',\n            description: 'Maximum number of users to return (default: 100)',\n            default: 100,\n          },\n        },\n      },\n      execute: async (params) => {\n        const users = await client.listUsers({\n          limit: params.limit as number || 100,\n        });\n        return {\n          count: users.length,\n          users: users.map((u) => ({\n            id: u.id,\n            email: u.email,\n            name: u.name,\n            role: u.role,\n          })),\n        };\n      },\n    },\n  ];\n}\n"]}
|