mirror of https://github.com/nocodb/nocodb
You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
150 lines
3.8 KiB
150 lines
3.8 KiB
import { isString } from '@vue/shared' |
|
import type { Roles, RolesObj, WorkspaceUserRoles } from 'nocodb-sdk' |
|
import { extractRolesObj } from 'nocodb-sdk' |
|
|
|
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 bases |
|
* * `baseRoles` - the roles a user has in the current base (if one was loaded) |
|
* * `allRoles` - all roles a user has (userRoles + baseRoles) |
|
* * `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 baseRoles = user.value?.base_roles ?? {} |
|
|
|
baseRoles = extractRolesObj(baseRoles) |
|
|
|
return { |
|
...orgRoles, |
|
...baseRoles, |
|
} |
|
}) |
|
|
|
const orgRoles = computed<RolesObj | null>(() => { |
|
let orgRoles = user.value?.roles ?? {} |
|
|
|
orgRoles = extractRolesObj(orgRoles) |
|
|
|
return orgRoles |
|
}) |
|
|
|
const baseRoles = computed<RolesObj | null>(() => { |
|
let baseRoles = user.value?.base_roles ?? {} |
|
|
|
if (Object.keys(baseRoles).length === 0) { |
|
baseRoles = user.value?.roles ?? {} |
|
} |
|
|
|
baseRoles = extractRolesObj(baseRoles) |
|
|
|
return baseRoles |
|
}) |
|
|
|
const workspaceRoles = computed<RolesObj | null>(() => { |
|
return null |
|
}) |
|
|
|
async function loadRoles( |
|
baseId?: string, |
|
options: { isSharedBase?: boolean; sharedBaseId?: string; isSharedErd?: boolean; sharedErdId?: string } = {}, |
|
) { |
|
if (options?.isSharedBase) { |
|
const res = await api.auth.me( |
|
{ |
|
base_id: baseId, |
|
}, |
|
{ |
|
headers: { |
|
'xc-shared-base-id': options?.sharedBaseId, |
|
}, |
|
}, |
|
) |
|
|
|
user.value = { |
|
...user.value, |
|
roles: res.roles, |
|
base_roles: res.base_roles, |
|
} as User |
|
} else if (options?.isSharedErd) { |
|
const res = await api.auth.me( |
|
{ |
|
base_id: baseId, |
|
}, |
|
{ |
|
headers: { |
|
'xc-shared-erd-id': options?.sharedErdId, |
|
}, |
|
}, |
|
) |
|
|
|
user.value = { |
|
...user.value, |
|
roles: res.roles, |
|
base_roles: res.base_roles, |
|
} as User |
|
} else if (baseId) { |
|
const res = await api.auth.me({ base_id: baseId }) |
|
|
|
user.value = { |
|
...user.value, |
|
roles: res.roles, |
|
base_roles: res.base_roles, |
|
display_name: res.display_name, |
|
} as User |
|
} else { |
|
const res = await api.auth.me({}) |
|
|
|
user.value = { |
|
...user.value, |
|
roles: res.roles, |
|
base_roles: res.base_roles, |
|
display_name: res.display_name, |
|
} as 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, baseRoles, loadRoles, isUIAllowed } |
|
})
|
|
|