ERP-node/frontend/lib/api/dashboard.ts

282 lines
7.7 KiB
TypeScript
Raw Normal View History

/**
* API
*/
import { DashboardElement } from '@/components/admin/dashboard/types';
// API 기본 설정
const API_BASE_URL = process.env.NEXT_PUBLIC_API_URL || 'http://localhost:3001/api';
// 토큰 가져오기 (실제 인증 시스템에 맞게 수정)
function getAuthToken(): string | null {
if (typeof window === 'undefined') return null;
return localStorage.getItem('authToken') || sessionStorage.getItem('authToken');
}
// API 요청 헬퍼
async function apiRequest<T>(
endpoint: string,
options: RequestInit = {}
): Promise<{ success: boolean; data?: T; message?: string; pagination?: any }> {
const token = getAuthToken();
const config: RequestInit = {
headers: {
'Content-Type': 'application/json',
...(token && { 'Authorization': `Bearer ${token}` }),
...options.headers,
},
...options,
};
try {
const response = await fetch(`${API_BASE_URL}${endpoint}`, config);
// 응답이 JSON이 아닐 수도 있으므로 안전하게 처리
let result;
try {
result = await response.json();
} catch (jsonError) {
console.error('JSON Parse Error:', jsonError);
throw new Error(`서버 응답을 파싱할 수 없습니다. Status: ${response.status}`);
}
if (!response.ok) {
console.error('API Error Response:', {
status: response.status,
statusText: response.statusText,
result
});
throw new Error(result.message || `HTTP ${response.status}: ${response.statusText}`);
}
return result;
} catch (error: any) {
console.error('API Request Error:', {
endpoint,
error: error?.message || error,
errorObj: error,
config
});
throw error;
}
}
// 대시보드 타입 정의
export interface Dashboard {
id: string;
title: string;
description?: string;
thumbnailUrl?: string;
isPublic: boolean;
createdBy: string;
createdAt: string;
updatedAt: string;
tags?: string[];
category?: string;
viewCount: number;
elementsCount?: number;
creatorName?: string;
elements?: DashboardElement[];
}
export interface CreateDashboardRequest {
title: string;
description?: string;
isPublic?: boolean;
elements: DashboardElement[];
tags?: string[];
category?: string;
}
export interface DashboardListQuery {
page?: number;
limit?: number;
search?: string;
category?: string;
isPublic?: boolean;
}
// 대시보드 API 함수들
export const dashboardApi = {
/**
*
*/
async createDashboard(data: CreateDashboardRequest): Promise<Dashboard> {
const result = await apiRequest<Dashboard>('/dashboards', {
method: 'POST',
body: JSON.stringify(data),
});
if (!result.success || !result.data) {
throw new Error(result.message || '대시보드 생성에 실패했습니다.');
}
return result.data;
},
/**
*
*/
async getDashboards(query: DashboardListQuery = {}) {
const params = new URLSearchParams();
if (query.page) params.append('page', query.page.toString());
if (query.limit) params.append('limit', query.limit.toString());
if (query.search) params.append('search', query.search);
if (query.category) params.append('category', query.category);
if (typeof query.isPublic === 'boolean') params.append('isPublic', query.isPublic.toString());
const queryString = params.toString();
const endpoint = `/dashboards${queryString ? `?${queryString}` : ''}`;
const result = await apiRequest<Dashboard[]>(endpoint);
if (!result.success) {
throw new Error(result.message || '대시보드 목록 조회에 실패했습니다.');
}
return {
dashboards: result.data || [],
pagination: result.pagination
};
},
/**
*
*/
async getMyDashboards(query: DashboardListQuery = {}) {
const params = new URLSearchParams();
if (query.page) params.append('page', query.page.toString());
if (query.limit) params.append('limit', query.limit.toString());
if (query.search) params.append('search', query.search);
if (query.category) params.append('category', query.category);
const queryString = params.toString();
const endpoint = `/dashboards/my${queryString ? `?${queryString}` : ''}`;
const result = await apiRequest<Dashboard[]>(endpoint);
if (!result.success) {
throw new Error(result.message || '내 대시보드 목록 조회에 실패했습니다.');
}
return {
dashboards: result.data || [],
pagination: result.pagination
};
},
/**
*
*/
async getDashboard(id: string): Promise<Dashboard> {
const result = await apiRequest<Dashboard>(`/dashboards/${id}`);
if (!result.success || !result.data) {
throw new Error(result.message || '대시보드 조회에 실패했습니다.');
}
return result.data;
},
/**
* ( )
*/
async getPublicDashboard(id: string): Promise<Dashboard> {
const result = await apiRequest<Dashboard>(`/dashboards/public/${id}`);
if (!result.success || !result.data) {
throw new Error(result.message || '대시보드 조회에 실패했습니다.');
}
return result.data;
},
/**
*
*/
async updateDashboard(id: string, data: Partial<CreateDashboardRequest>): Promise<Dashboard> {
const result = await apiRequest<Dashboard>(`/dashboards/${id}`, {
method: 'PUT',
body: JSON.stringify(data),
});
if (!result.success || !result.data) {
throw new Error(result.message || '대시보드 수정에 실패했습니다.');
}
return result.data;
},
/**
*
*/
async deleteDashboard(id: string): Promise<void> {
const result = await apiRequest(`/dashboards/${id}`, {
method: 'DELETE',
});
if (!result.success) {
throw new Error(result.message || '대시보드 삭제에 실패했습니다.');
}
},
/**
* ( )
*/
async getPublicDashboards(query: DashboardListQuery = {}) {
const params = new URLSearchParams();
if (query.page) params.append('page', query.page.toString());
if (query.limit) params.append('limit', query.limit.toString());
if (query.search) params.append('search', query.search);
if (query.category) params.append('category', query.category);
const queryString = params.toString();
const endpoint = `/dashboards/public${queryString ? `?${queryString}` : ''}`;
const result = await apiRequest<Dashboard[]>(endpoint);
if (!result.success) {
throw new Error(result.message || '공개 대시보드 목록 조회에 실패했습니다.');
}
return {
dashboards: result.data || [],
pagination: result.pagination
};
},
/**
* ( )
*/
async executeQuery(query: string): Promise<{ columns: string[]; rows: any[]; rowCount: number }> {
const result = await apiRequest<{ columns: string[]; rows: any[]; rowCount: number }>('/dashboards/execute-query', {
method: 'POST',
body: JSON.stringify({ query }),
});
if (!result.success || !result.data) {
throw new Error(result.message || '쿼리 실행에 실패했습니다.');
}
return result.data;
}
};
// 에러 처리 유틸리티
export function handleApiError(error: any): string {
if (error.message) {
return error.message;
}
if (typeof error === 'string') {
return error;
}
return '알 수 없는 오류가 발생했습니다.';
}