mirror of
https://github.com/logto-io/logto.git
synced 2025-03-31 22:51:25 -05:00
chore(console,core): add/replace SSO connector logo resources (#4963)
This commit is contained in:
parent
0d636d5607
commit
ac099b1e64
23 changed files with 143 additions and 68 deletions
|
@ -8,12 +8,6 @@
|
|||
padding: _.unit(6) _.unit(8);
|
||||
gap: _.unit(6);
|
||||
|
||||
.icon {
|
||||
width: 60px;
|
||||
height: 60px;
|
||||
flex-shrink: 0;
|
||||
}
|
||||
|
||||
.copyId {
|
||||
flex: 1;
|
||||
}
|
||||
|
@ -56,3 +50,9 @@
|
|||
}
|
||||
}
|
||||
}
|
||||
|
||||
.icon {
|
||||
width: 60px;
|
||||
height: 60px;
|
||||
flex-shrink: 0;
|
||||
}
|
||||
|
|
|
@ -0,0 +1,11 @@
|
|||
.container {
|
||||
background-color: var(--color-hover);
|
||||
display: flex;
|
||||
align-items: center;
|
||||
justify-content: center;
|
||||
flex-shrink: 0;
|
||||
}
|
||||
|
||||
.logo {
|
||||
flex-shrink: 0;
|
||||
}
|
|
@ -0,0 +1,42 @@
|
|||
import { Theme } from '@logto/schemas';
|
||||
import type { SsoConnectorWithProviderConfig } from '@logto/schemas';
|
||||
import classNames from 'classnames';
|
||||
|
||||
import ImageWithErrorFallback from '@/ds-components/ImageWithErrorFallback';
|
||||
import useTheme from '@/hooks/use-theme';
|
||||
|
||||
import * as styles from './index.module.scss';
|
||||
|
||||
type Props = {
|
||||
className?: string;
|
||||
containerClassName?: string;
|
||||
data: Pick<SsoConnectorWithProviderConfig, 'providerLogo' | 'providerLogoDark' | 'branding'>;
|
||||
};
|
||||
|
||||
const pickLogoForCurrentTheme = (
|
||||
isDarkMode: boolean,
|
||||
{ logo, logoDark }: { logo: string; logoDark: string },
|
||||
branding: SsoConnectorWithProviderConfig['branding']
|
||||
): string => {
|
||||
if (isDarkMode) {
|
||||
return branding.darkLogo ?? logoDark;
|
||||
}
|
||||
return branding.logo ?? logo;
|
||||
};
|
||||
|
||||
function SsoConnectorLogo({ className, containerClassName, data }: Props) {
|
||||
const theme = useTheme();
|
||||
const isDarkMode = theme === Theme.Dark;
|
||||
const { providerLogo: logo, providerLogoDark: logoDark, branding } = data;
|
||||
|
||||
return (
|
||||
<ImageWithErrorFallback
|
||||
containerClassName={classNames(styles.container, containerClassName)}
|
||||
className={classNames(styles.logo, className)}
|
||||
alt="logo"
|
||||
src={pickLogoForCurrentTheme(isDarkMode, { logo, logoDark }, branding)}
|
||||
/>
|
||||
);
|
||||
}
|
||||
|
||||
export default SsoConnectorLogo;
|
|
@ -1,7 +1,8 @@
|
|||
import { type SsoConnectorFactoryDetail } from '@logto/schemas';
|
||||
import { type SsoConnectorFactoryDetail, Theme } from '@logto/schemas';
|
||||
import classNames from 'classnames';
|
||||
|
||||
import ImageWithErrorFallback from '@/ds-components/ImageWithErrorFallback';
|
||||
import useTheme from '@/hooks/use-theme';
|
||||
|
||||
import * as styles from './index.module.scss';
|
||||
|
||||
|
@ -9,14 +10,15 @@ type Props = {
|
|||
data: SsoConnectorFactoryDetail;
|
||||
};
|
||||
|
||||
function SsoConnectorRadio({ data: { providerName, logo, description } }: Props) {
|
||||
function SsoConnectorRadio({ data: { providerName, logo, logoDark, description } }: Props) {
|
||||
const theme = useTheme();
|
||||
return (
|
||||
<div className={styles.ssoConnector}>
|
||||
<ImageWithErrorFallback
|
||||
containerClassName={styles.container}
|
||||
className={styles.logo}
|
||||
alt="logo"
|
||||
src={logo}
|
||||
src={theme === Theme.Light ? logo : logoDark}
|
||||
/>
|
||||
<div className={styles.content}>
|
||||
<div className={classNames(styles.name)}>
|
||||
|
|
|
@ -7,18 +7,18 @@
|
|||
.container {
|
||||
width: 40px;
|
||||
height: 40px;
|
||||
border-radius: 8px;
|
||||
background-color: var(--color-hover);
|
||||
display: flex;
|
||||
align-items: center;
|
||||
justify-content: center;
|
||||
flex-shrink: 0;
|
||||
border-radius: _.unit(2);
|
||||
|
||||
> img {
|
||||
width: 28px;
|
||||
height: 28px;
|
||||
}
|
||||
}
|
||||
|
||||
.logo {
|
||||
width: 40px;
|
||||
height: 40px;
|
||||
flex-shrink: 0;
|
||||
width: 28px;
|
||||
height: 28px;
|
||||
}
|
||||
|
||||
.domains {
|
||||
|
|
|
@ -1,5 +1,5 @@
|
|||
import { withAppInsights } from '@logto/app-insights/react';
|
||||
import { type SsoConnectorWithProviderConfig, Theme, SsoProviderName } from '@logto/schemas';
|
||||
import { type SsoConnectorWithProviderConfig, SsoProviderName } from '@logto/schemas';
|
||||
import { conditional } from '@silverhand/essentials';
|
||||
import { useEffect, useState } from 'react';
|
||||
import { useTranslation } from 'react-i18next';
|
||||
|
@ -13,16 +13,15 @@ import ItemPreview from '@/components/ItemPreview';
|
|||
import ListPage from '@/components/ListPage';
|
||||
import { defaultPageSize } from '@/consts';
|
||||
import Button from '@/ds-components/Button';
|
||||
import ImageWithErrorFallback from '@/ds-components/ImageWithErrorFallback';
|
||||
import TablePlaceholder from '@/ds-components/Table/TablePlaceholder';
|
||||
import Tag from '@/ds-components/Tag';
|
||||
import type { RequestError } from '@/hooks/use-api';
|
||||
import useSearchParametersWatcher from '@/hooks/use-search-parameters-watcher';
|
||||
import useTenantPathname from '@/hooks/use-tenant-pathname';
|
||||
import useTheme from '@/hooks/use-theme';
|
||||
import { buildUrl } from '@/utils/url';
|
||||
|
||||
import Guide from './Guide';
|
||||
import SsoConnectorLogo from './SsoConnectorLogo';
|
||||
import SsoCreationModal from './SsoCreationModal';
|
||||
import * as styles from './index.module.scss';
|
||||
import { type SsoConnectorWithProviderConfigWithGeneric } from './types';
|
||||
|
@ -34,8 +33,6 @@ const buildDetailsPathname = (id: string) => `${enterpriseSsoPathname}/${id}`;
|
|||
const buildGuidePathname = (id: string) => `${buildDetailsPathname(id)}/guide`;
|
||||
|
||||
function EnterpriseSsoConnectors() {
|
||||
const theme = useTheme();
|
||||
|
||||
const { pathname } = useLocation();
|
||||
const { navigate } = useTenantPathname();
|
||||
const { ssoConnectorId: id } = useParams();
|
||||
|
@ -91,19 +88,14 @@ function EnterpriseSsoConnectors() {
|
|||
title: t('enterprise_sso.col_connector_name'),
|
||||
dataIndex: 'name',
|
||||
colSpan: 256,
|
||||
render: ({ id, connectorName, providerLogo, branding }) => (
|
||||
render: ({ id, connectorName, ...rest }) => (
|
||||
<ItemPreview
|
||||
title={connectorName}
|
||||
icon={
|
||||
<ImageWithErrorFallback
|
||||
<SsoConnectorLogo
|
||||
className={styles.logo}
|
||||
containerClassName={styles.container}
|
||||
alt="logo"
|
||||
src={
|
||||
(theme === Theme.Dark && branding.darkLogo
|
||||
? branding.darkLogo
|
||||
: branding.logo) ?? providerLogo
|
||||
}
|
||||
data={rest}
|
||||
/>
|
||||
}
|
||||
to={buildDetailsPathname(id)}
|
||||
|
|
|
@ -10,16 +10,15 @@
|
|||
.container {
|
||||
width: 60px;
|
||||
height: 60px;
|
||||
border-radius: 12px;
|
||||
background-color: var(--color-hover);
|
||||
display: flex;
|
||||
align-items: center;
|
||||
justify-content: center;
|
||||
flex-shrink: 0;
|
||||
border-radius: _.unit(3);
|
||||
|
||||
> img {
|
||||
width: 42px;
|
||||
height: 42px;
|
||||
}
|
||||
}
|
||||
|
||||
.logo {
|
||||
width: 60px;
|
||||
height: 60px;
|
||||
flex-shrink: 0;
|
||||
width: 42px;
|
||||
height: 42px;
|
||||
}
|
||||
|
|
|
@ -1,5 +1,6 @@
|
|||
import { withAppInsights } from '@logto/app-insights/react';
|
||||
import { Theme, SsoProviderName } from '@logto/schemas';
|
||||
import { SsoProviderName } from '@logto/schemas';
|
||||
import { pick } from '@silverhand/essentials';
|
||||
import { useCallback, useEffect, useState } from 'react';
|
||||
import { toast } from 'react-hot-toast';
|
||||
import { useTranslation } from 'react-i18next';
|
||||
|
@ -16,13 +17,13 @@ import PageMeta from '@/components/PageMeta';
|
|||
import { EnterpriseSsoDetailsTabs } from '@/consts';
|
||||
import ConfirmModal from '@/ds-components/ConfirmModal';
|
||||
import DynamicT from '@/ds-components/DynamicT';
|
||||
import ImageWithErrorFallback from '@/ds-components/ImageWithErrorFallback';
|
||||
import TabNav, { TabNavItem } from '@/ds-components/TabNav';
|
||||
import type { RequestError } from '@/hooks/use-api';
|
||||
import useApi from '@/hooks/use-api';
|
||||
import useTenantPathname from '@/hooks/use-tenant-pathname';
|
||||
import useTheme from '@/hooks/use-theme';
|
||||
|
||||
import SsoConnectorLogo from '../EnterpriseSso/SsoConnectorLogo';
|
||||
import {
|
||||
type SsoConnectorWithProviderConfigWithGeneric,
|
||||
type ParsedSsoIdentityProviderConfig,
|
||||
|
@ -119,15 +120,10 @@ function EnterpriseSsoConnectorDetails<T extends SsoProviderName>() {
|
|||
<>
|
||||
<DetailsPageHeader
|
||||
icon={
|
||||
<ImageWithErrorFallback
|
||||
<SsoConnectorLogo
|
||||
className={styles.logo}
|
||||
containerClassName={styles.container}
|
||||
alt="logo"
|
||||
src={
|
||||
(theme === Theme.Dark && ssoConnector.branding.darkLogo
|
||||
? ssoConnector.branding.darkLogo
|
||||
: ssoConnector.branding.logo) ?? ssoConnector.providerLogo
|
||||
}
|
||||
data={pick(ssoConnector, 'providerLogo', 'providerLogoDark', 'branding')}
|
||||
/>
|
||||
}
|
||||
title={ssoConnector.connectorName}
|
||||
|
|
|
@ -4,9 +4,10 @@ import { useTranslation } from 'react-i18next';
|
|||
import useSWR from 'swr';
|
||||
|
||||
import CopyToClipboard from '@/ds-components/CopyToClipboard';
|
||||
import ImageWithErrorFallback from '@/ds-components/ImageWithErrorFallback';
|
||||
import Table from '@/ds-components/Table';
|
||||
import type { RequestError } from '@/hooks/use-api';
|
||||
import useTheme from '@/hooks/use-theme';
|
||||
import SsoConnectorLogo from '@/pages/EnterpriseSso/SsoConnectorLogo';
|
||||
|
||||
import * as styles from '../UserSocialIdentities/index.module.scss';
|
||||
|
||||
|
@ -15,13 +16,16 @@ type Props = {
|
|||
};
|
||||
|
||||
type DisplayConnector = {
|
||||
logo: SsoConnectorWithProviderConfig['providerLogo'];
|
||||
providerLogo: SsoConnectorWithProviderConfig['providerLogo'];
|
||||
providerLogoDark: SsoConnectorWithProviderConfig['providerLogoDark'];
|
||||
branding: SsoConnectorWithProviderConfig['branding'];
|
||||
name: SsoConnectorWithProviderConfig['connectorName'];
|
||||
userIdentity: UserSsoIdentity['identityId'];
|
||||
issuer: UserSsoIdentity['issuer'];
|
||||
};
|
||||
|
||||
function UserSsoIdentities({ ssoIdentities }: Props) {
|
||||
const theme = useTheme();
|
||||
const { t } = useTranslation(undefined, {
|
||||
keyPrefix: 'admin_console',
|
||||
});
|
||||
|
@ -40,15 +44,18 @@ function UserSsoIdentities({ ssoIdentities }: Props) {
|
|||
return (
|
||||
ssoIdentities
|
||||
.map((identity) => {
|
||||
const { providerLogo: logo, connectorName: name } =
|
||||
data.find((ssoConnector) => ssoConnector.id === identity.ssoConnectorId) ?? {};
|
||||
const {
|
||||
providerLogo,
|
||||
providerLogoDark,
|
||||
connectorName: name,
|
||||
} = data.find((ssoConnector) => ssoConnector.id === identity.ssoConnectorId) ?? {};
|
||||
const { identityId: userIdentity, issuer } = identity;
|
||||
|
||||
if (!(logo && name)) {
|
||||
if (!(providerLogo && providerLogoDark && name)) {
|
||||
return;
|
||||
}
|
||||
|
||||
return { logo, name, userIdentity, issuer };
|
||||
return { providerLogo, providerLogoDark, name, userIdentity, issuer };
|
||||
})
|
||||
// eslint-disable-next-line unicorn/prefer-native-coercion-functions
|
||||
.filter((identity): identity is DisplayConnector => Boolean(identity))
|
||||
|
@ -80,9 +87,9 @@ function UserSsoIdentities({ ssoIdentities }: Props) {
|
|||
title: t('user_details.sso_connectors.connectors'),
|
||||
dataIndex: 'name',
|
||||
colSpan: 5,
|
||||
render: ({ logo, name }) => (
|
||||
render: ({ providerLogo, providerLogoDark, branding, name }) => (
|
||||
<div className={styles.connectorName}>
|
||||
<ImageWithErrorFallback className={styles.icon} src={logo} alt="logo" />
|
||||
<SsoConnectorLogo data={{ providerLogo, providerLogoDark, branding }} />
|
||||
<div className={styles.name}>
|
||||
<span>{name}</span>
|
||||
</div>
|
||||
|
|
|
@ -161,7 +161,7 @@ describe('getFullSignInExperience()', () => {
|
|||
id: wellConfiguredSsoConnector.id,
|
||||
connectorName: wellConfiguredSsoConnector.connectorName,
|
||||
logo: ssoConnectorFactories[wellConfiguredSsoConnector.providerName].logo,
|
||||
darkLogo: undefined,
|
||||
darkLogo: ssoConnectorFactories[wellConfiguredSsoConnector.providerName].logoDark,
|
||||
},
|
||||
],
|
||||
isDevelopmentTenant: false,
|
||||
|
@ -185,7 +185,7 @@ describe('get sso connectors', () => {
|
|||
id: wellConfiguredSsoConnector.id,
|
||||
connectorName: wellConfiguredSsoConnector.connectorName,
|
||||
logo: ssoConnectorFactories[wellConfiguredSsoConnector.providerName].logo,
|
||||
darkLogo: undefined,
|
||||
darkLogo: ssoConnectorFactories[wellConfiguredSsoConnector.providerName].logoDark,
|
||||
},
|
||||
]);
|
||||
});
|
||||
|
|
|
@ -79,7 +79,7 @@ export const createSignInExperienceLibrary = (
|
|||
id,
|
||||
connectorName,
|
||||
logo: branding.logo ?? factory.logo,
|
||||
darkLogo: branding.darkLogo,
|
||||
darkLogo: branding.darkLogo ?? factory.logoDark,
|
||||
};
|
||||
}
|
||||
);
|
||||
|
|
|
@ -20,12 +20,13 @@ const mockTenantId = 'mock_tenant_id';
|
|||
|
||||
describe('parseFactoryDetail', () => {
|
||||
it.each(Object.values(SsoProviderName))('should return correct detail for %s', (providerName) => {
|
||||
const { logo, description } = ssoConnectorFactories[providerName];
|
||||
const { logo, logoDark, description } = ssoConnectorFactories[providerName];
|
||||
const detail = parseFactoryDetail(ssoConnectorFactories[providerName], 'en');
|
||||
|
||||
expect(detail).toEqual({
|
||||
providerName,
|
||||
logo,
|
||||
logoDark,
|
||||
description: description.en,
|
||||
});
|
||||
});
|
||||
|
@ -33,12 +34,13 @@ describe('parseFactoryDetail', () => {
|
|||
it.each(Object.values(SsoProviderName))(
|
||||
'should return correct detail for %s with unknown locale',
|
||||
(providerName) => {
|
||||
const { logo, description } = ssoConnectorFactories[providerName];
|
||||
const { logo, logoDark, description } = ssoConnectorFactories[providerName];
|
||||
const detail = parseFactoryDetail(ssoConnectorFactories[providerName], 'zh');
|
||||
|
||||
expect(detail).toEqual({
|
||||
providerName,
|
||||
logo,
|
||||
logoDark,
|
||||
description: description.en,
|
||||
});
|
||||
}
|
||||
|
@ -92,6 +94,7 @@ describe('fetchConnectorProviderDetails', () => {
|
|||
expect.objectContaining({
|
||||
...connector,
|
||||
providerLogo: ssoConnectorFactories[connector.providerName].logo,
|
||||
providerLogoDark: ssoConnectorFactories[connector.providerName].logoDark,
|
||||
providerConfig: {
|
||||
...connector.config,
|
||||
scope: 'openid profile email', // Default scope
|
||||
|
|
|
@ -18,11 +18,12 @@ export const parseFactoryDetail = (
|
|||
factory: SingleSignOnFactory<SsoProviderName>,
|
||||
locale: string
|
||||
) => {
|
||||
const { providerName, logo, description } = factory;
|
||||
const { providerName, logo, logoDark, description } = factory;
|
||||
|
||||
return {
|
||||
providerName,
|
||||
logo,
|
||||
logoDark,
|
||||
// eslint-disable-next-line @typescript-eslint/prefer-nullish-coalescing -- falsy value expected
|
||||
description: (isKeyOfI18nPhrases(locale, description) && description[locale]) || description.en,
|
||||
};
|
||||
|
@ -55,7 +56,7 @@ export const fetchConnectorProviderDetails = async (
|
|||
): Promise<SsoConnectorWithProviderConfig> => {
|
||||
const { providerName } = connector;
|
||||
|
||||
const { logo, constructor } = ssoConnectorFactories[providerName];
|
||||
const { logo, logoDark, constructor } = ssoConnectorFactories[providerName];
|
||||
|
||||
/*
|
||||
Safely fetch and parse the detailed connector config from provider.
|
||||
|
@ -69,6 +70,7 @@ export const fetchConnectorProviderDetails = async (
|
|||
return {
|
||||
...connector,
|
||||
providerLogo: logo,
|
||||
providerLogoDark: logoDark,
|
||||
providerConfig,
|
||||
};
|
||||
};
|
||||
|
|
|
@ -8,7 +8,9 @@ export class AzureAdSsoConnector extends SamlSsoConnector {}
|
|||
|
||||
export const azureAdSsoConnectorFactory: SingleSignOnFactory<SsoProviderName.AZURE_AD> = {
|
||||
providerName: SsoProviderName.AZURE_AD,
|
||||
logo: 'https://logtoeu.blob.core.windows.net/public-blobs/risa4g/aAYaRZOiGoxS/2023/09/03/zqdr28er/azure.png',
|
||||
logo: '',
|
||||
logoDark:
|
||||
'',
|
||||
description: {
|
||||
en: 'This connector is used to connect with Azure AD Single Sign-On.',
|
||||
},
|
||||
|
|
|
@ -49,7 +49,9 @@ export const googleWorkspaceSsoConnectorConfigGuard = basicOidcConnectorConfigGu
|
|||
export const googleWorkSpaceSsoConnectorFactory: SingleSignOnFactory<SsoProviderName.GOOGLE_WORKSPACE> =
|
||||
{
|
||||
providerName: SsoProviderName.GOOGLE_WORKSPACE,
|
||||
logo: '',
|
||||
logo: '',
|
||||
logoDark:
|
||||
'',
|
||||
description: {
|
||||
en: 'This connector is used to connect with Google Workspace Single Sign-On.',
|
||||
},
|
||||
|
|
|
@ -28,7 +28,9 @@ export class OidcSsoConnector extends OidcConnector implements SingleSignOn {
|
|||
|
||||
export const oidcSsoConnectorFactory: SingleSignOnFactory<SsoProviderName.OIDC> = {
|
||||
providerName: SsoProviderName.OIDC,
|
||||
logo: 'https://freesvg.org/img/techtonik_OpenID.png',
|
||||
logo: '',
|
||||
logoDark:
|
||||
'',
|
||||
description: {
|
||||
en: 'This connector is used to connect with OIDC single sign-on identity provider.',
|
||||
},
|
||||
|
|
5
packages/core/src/sso/OktaSsoConnector/consts.ts
Normal file
5
packages/core/src/sso/OktaSsoConnector/consts.ts
Normal file
File diff suppressed because one or more lines are too long
|
@ -12,6 +12,8 @@ import { basicOidcConnectorConfigGuard } from '../types/oidc.js';
|
|||
import { type ExtendedSocialUserInfo } from '../types/saml.js';
|
||||
import { type SingleSignOnConnectorSession } from '../types/session.js';
|
||||
|
||||
import { logoBase64, logoDarkBase64 } from './consts.js';
|
||||
|
||||
export class OktaSsoConnector extends OidcSsoConnector {
|
||||
/**
|
||||
* Override the getUserInfo method from the OidcSsoConnector class
|
||||
|
@ -54,7 +56,8 @@ export class OktaSsoConnector extends OidcSsoConnector {
|
|||
|
||||
export const oktaSsoConnectorFactory: SingleSignOnFactory<SsoProviderName.OKTA> = {
|
||||
providerName: SsoProviderName.OKTA,
|
||||
logo: 'https://logtodev.blob.core.windows.net/public-blobs/admin/r2a6qctI3lmG/2023/11/22/8bvg68e7/OKTA.D.png',
|
||||
logo: logoBase64,
|
||||
logoDark: logoDarkBase64,
|
||||
description: {
|
||||
en: 'This connector is used to connect with Okta Single Sign-On.',
|
||||
},
|
||||
|
|
|
@ -98,7 +98,9 @@ export class SamlSsoConnector extends SamlConnector implements SingleSignOn {
|
|||
|
||||
export const samlSsoConnectorFactory: SingleSignOnFactory<SsoProviderName.SAML> = {
|
||||
providerName: SsoProviderName.SAML,
|
||||
logo: '',
|
||||
logo: '',
|
||||
logoDark:
|
||||
'',
|
||||
description: {
|
||||
en: 'This connector is used to connect to SAML single sign-on identity provider.',
|
||||
},
|
||||
|
|
|
@ -35,6 +35,7 @@ type SingleSignOnConnectorConfig = {
|
|||
export type SingleSignOnFactory<T extends SsoProviderName> = {
|
||||
providerName: T;
|
||||
logo: string;
|
||||
logoDark: string;
|
||||
description: I18nPhrases;
|
||||
configGuard: SingleSignOnConnectorConfig[T];
|
||||
constructor: SingleSignOnConstructor[T];
|
||||
|
|
|
@ -15,6 +15,7 @@ export type ConnectorFactoryResponse = {
|
|||
|
||||
export type SsoConnectorWithProviderConfig = SsoConnector & {
|
||||
providerLogo: string;
|
||||
providerLogoDark: string;
|
||||
providerConfig?: Record<string, unknown>;
|
||||
};
|
||||
|
||||
|
|
|
@ -112,6 +112,7 @@ describe('get sso-connectors', () => {
|
|||
|
||||
expect(connector).toBeDefined();
|
||||
expect(connector?.providerLogo).toBeDefined();
|
||||
expect(connector?.providerLogoDark).toBeDefined();
|
||||
|
||||
// Empty config object is a valid SAML config.
|
||||
if (providerName === 'OIDC') {
|
||||
|
|
|
@ -51,6 +51,7 @@ export type SupportedSsoConnector = Omit<SsoConnector, 'providerName'> & {
|
|||
const ssoConnectorFactoryDetailGuard = z.object({
|
||||
providerName: z.nativeEnum(SsoProviderName),
|
||||
logo: z.string(),
|
||||
logoDark: z.string(),
|
||||
description: z.string(),
|
||||
});
|
||||
|
||||
|
@ -70,6 +71,7 @@ export const ssoConnectorWithProviderConfigGuard = SsoConnectors.guard
|
|||
z.object({
|
||||
providerName: z.nativeEnum(SsoProviderName),
|
||||
providerLogo: z.string(),
|
||||
providerLogoDark: z.string(),
|
||||
providerConfig: z.record(z.unknown()).optional(),
|
||||
})
|
||||
);
|
||||
|
|
Loading…
Add table
Reference in a new issue