open-webui/src/lib/components/admin/Users/Groups.svelte

231 lines
6.4 KiB
Svelte
Raw Normal View History

2024-11-13 18:00:00 +08:00
<script>
import { toast } from 'svelte-sonner';
import dayjs from 'dayjs';
import relativeTime from 'dayjs/plugin/relativeTime';
dayjs.extend(relativeTime);
import { onMount, getContext } from 'svelte';
import { goto } from '$app/navigation';
2024-11-14 19:16:26 +08:00
import { WEBUI_NAME, config, user, showSidebar, knowledge } from '$lib/stores';
2024-11-13 18:00:00 +08:00
import { WEBUI_BASE_URL } from '$lib/constants';
import Tooltip from '$lib/components/common/Tooltip.svelte';
import Plus from '$lib/components/icons/Plus.svelte';
2024-11-13 19:09:46 +08:00
import Badge from '$lib/components/common/Badge.svelte';
import UsersSolid from '$lib/components/icons/UsersSolid.svelte';
import ChevronRight from '$lib/components/icons/ChevronRight.svelte';
import EllipsisHorizontal from '$lib/components/icons/EllipsisHorizontal.svelte';
2025-06-26 01:42:18 +08:00
import Search from '$lib/components/icons/Search.svelte';
2024-11-13 19:09:46 +08:00
import User from '$lib/components/icons/User.svelte';
import UserCircleSolid from '$lib/components/icons/UserCircleSolid.svelte';
2024-11-14 18:20:34 +08:00
import GroupModal from './Groups/EditGroupModal.svelte';
import Pencil from '$lib/components/icons/Pencil.svelte';
import GroupItem from './Groups/GroupItem.svelte';
import AddGroupModal from './Groups/AddGroupModal.svelte';
2024-11-15 10:37:42 +08:00
import { createNewGroup, getGroups } from '$lib/apis/groups';
import { getAllUsers, updateUserDefaultPermissions } from '$lib/apis/users';
2024-11-13 18:00:00 +08:00
const i18n = getContext('i18n');
let loaded = false;
2025-04-30 20:58:38 +08:00
let users = [];
let total = 0;
2024-11-14 18:20:34 +08:00
2024-11-13 18:00:00 +08:00
let groups = [];
let filteredGroups;
$: filteredGroups = groups.filter((user) => {
if (search === '') {
return true;
} else {
let name = user.name.toLowerCase();
const query = search.toLowerCase();
return name.includes(query);
}
});
let search = '';
let defaultPermissions = {};
2024-11-13 18:00:00 +08:00
let showCreateGroupModal = false;
2024-11-14 18:20:34 +08:00
let showDefaultPermissionsModal = false;
2024-11-13 18:00:00 +08:00
2024-11-15 12:51:49 +08:00
const setGroups = async () => {
const allGroups = await getGroups(localStorage.token);
const userGroup = allGroups.find((g) => g.name.toLowerCase() === 'user');
if (userGroup) {
defaultPermissions = userGroup.permissions;
}
groups = allGroups.filter((g) => g.name.toLowerCase() !== 'user');
2024-11-15 12:51:49 +08:00
};
const addGroupHandler = async (group) => {
const res = await createNewGroup(localStorage.token, group).catch((error) => {
2025-01-21 14:41:32 +08:00
toast.error(`${error}`);
2024-11-15 12:51:49 +08:00
return null;
});
if (res) {
toast.success($i18n.t('Group created successfully'));
groups = await getGroups(localStorage.token);
}
};
2024-11-15 18:05:43 +08:00
const updateDefaultPermissionsHandler = async (group) => {
console.debug(group.permissions);
2024-11-15 18:05:43 +08:00
2024-11-16 17:24:34 +08:00
const res = await updateUserDefaultPermissions(localStorage.token, group.permissions).catch(
2024-11-15 18:05:43 +08:00
(error) => {
2025-01-21 14:41:32 +08:00
toast.error(`${error}`);
2024-11-15 18:05:43 +08:00
return null;
}
);
if (res) {
toast.success($i18n.t('Default permissions updated successfully'));
2024-11-17 07:26:01 +08:00
defaultPermissions = await getUserDefaultPermissions(localStorage.token);
2024-11-15 18:05:43 +08:00
}
2024-11-15 12:51:49 +08:00
};
2024-11-13 18:00:00 +08:00
onMount(async () => {
if ($user?.role !== 'admin') {
await goto('/');
2025-04-30 20:58:38 +08:00
return;
2024-11-13 18:00:00 +08:00
}
2025-04-30 20:58:38 +08:00
const res = await getAllUsers(localStorage.token).catch((error) => {
toast.error(`${error}`);
return null;
});
if (res) {
users = res.users;
total = res.total;
}
await setGroups();
2024-11-13 18:00:00 +08:00
loaded = true;
});
</script>
{#if loaded}
2024-11-15 12:51:49 +08:00
<AddGroupModal bind:show={showCreateGroupModal} onSubmit={addGroupHandler} />
2024-11-13 18:00:00 +08:00
<div class="mt-0.5 mb-2 gap-1 flex flex-col md:flex-row justify-between">
<div class="flex md:self-center text-lg font-medium px-0.5">
{$i18n.t('Groups')}
<div class="flex self-center w-[1px] h-6 mx-2.5 bg-gray-50 dark:bg-gray-850" />
<span class="text-lg font-medium text-gray-500 dark:text-gray-300">{groups.length}</span>
</div>
<div class="flex gap-1">
<div class=" flex w-full space-x-2">
<div class="flex flex-1">
<div class=" self-center ml-1 mr-3">
2025-06-26 02:04:40 +08:00
<Search />
2024-11-13 18:00:00 +08:00
</div>
<input
2025-02-16 11:27:25 +08:00
class=" w-full text-sm pr-4 py-1 rounded-r-xl outline-hidden bg-transparent"
2024-11-13 18:00:00 +08:00
bind:value={search}
placeholder={$i18n.t('Search')}
/>
</div>
<div>
<Tooltip content={$i18n.t('Create Group')}>
<button
class=" p-2 rounded-xl hover:bg-gray-100 dark:bg-gray-900 dark:hover:bg-gray-850 transition font-medium text-sm flex items-center space-x-1"
on:click={() => {
showCreateGroupModal = !showCreateGroupModal;
}}
>
<Plus className="size-3.5" />
</button>
</Tooltip>
</div>
</div>
</div>
</div>
<div>
{#if filteredGroups.length === 0}
<div class="flex flex-col items-center justify-center h-40">
<div class=" text-xl font-medium">
{$i18n.t('Organize your users')}
</div>
<div class="mt-1 text-sm dark:text-gray-300">
{$i18n.t('Use groups to group your users and assign permissions.')}
</div>
<div class="mt-3">
<button
class=" px-4 py-1.5 text-sm rounded-full bg-black hover:bg-gray-800 text-white dark:bg-white dark:text-black dark:hover:bg-gray-100 transition font-medium flex items-center space-x-1"
aria-label={$i18n.t('Create Group')}
on:click={() => {
showCreateGroupModal = true;
}}
>
{$i18n.t('Create Group')}
</button>
</div>
</div>
{:else}
2024-11-13 19:09:46 +08:00
<div>
2025-09-16 04:53:11 +08:00
<div class=" flex items-center gap-3 justify-between text-xs uppercase px-1 font-semibold">
2025-08-20 02:39:17 +08:00
<div class="w-full basis-3/5">{$i18n.t('Group')}</div>
2024-11-13 19:09:46 +08:00
2025-08-20 02:39:17 +08:00
<div class="w-full basis-2/5 text-right">{$i18n.t('Users')}</div>
2024-11-13 19:09:46 +08:00
</div>
2025-02-16 11:50:40 +08:00
<hr class="mt-1.5 border-gray-100 dark:border-gray-850" />
2024-11-13 19:09:46 +08:00
{#each filteredGroups as group}
2024-11-14 19:16:26 +08:00
<div class="my-2">
<GroupItem {group} {users} {setGroups} {defaultPermissions} />
2024-11-14 19:16:26 +08:00
</div>
2024-11-13 19:09:46 +08:00
{/each}
</div>
2024-11-13 18:00:00 +08:00
{/if}
2024-11-13 19:09:46 +08:00
2025-02-16 11:50:40 +08:00
<hr class="mb-2 border-gray-100 dark:border-gray-850" />
2024-11-13 19:09:46 +08:00
2024-11-15 12:51:49 +08:00
<GroupModal
bind:show={showDefaultPermissionsModal}
tabs={['permissions']}
2024-11-17 07:26:01 +08:00
bind:permissions={defaultPermissions}
2024-11-15 12:51:49 +08:00
custom={false}
onSubmit={updateDefaultPermissionsHandler}
/>
2024-11-14 18:20:34 +08:00
<button
class="flex items-center justify-between rounded-lg w-full transition pt-1"
on:click={() => {
showDefaultPermissionsModal = true;
}}
>
2024-11-13 19:09:46 +08:00
<div class="flex items-center gap-2.5">
<div class="p-1.5 bg-black/5 dark:bg-white/10 rounded-full">
<UsersSolid className="size-4" />
</div>
<div class="text-left">
<div class=" text-sm font-medium">{$i18n.t('Default permissions')}</div>
<div class="flex text-xs mt-0.5">
{$i18n.t('applies to all users with the "user" role')}
</div>
</div>
</div>
<div>
<ChevronRight strokeWidth="2.5" />
</div>
</button>
2024-11-13 18:00:00 +08:00
</div>
{/if}