mirror of
https://github.com/logto-io/logto.git
synced 2025-01-13 21:30:30 -05:00
feat(core): connect console (#306)
* feat(core): connect console * fix(core): prod spa dist * fix(core): test * test(core): refactor spa proxy tests
This commit is contained in:
parent
5734333dea
commit
365c63b2c7
10 changed files with 149 additions and 142 deletions
|
@ -7,7 +7,7 @@
|
||||||
"lerna": "lerna",
|
"lerna": "lerna",
|
||||||
"bootstrap": "lerna bootstrap",
|
"bootstrap": "lerna bootstrap",
|
||||||
"prepare": "if test \"$NODE_ENV\" != \"production\" && test \"$CI\" != \"true\" ; then husky install ; fi",
|
"prepare": "if test \"$NODE_ENV\" != \"production\" && test \"$CI\" != \"true\" ; then husky install ; fi",
|
||||||
"dev": "lerna run --stream prepack && lerna --scope=@logto/{core,ui} exec -- pnpm dev"
|
"dev": "lerna run --stream prepack && lerna --scope=@logto/{core,ui,console} exec -- pnpm dev"
|
||||||
},
|
},
|
||||||
"devDependencies": {
|
"devDependencies": {
|
||||||
"@commitlint/cli": "^13.2.1",
|
"@commitlint/cli": "^13.2.1",
|
||||||
|
|
|
@ -9,8 +9,9 @@
|
||||||
"preinstall": "npx only-allow pnpm",
|
"preinstall": "npx only-allow pnpm",
|
||||||
"precommit": "lint-staged",
|
"precommit": "lint-staged",
|
||||||
"start": "parcel src/index.html",
|
"start": "parcel src/index.html",
|
||||||
|
"dev": "PORT=5002 parcel src/index.html --public-url /console --no-hmr",
|
||||||
"check": "tsc --noEmit",
|
"check": "tsc --noEmit",
|
||||||
"build": "pnpm check && rm -rf dist && parcel build src/index.html --no-autoinstall",
|
"build": "pnpm check && rm -rf dist && parcel build src/index.html --no-autoinstall --public-url /console",
|
||||||
"lint": "eslint --ext .ts --ext .tsx src",
|
"lint": "eslint --ext .ts --ext .tsx src",
|
||||||
"stylelint": "stylelint \"src/**/*.scss\""
|
"stylelint": "stylelint \"src/**/*.scss\""
|
||||||
},
|
},
|
||||||
|
|
|
@ -4,7 +4,7 @@ import * as koaErrorHandler from '@/middleware/koa-error-handler';
|
||||||
import * as koaI18next from '@/middleware/koa-i18next';
|
import * as koaI18next from '@/middleware/koa-i18next';
|
||||||
import * as koaOIDCErrorHandler from '@/middleware/koa-oidc-error-handler';
|
import * as koaOIDCErrorHandler from '@/middleware/koa-oidc-error-handler';
|
||||||
import * as koaSlonikErrorHandler from '@/middleware/koa-slonik-error-handler';
|
import * as koaSlonikErrorHandler from '@/middleware/koa-slonik-error-handler';
|
||||||
import * as koaUIProxy from '@/middleware/koa-ui-proxy';
|
import * as koaSpaProxy from '@/middleware/koa-spa-proxy';
|
||||||
import * as koaUserLog from '@/middleware/koa-user-log';
|
import * as koaUserLog from '@/middleware/koa-user-log';
|
||||||
import * as initOidc from '@/oidc/init';
|
import * as initOidc from '@/oidc/init';
|
||||||
import * as initRouter from '@/routes/init';
|
import * as initRouter from '@/routes/init';
|
||||||
|
@ -20,7 +20,7 @@ describe('App Init', () => {
|
||||||
koaI18next,
|
koaI18next,
|
||||||
koaOIDCErrorHandler,
|
koaOIDCErrorHandler,
|
||||||
koaSlonikErrorHandler,
|
koaSlonikErrorHandler,
|
||||||
koaUIProxy,
|
koaSpaProxy,
|
||||||
koaUserLog,
|
koaUserLog,
|
||||||
];
|
];
|
||||||
const initMethods = [initRouter, initOidc];
|
const initMethods = [initRouter, initOidc];
|
||||||
|
|
|
@ -3,14 +3,15 @@ import https from 'https';
|
||||||
|
|
||||||
import Koa from 'koa';
|
import Koa from 'koa';
|
||||||
import koaLogger from 'koa-logger';
|
import koaLogger from 'koa-logger';
|
||||||
|
import mount from 'koa-mount';
|
||||||
|
|
||||||
import { port } from '@/env/consts';
|
import { MountedApps, port } from '@/env/consts';
|
||||||
import koaConnectorErrorHandler from '@/middleware/koa-connector-error-handle';
|
import koaConnectorErrorHandler from '@/middleware/koa-connector-error-handle';
|
||||||
import koaErrorHandler from '@/middleware/koa-error-handler';
|
import koaErrorHandler from '@/middleware/koa-error-handler';
|
||||||
import koaI18next from '@/middleware/koa-i18next';
|
import koaI18next from '@/middleware/koa-i18next';
|
||||||
import koaOIDCErrorHandler from '@/middleware/koa-oidc-error-handler';
|
import koaOIDCErrorHandler from '@/middleware/koa-oidc-error-handler';
|
||||||
import koaSlonikErrorHandler from '@/middleware/koa-slonik-error-handler';
|
import koaSlonikErrorHandler from '@/middleware/koa-slonik-error-handler';
|
||||||
import koaUIProxy from '@/middleware/koa-ui-proxy';
|
import koaSpaProxy from '@/middleware/koa-spa-proxy';
|
||||||
import koaUserLog from '@/middleware/koa-user-log';
|
import koaUserLog from '@/middleware/koa-user-log';
|
||||||
import initOidc from '@/oidc/init';
|
import initOidc from '@/oidc/init';
|
||||||
import initRouter from '@/routes/init';
|
import initRouter from '@/routes/init';
|
||||||
|
@ -29,7 +30,10 @@ export default async function initApp(app: Koa): Promise<void> {
|
||||||
const provider = await initOidc(app);
|
const provider = await initOidc(app);
|
||||||
initRouter(app, provider);
|
initRouter(app, provider);
|
||||||
|
|
||||||
app.use(koaUIProxy());
|
app.use(
|
||||||
|
mount('/' + MountedApps.Console, koaSpaProxy(MountedApps.Console, 5002, MountedApps.Console))
|
||||||
|
);
|
||||||
|
app.use(koaSpaProxy());
|
||||||
|
|
||||||
const { HTTPS_CERT, HTTPS_KEY } = process.env;
|
const { HTTPS_CERT, HTTPS_KEY } = process.env;
|
||||||
|
|
||||||
|
|
6
packages/core/src/env/consts.ts
vendored
6
packages/core/src/env/consts.ts
vendored
|
@ -3,7 +3,11 @@ import { assertEnv, getEnv } from '@silverhand/essentials';
|
||||||
export const signIn = assertEnv('UI_SIGN_IN_ROUTE');
|
export const signIn = assertEnv('UI_SIGN_IN_ROUTE');
|
||||||
export const isProduction = getEnv('NODE_ENV') === 'production';
|
export const isProduction = getEnv('NODE_ENV') === 'production';
|
||||||
export const port = Number(getEnv('PORT', '3001'));
|
export const port = Number(getEnv('PORT', '3001'));
|
||||||
export const mountedApps = Object.freeze(['api', 'oidc']);
|
export enum MountedApps {
|
||||||
|
Api = 'api',
|
||||||
|
Oidc = 'oidc',
|
||||||
|
Console = 'console',
|
||||||
|
}
|
||||||
export const developmentUserId = getEnv('DEVELOPMENT_USER_ID');
|
export const developmentUserId = getEnv('DEVELOPMENT_USER_ID');
|
||||||
|
|
||||||
// Trusting TLS offloading proxies: https://github.com/panva/node-oidc-provider/blob/main/docs/README.md#trusting-tls-offloading-proxies
|
// Trusting TLS offloading proxies: https://github.com/panva/node-oidc-provider/blob/main/docs/README.md#trusting-tls-offloading-proxies
|
||||||
|
|
72
packages/core/src/middleware/koa-spa-proxy.test.ts
Normal file
72
packages/core/src/middleware/koa-spa-proxy.test.ts
Normal file
|
@ -0,0 +1,72 @@
|
||||||
|
import { MountedApps } from '@/env/consts';
|
||||||
|
import { createContextWithRouteParameters } from '@/utils/test-utils';
|
||||||
|
|
||||||
|
import koaSpaProxy from './koa-spa-proxy';
|
||||||
|
|
||||||
|
const mockProxyMiddleware = jest.fn();
|
||||||
|
const mockStaticMiddleware = jest.fn();
|
||||||
|
|
||||||
|
jest.mock('fs/promises', () => ({
|
||||||
|
...jest.requireActual('fs/promises'),
|
||||||
|
readdir: jest.fn().mockResolvedValue(['sign-in']),
|
||||||
|
}));
|
||||||
|
|
||||||
|
jest.mock('koa-proxies', () => jest.fn(() => mockProxyMiddleware));
|
||||||
|
jest.mock('koa-static', () => jest.fn(() => mockStaticMiddleware));
|
||||||
|
|
||||||
|
describe('koaSpaProxy middleware', () => {
|
||||||
|
const envBackup = process.env;
|
||||||
|
|
||||||
|
beforeEach(() => {
|
||||||
|
process.env = { ...envBackup };
|
||||||
|
jest.clearAllMocks();
|
||||||
|
jest.resetModules();
|
||||||
|
});
|
||||||
|
|
||||||
|
const next = jest.fn();
|
||||||
|
|
||||||
|
for (const app of Object.values(MountedApps)) {
|
||||||
|
// eslint-disable-next-line @typescript-eslint/no-loop-func
|
||||||
|
it(`${app} path should not call dev proxy`, async () => {
|
||||||
|
const ctx = createContextWithRouteParameters({
|
||||||
|
url: `/${app}/foo`,
|
||||||
|
});
|
||||||
|
|
||||||
|
await koaSpaProxy()(ctx, next);
|
||||||
|
|
||||||
|
expect(mockProxyMiddleware).not.toBeCalled();
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
it('dev env should call dev proxy for SPA paths', async () => {
|
||||||
|
const ctx = createContextWithRouteParameters();
|
||||||
|
await koaSpaProxy()(ctx, next);
|
||||||
|
expect(mockProxyMiddleware).toBeCalled();
|
||||||
|
});
|
||||||
|
|
||||||
|
it('production env should overwrite the request path to root if no target ui file are detected', async () => {
|
||||||
|
process.env.NODE_ENV = 'production';
|
||||||
|
|
||||||
|
const ctx = createContextWithRouteParameters({
|
||||||
|
url: '/foo',
|
||||||
|
});
|
||||||
|
|
||||||
|
const { default: proxy } = await import('./koa-spa-proxy');
|
||||||
|
await proxy()(ctx, next);
|
||||||
|
|
||||||
|
expect(mockStaticMiddleware).toBeCalled();
|
||||||
|
expect(ctx.request.path).toEqual('/');
|
||||||
|
});
|
||||||
|
|
||||||
|
it('production env should call the static middleware if path hit the ui file directory', async () => {
|
||||||
|
process.env.NODE_ENV = 'production';
|
||||||
|
|
||||||
|
const { default: proxy } = await import('./koa-spa-proxy');
|
||||||
|
const ctx = createContextWithRouteParameters({
|
||||||
|
url: '/sign-in',
|
||||||
|
});
|
||||||
|
|
||||||
|
await proxy()(ctx, next);
|
||||||
|
expect(mockStaticMiddleware).toBeCalled();
|
||||||
|
});
|
||||||
|
});
|
59
packages/core/src/middleware/koa-spa-proxy.ts
Normal file
59
packages/core/src/middleware/koa-spa-proxy.ts
Normal file
|
@ -0,0 +1,59 @@
|
||||||
|
import fs from 'fs/promises';
|
||||||
|
import path from 'path';
|
||||||
|
|
||||||
|
import { MiddlewareType } from 'koa';
|
||||||
|
import proxy from 'koa-proxies';
|
||||||
|
import { IRouterParamContext } from 'koa-router';
|
||||||
|
import serveStatic from 'koa-static';
|
||||||
|
|
||||||
|
import { isProduction, MountedApps } from '@/env/consts';
|
||||||
|
|
||||||
|
export default function koaSpaProxy<StateT, ContextT extends IRouterParamContext, ResponseBodyT>(
|
||||||
|
packagePath = 'ui',
|
||||||
|
port = 5001,
|
||||||
|
prefix = ''
|
||||||
|
): MiddlewareType<StateT, ContextT, ResponseBodyT> {
|
||||||
|
type Middleware = MiddlewareType<StateT, ContextT, ResponseBodyT>;
|
||||||
|
|
||||||
|
const distPath = path.join('..', packagePath, 'dist');
|
||||||
|
|
||||||
|
const spaProxy: Middleware = isProduction
|
||||||
|
? serveStatic(distPath)
|
||||||
|
: proxy('*', {
|
||||||
|
target: `http://localhost:${port}`,
|
||||||
|
changeOrigin: true,
|
||||||
|
logs: true,
|
||||||
|
rewrite: (requestPath) => {
|
||||||
|
// Static files
|
||||||
|
if (requestPath.includes('.')) {
|
||||||
|
return '/' + path.join(prefix, requestPath);
|
||||||
|
}
|
||||||
|
|
||||||
|
// In-app routes
|
||||||
|
return requestPath;
|
||||||
|
},
|
||||||
|
});
|
||||||
|
|
||||||
|
return async (ctx, next) => {
|
||||||
|
const requestPath = ctx.request.path;
|
||||||
|
|
||||||
|
// Route has been handled by one of mounted apps
|
||||||
|
if (
|
||||||
|
Object.values(MountedApps).some((app) => app !== prefix && requestPath.startsWith(`/${app}`))
|
||||||
|
) {
|
||||||
|
return next();
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!isProduction) {
|
||||||
|
return spaProxy(ctx, next);
|
||||||
|
}
|
||||||
|
|
||||||
|
const spaDistFiles = await fs.readdir(distPath);
|
||||||
|
|
||||||
|
if (!spaDistFiles.some((file) => requestPath.startsWith('/' + file))) {
|
||||||
|
ctx.request.path = '/';
|
||||||
|
}
|
||||||
|
|
||||||
|
return spaProxy(ctx, next);
|
||||||
|
};
|
||||||
|
}
|
|
@ -1,88 +0,0 @@
|
||||||
import { mountedApps } from '@/env/consts';
|
|
||||||
import { createContextWithRouteParameters } from '@/utils/test-utils';
|
|
||||||
|
|
||||||
import koaUIProxy from './koa-ui-proxy';
|
|
||||||
|
|
||||||
const mockProxyMiddleware = jest.fn();
|
|
||||||
const mockStaticMiddleware = jest.fn();
|
|
||||||
|
|
||||||
jest.mock('fs/promises', () => ({
|
|
||||||
...jest.requireActual('fs/promises'),
|
|
||||||
readdir: jest.fn().mockResolvedValue(['sign-in']),
|
|
||||||
}));
|
|
||||||
|
|
||||||
jest.mock('koa-proxies', () => jest.fn(() => mockProxyMiddleware));
|
|
||||||
jest.mock('koa-static', () => jest.fn(() => mockStaticMiddleware));
|
|
||||||
|
|
||||||
describe('koaUIProxy middleware', () => {
|
|
||||||
beforeEach(() => {
|
|
||||||
jest.clearAllMocks();
|
|
||||||
jest.resetModules();
|
|
||||||
});
|
|
||||||
|
|
||||||
const next = jest.fn();
|
|
||||||
|
|
||||||
for (const app of mountedApps) {
|
|
||||||
// eslint-disable-next-line @typescript-eslint/no-loop-func
|
|
||||||
it(`${app} path should not call uiProxy`, async () => {
|
|
||||||
const ctx = createContextWithRouteParameters({
|
|
||||||
url: `/${app}/foo`,
|
|
||||||
});
|
|
||||||
|
|
||||||
await koaUIProxy()(ctx, next);
|
|
||||||
|
|
||||||
expect(mockProxyMiddleware).not.toBeCalled();
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
it('dev env should call proxy middleware for ui paths', async () => {
|
|
||||||
const ctx = createContextWithRouteParameters();
|
|
||||||
await koaUIProxy()(ctx, next);
|
|
||||||
expect(mockProxyMiddleware).toBeCalled();
|
|
||||||
});
|
|
||||||
|
|
||||||
it('production env should overwrite the request path to root if no target ui file are detected', async () => {
|
|
||||||
// Mock the @/env/consts
|
|
||||||
jest.mock('@/env/consts', () => ({
|
|
||||||
...jest.requireActual('@/env/consts'),
|
|
||||||
isProduction: true,
|
|
||||||
}));
|
|
||||||
|
|
||||||
/* eslint-disable @typescript-eslint/no-require-imports */
|
|
||||||
/* eslint-disable @typescript-eslint/no-var-requires */
|
|
||||||
/* eslint-disable unicorn/prefer-module */
|
|
||||||
const koaUIProxyModule = require('./koa-ui-proxy') as { default: typeof koaUIProxy };
|
|
||||||
/* eslint-enable @typescript-eslint/no-require-imports */
|
|
||||||
/* eslint-enable @typescript-eslint/no-var-requires */
|
|
||||||
/* eslint-enable unicorn/prefer-module */
|
|
||||||
const ctx = createContextWithRouteParameters({
|
|
||||||
url: '/foo',
|
|
||||||
});
|
|
||||||
|
|
||||||
await koaUIProxyModule.default()(ctx, next);
|
|
||||||
expect(mockStaticMiddleware).toBeCalled();
|
|
||||||
expect(ctx.request.path).toEqual('/');
|
|
||||||
});
|
|
||||||
|
|
||||||
it('production env should call the static middleware if path hit the ui file directory', async () => {
|
|
||||||
// Mock the @/env/consts
|
|
||||||
jest.mock('@/env/consts', () => ({
|
|
||||||
...jest.requireActual('@/env/consts'),
|
|
||||||
isProduction: true,
|
|
||||||
}));
|
|
||||||
|
|
||||||
/* eslint-disable @typescript-eslint/no-require-imports */
|
|
||||||
/* eslint-disable @typescript-eslint/no-var-requires */
|
|
||||||
/* eslint-disable unicorn/prefer-module */
|
|
||||||
const koaUIProxyModule = require('./koa-ui-proxy') as { default: typeof koaUIProxy };
|
|
||||||
/* eslint-enable @typescript-eslint/no-require-imports */
|
|
||||||
/* eslint-enable @typescript-eslint/no-var-requires */
|
|
||||||
/* eslint-enable unicorn/prefer-module */
|
|
||||||
const ctx = createContextWithRouteParameters({
|
|
||||||
url: '/sign-in',
|
|
||||||
});
|
|
||||||
|
|
||||||
await koaUIProxyModule.default()(ctx, next);
|
|
||||||
expect(mockStaticMiddleware).toBeCalled();
|
|
||||||
});
|
|
||||||
});
|
|
|
@ -1,45 +0,0 @@
|
||||||
import fs from 'fs/promises';
|
|
||||||
|
|
||||||
import { MiddlewareType } from 'koa';
|
|
||||||
import proxy from 'koa-proxies';
|
|
||||||
import { IRouterParamContext } from 'koa-router';
|
|
||||||
import serveStatic from 'koa-static';
|
|
||||||
|
|
||||||
import { isProduction, mountedApps } from '@/env/consts';
|
|
||||||
|
|
||||||
const PATH_TO_UI_DIST = '../ui/dist';
|
|
||||||
|
|
||||||
export default function koaUIProxy<
|
|
||||||
StateT,
|
|
||||||
ContextT extends IRouterParamContext,
|
|
||||||
ResponseBodyT
|
|
||||||
>(): MiddlewareType<StateT, ContextT, ResponseBodyT> {
|
|
||||||
type Middleware = MiddlewareType<StateT, ContextT, ResponseBodyT>;
|
|
||||||
|
|
||||||
const uiProxy: Middleware = isProduction
|
|
||||||
? serveStatic(PATH_TO_UI_DIST)
|
|
||||||
: proxy('*', {
|
|
||||||
target: 'http://localhost:5001',
|
|
||||||
changeOrigin: true,
|
|
||||||
logs: true,
|
|
||||||
});
|
|
||||||
|
|
||||||
return async (ctx, next) => {
|
|
||||||
// Route has been handled by one of mounted apps
|
|
||||||
if (mountedApps.some((app) => ctx.request.path.startsWith(`/${app}`))) {
|
|
||||||
return next();
|
|
||||||
}
|
|
||||||
|
|
||||||
if (!isProduction) {
|
|
||||||
return uiProxy(ctx, next);
|
|
||||||
}
|
|
||||||
|
|
||||||
const uiDistFiles = await fs.readdir(PATH_TO_UI_DIST);
|
|
||||||
|
|
||||||
if (!uiDistFiles.some((file) => ctx.request.path.startsWith(`/${file}`))) {
|
|
||||||
ctx.request.path = '/';
|
|
||||||
}
|
|
||||||
|
|
||||||
return uiProxy(ctx, next);
|
|
||||||
};
|
|
||||||
}
|
|
|
@ -7,7 +7,7 @@
|
||||||
"preinstall": "npx only-allow pnpm",
|
"preinstall": "npx only-allow pnpm",
|
||||||
"precommit": "lint-staged",
|
"precommit": "lint-staged",
|
||||||
"start": "parcel src/index.html",
|
"start": "parcel src/index.html",
|
||||||
"dev": "PORT=5001 pnpm start",
|
"dev": "PORT=5001 parcel src/index.html --no-hmr",
|
||||||
"check": "tsc --noEmit",
|
"check": "tsc --noEmit",
|
||||||
"build": "pnpm check && rm -rf dist && parcel build src/index.html --no-autoinstall",
|
"build": "pnpm check && rm -rf dist && parcel build src/index.html --no-autoinstall",
|
||||||
"lint": "eslint --ext .ts --ext .tsx src",
|
"lint": "eslint --ext .ts --ext .tsx src",
|
||||||
|
|
Loading…
Add table
Reference in a new issue