homepage-plus
Some checks failed
Docker CI / Linting Checks (push) Has been cancelled
Docker CI / Docker Build & Push (push) Has been cancelled

This commit is contained in:
Dominik 2024-12-12 15:15:24 +01:00
parent 2376184b14
commit 1370cd195a
16 changed files with 265 additions and 51 deletions

View file

@ -0,0 +1,70 @@
import ProxyIdentityProvider from "./proxy";
import NullIdentityProvider from "./null";
const IdentityProviders = {
null: NullIdentityProvider,
proxy: ProxyIdentityProvider,
};
function getProviderByKey(key) {
return IdentityProviders[key] || NullIdentityProvider;
}
function identityAllow({ user, groups }, item) {
const groupAllow =
"allowGroups" in item && item.allowGroups && groups.some((group) => item.allowGroups.includes(group));
const userAllow = "allowUsers" in item && item.allowUsers && item.allowUsers.includes(user);
const allowAll = !("allowGroups" in item && item.allowGroups) && !("allowUsers" in item && item.allowUsers);
return userAllow || groupAllow || allowAll;
}
export function checkAllowedGroup(perms, idGroups, groupName) {
const testGroup = idGroups.find((group) => group.name === groupName);
return testGroup ? identityAllow(perms, testGroup) : true;
}
function filterAllowedItems(perms, idGroups, groups, groupKey) {
return groups
.filter((group) => checkAllowedGroup(perms, idGroups, group.name))
.map((group) => ({
name: group.name,
[groupKey]: group[groupKey].filter((item) => identityAllow(perms, item)),
}))
.filter((group) => group[groupKey].length);
}
export function readIdentitySettings({ provider, groups } = {}) {
let groupArray = [];
if (groups) {
if (Array.isArray(groups)) {
groupArray = groups.map((group) => ({
name: Object.keys(group)[0],
allowUsers: group.allowUsers,
allowGroups: group.allowGroups,
}));
} else {
groupArray = Object.keys(groups).map((group) => ({
name: group,
allowUsers: groups[group].allowUsers,
allowGroups: groups[group].allowGroups,
}));
}
}
return {
provider: provider ? getProviderByKey(provider.type).create(provider) : NullIdentityProvider.create(),
groups: groupArray,
};
}
export async function fetchWithIdentity(key, context) {
return getProviderByKey(context.provider).fetch([key, context]);
}
export const filterAllowedServices = (perms, idGroups, services) =>
filterAllowedItems(perms, idGroups, services, "services");
export const filterAllowedBookmarks = (perms, idGroups, bookmarks) =>
filterAllowedItems(perms, idGroups, bookmarks, "bookmarks");
export const filterAllowedWidgets = (perms, widgets) =>
widgets.filter((widget) => identityAllow(perms, widget.options));

View file

@ -0,0 +1,21 @@
const NullIdentity = { user: null, groups: [] };
function createNullIdentity() {
return {
getIdentity: () => NullIdentity,
getContext: () => ({
provider: "null",
}),
};
}
async function fetchNullIdentity([key]) {
return fetch(key).then((res) => res.json());
}
const NullIdentityProvider = {
create: createNullIdentity,
fetch: fetchNullIdentity,
};
export default NullIdentityProvider;

View file

@ -0,0 +1,34 @@
// 'proxy' identity provider is meant to be used by a reverse proxy that injects permission headers into the origin
// request. In this case we are relying on our proxy to authenitcate our users and validate their identity.
function getProxyPermissions(userHeader, groupHeader, groupSeparator, request) {
const user =
userHeader && request.headers[userHeader.toLowerCase()] ? request.headers[userHeader.toLowerCase()] : null;
const groupsString =
groupHeader && request.headers[groupHeader.toLowerCase()] ? request.headers[groupHeader.toLowerCase()] : "";
return { user, groups: groupsString ? groupsString.split(groupSeparator ?? "|").map((v) => v.trim()) : [] };
}
function createProxyIdentity({ groupHeader, groupSeparator, userHeader }) {
return {
getContext: (request) => ({
provider: "proxy",
...(userHeader &&
request.headers[userHeader] && { [userHeader.toLowerCase()]: request.headers[userHeader.toLowerCase()] }),
...(groupHeader &&
request.headers[groupHeader] && { [groupHeader.toLowerCase()]: request.headers[groupHeader.toLowerCase()] }),
}),
getIdentity: (request) => getProxyPermissions(userHeader, groupHeader, groupSeparator, request),
};
}
async function fetchProxyIdentity([key, context]) {
return fetch(key, { headers: context.headers }).then((res) => res.json());
}
const ProxyIdentityProvider = {
create: createProxyIdentity,
fetch: fetchProxyIdentity,
};
export default ProxyIdentityProvider;