mirror of
https://github.com/logto-io/logto.git
synced 2024-12-30 20:33:54 -05:00
chore(schemas,test,core): add IT to ensure mock connectors are installed (#3762)
This commit is contained in:
parent
858854b9b1
commit
6e5b48bdcb
4 changed files with 71 additions and 21 deletions
|
@ -1,6 +1,6 @@
|
||||||
import { buildRawConnector } from '@logto/cli/lib/connector/index.js';
|
import { buildRawConnector } from '@logto/cli/lib/connector/index.js';
|
||||||
import { demoConnectorIds, validateConfig } from '@logto/connector-kit';
|
import { demoConnectorIds, validateConfig } from '@logto/connector-kit';
|
||||||
import { Connectors, ConnectorType } from '@logto/schemas';
|
import { connectorFactoryResponseGuard, Connectors, ConnectorType } from '@logto/schemas';
|
||||||
import { buildIdGenerator } from '@logto/shared';
|
import { buildIdGenerator } from '@logto/shared';
|
||||||
import cleanDeep from 'clean-deep';
|
import cleanDeep from 'clean-deep';
|
||||||
import { string, object } from 'zod';
|
import { string, object } from 'zod';
|
||||||
|
@ -200,18 +200,29 @@ export default function connectorRoutes<T extends AuthedRouter>(
|
||||||
}
|
}
|
||||||
);
|
);
|
||||||
|
|
||||||
router.get('/connector-factories', async (ctx, next) => {
|
router.get(
|
||||||
|
'/connector-factories',
|
||||||
|
koaGuard({
|
||||||
|
response: connectorFactoryResponseGuard.array(),
|
||||||
|
status: [200],
|
||||||
|
}),
|
||||||
|
async (ctx, next) => {
|
||||||
const connectorFactories = await loadConnectorFactories();
|
const connectorFactories = await loadConnectorFactories();
|
||||||
ctx.body = connectorFactories.map((connectorFactory) =>
|
ctx.body = connectorFactories.map((connectorFactory) =>
|
||||||
transpileConnectorFactory(connectorFactory)
|
transpileConnectorFactory(connectorFactory)
|
||||||
);
|
);
|
||||||
|
|
||||||
return next();
|
return next();
|
||||||
});
|
}
|
||||||
|
);
|
||||||
|
|
||||||
router.get(
|
router.get(
|
||||||
'/connector-factories/:id',
|
'/connector-factories/:id',
|
||||||
koaGuard({ params: object({ id: string().min(1) }) }),
|
koaGuard({
|
||||||
|
params: object({ id: string().min(1) }),
|
||||||
|
response: connectorFactoryResponseGuard,
|
||||||
|
status: [200],
|
||||||
|
}),
|
||||||
async (ctx, next) => {
|
async (ctx, next) => {
|
||||||
const {
|
const {
|
||||||
params: { id },
|
params: { id },
|
||||||
|
|
|
@ -1,4 +1,9 @@
|
||||||
import type { Connector, ConnectorResponse, CreateConnector } from '@logto/schemas';
|
import type {
|
||||||
|
Connector,
|
||||||
|
ConnectorFactoryResponse,
|
||||||
|
ConnectorResponse,
|
||||||
|
CreateConnector,
|
||||||
|
} from '@logto/schemas';
|
||||||
|
|
||||||
import { authedAdminApi } from './api.js';
|
import { authedAdminApi } from './api.js';
|
||||||
|
|
||||||
|
@ -8,6 +13,12 @@ export const listConnectors = async () =>
|
||||||
export const getConnector = async (connectorId: string) =>
|
export const getConnector = async (connectorId: string) =>
|
||||||
authedAdminApi.get(`connectors/${connectorId}`).json<ConnectorResponse>();
|
authedAdminApi.get(`connectors/${connectorId}`).json<ConnectorResponse>();
|
||||||
|
|
||||||
|
export const listConnectorFactories = async () =>
|
||||||
|
authedAdminApi.get('connector-factories').json<ConnectorFactoryResponse[]>();
|
||||||
|
|
||||||
|
export const getConnectorFactory = async (connectorId: string) =>
|
||||||
|
authedAdminApi.get(`connector-factories/${connectorId}`).json<ConnectorFactoryResponse>();
|
||||||
|
|
||||||
// FIXME @Darcy: correct use of `id` and `connectorId`.
|
// FIXME @Darcy: correct use of `id` and `connectorId`.
|
||||||
export const postConnector = async (
|
export const postConnector = async (
|
||||||
payload: Pick<CreateConnector, 'connectorId' | 'config' | 'metadata' | 'syncProfile'>
|
payload: Pick<CreateConnector, 'connectorId' | 'config' | 'metadata' | 'syncProfile'>
|
||||||
|
|
|
@ -18,16 +18,43 @@ import {
|
||||||
sendEmailTestMessage,
|
sendEmailTestMessage,
|
||||||
sendSmsTestMessage,
|
sendSmsTestMessage,
|
||||||
updateConnectorConfig,
|
updateConnectorConfig,
|
||||||
|
listConnectorFactories,
|
||||||
|
getConnectorFactory,
|
||||||
} from '#src/api/connector.js';
|
} from '#src/api/connector.js';
|
||||||
|
|
||||||
const connectorIdMap = new Map<string, string>();
|
const connectorIdMap = new Map<string, string>();
|
||||||
|
|
||||||
|
const mockConnectorSetups = [
|
||||||
|
{ connectorId: mockSmsConnectorId, config: mockSmsConnectorConfig },
|
||||||
|
{ connectorId: mockEmailConnectorId, config: mockEmailConnectorConfig },
|
||||||
|
{ connectorId: mockSocialConnectorId, config: mockSocialConnectorConfig },
|
||||||
|
];
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* We'd better only use mock connectors in integration tests.
|
* We'd better only use mock connectors in integration tests.
|
||||||
* Since we will refactor connectors soon, keep using some real connectors
|
* Since we will refactor connectors soon, keep using some real connectors
|
||||||
* for testing updating configs and enabling/disabling for now.
|
* for testing updating configs and enabling/disabling for now.
|
||||||
*/
|
*/
|
||||||
test('connector set-up flow', async () => {
|
test('connector set-up flow', async () => {
|
||||||
|
/**
|
||||||
|
* Check whether mock connector factories are properly installed using
|
||||||
|
* both connector factory APIs:
|
||||||
|
* 1. GET /connector-factories (listConnectorFactories()).
|
||||||
|
* 2. GET /connector-factories/:connectorId (getConnectorFactory()).
|
||||||
|
*/
|
||||||
|
const connectorFactories = await listConnectorFactories();
|
||||||
|
await Promise.all(
|
||||||
|
mockConnectorSetups.map(async ({ connectorId }) => {
|
||||||
|
expect(connectorFactories.find(({ id }) => id === connectorId)).toBeDefined();
|
||||||
|
|
||||||
|
const connectorFactory = await getConnectorFactory(connectorId);
|
||||||
|
expect(connectorFactory).toBeDefined();
|
||||||
|
})
|
||||||
|
);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Clean up `connector` table
|
||||||
|
*/
|
||||||
const connectors = await listConnectors();
|
const connectors = await listConnectors();
|
||||||
await Promise.all(
|
await Promise.all(
|
||||||
connectors.map(async ({ id }) => {
|
connectors.map(async ({ id }) => {
|
||||||
|
@ -40,11 +67,7 @@ test('connector set-up flow', async () => {
|
||||||
* Set up social/SMS/email connectors
|
* Set up social/SMS/email connectors
|
||||||
*/
|
*/
|
||||||
await Promise.all(
|
await Promise.all(
|
||||||
[
|
mockConnectorSetups.map(async ({ connectorId, config }) => {
|
||||||
{ connectorId: mockSmsConnectorId, config: mockSmsConnectorConfig },
|
|
||||||
{ connectorId: mockEmailConnectorId, config: mockEmailConnectorConfig },
|
|
||||||
{ connectorId: mockSocialConnectorId, config: mockSocialConnectorConfig },
|
|
||||||
].map(async ({ connectorId, config }) => {
|
|
||||||
// @darcy FIXME: should call post method directly
|
// @darcy FIXME: should call post method directly
|
||||||
const { id } = await postConnector({ connectorId });
|
const { id } = await postConnector({ connectorId });
|
||||||
connectorIdMap.set(connectorId, id);
|
connectorIdMap.set(connectorId, id);
|
||||||
|
|
|
@ -1,4 +1,6 @@
|
||||||
import type { BaseConnector, ConnectorMetadata, ConnectorType } from '@logto/connector-kit';
|
import type { BaseConnector, ConnectorMetadata } from '@logto/connector-kit';
|
||||||
|
import { ConnectorType, connectorMetadataGuard } from '@logto/connector-kit';
|
||||||
|
import { z } from 'zod';
|
||||||
|
|
||||||
import type { Connector } from '../db-entries/index.js';
|
import type { Connector } from '../db-entries/index.js';
|
||||||
|
|
||||||
|
@ -12,8 +14,11 @@ export type ConnectorResponse = Pick<
|
||||||
Omit<BaseConnector<ConnectorType>, 'configGuard' | 'metadata'> &
|
Omit<BaseConnector<ConnectorType>, 'configGuard' | 'metadata'> &
|
||||||
ConnectorMetadata & { isDemo?: boolean };
|
ConnectorMetadata & { isDemo?: boolean };
|
||||||
|
|
||||||
export type ConnectorFactoryResponse = Omit<
|
export const connectorFactoryResponseGuard = z
|
||||||
BaseConnector<ConnectorType>,
|
.object({
|
||||||
'configGuard' | 'metadata'
|
type: z.nativeEnum(ConnectorType), // Omit<BaseConnector<ConnectorType>, 'configGuard' | 'metadata'>
|
||||||
> &
|
isDemo: z.boolean().optional(),
|
||||||
ConnectorMetadata & { isDemo?: boolean };
|
})
|
||||||
|
.merge(connectorMetadataGuard);
|
||||||
|
|
||||||
|
export type ConnectorFactoryResponse = z.infer<typeof connectorFactoryResponseGuard>;
|
||||||
|
|
Loading…
Reference in a new issue