|
|
|
import { isString } from '@vue/shared'
|
|
|
|
import type { Roles, RolesObj, WorkspaceUserRoles } from 'nocodb-sdk'
|
|
|
|
import { extractRolesObj } from 'nocodb-sdk'
|
|
|
|
import { computed, createSharedComposable, rolePermissions, useApi, useGlobal } from '#imports'
|
|
|
|
import type { Permission } from '#imports'
|
|
|
|
|
|
|
|
const hasPermission = (role: Exclude<Roles, WorkspaceUserRoles>, hasRole: boolean, permission: Permission | string) => {
|
|
|
|
const rolePermission = rolePermissions[role]
|
|
|
|
|
|
|
|
if (!hasRole || !rolePermission) return false
|
|
|
|
|
|
|
|
if (isString(rolePermission) && rolePermission === '*') return true
|
|
|
|
|
|
|
|
if ('include' in rolePermission && rolePermission.include) {
|
|
|
|
return !!rolePermission.include[permission as keyof typeof rolePermission.include]
|
|
|
|
}
|
|
|
|
|
|
|
|
return rolePermission[permission as keyof typeof rolePermission]
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Provides the roles a user currently has
|
|
|
|
*
|
|
|
|
* * `userRoles` - the roles a user has outside of projects
|
|
|
|
* * `projectRoles` - the roles a user has in the current project (if one was loaded)
|
|
|
|
* * `allRoles` - all roles a user has (userRoles + projectRoles)
|
|
|
|
* * `loadRoles` - a function to load reload user roles for scope
|
|
|
|
*/
|
|
|
|
export const useRoles = createSharedComposable(() => {
|
|
|
|
const { user } = useGlobal()
|
|
|
|
|
|
|
|
const { api } = useApi()
|
|
|
|
|
|
|
|
const allRoles = computed<RolesObj | null>(() => {
|
|
|
|
let orgRoles = user.value?.roles ?? {}
|
|
|
|
|
|
|
|
orgRoles = extractRolesObj(orgRoles)
|
|
|
|
|
|
|
|
let projectRoles = user.value?.project_roles ?? {}
|
|
|
|
|
|
|
|
projectRoles = extractRolesObj(projectRoles)
|
|
|
|
|
|
|
|
return {
|
|
|
|
...orgRoles,
|
|
|
|
...projectRoles,
|
|
|
|
}
|
|
|
|
})
|
|
|
|
|
|
|
|
const orgRoles = computed<RolesObj | null>(() => {
|
|
|
|
let orgRoles = user.value?.roles ?? {}
|
|
|
|
|
|
|
|
orgRoles = extractRolesObj(orgRoles)
|
|
|
|
|
|
|
|
return orgRoles
|
|
|
|
})
|
|
|
|
|
|
|
|
const projectRoles = computed<RolesObj | null>(() => {
|
|
|
|
let projectRoles = user.value?.project_roles ?? {}
|
|
|
|
|
|
|
|
if (Object.keys(projectRoles).length === 0) {
|
|
|
|
projectRoles = user.value?.roles ?? {}
|
|
|
|
}
|
|
|
|
|
|
|
|
projectRoles = extractRolesObj(projectRoles)
|
|
|
|
|
|
|
|
return projectRoles
|
|
|
|
})
|
|
|
|
|
|
|
|
const workspaceRoles = computed<RolesObj | null>(() => {
|
|
|
|
return null
|
|
|
|
})
|
|
|
|
|
|
|
|
async function loadRoles(
|
|
|
|
projectId?: string,
|
|
|
|
options: { isSharedBase?: boolean; sharedBaseId?: string; isSharedErd?: boolean; sharedErdId?: string } = {},
|
|
|
|
) {
|
|
|
|
if (options?.isSharedBase) {
|
|
|
|
const res = await api.auth.me(
|
|
|
|
{
|
|
|
|
project_id: projectId,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
headers: {
|
|
|
|
'xc-shared-base-id': options?.sharedBaseId,
|
|
|
|
},
|
|
|
|
},
|
|
|
|
)
|
|
|
|
|
|
|
|
user.value = {
|
|
|
|
...user.value,
|
|
|
|
roles: res.roles,
|
|
|
|
project_roles: res.project_roles,
|
|
|
|
} as typeof User
|
|
|
|
} else if (options?.isSharedErd) {
|
|
|
|
const res = await api.auth.me(
|
|
|
|
{
|
|
|
|
project_id: projectId,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
headers: {
|
|
|
|
'xc-shared-erd-id': options?.sharedErdId,
|
|
|
|
},
|
|
|
|
},
|
|
|
|
)
|
|
|
|
|
|
|
|
user.value = {
|
|
|
|
...user.value,
|
|
|
|
roles: res.roles,
|
|
|
|
project_roles: res.project_roles,
|
|
|
|
} as typeof User
|
|
|
|
} else if (projectId) {
|
|
|
|
const res = await api.auth.me({ project_id: projectId })
|
|
|
|
|
|
|
|
user.value = {
|
|
|
|
...user.value,
|
|
|
|
roles: res.roles,
|
|
|
|
project_roles: res.project_roles,
|
|
|
|
} as typeof User
|
|
|
|
} else {
|
|
|
|
const res = await api.auth.me({})
|
|
|
|
|
|
|
|
user.value = {
|
|
|
|
...user.value,
|
|
|
|
roles: res.roles,
|
|
|
|
project_roles: res.project_roles,
|
|
|
|
} as typeof User
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
const isUIAllowed = (
|
|
|
|
permission: Permission | string,
|
|
|
|
args: { roles?: string | Record<string, boolean> | string[] | null } = {},
|
|
|
|
) => {
|
|
|
|
const { roles } = args
|
|
|
|
|
|
|
|
let checkRoles: Record<string, boolean> = {}
|
|
|
|
|
|
|
|
if (!roles) {
|
|
|
|
if (allRoles.value) checkRoles = allRoles.value
|
|
|
|
} else {
|
|
|
|
checkRoles = extractRolesObj(roles)
|
|
|
|
}
|
|
|
|
|
|
|
|
return Object.entries(checkRoles).some(([role, hasRole]) =>
|
|
|
|
hasPermission(role as Exclude<Roles, WorkspaceUserRoles>, hasRole, permission),
|
|
|
|
)
|
|
|
|
}
|
|
|
|
|
|
|
|
return { allRoles, orgRoles, workspaceRoles, projectRoles, loadRoles, isUIAllowed }
|
|
|
|
})
|