frontend/src/services/apiClient.js

133 lines
2.9 KiB
JavaScript
Raw Normal View History

import { useAuthStore } from '@/stores/authStore.js';
2025-09-09 23:19:17 +03:00
const BASE_URL = 'http://localhost:9000/api/v2';
let isRefreshing = false;
let refreshPromise = null;
function createConfig(options = {}) {
const authStore = useAuthStore();
const headers = {
2025-09-09 23:19:17 +03:00
'Content-Type': 'application/json',
...options.headers,
};
if (authStore.accessToken) {
headers['Authorization'] = `Bearer ${authStore.accessToken}`;
}
return {
headers,
credentials: 'include',
...options,
};
}
async function refreshAccessToken() {
const authStore = useAuthStore();
if (isRefreshing) return refreshPromise;
isRefreshing = true;
2025-09-14 16:18:13 +03:00
refreshPromise = fetch(`${BASE_URL}/auth/refresh`, {
method: 'POST',
credentials: 'include',
})
2025-09-14 16:18:13 +03:00
.then(async (res) => {
if (!res.ok) throw new Error('Failed to refresh access token');
2025-09-14 16:18:13 +03:00
const data = await res.json();
console.log("Refresh response data: ", data);
return data;
})
2025-09-14 16:18:13 +03:00
.then((data) => {
2025-09-14 16:18:13 +03:00
const token = data.access_token
if (!token) {
throw new Error('No access_token in refresh response');
}
authStore.setToken(data.access_token);
console.log('refreshed token', data.access_token);
return data;
})
2025-09-14 16:18:13 +03:00
.catch((error) => {
throw error;
})
2025-09-14 16:18:13 +03:00
.finally(() => {
isRefreshing = false;
refreshPromise = null;
});
return refreshPromise;
}
async function request(url, options = {}, isRetry = false) {
const config = createConfig(options);
const response = await fetch(`${BASE_URL}${url}`, config);
if (response.status === 401 && !isRetry) {
try {
const data = await refreshAccessToken();
2025-09-14 16:18:13 +03:00
const token = data.access_token;
if (!token) {
throw new Error('Refresh response did not contain access_token');
}
const newOptions = {
...options,
headers: {
...options.headers,
2025-09-14 16:18:13 +03:00
'Authorization': `Bearer ${token}`,
},
};
return await request(url, newOptions, true);
} catch (e) {
const authStore = useAuthStore();
authStore.forceLogout();
throw e;
}
2025-09-09 23:19:17 +03:00
}
if (!response.ok) {
let errorData;
try {
errorData = await response.json();
} catch {
errorData = {};
}
throw new Error(errorData.message || `HTTP Error: ${response.status}`);
}
const contentType = response.headers.get('content-type');
if (contentType?.includes('application/json')) {
try {
return await response.json();
} catch (e) {
console.warn('Failed to parse JSON response', e);
return null;
}
}
return null;
}
export const apiClient = {
get: (url) => request(url, { method: 'GET' }),
post: (url, data) => request(url, {
method: 'POST',
body: JSON.stringify(data),
}),
put: (url, data) => request(url, {
method: 'PUT',
body: JSON.stringify(data),
}),
delete: (url) => request(url, { method: 'DELETE' }),
};