tududi/frontend/utils/sharesService.ts
Chris e3dcb49efa
Fix bug 661 (#682)
* Limit project card text length

* Fix projects issues

* fixup! Fix projects issues
2025-12-08 16:14:10 +02:00

91 lines
2.6 KiB
TypeScript

import { getApiPath } from '../config/paths';
export type AccessLevel = 'ro' | 'rw';
export interface ShareGrantRequest {
resource_type: 'project' | 'task' | 'note' | 'area' | 'tag';
resource_uid: string;
target_user_email: string;
access_level: AccessLevel;
}
export async function grantShare(req: ShareGrantRequest): Promise<void> {
const res = await fetch(getApiPath('shares'), {
method: 'POST',
credentials: 'include',
headers: {
'Content-Type': 'application/json',
Accept: 'application/json',
},
body: JSON.stringify(req),
});
if (!res.ok) {
let message = 'Failed to share resource';
try {
const body = await res.json();
if (body?.error) message = body.error;
} catch {
// ignore non-JSON error bodies
}
throw new Error(message);
}
}
export interface ListSharesResponseRow {
user_id: number;
access_level: AccessLevel | 'owner';
created_at: string | null;
email?: string | null;
avatar_image?: string | null;
is_owner?: boolean;
}
export async function listShares(
resource_type: ShareGrantRequest['resource_type'],
resource_uid: string
): Promise<ListSharesResponseRow[]> {
const params = new URLSearchParams({ resource_type, resource_uid });
const res = await fetch(getApiPath(`shares?${params.toString()}`), {
method: 'GET',
credentials: 'include',
headers: { Accept: 'application/json' },
});
if (!res.ok) {
let message = 'Failed to load shares';
try {
const body = await res.json();
if (body?.error) message = body.error;
} catch {
// ignore non-JSON error bodies
}
throw new Error(message);
}
const data = await res.json();
return data.shares || [];
}
export async function revokeShare(
resource_type: ShareGrantRequest['resource_type'],
resource_uid: string,
target_user_id: number
): Promise<void> {
const res = await fetch(getApiPath('shares'), {
method: 'DELETE',
credentials: 'include',
headers: {
'Content-Type': 'application/json',
Accept: 'application/json',
},
body: JSON.stringify({ resource_type, resource_uid, target_user_id }),
});
if (!res.ok) {
let message = 'Failed to revoke share';
try {
const body = await res.json();
if (body?.error) message = body.error;
} catch {
// ignore non-JSON error bodies
}
throw new Error(message);
}
}