xpeditis2.0/apps/frontend/src/lib/api/client.ts
2026-01-27 19:33:51 +01:00

430 lines
11 KiB
TypeScript

/**
* API Client Base
*
* Core HTTP client with authentication and error handling
*/
const API_BASE_URL = process.env.NEXT_PUBLIC_API_URL || 'http://localhost:4000';
// Track if we're currently refreshing to avoid multiple simultaneous refresh requests
let isRefreshing = false;
let refreshSubscribers: Array<(token: string) => void> = [];
/**
* Get authentication token from localStorage
*/
export function getAuthToken(): string | null {
if (typeof window === 'undefined') return null;
return localStorage.getItem('access_token');
}
/**
* Get refresh token from localStorage
*/
export function getRefreshToken(): string | null {
if (typeof window === 'undefined') return null;
return localStorage.getItem('refresh_token');
}
/**
* Set authentication tokens
*/
export function setAuthTokens(accessToken: string, refreshToken: string): void {
if (typeof window === 'undefined') return;
localStorage.setItem('access_token', accessToken);
localStorage.setItem('refresh_token', refreshToken);
}
/**
* Clear authentication tokens
*/
export function clearAuthTokens(): void {
if (typeof window === 'undefined') return;
localStorage.removeItem('access_token');
localStorage.removeItem('refresh_token');
localStorage.removeItem('user');
}
/**
* Add subscriber to be notified when token is refreshed
*/
function subscribeTokenRefresh(callback: (token: string) => void): void {
refreshSubscribers.push(callback);
}
/**
* Notify all subscribers that token has been refreshed
*/
function onTokenRefreshed(token: string): void {
refreshSubscribers.forEach(callback => callback(token));
refreshSubscribers = [];
}
/**
* Refresh access token using refresh token
*/
async function refreshAccessToken(): Promise<string> {
const refreshToken = getRefreshToken();
if (!refreshToken) {
throw new Error('No refresh token available');
}
const response = await fetch(`${API_BASE_URL}/api/v1/auth/refresh`, {
method: 'POST',
headers: {
'Content-Type': 'application/json',
},
body: JSON.stringify({ refreshToken }),
});
if (!response.ok) {
// Refresh token invalid or expired, clear everything
clearAuthTokens();
// Redirect to login
if (typeof window !== 'undefined') {
window.location.href = '/login';
}
throw new Error('Failed to refresh token');
}
const data = await response.json();
const newAccessToken = data.accessToken;
// Update access token in localStorage (keep same refresh token)
if (typeof window !== 'undefined') {
localStorage.setItem('access_token', newAccessToken);
}
return newAccessToken;
}
/**
* Create headers with authentication
*/
export function createHeaders(includeAuth = true): HeadersInit {
const headers: HeadersInit = {
'Content-Type': 'application/json',
};
if (includeAuth) {
const token = getAuthToken();
if (token) {
headers['Authorization'] = `Bearer ${token}`;
}
}
return headers;
}
/**
* Create headers for multipart form data
*/
export function createMultipartHeaders(includeAuth = true): HeadersInit {
const headers: HeadersInit = {};
if (includeAuth) {
const token = getAuthToken();
if (token) {
headers['Authorization'] = `Bearer ${token}`;
}
}
return headers;
}
/**
* API Error
*/
export class ApiError extends Error {
constructor(
message: string,
public statusCode: number,
public response?: any
) {
super(message);
this.name = 'ApiError';
}
}
/**
* Make API request with automatic token refresh on 401
*/
export async function apiRequest<T>(
endpoint: string,
options: RequestInit = {},
isRetry = false
): Promise<T> {
const url = `${API_BASE_URL}${endpoint}`;
const response = await fetch(url, {
...options,
headers: {
...options.headers,
},
});
// Handle 401 Unauthorized - token expired
// Skip auto-redirect for auth endpoints (login, register, refresh) - they handle their own errors
const isAuthEndpoint = endpoint.includes('/auth/login') ||
endpoint.includes('/auth/register') ||
endpoint.includes('/auth/refresh');
if (response.status === 401 && !isRetry && !isAuthEndpoint) {
// Check if we have a refresh token
const refreshToken = getRefreshToken();
if (!refreshToken) {
// No refresh token, redirect to login
clearAuthTokens();
if (typeof window !== 'undefined') {
window.location.href = '/login';
}
throw new ApiError('Session expired', 401);
}
// Try to refresh the token
try {
if (!isRefreshing) {
isRefreshing = true;
const newAccessToken = await refreshAccessToken();
isRefreshing = false;
onTokenRefreshed(newAccessToken);
// Retry the original request with new token
const newHeaders = { ...options.headers };
if (newHeaders && typeof newHeaders === 'object' && 'Authorization' in newHeaders) {
(newHeaders as any)['Authorization'] = `Bearer ${newAccessToken}`;
}
return apiRequest<T>(
endpoint,
{
...options,
headers: newHeaders,
},
true
);
} else {
// Already refreshing, wait for the new token
return new Promise((resolve, reject) => {
subscribeTokenRefresh(async (newAccessToken: string) => {
const newHeaders = { ...options.headers };
if (newHeaders && typeof newHeaders === 'object' && 'Authorization' in newHeaders) {
(newHeaders as any)['Authorization'] = `Bearer ${newAccessToken}`;
}
try {
const result = await apiRequest<T>(
endpoint,
{
...options,
headers: newHeaders,
},
true
);
resolve(result);
} catch (error) {
reject(error);
}
});
});
}
} catch (refreshError) {
isRefreshing = false;
refreshSubscribers = [];
throw refreshError;
}
}
if (!response.ok) {
const error = await response.json().catch(() => ({}));
throw new ApiError(
error.message || `API request failed: ${response.statusText}`,
response.status,
error
);
}
// Handle 204 No Content
if (response.status === 204) {
return undefined as T;
}
return response.json();
}
/**
* GET request
*/
export async function get<T>(endpoint: string, includeAuth = true): Promise<T> {
return apiRequest<T>(endpoint, {
method: 'GET',
headers: createHeaders(includeAuth),
});
}
/**
* POST request
*/
export async function post<T>(endpoint: string, data?: any, includeAuth = true): Promise<T> {
return apiRequest<T>(endpoint, {
method: 'POST',
headers: createHeaders(includeAuth),
body: data ? JSON.stringify(data) : undefined,
});
}
/**
* PATCH request
*/
export async function patch<T>(endpoint: string, data: any, includeAuth = true): Promise<T> {
return apiRequest<T>(endpoint, {
method: 'PATCH',
headers: createHeaders(includeAuth),
body: JSON.stringify(data),
});
}
/**
* DELETE request
*/
export async function del<T>(endpoint: string, includeAuth = true): Promise<T> {
return apiRequest<T>(endpoint, {
method: 'DELETE',
headers: createHeaders(includeAuth),
});
}
/**
* Upload file (multipart/form-data)
*/
export async function upload<T>(
endpoint: string,
formData: FormData,
includeAuth = true
): Promise<T> {
const url = `${API_BASE_URL}${endpoint}`;
const response = await fetch(url, {
method: 'POST',
headers: createMultipartHeaders(includeAuth),
body: formData,
});
// Handle 401 Unauthorized for file uploads
if (response.status === 401) {
const refreshToken = getRefreshToken();
if (refreshToken) {
try {
const newAccessToken = await refreshAccessToken();
// Retry upload with new token
const retryResponse = await fetch(url, {
method: 'POST',
headers: {
...createMultipartHeaders(includeAuth),
Authorization: `Bearer ${newAccessToken}`,
},
body: formData,
});
if (!retryResponse.ok) {
const error = await retryResponse.json().catch(() => ({}));
throw new ApiError(
error.message || `Upload failed: ${retryResponse.statusText}`,
retryResponse.status,
error
);
}
return retryResponse.json();
} catch (refreshError) {
clearAuthTokens();
if (typeof window !== 'undefined') {
window.location.href = '/login';
}
throw refreshError;
}
}
}
if (!response.ok) {
const error = await response.json().catch(() => ({}));
throw new ApiError(
error.message || `Upload failed: ${response.statusText}`,
response.status,
error
);
}
return response.json();
}
/**
* Download file
*/
export async function download(
endpoint: string,
filename: string,
includeAuth = true
): Promise<void> {
const url = `${API_BASE_URL}${endpoint}`;
const response = await fetch(url, {
method: 'GET',
headers: createHeaders(includeAuth),
});
// Handle 401 Unauthorized for downloads
if (response.status === 401) {
const refreshToken = getRefreshToken();
if (refreshToken) {
try {
const newAccessToken = await refreshAccessToken();
// Retry download with new token
const retryResponse = await fetch(url, {
method: 'GET',
headers: {
...createHeaders(includeAuth),
Authorization: `Bearer ${newAccessToken}`,
},
});
if (!retryResponse.ok) {
throw new ApiError(`Download failed: ${retryResponse.statusText}`, retryResponse.status);
}
const blob = await retryResponse.blob();
const downloadUrl = window.URL.createObjectURL(blob);
const link = document.createElement('a');
link.href = downloadUrl;
link.download = filename;
document.body.appendChild(link);
link.click();
document.body.removeChild(link);
window.URL.revokeObjectURL(downloadUrl);
return;
} catch (refreshError) {
clearAuthTokens();
if (typeof window !== 'undefined') {
window.location.href = '/login';
}
throw refreshError;
}
}
}
if (!response.ok) {
throw new ApiError(`Download failed: ${response.statusText}`, response.status);
}
const blob = await response.blob();
const downloadUrl = window.URL.createObjectURL(blob);
const link = document.createElement('a');
link.href = downloadUrl;
link.download = filename;
document.body.appendChild(link);
link.click();
document.body.removeChild(link);
window.URL.revokeObjectURL(downloadUrl);
}