first commit
This commit is contained in:
		
						commit
						0a45a1f5fb
					
				
					 20 changed files with 992 additions and 0 deletions
				
			
		
							
								
								
									
										5
									
								
								f/app_custom/folder.meta.yaml
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										5
									
								
								f/app_custom/folder.meta.yaml
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,5 @@ | |||
| summary: null | ||||
| display_name: App Custom Components | ||||
| extra_perms: | ||||
|   g/all: false | ||||
| owners: [] | ||||
							
								
								
									
										5
									
								
								f/app_groups/folder.meta.yaml
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										5
									
								
								f/app_groups/folder.meta.yaml
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,5 @@ | |||
| summary: null | ||||
| display_name: App Groups | ||||
| extra_perms: | ||||
|   g/all: false | ||||
| owners: [] | ||||
							
								
								
									
										5
									
								
								f/app_themes/folder.meta.yaml
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										5
									
								
								f/app_themes/folder.meta.yaml
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,5 @@ | |||
| summary: null | ||||
| display_name: App Themes | ||||
| extra_perms: | ||||
|   g/all: false | ||||
| owners: [] | ||||
							
								
								
									
										155
									
								
								f/clickup/create_task.deno.ts
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										155
									
								
								f/clickup/create_task.deno.ts
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,155 @@ | |||
| // clickup/create_task.ts
 | ||||
| type TaskInput = { | ||||
|   // REQUIRED FIELDS
 | ||||
|   /** The ID of the list to put this task into */ | ||||
|   list_id: string; | ||||
|   /** Name of the task (max 99840 chars) */ | ||||
|   name: string; | ||||
| 
 | ||||
|   // OPTIONAL CORE FIELDS  
 | ||||
|   /** Detailed task description (supports markdown) */ | ||||
|   description?: string; | ||||
|   /** Task status must match existing status in List */ | ||||
|   status?: string; | ||||
|   /** Unix timestamp in milliseconds */ | ||||
|   due_date?: number; | ||||
|   /** Unix timestamp in milliseconds */ | ||||
|   start_date?: number; | ||||
|   /** Time estimate in milliseconds */ | ||||
|   time_estimate?: number; | ||||
|   /** Markdown-formatted description */ | ||||
|   markdown_description?: string; | ||||
|   /** Priority level (1-4) */ | ||||
|   priority?: 1 | 2 | 3 | 4; | ||||
|   /** Parent task ID for subtasks */ | ||||
|   parent?: string; | ||||
|   /** Custom task type ID */ | ||||
|   custom_item_id?: number; | ||||
| 
 | ||||
|   // PEOPLE MANAGEMENT  
 | ||||
|   /** Array of user IDs */ | ||||
|   assignees?: number[]; | ||||
|   /** Notify all task members */ | ||||
|   notify_all?: boolean; | ||||
| 
 | ||||
|   // TAGS & RELATIONSHIPS
 | ||||
|   /** Array of tag names */ | ||||
|   tags?: string[]; | ||||
|   /** Task dependencies */ | ||||
|   depends_on?: string[]; | ||||
|   /** Linked task IDs */ | ||||
|   links_to?: string; | ||||
| 
 | ||||
|   // CUSTOMIZATION
 | ||||
|   /** Array of custom field objects */ | ||||
|   custom_fields?: { | ||||
|     /** Custom field ID */ | ||||
|     id: string; | ||||
|     /** Field value (type-specific) */ | ||||
|     value: string | number | boolean | string[]; | ||||
|   }[]; | ||||
| 
 | ||||
|   // VALIDATION FLAGS
 | ||||
|   /** Enforce required custom fields */ | ||||
|   check_required_custom_fields?: boolean; | ||||
| 
 | ||||
|   // ADVANCED OPTIONS
 | ||||
|   /** Team ID for custom task IDs */ | ||||
|   team_id?: number; | ||||
|   /** Custom task ID (requires team_id) */ | ||||
|   custom_task_id?: string; | ||||
|   /** Task creation source (max 6 chars) */ | ||||
|   source?: string; | ||||
| }; | ||||
| 
 | ||||
| // All we need here is the access token.
 | ||||
| type ClickupCredentials = { | ||||
|   access_token: string; | ||||
| }; | ||||
| 
 | ||||
| export async function main( | ||||
|   params: TaskInput, | ||||
|   credentials: ClickupCredentials | ||||
| ) { | ||||
|   // 1. Input Validation
 | ||||
|   if (!params.list_id?.trim()) { | ||||
|     throw new Error("Missing required field: list_id"); | ||||
|   } | ||||
| 
 | ||||
|   if (!params.name?.trim()) { | ||||
|     throw new Error("Missing required field: name"); | ||||
|   } | ||||
| 
 | ||||
|   // 2. API Request Configuration
 | ||||
|   const response = await fetch(`https://api.clickup.com/api/v2/list/${params.list_id}/task`, { | ||||
|     method: "POST", | ||||
|     headers: { | ||||
|       "Content-Type": "application/json", | ||||
|       "accept": 'application/json', | ||||
|       "Authorization": "Bearer " + credentials.access_token | ||||
|     }, | ||||
|     body: JSON.stringify({ | ||||
|       // Required
 | ||||
|       name: params.name, | ||||
| 
 | ||||
|       // Core optional parameters
 | ||||
|       description: params.description, | ||||
|       status: params.status, | ||||
|       due_date: params.due_date, | ||||
|       start_date: params.start_date, | ||||
|       time_estimate: params.time_estimate, | ||||
|       markdown_description: params.markdown_description, | ||||
|       priority: params.priority, | ||||
|       parent: params.parent, | ||||
|       custom_item_id: params.custom_item_id, | ||||
| 
 | ||||
|       // People management
 | ||||
|       assignees: params.assignees, | ||||
|       notify_all: params.notify_all, | ||||
| 
 | ||||
|       // Tags & relationships
 | ||||
|       tags: params.tags, | ||||
|       depends_on: params.depends_on, | ||||
|       links_to: params.links_to, | ||||
| 
 | ||||
|       // Customization
 | ||||
|       custom_fields: params.custom_fields?.map(field => ({ | ||||
|         id: field.id, | ||||
|         value: field.value | ||||
|       })), | ||||
| 
 | ||||
|       // Validation flags
 | ||||
|       check_required_custom_fields: params.check_required_custom_fields, | ||||
| 
 | ||||
|       // Advanced configurations
 | ||||
|       team_id: params.team_id, | ||||
|       custom_task_id: params.custom_task_id, | ||||
|       source: params.source | ||||
|     }) | ||||
| 
 | ||||
|   }); | ||||
| 
 | ||||
|   // 3. Error Handling
 | ||||
|   if (!response.ok) { | ||||
|     const errorBody = await response.text(); | ||||
|     throw new Error(`ClickUp API Error (${response.status}): ${errorBody}`); | ||||
|   } | ||||
| 
 | ||||
|   // 4. Response Processing
 | ||||
|   const result = await response.json(); | ||||
| 
 | ||||
|   // Validate successful task creation
 | ||||
|   if (!result?.id) { | ||||
|     throw new Error("Failed to create task: No ID returned"); | ||||
|   } | ||||
| 
 | ||||
|   return { | ||||
|     task_id: result.id, | ||||
|     url: result.url, | ||||
|     status: result.status?.status || "created", | ||||
|     _metadata: { | ||||
|       clickup_space: result.space?.id, | ||||
|       created_at: new Date().toISOString() | ||||
|     } | ||||
|   }; | ||||
| } | ||||
							
								
								
									
										21
									
								
								f/clickup/create_task.script.yaml
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										21
									
								
								f/clickup/create_task.script.yaml
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,21 @@ | |||
| summary: Create Task | ||||
| description: '' | ||||
| lock: '' | ||||
| kind: script | ||||
| schema: | ||||
|   $schema: 'https://json-schema.org/draft/2020-12/schema' | ||||
|   type: object | ||||
|   properties: | ||||
|     credentials: | ||||
|       type: object | ||||
|       description: '' | ||||
|       default: null | ||||
|       format: resource-clickup_credentials | ||||
|     params: | ||||
|       type: object | ||||
|       description: '' | ||||
|       default: null | ||||
|       format: resource-task_input | ||||
|   required: | ||||
|     - params | ||||
|     - credentials | ||||
							
								
								
									
										7
									
								
								f/clickup/folder.meta.yaml
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										7
									
								
								f/clickup/folder.meta.yaml
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,7 @@ | |||
| summary: '' | ||||
| display_name: clickup | ||||
| extra_perms: | ||||
|   g/all: true | ||||
|   u/peter: true | ||||
| owners: | ||||
|   - u/peter | ||||
							
								
								
									
										205
									
								
								f/clickup/get_task.deno.ts
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										205
									
								
								f/clickup/get_task.deno.ts
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,205 @@ | |||
| // clickup/get_task.ts
 | ||||
| interface TaskResponse { | ||||
|   id: string; | ||||
|   name: string; | ||||
|   description?: string; | ||||
|   text_content?: string; | ||||
|   status?: { | ||||
|     status: string; | ||||
|     color: string; | ||||
|     orderindex: number; | ||||
|     type: "open" | "closed" | "custom"; | ||||
|   }; | ||||
|   orderindex?: string; | ||||
|   date_created?: string; | ||||
|   date_updated?: string; | ||||
|   date_closed?: string | null; | ||||
|   date_done?: string | null; | ||||
|   archived?: boolean; | ||||
|   creator?: { | ||||
|     id: number; | ||||
|     username: string; | ||||
|     email: string; | ||||
|     color: string; | ||||
|     profilePicture?: string; | ||||
|   }; | ||||
|   assignees?: Array<{ | ||||
|     id: number; | ||||
|     username: string; | ||||
|     email: string; | ||||
|     color: string; | ||||
|     initials: string; | ||||
|     profilePicture?: string; | ||||
|   }>; | ||||
|   watchers?: Array<{ | ||||
|     id: number; | ||||
|     username: string; | ||||
|     email: string; | ||||
|     color: string; | ||||
|     initials: string; | ||||
|     profilePicture?: string; | ||||
|   }>; | ||||
|   checklists?: Array<{ | ||||
|     id: string; | ||||
|     name: string; | ||||
|     orderindex: number; | ||||
|     resolved: number; | ||||
|     unresolved: number; | ||||
|     items: Array<{ | ||||
|       id: string; | ||||
|       name: string; | ||||
|       orderindex: number; | ||||
|       assignee?: number; | ||||
|       resolved: boolean; | ||||
|       parent?: string; | ||||
|       date_created: string; | ||||
|       children?: string[]; | ||||
|     }>; | ||||
|   }>; | ||||
|   tags?: Array<{ | ||||
|     name: string; | ||||
|     tag_fg: string; | ||||
|     tag_bg: string; | ||||
|     creator?: number; | ||||
|   }>; | ||||
|   parent?: string | null; | ||||
|   priority?: { | ||||
|     id: string; | ||||
|     priority: "urgent" | "high" | "normal" | "low"; | ||||
|     color: string; | ||||
|   }; | ||||
|   due_date?: string | null; | ||||
|   start_date?: string | null; | ||||
|   time_estimate?: number | null; | ||||
|   time_spent?: number | null; | ||||
|   custom_fields?: Array<{ | ||||
|     id: string; | ||||
|     name: string; | ||||
|     type: string; | ||||
|     type_config: Record<string, any>; | ||||
|     value: any; | ||||
|   }>; | ||||
|   list?: { | ||||
|     id: string; | ||||
|     name: string; | ||||
|     access: boolean; | ||||
|   }; | ||||
|   project?: { | ||||
|     id: string; | ||||
|     name: string; | ||||
|     hidden: boolean; | ||||
|     access: boolean; | ||||
|   }; | ||||
|   folder?: { | ||||
|     id: string; | ||||
|     name: string; | ||||
|     hidden: boolean; | ||||
|     access: boolean; | ||||
|   }; | ||||
|   space?: { | ||||
|     id: string; | ||||
|     name: string; | ||||
|   }; | ||||
|   url?: string; | ||||
|   permission_level?: string; | ||||
|   custom_item_id?: number | null; | ||||
|   custom_task_ids?: Array<{ | ||||
|     custom_task_id: string; | ||||
|     team_id: string; | ||||
|   }>; | ||||
|   dependencies?: Array<{ | ||||
|     task_id: string; | ||||
|     depends_on: string; | ||||
|     type: number; | ||||
|     date_created: string; | ||||
|     userid: string; | ||||
|   }>; | ||||
|   linked_tasks?: Array<{ | ||||
|     task_id: string; | ||||
|     link_id: string; | ||||
|     date_created: string; | ||||
|     userid: string; | ||||
|   }>; | ||||
|   team_id?: string; | ||||
|   custom_id?: string | null; | ||||
|   attachments?: Array<{ | ||||
|     id: string; | ||||
|     version: string; | ||||
|     date: string; | ||||
|     title: string; | ||||
|     extension: string; | ||||
|     thumbnail_small: string; | ||||
|     thumbnail_large: string; | ||||
|     size: number; | ||||
|   }>; | ||||
|   shared?: Array<{ | ||||
|     id: string; | ||||
|     name: string; | ||||
|     type: string; | ||||
|     access_level: string; | ||||
|     team_id: string; | ||||
|   }>; | ||||
|   followers?: Array<{ | ||||
|     id: number; | ||||
|     username: string; | ||||
|     email: string; | ||||
|     color: string; | ||||
|     initials: string; | ||||
|     profilePicture?: string; | ||||
|   }>; | ||||
|   [key: string]: any; // For future API additions
 | ||||
| } | ||||
| 
 | ||||
| type FieldSelector = Array<keyof TaskResponse | string>; | ||||
| 
 | ||||
| // All we need here is the access token.
 | ||||
| type ClickupCredentials = { | ||||
|   access_token: string; | ||||
| }; | ||||
| 
 | ||||
| export async function main( | ||||
|   task_id: string, | ||||
|   fields: FieldSelector = ["id", "name", "status", "date_created"], | ||||
|   credentials: ClickupCredentials, | ||||
| ) { | ||||
|   // Input validation
 | ||||
|   if (!task_id?.trim()) throw new Error("Task ID is required"); | ||||
| 
 | ||||
|   // API call
 | ||||
|   const response = await fetch( | ||||
|     `https://api.clickup.com/api/v2/task/${task_id}`, | ||||
|     { | ||||
|       headers: { | ||||
|         "Authorization": credentials.access_token, | ||||
|       }, | ||||
|     }, | ||||
|   ); | ||||
| 
 | ||||
|   if (!response.ok) { | ||||
|     const error = await response.text(); | ||||
|     throw new Error(`ClickUp Error ${response.status}: ${error}`); | ||||
|   } | ||||
| 
 | ||||
|   // Full response parsing
 | ||||
|   const fullTask = await response.json(); | ||||
| 
 | ||||
|   // Field filtering
 | ||||
|   return fields.length > 0 ? filterFields(fullTask, fields) : fullTask; | ||||
| } | ||||
| 
 | ||||
| // Recursive field filtering function
 | ||||
| function filterFields(obj: any, fields: FieldSelector): any { | ||||
|   return fields.reduce((acc, field) => { | ||||
|     const [root, ...nested] = (field as string).split("."); | ||||
| 
 | ||||
|     if (obj[root] !== undefined) { | ||||
|       if (nested.length > 0 && typeof obj[root] === "object") { | ||||
|         acc[root] = filterFields(obj[root], [nested.join(".")]); | ||||
|       } else { | ||||
|         acc[root] = obj[root]; | ||||
|       } | ||||
|     } | ||||
| 
 | ||||
|     return acc; | ||||
|   }, {} as Record<string, any>); | ||||
| } | ||||
							
								
								
									
										35
									
								
								f/clickup/get_task.script.yaml
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										35
									
								
								f/clickup/get_task.script.yaml
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,35 @@ | |||
| summary: Get Task | ||||
| description: '' | ||||
| lock: '' | ||||
| kind: script | ||||
| schema: | ||||
|   $schema: 'https://json-schema.org/draft/2020-12/schema' | ||||
|   type: object | ||||
|   properties: | ||||
|     credentials: | ||||
|       type: object | ||||
|       description: '' | ||||
|       default: null | ||||
|       format: resource-clickup_credentials | ||||
|       properties: | ||||
|         access_token: | ||||
|           type: string | ||||
|           description: '' | ||||
|           originalType: string | ||||
|     fields: | ||||
|       type: object | ||||
|       description: '' | ||||
|       default: | ||||
|         - id | ||||
|         - name | ||||
|         - status | ||||
|         - date_created | ||||
|       format: resource-field_selector | ||||
|     task_id: | ||||
|       type: string | ||||
|       description: '' | ||||
|       default: null | ||||
|       originalType: string | ||||
|   required: | ||||
|     - task_id | ||||
|     - credentials | ||||
							
								
								
									
										132
									
								
								f/clickup/rotate_tokens.deno.ts
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										132
									
								
								f/clickup/rotate_tokens.deno.ts
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,132 @@ | |||
| import * as wmill from "jsr:@windmill/windmill"; | ||||
| 
 | ||||
| type ClickupCredentials = { | ||||
|   client_id: string; | ||||
|   expires_at: number; | ||||
|   access_token: string; | ||||
|   client_secret: string; | ||||
| }; | ||||
| 
 | ||||
| // Function to initiate OAuth flow and get initial tokens
 | ||||
| export async function get_initial_token( | ||||
|   client_id: string, | ||||
|   client_secret: string, | ||||
|   redirect_uri: string, | ||||
| ) { | ||||
|   const state = crypto.randomUUID(); | ||||
| 
 | ||||
|   const authUrl = `https://app.clickup.com/api?client_id=${encodeURIComponent(client_id) | ||||
|     }&redirect_uri=${encodeURIComponent(redirect_uri)}&state=${encodeURIComponent(state) | ||||
|     }&response_type=code&scope=*`;
 | ||||
| 
 | ||||
|   console.log("Please visit this URL to authorize the application:", authUrl); | ||||
| 
 | ||||
|   return { | ||||
|     auth_url: authUrl, | ||||
|     state: state, | ||||
|     next_step: | ||||
|       "After authorization, you will receive a code in the redirect URI. Use this code with exchange_code_for_token function and verify the state parameter matches.", | ||||
|   }; | ||||
| } | ||||
| 
 | ||||
| // Function to exchange authorization code for tokens
 | ||||
| export async function exchange_code_for_token( | ||||
|   code: string, | ||||
|   client_id: string, | ||||
|   client_secret: string, | ||||
|   expected_state?: string, | ||||
|   received_state?: string, | ||||
| ) { | ||||
|   if (expected_state && received_state) { | ||||
|     if (expected_state !== received_state) { | ||||
|       throw new Error("State parameter mismatch - possible CSRF attack"); | ||||
|     } | ||||
|   } | ||||
| 
 | ||||
|   const formData = new URLSearchParams({ | ||||
|     client_id: client_id, | ||||
|     client_secret: client_secret, | ||||
|     code: code, | ||||
|     grant_type: "authorization_code" | ||||
|   }); | ||||
| 
 | ||||
|   try { | ||||
|     const response = await fetch("https://api.clickup.com/api/v2/oauth/token", { | ||||
|       method: "POST", | ||||
|       headers: { | ||||
|         "Content-Type": "application/x-www-form-urlencoded", | ||||
|       }, | ||||
|       body: formData.toString(), | ||||
|     }); | ||||
| 
 | ||||
|     if (!response.ok) { | ||||
|       const errorText = await response.text(); | ||||
|       console.error("Token exchange error details:", { | ||||
|         status: response.status, | ||||
|         statusText: response.statusText, | ||||
|         error: errorText, | ||||
|         requestBody: formData.toString(), | ||||
|       }); | ||||
|       throw new Error( | ||||
|         `Token exchange failed: ${response.status} ${errorText}`, | ||||
|       ); | ||||
|     } | ||||
| 
 | ||||
|     const tokens = await response.json(); | ||||
| 
 | ||||
|     const credentials: ClickupCredentials = { | ||||
|       client_id, | ||||
|       client_secret, | ||||
|       access_token: tokens.access_token, | ||||
|       expires_at: Date.now() + (tokens.expires_in * 1000), | ||||
|     }; | ||||
| 
 | ||||
|     // Using setResource with string path instead of object
 | ||||
|     await wmill.setResource("f/clickup/clickup_prod_creds", JSON.stringify(credentials)); | ||||
| 
 | ||||
|     return { | ||||
|       status: "initialized", | ||||
|       expires_at: new Date(credentials.expires_at).toISOString(), | ||||
|     }; | ||||
|   } catch (error) { | ||||
|     console.error("Detailed error:", error); | ||||
|     throw error; | ||||
|   } | ||||
| } | ||||
| 
 | ||||
| export async function main( | ||||
|   clickupResource: ClickupCredentials, | ||||
|   mode: "check" | "refresh" | "initialize" = "check", | ||||
|   code?: string, | ||||
|   redirect_uri?: string, | ||||
|   state?: string, | ||||
|   expected_state?: string, | ||||
| ) { | ||||
|   if (mode === "initialize") { | ||||
|     if (!clickupResource.client_id || !clickupResource.client_secret) { | ||||
|       throw new Error( | ||||
|         "Client ID and Client Secret are required for initialization", | ||||
|       ); | ||||
|     } | ||||
|     if (!redirect_uri) { | ||||
|       throw new Error("Redirect URI is required for initialization"); | ||||
|     } | ||||
|     return await get_initial_token( | ||||
|       clickupResource.client_id, | ||||
|       clickupResource.client_secret, | ||||
|       redirect_uri, | ||||
|     ); | ||||
|   } | ||||
| 
 | ||||
|   if (!code) { | ||||
|     throw new Error("Clickup oAuth CODE required to get token. Use Initialize step first."); | ||||
|   } | ||||
| 
 | ||||
|   return await exchange_code_for_token( | ||||
|     code, | ||||
|     clickupResource.client_id, | ||||
|     clickupResource.client_secret, | ||||
|     expected_state, | ||||
|     state, | ||||
|   ); | ||||
| } | ||||
							
								
								
									
										11
									
								
								f/clickup/rotate_tokens.script.lock
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										11
									
								
								f/clickup/rotate_tokens.script.lock
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,11 @@ | |||
| { | ||||
|   "version": "4", | ||||
|   "specifiers": { | ||||
|     "jsr:@windmill/windmill@*": "1.458.2" | ||||
|   }, | ||||
|   "jsr": { | ||||
|     "@windmill/windmill@1.458.2": { | ||||
|       "integrity": "c3188c099b9553cd92c5f3aa2b9a4f500feef31d8222237b527b7751de49558c" | ||||
|     } | ||||
|   } | ||||
| } | ||||
							
								
								
									
										46
									
								
								f/clickup/rotate_tokens.script.yaml
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										46
									
								
								f/clickup/rotate_tokens.script.yaml
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,46 @@ | |||
| summary: Rotate Tokens | ||||
| description: '' | ||||
| lock: '!inline f/clickup/rotate_tokens.script.lock' | ||||
| concurrency_time_window_s: 0 | ||||
| kind: script | ||||
| schema: | ||||
|   $schema: 'https://json-schema.org/draft/2020-12/schema' | ||||
|   type: object | ||||
|   order: [] | ||||
|   properties: | ||||
|     clickupResource: | ||||
|       type: object | ||||
|       description: '' | ||||
|       default: null | ||||
|       format: resource-clickup_credentials | ||||
|     code: | ||||
|       type: string | ||||
|       description: '' | ||||
|       default: null | ||||
|       originalType: string | ||||
|     expected_state: | ||||
|       type: string | ||||
|       description: '' | ||||
|       default: null | ||||
|       originalType: string | ||||
|     mode: | ||||
|       type: string | ||||
|       description: '' | ||||
|       default: check | ||||
|       enum: | ||||
|         - check | ||||
|         - refresh | ||||
|         - initialize | ||||
|       originalType: enum | ||||
|     redirect_uri: | ||||
|       type: string | ||||
|       description: '' | ||||
|       default: null | ||||
|       originalType: string | ||||
|     state: | ||||
|       type: string | ||||
|       description: '' | ||||
|       default: null | ||||
|       originalType: string | ||||
|   required: | ||||
|     - clickupResource | ||||
							
								
								
									
										64
									
								
								f/clickup/utils/filter-task-fields.deno.ts
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										64
									
								
								f/clickup/utils/filter-task-fields.deno.ts
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,64 @@ | |||
| // utils/filter-task-fields.ts
 | ||||
| import type { TaskResponse } from '../types/clickup'; | ||||
| 
 | ||||
| type FieldSelector = Array<string | NestedField>; | ||||
| type NestedField = { | ||||
|   field: string; | ||||
|   subfields?: FieldSelector; | ||||
| }; | ||||
| 
 | ||||
| export function filterTaskFields( | ||||
|   task: TaskResponse, | ||||
|   fields: FieldSelector | ||||
| ): Partial<TaskResponse> { | ||||
|   return fields.reduce((acc, selector) => { | ||||
|     if (typeof selector === 'string') { | ||||
|       handleStringSelector(acc, task, selector); | ||||
|     } else if (selector.subfields) { | ||||
|       handleNestedSelector(acc, task, selector); | ||||
|     } | ||||
|     return acc; | ||||
|   }, {} as Partial<TaskResponse>); | ||||
| } | ||||
| 
 | ||||
| function handleStringSelector( | ||||
|   acc: Partial<TaskResponse>, | ||||
|   task: TaskResponse, | ||||
|   selector: string | ||||
| ) { | ||||
|   const [root, ...nested] = selector.split('.'); | ||||
| 
 | ||||
|   if (task[root as keyof TaskResponse] !== undefined) { | ||||
|     if (nested.length > 0) { | ||||
|       acc[root as keyof TaskResponse] = processNestedField( | ||||
|         task[root as keyof TaskResponse], | ||||
|         nested.join('.') | ||||
|       ); | ||||
|     } else { | ||||
|       acc[root as keyof TaskResponse] = task[root as keyof TaskResponse]; | ||||
|     } | ||||
|   } | ||||
| } | ||||
| 
 | ||||
| function handleNestedSelector( | ||||
|   acc: Partial<TaskResponse>, | ||||
|   task: TaskResponse, | ||||
|   selector: NestedField | ||||
| ) { | ||||
|   const value = task[selector.field as keyof TaskResponse]; | ||||
|   if (value && typeof value === 'object') { | ||||
|     acc[selector.field as keyof TaskResponse] = Array.isArray(value) | ||||
|       ? value.map(item => filterTaskFields(item as TaskResponse, selector.subfields!)) | ||||
|       : filterTaskFields(value as TaskResponse, selector.subfields!); | ||||
|   } | ||||
| } | ||||
| 
 | ||||
| function processNestedField(value: any, nestedPath: string): any { | ||||
|   const [current, ...remaining] = nestedPath.split('.'); | ||||
| 
 | ||||
|   if (value[current] === undefined) return undefined; | ||||
| 
 | ||||
|   return remaining.length > 0 | ||||
|     ? processNestedField(value[current], remaining.join('.')) | ||||
|     : value[current]; | ||||
| } | ||||
							
								
								
									
										10
									
								
								f/clickup/utils/filter-task-fields.script.yaml
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										10
									
								
								f/clickup/utils/filter-task-fields.script.yaml
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,10 @@ | |||
| summary: Filter Task Fields | ||||
| description: '' | ||||
| lock: '' | ||||
| kind: script | ||||
| no_main_func: true | ||||
| schema: | ||||
|   $schema: 'https://json-schema.org/draft/2020-12/schema' | ||||
|   type: object | ||||
|   properties: {} | ||||
|   required: [] | ||||
							
								
								
									
										6
									
								
								f/clickup_utils/folder.meta.yaml
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										6
									
								
								f/clickup_utils/folder.meta.yaml
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,6 @@ | |||
| summary: null | ||||
| display_name: clickup_utils | ||||
| extra_perms: | ||||
|   u/peter: true | ||||
| owners: | ||||
|   - u/peter | ||||
							
								
								
									
										241
									
								
								f/types/clickup.deno.ts
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										241
									
								
								f/types/clickup.deno.ts
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,241 @@ | |||
| // types/clickup.d.ts
 | ||||
| 
 | ||||
| /** | ||||
|  * Base Task Interface - Shared mutable fields across operations | ||||
|  * Contains all fields that can be SENT in create/update requests | ||||
|  * and RECEIVED in responses (excluding read-only metadata) | ||||
|  */ | ||||
| interface TaskBase { | ||||
|   // Core Content
 | ||||
|   name: string; | ||||
|   description?: string; | ||||
|   text_content?: string; | ||||
| 
 | ||||
|   // Status & Priority
 | ||||
|   status?: string; | ||||
|   priority?: 1 | 2 | 3 | 4; | ||||
| 
 | ||||
|   // Time Management
 | ||||
|   due_date?: number | null; | ||||
|   start_date?: number | null; | ||||
|   time_estimate?: number | null; | ||||
|   time_spent?: number | null; | ||||
| 
 | ||||
|   // People & Assignments
 | ||||
|   assignees?: number[]; | ||||
|   tags?: string[]; | ||||
|   custom_fields?: Array<{ | ||||
|     id: string; | ||||
|     value: any; | ||||
|   }>; | ||||
| 
 | ||||
|   // Task Relationships
 | ||||
|   parent?: string | null; | ||||
|   links_to?: string; | ||||
| 
 | ||||
|   // Advanced Features
 | ||||
|   markdown_description?: string; | ||||
|   notify_all?: boolean; | ||||
|   check_required_custom_fields?: boolean; | ||||
|   custom_item_id?: number | null; | ||||
|   source?: string; | ||||
| } | ||||
| 
 | ||||
| /** | ||||
|  * Task Creation Parameters | ||||
|  * @see https://developer.clickup.com/reference/createtask
 | ||||
|  */ | ||||
| interface TaskCreate extends TaskBase { | ||||
|   /** Required Fields */ | ||||
|   list_id: string; | ||||
|   name: string; | ||||
| 
 | ||||
|   /** Creation-Specific Fields */ | ||||
|   custom_task_id?: string; | ||||
|   team_id?: number; | ||||
| } | ||||
| 
 | ||||
| /** | ||||
|  * Task Update Parameters | ||||
|  * @see https://developer.clickup.com/reference/updatetask
 | ||||
|  */ | ||||
| interface TaskUpdate extends Partial<TaskBase> { | ||||
|   /** Required Identifier */ | ||||
|   id: string; | ||||
| 
 | ||||
|   /** Update-Specific Fields */ | ||||
|   task_revision?: number; | ||||
|   task_template_id?: string; | ||||
|   idempotency_key?: string; | ||||
| } | ||||
| 
 | ||||
| /** | ||||
|  * Full Task Response | ||||
|  * @see https://developer.clickup.com/reference/gettask
 | ||||
|  */ | ||||
| interface TaskResponse extends TaskBase { | ||||
|   /** Read-Only Metadata */ | ||||
|   id: string; | ||||
|   url: string; | ||||
|   orderindex: string; | ||||
|   date_created: string; | ||||
|   date_updated: string; | ||||
|   date_closed?: string | null; | ||||
|   date_done?: string | null; | ||||
|   archived: boolean; | ||||
| 
 | ||||
|   /** System Relationships */ | ||||
|   list: { | ||||
|     id: string; | ||||
|     name: string; | ||||
|     access: boolean; | ||||
|   }; | ||||
|   space: { | ||||
|     id: string; | ||||
|     name: string; | ||||
|   }; | ||||
|   folder: { | ||||
|     id: string; | ||||
|     name: string; | ||||
|     hidden: boolean; | ||||
|     access: boolean; | ||||
|   }; | ||||
| 
 | ||||
|   /** People & Permissions */ | ||||
|   creator: User; | ||||
|   watchers: User[]; | ||||
|   followers: User[]; | ||||
|   permission_level: string; | ||||
|   shared: SharedAccess[]; | ||||
| 
 | ||||
|   /** Complex Structures */ | ||||
|   checklists: Checklist[]; | ||||
|   dependencies: Dependency[]; | ||||
|   linked_tasks: LinkedTask[]; | ||||
|   subtasks: TaskResponse[]; | ||||
|   attachments: Attachment[]; | ||||
| 
 | ||||
|   /** Status Details */ | ||||
|   status: { | ||||
|     status: string; | ||||
|     color: string; | ||||
|     orderindex: number; | ||||
|     type: 'open' | 'closed' | 'custom'; | ||||
|   }; | ||||
| 
 | ||||
|   /** Priority Details */ | ||||
|   priority: { | ||||
|     id: string; | ||||
|     priority: 'urgent' | 'high' | 'normal' | 'low'; | ||||
|     color: string; | ||||
|   } | null; | ||||
| } | ||||
| 
 | ||||
| // Supporting Interfaces
 | ||||
| interface User { | ||||
|   id: number; | ||||
|   username: string; | ||||
|   email: string; | ||||
|   color: string; | ||||
|   initials: string; | ||||
|   profilePicture?: string; | ||||
| } | ||||
| 
 | ||||
| interface Checklist { | ||||
|   id: string; | ||||
|   name: string; | ||||
|   orderindex: number; | ||||
|   resolved: number; | ||||
|   unresolved: number; | ||||
|   items: ChecklistItem[]; | ||||
| } | ||||
| 
 | ||||
| interface ChecklistItem { | ||||
|   id: string; | ||||
|   name: string; | ||||
|   orderindex: number; | ||||
|   assignee?: number; | ||||
|   resolved: boolean; | ||||
|   parent?: string; | ||||
|   date_created: string; | ||||
|   children?: string[]; | ||||
| } | ||||
| 
 | ||||
| interface Tag { | ||||
|   name: string; | ||||
|   tag_fg: string; | ||||
|   tag_bg: string; | ||||
|   creator?: number; | ||||
| } | ||||
| 
 | ||||
| interface Dependency { | ||||
|   task_id: string; | ||||
|   depends_on: string; | ||||
|   type: number; | ||||
|   date_created: string; | ||||
|   userid: string; | ||||
| } | ||||
| 
 | ||||
| interface LinkedTask { | ||||
|   task_id: string; | ||||
|   link_id: string; | ||||
|   date_created: string; | ||||
|   userid: string; | ||||
| } | ||||
| 
 | ||||
| interface CustomField { | ||||
|   id: string; | ||||
|   name: string; | ||||
|   type: string; | ||||
|   type_config: { | ||||
|     [key: string]: any; | ||||
|     options?: Array<{ | ||||
|       id: string; | ||||
|       name: string; | ||||
|       color?: string; | ||||
|       orderindex?: number; | ||||
|     }>; | ||||
|   }; | ||||
|   value: any; | ||||
| } | ||||
| 
 | ||||
| interface Attachment { | ||||
|   id: string; | ||||
|   version: string; | ||||
|   date: string; | ||||
|   title: string; | ||||
|   extension: string; | ||||
|   thumbnail_small: string; | ||||
|   thumbnail_large: string; | ||||
|   size: number; | ||||
| } | ||||
| 
 | ||||
| interface SharedAccess { | ||||
|   id: string; | ||||
|   name: string; | ||||
|   type: string; | ||||
|   access_level: string; | ||||
|   team_id: string; | ||||
| } | ||||
| 
 | ||||
| /** | ||||
|  * Full task hierarchy types | ||||
|  * Ref: https://developer.clickup.com/docs
 | ||||
|  */ | ||||
| interface ClickUpHierarchy { | ||||
|   workspace_id: string; | ||||
|   space_id: string; | ||||
|   folder_id?: string; | ||||
|   list_id: string; | ||||
|   task_id: string; | ||||
| } | ||||
| 
 | ||||
| /** | ||||
|  * API error response format | ||||
|  * Ref: https://developer.clickup.com/docs/errors
 | ||||
|  */ | ||||
| interface ClickUpError { | ||||
|   err: string; | ||||
|   code: number; | ||||
|   message?: string; | ||||
| } | ||||
							
								
								
									
										16
									
								
								f/types/clickup.script.yaml
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										16
									
								
								f/types/clickup.script.yaml
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,16 @@ | |||
| summary: clickup | ||||
| description: '' | ||||
| lock: '' | ||||
| concurrency_time_window_s: 0 | ||||
| kind: script | ||||
| no_main_func: true | ||||
| schema: | ||||
|   $schema: 'https://json-schema.org/draft/2020-12/schema' | ||||
|   type: object | ||||
|   order: | ||||
|     - a | ||||
|     - b | ||||
|     - d | ||||
|     - e | ||||
|   properties: {} | ||||
|   required: [] | ||||
							
								
								
									
										6
									
								
								f/types/folder.meta.yaml
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										6
									
								
								f/types/folder.meta.yaml
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,6 @@ | |||
| summary: null | ||||
| display_name: types | ||||
| extra_perms: | ||||
|   u/peter: true | ||||
| owners: | ||||
|   - u/peter | ||||
							
								
								
									
										6
									
								
								f/utils/folder.meta.yaml
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										6
									
								
								f/utils/folder.meta.yaml
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,6 @@ | |||
| summary: null | ||||
| display_name: utils | ||||
| extra_perms: | ||||
|   u/peter: true | ||||
| owners: | ||||
|   - u/peter | ||||
							
								
								
									
										6
									
								
								wmill-lock.yaml
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										6
									
								
								wmill-lock.yaml
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,6 @@ | |||
| locks: | ||||
|   f/clickup/create_task: 43a306643623f8d42f2e9432175d2061cb05bd4ac0825972c93559163b773ad0 | ||||
|   f/clickup/get_task: e1ac7c952de428bd53df01621c05ecb3995e38cf5f78de1ab03c6baeda9da760 | ||||
|   f/clickup/rotate_tokens: d0a53e8f1855ab682ebf0cc89d093e8e98db761d9225831c7c9c2d0014f4f6bb | ||||
|   f/clickup/utils/filter-task-fields: a160d19c2fa752fa3828f2c87b498be1e79f383ac271678a7dc51c63a4237b4a | ||||
|   f/types/clickup: eb03485311a666b6410b7a621f063dcc84aee0e3a198a721165abdccdf690fa4 | ||||
							
								
								
									
										10
									
								
								wmill.yaml
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										10
									
								
								wmill.yaml
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,10 @@ | |||
| defaultTs: bun | ||||
| includes: | ||||
|   - f/** | ||||
| excludes: [] | ||||
| codebases: [] | ||||
| skipVariables: true | ||||
| skipResources: true | ||||
| skipSecrets: true | ||||
| includeSchedules: false | ||||
| includeTriggers: false | ||||
		Loading…
	
	Add table
		Add a link
		
	
		Reference in a new issue