MingruiZhang's picture
done (#4)
f80b091 unverified
raw
history blame
3.51 kB
import { auth } from '@/auth';
import { v5 as uuidV5 } from 'uuid';
interface ApiResponse<T> {
code: number; // code from server 0
message: string;
data: T;
}
const clefApiBuilder = <Params extends object | void, Resp>(
path: string,
options?: {
// default to GET
method: 'GET' | 'POST';
// default to 'app'
prefix: 'api' | 'app' | 'api.dev' | 'app.dev';
},
) => {
return async (params: Params): Promise<Resp> => {
const session = await auth();
const email = session?.user?.email;
if (!email || !email.endsWith('@landing.ai')) {
throw new Response('Unauthorized', {
status: 401,
});
}
const sessionUser = {
id: uuidV5(email, uuidV5.URL),
orgId: '-1024',
email: email,
username: email.split('@')[0],
userRole: 'adminPortal',
bucket: 'fake_bucket',
};
const prefix = options?.prefix ?? 'app';
const baseURL = `https://${prefix}.landing.ai/${path}`;
const method = options?.method ?? 'GET';
// Create a URL object with query params
const url = new URL(baseURL);
if (method === 'GET') {
Object.entries(params ?? {}).forEach(([key, value]) =>
url.searchParams.append(key, value),
);
}
const fetchParams: RequestInit = {
method: options?.method ?? 'GET',
headers: {
sessionuser: JSON.stringify(sessionUser), // faked production user
apikey: 'land_sk_DKeoYtaZZrYqJ9TMMiXe4BIQgJcZ0s3XAoB0JT3jv73FFqnr6k', // dev key
'X-ROUTE': 'mingruizhang-landing',
},
};
if (method === 'POST' && params instanceof Object) {
const formData = new FormData();
for (const [key, value] of Object.entries(params)) {
formData.append(key, value);
}
fetchParams.body = formData;
}
const res = await fetch(url.toString(), fetchParams);
if (!res.ok) {
console.error('ERROR: fetch data failure', res.status, res.statusText);
// This will activate the closest `error.js` Error Boundary
throw new Error('Failed to fetch data');
}
const { data }: ApiResponse<Resp> = await res.json();
return data;
};
};
export type ProjectBaseInfo = {
id: number;
name: string;
created_at: Date;
label_type: string;
organization: {
id: number;
name: string;
};
};
/**
* Fetch recent projects from all organizations from past 30 days, excluding
* 1. test organization such as bdd, cypress
* 2. internal organization such as landing, landing-ai, orgId=1
* 3. projects not containing media or only contain sample media
* @author https://github.com/landing-ai/landing-platform/blob/mingrui-04-08-meaningful-project/packages/server-clef/src/main_app/controllers/admin/get_admin_meaningful_project_controller.ts
*/
export const fetchRecentProjectList = clefApiBuilder<void, ProjectBaseInfo[]>(
'api/admin/projects/recent',
);
export type MediaDetails = {
id: number;
name: string;
path: string;
url: string;
projectId: number;
thumbnails: string[];
properties: {
width: number;
height: number;
format: string;
orientation: number;
};
};
/**
* Randomly fetch 10 media from a given project
* @author https://github.com/landing-ai/landing-platform/blob/mingrui-04-08-meaningful-project/packages/server-clef/src/main_app/controllers/admin/get_admin_meaningful_project_controller.ts
*/
export const fetchProjectMedia = clefApiBuilder<
{ projectId: number },
MediaDetails[]
>('api/admin/project/media');