mirror of
https://github.com/simstudioai/sim.git
synced 2026-02-06 04:35:03 -05:00
* fix(linear): align tool outputs, queries, and pagination with API * fix(linear): coerce first param to number, remove duplicate conditions, add null guard
179 lines
4.5 KiB
TypeScript
179 lines
4.5 KiB
TypeScript
import type { LinearSearchIssuesParams, LinearSearchIssuesResponse } from '@/tools/linear/types'
|
|
import { ISSUE_OUTPUT_PROPERTIES, PAGE_INFO_OUTPUT } from '@/tools/linear/types'
|
|
import type { ToolConfig } from '@/tools/types'
|
|
|
|
export const linearSearchIssuesTool: ToolConfig<
|
|
LinearSearchIssuesParams,
|
|
LinearSearchIssuesResponse
|
|
> = {
|
|
id: 'linear_search_issues',
|
|
name: 'Linear Search Issues',
|
|
description: 'Search for issues in Linear using full-text search',
|
|
version: '1.0.0',
|
|
|
|
oauth: {
|
|
required: true,
|
|
provider: 'linear',
|
|
},
|
|
|
|
params: {
|
|
query: {
|
|
type: 'string',
|
|
required: true,
|
|
visibility: 'user-or-llm',
|
|
description: 'Search query string',
|
|
},
|
|
teamId: {
|
|
type: 'string',
|
|
required: false,
|
|
visibility: 'user-or-llm',
|
|
description: 'Filter by team ID',
|
|
},
|
|
includeArchived: {
|
|
type: 'boolean',
|
|
required: false,
|
|
visibility: 'user-or-llm',
|
|
description: 'Include archived issues in search results',
|
|
},
|
|
first: {
|
|
type: 'number',
|
|
required: false,
|
|
visibility: 'user-or-llm',
|
|
description: 'Number of results to return (default: 50)',
|
|
},
|
|
after: {
|
|
type: 'string',
|
|
required: false,
|
|
visibility: 'user-or-llm',
|
|
description: 'Cursor for pagination',
|
|
},
|
|
},
|
|
|
|
request: {
|
|
url: 'https://api.linear.app/graphql',
|
|
method: 'POST',
|
|
headers: (params) => {
|
|
if (!params.accessToken) {
|
|
throw new Error('Missing access token for Linear API request')
|
|
}
|
|
return {
|
|
'Content-Type': 'application/json',
|
|
Authorization: `Bearer ${params.accessToken}`,
|
|
}
|
|
},
|
|
body: (params) => {
|
|
const filter: Record<string, any> = {}
|
|
if (params.teamId) {
|
|
filter.team = { id: { eq: params.teamId } }
|
|
}
|
|
|
|
return {
|
|
query: `
|
|
query SearchIssues($term: String!, $filter: IssueFilter, $first: Int, $after: String, $includeArchived: Boolean) {
|
|
searchIssues(term: $term, filter: $filter, first: $first, after: $after, includeArchived: $includeArchived) {
|
|
nodes {
|
|
id
|
|
title
|
|
description
|
|
priority
|
|
estimate
|
|
url
|
|
createdAt
|
|
updatedAt
|
|
archivedAt
|
|
state {
|
|
id
|
|
name
|
|
type
|
|
}
|
|
assignee {
|
|
id
|
|
name
|
|
email
|
|
}
|
|
team {
|
|
id
|
|
name
|
|
}
|
|
project {
|
|
id
|
|
name
|
|
}
|
|
labels {
|
|
nodes {
|
|
id
|
|
name
|
|
color
|
|
}
|
|
}
|
|
}
|
|
pageInfo {
|
|
hasNextPage
|
|
endCursor
|
|
}
|
|
}
|
|
}
|
|
`,
|
|
variables: {
|
|
term: params.query,
|
|
filter: Object.keys(filter).length > 0 ? filter : undefined,
|
|
first: params.first ? Number(params.first) : 50,
|
|
after: params.after,
|
|
includeArchived: params.includeArchived || false,
|
|
},
|
|
}
|
|
},
|
|
},
|
|
|
|
transformResponse: async (response) => {
|
|
const data = await response.json()
|
|
|
|
if (data.errors) {
|
|
return {
|
|
success: false,
|
|
error: data.errors[0]?.message || 'Failed to search issues',
|
|
output: {},
|
|
}
|
|
}
|
|
|
|
const result = data.data.searchIssues
|
|
return {
|
|
success: true,
|
|
output: {
|
|
issues: result.nodes.map((issue: any) => ({
|
|
id: issue.id,
|
|
title: issue.title,
|
|
description: issue.description,
|
|
priority: issue.priority,
|
|
estimate: issue.estimate,
|
|
url: issue.url,
|
|
createdAt: issue.createdAt,
|
|
updatedAt: issue.updatedAt,
|
|
archivedAt: issue.archivedAt,
|
|
state: issue.state,
|
|
assignee: issue.assignee,
|
|
teamId: issue.team?.id,
|
|
projectId: issue.project?.id,
|
|
labels: issue.labels?.nodes || [],
|
|
})),
|
|
pageInfo: {
|
|
hasNextPage: result.pageInfo.hasNextPage,
|
|
endCursor: result.pageInfo.endCursor,
|
|
},
|
|
},
|
|
}
|
|
},
|
|
|
|
outputs: {
|
|
issues: {
|
|
type: 'array',
|
|
description: 'Array of matching issues',
|
|
items: {
|
|
type: 'object',
|
|
properties: ISSUE_OUTPUT_PROPERTIES,
|
|
},
|
|
},
|
|
pageInfo: PAGE_INFO_OUTPUT,
|
|
},
|
|
}
|