mirror of
https://github.com/simstudioai/sim.git
synced 2026-02-07 05:05:15 -05:00
* feat(tools): added calcom * added more triggers, tested * updated regex in script for release to be more lenient * fix(tag-dropdown): performance improvements and scroll bug fixes - Add flatTagIndexMap for O(1) tag lookups (replaces O(n²) findIndex calls) - Memoize caret position calculation to avoid DOM manipulation on every render - Use refs for inputValue/cursorPosition to keep handleTagSelect callback stable - Change itemRefs from index-based to tag-based keys to prevent stale refs - Fix scroll jump in nested folders by removing scroll reset from registerFolder - Add onFolderEnter callback for scroll reset when entering folder via keyboard - Disable keyboard navigation wrap-around at boundaries - Simplify selection reset to single effect on flatTagList.length change Also: - Add safeCompare utility for timing-safe string comparison - Refactor webhook signature validation to use safeCompare Co-Authored-By: Claude Opus 4.5 <noreply@anthropic.com> * updated types * fix(calcom): simplify required field constraints for booking attendee The condition field already restricts these to calcom_create_booking, so simplified to required: true. Per Cal.com API docs, email is optional while name and timeZone are required. Co-Authored-By: Claude Opus 4.5 <noreply@anthropic.com> * added tests * updated folder multi select, updated calcom and github tools and docs generator script * updated drag, updated outputs for tools, regen docs with nested docs script * updated setup instructions links, destructure trigger outputs, fix text subblock styling * updated docs gen script * updated docs script * updated docs script * updated script * remove destructuring of stripe webhook * expanded wand textarea, updated calcom tools --------- Co-authored-by: Claude Opus 4.5 <noreply@anthropic.com>
124 lines
3.0 KiB
TypeScript
124 lines
3.0 KiB
TypeScript
import type { LinearListCyclesParams, LinearListCyclesResponse } from '@/tools/linear/types'
|
|
import { CYCLE_FULL_OUTPUT_PROPERTIES, PAGE_INFO_OUTPUT } from '@/tools/linear/types'
|
|
import type { ToolConfig } from '@/tools/types'
|
|
|
|
export const linearListCyclesTool: ToolConfig<LinearListCyclesParams, LinearListCyclesResponse> = {
|
|
id: 'linear_list_cycles',
|
|
name: 'Linear List Cycles',
|
|
description: 'List cycles (sprints/iterations) in Linear',
|
|
version: '1.0.0',
|
|
|
|
oauth: {
|
|
required: true,
|
|
provider: 'linear',
|
|
},
|
|
|
|
params: {
|
|
teamId: {
|
|
type: 'string',
|
|
required: false,
|
|
visibility: 'user-or-llm',
|
|
description: 'Filter by team ID',
|
|
},
|
|
first: {
|
|
type: 'number',
|
|
required: false,
|
|
visibility: 'user-or-llm',
|
|
description: 'Number of cycles 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 ListCycles($filter: CycleFilter, $first: Int, $after: String) {
|
|
cycles(filter: $filter, first: $first, after: $after) {
|
|
nodes {
|
|
id
|
|
number
|
|
name
|
|
startsAt
|
|
endsAt
|
|
completedAt
|
|
progress
|
|
team {
|
|
id
|
|
name
|
|
}
|
|
}
|
|
pageInfo {
|
|
hasNextPage
|
|
endCursor
|
|
}
|
|
}
|
|
}
|
|
`,
|
|
variables: {
|
|
filter: Object.keys(filter).length > 0 ? filter : undefined,
|
|
first: params.first ? Number(params.first) : 50,
|
|
after: params.after,
|
|
},
|
|
}
|
|
},
|
|
},
|
|
|
|
transformResponse: async (response) => {
|
|
const data = await response.json()
|
|
|
|
if (data.errors) {
|
|
return {
|
|
success: false,
|
|
error: data.errors[0]?.message || 'Failed to list cycles',
|
|
output: {},
|
|
}
|
|
}
|
|
|
|
const result = data.data.cycles
|
|
return {
|
|
success: true,
|
|
output: {
|
|
cycles: result.nodes,
|
|
pageInfo: {
|
|
hasNextPage: result.pageInfo.hasNextPage,
|
|
endCursor: result.pageInfo.endCursor,
|
|
},
|
|
},
|
|
}
|
|
},
|
|
|
|
outputs: {
|
|
cycles: {
|
|
type: 'array',
|
|
description: 'Array of cycles',
|
|
items: {
|
|
type: 'object',
|
|
properties: CYCLE_FULL_OUTPUT_PROPERTIES,
|
|
},
|
|
},
|
|
pageInfo: PAGE_INFO_OUTPUT,
|
|
},
|
|
}
|