mirror of
https://github.com/logto-io/logto.git
synced 2025-03-31 22:51:25 -05:00
refactor(core): fix segment matching for path-based
This commit is contained in:
parent
d29432f45d
commit
a8355eb6a3
2 changed files with 47 additions and 15 deletions
|
@ -77,7 +77,7 @@ describe('getTenantId()', () => {
|
|||
ADMIN_DISABLE_LOCALHOST: '1',
|
||||
};
|
||||
|
||||
expect(getTenantId(new URL('http://localhost:5000/app///asdasd'))).toBe(defaultTenantId);
|
||||
expect(getTenantId(new URL('http://localhost:5000/app///asdasd'))).toBe(undefined);
|
||||
expect(getTenantId(new URL('http://localhost:3002/app///asdasd'))).toBe(undefined);
|
||||
expect(getTenantId(new URL('https://user.foo.logto.mock/app'))).toBe('foo');
|
||||
expect(getTenantId(new URL('https://user.admin.logto.mock/app//'))).toBe(undefined); // Admin endpoint is explicitly set
|
||||
|
@ -92,4 +92,21 @@ describe('getTenantId()', () => {
|
|||
};
|
||||
expect(getTenantId(new URL('https://user.admin.logto.mock/app//'))).toBe('admin');
|
||||
});
|
||||
|
||||
it('should resolve proper tenant ID for path-based multi-tenancy', async () => {
|
||||
process.env = {
|
||||
...backupEnv,
|
||||
NODE_ENV: 'production',
|
||||
PORT: '5000',
|
||||
ENDPOINT: 'https://user.logto.mock/app',
|
||||
PATH_BASED_MULTI_TENANCY: '1',
|
||||
};
|
||||
|
||||
expect(getTenantId(new URL('http://localhost:5000/app///asdasd'))).toBe('app');
|
||||
expect(getTenantId(new URL('http://localhost:3002///bar///asdasd'))).toBe(adminTenantId);
|
||||
expect(getTenantId(new URL('https://user.foo.logto.mock/app'))).toBe(undefined);
|
||||
expect(getTenantId(new URL('https://user.admin.logto.mock/app//'))).toBe(undefined);
|
||||
expect(getTenantId(new URL('https://user.logto.mock/app'))).toBe(undefined);
|
||||
expect(getTenantId(new URL('https://user.logto.mock/app/admin'))).toBe('admin');
|
||||
});
|
||||
});
|
||||
|
|
|
@ -1,6 +1,7 @@
|
|||
import { adminTenantId, defaultTenantId } from '@logto/schemas';
|
||||
import { conditionalString } from '@silverhand/essentials';
|
||||
|
||||
import type UrlSet from '#src/env-set/UrlSet.js';
|
||||
import { EnvSet, getTenantEndpoint } from '#src/env-set/index.js';
|
||||
|
||||
const normalizePathname = (pathname: string) =>
|
||||
|
@ -14,6 +15,32 @@ const isEndpointOf = (current: URL, endpoint: URL) => {
|
|||
);
|
||||
};
|
||||
|
||||
const matchDomainBasedTenantId = (pattern: URL, url: URL) => {
|
||||
const toMatch = pattern.hostname.replace('*', '([^.]*)');
|
||||
const matchedId = new RegExp(toMatch).exec(url.hostname)?.[1];
|
||||
|
||||
if (!matchedId || matchedId === '*') {
|
||||
return;
|
||||
}
|
||||
|
||||
if (isEndpointOf(url, getTenantEndpoint(matchedId, EnvSet.values))) {
|
||||
return matchedId;
|
||||
}
|
||||
};
|
||||
|
||||
const matchPathBasedTenantId = (urlSet: UrlSet, url: URL) => {
|
||||
const found = urlSet.deduplicated().find((value) => isEndpointOf(url, value));
|
||||
|
||||
if (!found) {
|
||||
return;
|
||||
}
|
||||
|
||||
const urlSegments = url.pathname.split('/');
|
||||
const endpointSegments = found.pathname.split('/');
|
||||
|
||||
return urlSegments[found.pathname === '/' ? 1 : endpointSegments.length];
|
||||
};
|
||||
|
||||
export const getTenantId = (url: URL) => {
|
||||
const {
|
||||
isDomainBasedMultiTenancy,
|
||||
|
@ -40,20 +67,8 @@ export const getTenantId = (url: URL) => {
|
|||
}
|
||||
|
||||
if (isPathBasedMultiTenancy) {
|
||||
const urlSegments = url.pathname.split('/');
|
||||
const endpointSegments = urlSet.endpoint.pathname.split('/');
|
||||
|
||||
return urlSegments[endpointSegments.length - 1];
|
||||
return matchPathBasedTenantId(urlSet, url);
|
||||
}
|
||||
|
||||
const toMatch = urlSet.endpoint.hostname.replace('*', '([^.]*)');
|
||||
const matchedId = new RegExp(toMatch).exec(url.hostname)?.[1];
|
||||
|
||||
if (!matchedId || matchedId === '*') {
|
||||
return;
|
||||
}
|
||||
|
||||
if (isEndpointOf(url, getTenantEndpoint(matchedId, EnvSet.values))) {
|
||||
return matchedId;
|
||||
}
|
||||
return matchDomainBasedTenantId(urlSet.endpoint, url);
|
||||
};
|
||||
|
|
Loading…
Add table
Reference in a new issue