Files
jobtrackingapp/job-tracker-ui/src/api.ts
T

73 lines
2.7 KiB
TypeScript

import axios from "axios";
import { clearAuthClientState, getCsrfToken } from "./auth";
function looksLikeHtml(value: string) {
return /<\s*html\b|<\s*body\b|<\s*head\b|<\s*title\b|<\s*!doctype\b/i.test(value);
}
function sanitizeServerMessage(value: string, fallback: string) {
const text = value.trim();
if (!text) return fallback;
if (looksLikeHtml(text)) return fallback;
return text.length > 300 ? `${text.slice(0, 297).trimEnd()}...` : text;
}
export function getApiErrorMessage(error: any, fallback = "Request failed.") {
const data = error?.response?.data;
if (typeof data === "string" && data.trim()) return sanitizeServerMessage(data, fallback);
if (typeof data?.message === "string" && data.message.trim()) return sanitizeServerMessage(data.message, fallback);
if (typeof data?.detail === "string" && data.detail.trim()) return sanitizeServerMessage(data.detail, fallback);
if (typeof data?.title === "string" && data.title.trim()) return sanitizeServerMessage(data.title, fallback);
if (Array.isArray(data?.errors)) {
const first = data.errors.find((value: unknown) => typeof value === "string" && value.trim());
if (first) return sanitizeServerMessage(first, fallback);
}
if (data?.errors && typeof data.errors === "object") {
for (const value of Object.values(data.errors)) {
if (Array.isArray(value)) {
const first = value.find((item: unknown) => typeof item === "string" && item.trim());
if (first) return sanitizeServerMessage(first, fallback);
}
if (typeof value === "string" && value.trim()) return sanitizeServerMessage(value, fallback);
}
}
if (typeof error?.message === "string" && error.message.trim()) return sanitizeServerMessage(error.message, fallback);
return fallback;
}
const envBaseUrl = process.env.REACT_APP_API_BASE_URL;
const defaultBaseUrl =
window.location.hostname === "localhost"
? "http://localhost:5202/api"
: "/api";
export const api = axios.create({
baseURL: envBaseUrl && envBaseUrl.trim().length > 0 ? envBaseUrl : defaultBaseUrl,
withCredentials: true,
xsrfCookieName: "XSRF-TOKEN",
xsrfHeaderName: "X-CSRF-TOKEN",
});
api.interceptors.request.use((config) => {
const method = (config.method ?? "get").toUpperCase();
if (["POST", "PUT", "PATCH", "DELETE"].includes(method)) {
const csrfToken = getCsrfToken();
if (csrfToken) {
config.headers = config.headers ?? {};
config.headers["X-CSRF-TOKEN"] = csrfToken;
}
}
return config;
});
api.interceptors.response.use(
(r) => r,
(err) => {
const status = err?.response?.status;
if (status === 401) {
clearAuthClientState();
}
return Promise.reject(err);
},
);