Skip to main content
Glama
feature.ts4.45 kB
import { WebApi } from 'azure-devops-node-api'; import { TimelineRecord, TimelineRecordState, TaskResult, } from 'azure-devops-node-api/interfaces/BuildInterfaces'; import { AzureDevOpsAuthenticationError, AzureDevOpsError, AzureDevOpsResourceNotFoundError, } from '../../../shared/errors'; import { defaultProject } from '../../../utils/environment'; import { GetPipelineTimelineOptions, PipelineTimeline } from '../types'; const API_VERSION = '7.1'; export async function getPipelineTimeline( connection: WebApi, options: GetPipelineTimelineOptions, ): Promise<PipelineTimeline> { try { const buildApi = await connection.getBuildApi(); const projectId = options.projectId ?? defaultProject; const { runId, timelineId, state, result } = options; const route = `${encodeURIComponent(projectId)}/_apis/build/builds/${runId}/timeline`; const baseUrl = connection.serverUrl.replace(/\/+$/, ''); const url = new URL(`${route}`, `${baseUrl}/`); url.searchParams.set('api-version', API_VERSION); if (timelineId) { url.searchParams.set('timelineId', timelineId); } const requestOptions = buildApi.createRequestOptions( 'application/json', API_VERSION, ); const response = await buildApi.rest.get<PipelineTimeline | null>( url.toString(), requestOptions, ); if (response.statusCode === 404 || !response.result) { throw new AzureDevOpsResourceNotFoundError( `Timeline not found for run ${runId} in project ${projectId}`, ); } const timeline = response.result as PipelineTimeline & { records?: TimelineRecord[]; }; const stateFilters = normalizeFilter(state); const resultFilters = normalizeFilter(result); if (Array.isArray(timeline.records) && (stateFilters || resultFilters)) { const filteredRecords = timeline.records.filter((record) => { const recordState = stateToString(record.state); const recordResult = resultToString(record.result); const stateMatch = !stateFilters || (recordState && stateFilters.has(recordState)); const resultMatch = !resultFilters || (recordResult && resultFilters.has(recordResult)); return stateMatch && resultMatch; }); return { ...timeline, records: filteredRecords, } as PipelineTimeline; } return timeline; } catch (error) { if (error instanceof AzureDevOpsError) { throw error; } if (error instanceof Error) { const message = error.message.toLowerCase(); if ( message.includes('authentication') || message.includes('unauthorized') || message.includes('401') ) { throw new AzureDevOpsAuthenticationError( `Failed to authenticate: ${error.message}`, ); } if ( message.includes('not found') || message.includes('does not exist') || message.includes('404') ) { throw new AzureDevOpsResourceNotFoundError( `Pipeline timeline or project not found: ${error.message}`, ); } } throw new AzureDevOpsError( `Failed to retrieve pipeline timeline: ${ error instanceof Error ? error.message : String(error) }`, ); } } function normalizeFilter(value?: string | string[]): Set<string> | undefined { if (!value) { return undefined; } const values = Array.isArray(value) ? value : [value]; const normalized = values .map((item) => (typeof item === 'string' ? item.trim().toLowerCase() : '')) .filter((item) => item.length > 0); return normalized.length > 0 ? new Set(normalized) : undefined; } function stateToString( state?: TimelineRecordState | string, ): string | undefined { if (typeof state === 'number') { const stateName = TimelineRecordState[state]; return typeof stateName === 'string' ? stateName.toLowerCase() : undefined; } if (typeof state === 'string' && state.length > 0) { return state.toLowerCase(); } return undefined; } function resultToString(result?: TaskResult | string): string | undefined { if (typeof result === 'number') { const resultName = TaskResult[result]; return typeof resultName === 'string' ? resultName.toLowerCase() : undefined; } if (typeof result === 'string' && result.length > 0) { return result.toLowerCase(); } return undefined; }

Latest Blog Posts

MCP directory API

We provide all the information about MCP servers via our MCP API.

curl -X GET 'https://glama.ai/api/mcp/v1/servers/Tiberriver256/mcp-server-azure-devops'

If you have feedback or need assistance with the MCP directory API, please join our Discord server