<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'; import { WEBUI_NAME, config, user, showSidebar, knowledge } from '$lib/stores'; import { WEBUI_BASE_URL } from '$lib/constants'; import Tooltip from '$lib/components/common/Tooltip.svelte'; import Plus from '$lib/components/icons/Plus.svelte'; 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'; import User from '$lib/components/icons/User.svelte'; import UserCircleSolid from '$lib/components/icons/UserCircleSolid.svelte'; 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'; import { createNewGroup, getGroups } from '$lib/apis/groups'; import { getUserDefaultPermissions, updateUserDefaultPermissions } from '$lib/apis/users'; const i18n = getContext('i18n'); let loaded = false; export let users = []; 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 = { workspace: { models: false, knowledge: false, prompts: false, tools: false }, chat: { file_upload: true, delete: true, edit: true, temporary: true } }; let showCreateGroupModal = false; let showDefaultPermissionsModal = false; const setGroups = async () => { groups = await getGroups(localStorage.token); }; const addGroupHandler = async (group) => { const res = await createNewGroup(localStorage.token, group).catch((error) => { toast.error(error); return null; }); if (res) { toast.success($i18n.t('Group created successfully')); groups = await getGroups(localStorage.token); } }; const updateDefaultPermissionsHandler = async (group) => { console.log(group.permissions); const res = await updateUserDefaultPermissions(localStorage.token, group.permissions).catch( (error) => { toast.error(error); return null; } ); if (res) { toast.success($i18n.t('Default permissions updated successfully')); defaultPermissions = await getUserDefaultPermissions(localStorage.token); } }; onMount(async () => { if ($user?.role !== 'admin') { await goto('/'); } else { await setGroups(); defaultPermissions = await getUserDefaultPermissions(localStorage.token); } loaded = true; }); </script> {#if loaded} <AddGroupModal bind:show={showCreateGroupModal} onSubmit={addGroupHandler} /> <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"> <svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 20 20" fill="currentColor" class="w-4 h-4" > <path fill-rule="evenodd" d="M9 3.5a5.5 5.5 0 100 11 5.5 5.5 0 000-11zM2 9a7 7 0 1112.452 4.391l3.328 3.329a.75.75 0 11-1.06 1.06l-3.329-3.328A7 7 0 012 9z" clip-rule="evenodd" /> </svg> </div> <input class=" w-full text-sm pr-4 py-1 rounded-r-xl outline-none bg-transparent" 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} <div> <div class=" flex items-center gap-3 justify-between text-xs uppercase px-1 font-bold"> <div class="w-full">Group</div> <div class="w-full">Users</div> <div class="w-full"></div> </div> <hr class="mt-1.5 border-gray-50 dark:border-gray-850" /> {#each filteredGroups as group} <div class="my-2"> <GroupItem {group} {users} {setGroups} /> </div> {/each} </div> {/if} <hr class="mb-2 border-gray-50 dark:border-gray-850" /> <GroupModal bind:show={showDefaultPermissionsModal} tabs={['permissions']} bind:permissions={defaultPermissions} custom={false} onSubmit={updateDefaultPermissionsHandler} /> <button class="flex items-center justify-between rounded-lg w-full transition pt-1" on:click={() => { showDefaultPermissionsModal = true; }} > <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> </div> {/if}