1
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
 
 
 
 
 
 

155 lines
5.0 KiB

import { lastValueFrom } from 'rxjs';
import { FetchResponse, getBackendSrv } from '@grafana/runtime';
import { PostableRulerRuleGroupDTO, RulerRuleGroupDTO, RulerRulesConfigDTO } from 'app/types/unified-alerting-dto';
import { getDatasourceAPIId, GRAFANA_RULES_SOURCE_NAME } from '../utils/datasource';
import { RULER_NOT_SUPPORTED_MSG } from '../utils/constants';
interface ErrorResponseMessage {
message?: string;
error?: string;
}
// upsert a rule group. use this to update rules
export async function setRulerRuleGroup(
dataSourceName: string,
namespace: string,
group: PostableRulerRuleGroupDTO
): Promise<void> {
await lastValueFrom(
getBackendSrv().fetch<unknown>({
method: 'POST',
url: `/api/ruler/${getDatasourceAPIId(dataSourceName)}/api/v1/rules/${encodeURIComponent(namespace)}`,
data: group,
showErrorAlert: false,
showSuccessAlert: false,
})
);
}
export interface FetchRulerRulesFilter {
dashboardUID: string;
panelId?: number;
}
// fetch all ruler rule namespaces and included groups
export async function fetchRulerRules(dataSourceName: string, filter?: FetchRulerRulesFilter) {
if (filter?.dashboardUID && dataSourceName !== GRAFANA_RULES_SOURCE_NAME) {
throw new Error('Filtering by dashboard UID is not supported for cloud rules sources.');
}
const params: Record<string, string> = {};
if (filter?.dashboardUID) {
params['dashboard_uid'] = filter.dashboardUID;
if (filter.panelId) {
params['panel_id'] = String(filter.panelId);
}
}
return rulerGetRequest<RulerRulesConfigDTO>(
`/api/ruler/${getDatasourceAPIId(dataSourceName)}/api/v1/rules`,
{},
params
);
}
// fetch rule groups for a particular namespace
// will throw with { status: 404 } if namespace does not exist
export async function fetchRulerRulesNamespace(dataSourceName: string, namespace: string) {
const result = await rulerGetRequest<Record<string, RulerRuleGroupDTO[]>>(
`/api/ruler/${getDatasourceAPIId(dataSourceName)}/api/v1/rules/${encodeURIComponent(namespace)}`,
{}
);
return result[namespace] || [];
}
// fetch a particular rule group
// will throw with { status: 404 } if rule group does not exist
export async function fetchRulerRulesGroup(
dataSourceName: string,
namespace: string,
group: string
): Promise<RulerRuleGroupDTO | null> {
return rulerGetRequest<RulerRuleGroupDTO | null>(
`/api/ruler/${getDatasourceAPIId(dataSourceName)}/api/v1/rules/${encodeURIComponent(
namespace
)}/${encodeURIComponent(group)}`,
null
);
}
export async function deleteRulerRulesGroup(dataSourceName: string, namespace: string, groupName: string) {
await lastValueFrom(
getBackendSrv().fetch({
url: `/api/ruler/${getDatasourceAPIId(dataSourceName)}/api/v1/rules/${encodeURIComponent(
namespace
)}/${encodeURIComponent(groupName)}`,
method: 'DELETE',
showSuccessAlert: false,
showErrorAlert: false,
})
);
}
// false in case ruler is not supported. this is weird, but we'll work on it
async function rulerGetRequest<T>(url: string, empty: T, params?: Record<string, string>): Promise<T> {
try {
const response = await lastValueFrom(
getBackendSrv().fetch<T>({
url,
showErrorAlert: false,
showSuccessAlert: false,
params,
})
);
return response.data;
} catch (error) {
if (!isResponseError(error)) {
throw error;
}
const notFoundError = error.status === 404;
const rulerNotSupported =
error.status === 500 &&
error.data.error?.includes('unexpected content type from upstream. expected YAML, got text/html');
if (notFoundError) {
// the endpoint will return 404 but confirm that it's a Cortex endpoint
if (isCortexErrorResponse(error)) {
return empty;
}
// any other 404 should throw an exception
throw new Error('404 from rules config endpoint. Perhaps ruler API is not enabled?');
} else if (rulerNotSupported) {
// assert if the endoint is not supported at all
throw {
...error,
data: {
...error.data,
message: RULER_NOT_SUPPORTED_MSG,
},
};
}
throw error;
}
}
function isResponseError(error: unknown): error is FetchResponse<ErrorResponseMessage> {
const hasErrorMessage = (error as FetchResponse<ErrorResponseMessage>).data != null;
const hasErrorCode = Number.isFinite((error as FetchResponse<ErrorResponseMessage>).status);
return hasErrorCode && hasErrorMessage;
}
function isCortexErrorResponse(error: FetchResponse<ErrorResponseMessage>) {
return error.data.error?.includes('group does not exist') || error.data.error?.includes('no rule groups found');
}
export async function deleteNamespace(dataSourceName: string, namespace: string): Promise<void> {
await lastValueFrom(
getBackendSrv().fetch<unknown>({
method: 'DELETE',
url: `/api/ruler/${getDatasourceAPIId(dataSourceName)}/api/v1/rules/${encodeURIComponent(namespace)}`,
showErrorAlert: false,
showSuccessAlert: false,
})
);
}