Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Onboard OAuth support for Email Provider #7917

Draft
wants to merge 1 commit into
base: master
Choose a base branch
from
Draft
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
3 changes: 2 additions & 1 deletion features/admin.core.v1/configs/app.ts
Original file line number Diff line number Diff line change
Expand Up @@ -228,7 +228,8 @@ export class Config {
I18nConstants.POLICY_ADMINISTRATION_NAMESPACE,
I18nConstants.REMOTE_USER_STORES_NAMESPACE,
I18nConstants.RULES_NAMESPACE,
I18nConstants.PUSH_PROVIDERS_NAMESPACE
I18nConstants.PUSH_PROVIDERS_NAMESPACE,
I18nConstants.EMAIL_PROVIDERS_NAMESPACE
],
preload: []
};
Expand Down
8 changes: 7 additions & 1 deletion features/admin.core.v1/constants/i18n-constants.ts
Original file line number Diff line number Diff line change
Expand Up @@ -197,6 +197,11 @@ export class I18nConstants {
*/
public static readonly PUSH_PROVIDERS_NAMESPACE: string = I18nModuleConstants.PUSH_PROVIDERS_NAMESPACE;

/**
* Push Providers namespace.
*/
public static readonly EMAIL_PROVIDERS_NAMESPACE: string = I18nModuleConstants.EMAIL_PROVIDERS_NAMESPACE;

/**
* SMS Templates namespace.
*/
Expand Down Expand Up @@ -357,7 +362,8 @@ export class I18nConstants {
[ I18nConstants.POLICY_ADMINISTRATION_NAMESPACE, "portals" ],
[ I18nConstants.REMOTE_USER_STORES_NAMESPACE, "portals" ],
[ I18nConstants.RULES_NAMESPACE, "portals" ],
[ I18nConstants.PUSH_PROVIDERS_NAMESPACE, "portals" ]
[ I18nConstants.PUSH_PROVIDERS_NAMESPACE, "portals" ],
[ I18nConstants.EMAIL_PROVIDERS_NAMESPACE, "portals" ]
]);

/**
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -18,6 +18,23 @@

import { IdentityAppsError } from "@wso2is/core/errors";

/**
* Interface for the authentication type dropdown options.
*/
export interface AuthenticationTypeDropdownOption {
key: AuthenticationType;
text: string;
value: AuthenticationType;
}

/**
* Enum for the authentication types.
*/
export enum AuthenticationType {
BASIC = "BASIC",
CLIENT_CREDENTIAL = "CLIENT CREDENTIAL"
}

export class EmailProviderConstants {

private constructor() { }
Expand All @@ -26,10 +43,10 @@
public static readonly REPLY_TO_ADDRESS_KEY: string = "mail.smtp.replyTo";
public static readonly SIGNATURE_KEY: string = "mail.smtp.signature";

public static readonly EMAIL_PROVIDER_CONFIG_FETCH_ERROR_CODE: string = "ASG-EPC-00001";

Check warning on line 46 in features/admin.email-providers.v1/constants/email-provider-constants.ts

View workflow job for this annotation

GitHub Actions / ⬣ ESLint (STATIC ANALYSIS) (lts/*, 8.7.4)

Expected indentation of 4 spaces but found 1 tab
public static readonly EMAIL_PROVIDER_CONFIG_FETCH_INVALID_STATUS_CODE_ERROR_CODE: string = "ASG-EPC-00002";
public static readonly EMAIL_PROVIDER_CONFIG_UPDATE_ERROR_CODE: string = "ASG-EP-00003";

Check warning on line 48 in features/admin.email-providers.v1/constants/email-provider-constants.ts

View workflow job for this annotation

GitHub Actions / ⬣ ESLint (STATIC ANALYSIS) (lts/*, 8.7.4)

Expected indentation of 4 spaces but found 1 tab
public static readonly EMAIL_PROVIDER_CONFIG_DELETE_ERROR_CODE: string = "ASG-EP-60004";

Check warning on line 49 in features/admin.email-providers.v1/constants/email-provider-constants.ts

View workflow job for this annotation

GitHub Actions / ⬣ ESLint (STATIC ANALYSIS) (lts/*, 8.7.4)

Expected indentation of 4 spaces but found 1 tab
public static readonly EMAIL_PROVIDER_CONFIG_NOT_FOUND_ERROR_CODE: string = "NSM-60006";

public static readonly EMAIL_PROVIDER_CONFIG_FIELD_MIN_LENGTH: number = 0;
Expand All @@ -37,18 +54,18 @@
public static readonly EMAIL_PROVIDER_SERVER_PORT_MAX_LENGTH: number = 6;
public static readonly EMAIL_REGEX: RegExp = /^[^\s@]+@[^\s@]+\.[^\s@]+$/;

public static ErrorMessages: {

Check warning on line 57 in features/admin.email-providers.v1/constants/email-provider-constants.ts

View workflow job for this annotation

GitHub Actions / ⬣ ESLint (STATIC ANALYSIS) (lts/*, 8.7.4)

Expected indentation of 4 spaces but found 1 tab
EMAIL_PROVIDER_CONFIG_FETCH_ERROR_CODE: IdentityAppsError;
EMAIL_PROVIDER_CONFIG_FETCH_INVALID_STATUS_CODE_ERROR_CODE: IdentityAppsError;
EMAIL_PROVIDER_CONFIG_UPDATE_ERROR_CODE: IdentityAppsError;
EMAIL_PROVIDER_CONFIG_DELETE_ERROR_CODE: IdentityAppsError;
} = {
EMAIL_PROVIDER_CONFIG_DELETE_ERROR_CODE: new IdentityAppsError(

Check warning on line 63 in features/admin.email-providers.v1/constants/email-provider-constants.ts

View workflow job for this annotation

GitHub Actions / ⬣ ESLint (STATIC ANALYSIS) (lts/*, 8.7.4)

Expected indentation of 12 spaces but found 8
EmailProviderConstants.EMAIL_PROVIDER_CONFIG_DELETE_ERROR_CODE,

Check warning on line 64 in features/admin.email-providers.v1/constants/email-provider-constants.ts

View workflow job for this annotation

GitHub Actions / ⬣ ESLint (STATIC ANALYSIS) (lts/*, 8.7.4)

Expected indentation of 16 spaces but found 12
"An error occurred while deleting the email provider configurations.",

Check warning on line 65 in features/admin.email-providers.v1/constants/email-provider-constants.ts

View workflow job for this annotation

GitHub Actions / ⬣ ESLint (STATIC ANALYSIS) (lts/*, 8.7.4)

Expected indentation of 16 spaces but found 12
"Error while deleting email provider configurations",

Check warning on line 66 in features/admin.email-providers.v1/constants/email-provider-constants.ts

View workflow job for this annotation

GitHub Actions / ⬣ ESLint (STATIC ANALYSIS) (lts/*, 8.7.4)

Expected indentation of 16 spaces but found 12
null

Check warning on line 67 in features/admin.email-providers.v1/constants/email-provider-constants.ts

View workflow job for this annotation

GitHub Actions / ⬣ ESLint (STATIC ANALYSIS) (lts/*, 8.7.4)

Expected indentation of 16 spaces but found 12
),

Check warning on line 68 in features/admin.email-providers.v1/constants/email-provider-constants.ts

View workflow job for this annotation

GitHub Actions / ⬣ ESLint (STATIC ANALYSIS) (lts/*, 8.7.4)

Expected indentation of 12 spaces but found 8
EMAIL_PROVIDER_CONFIG_FETCH_ERROR_CODE: new IdentityAppsError(
EmailProviderConstants.EMAIL_PROVIDER_CONFIG_FETCH_ERROR_CODE,
"An error occurred while fetching the email provider configurations.",
Expand All @@ -68,4 +85,17 @@
null
)
};

public static readonly AUTH_TYPES: AuthenticationTypeDropdownOption[] = [
{
key: AuthenticationType.BASIC,
text: "emailProviders:fields.authentication.types.basic.name",
value: AuthenticationType.BASIC
},
{
key: AuthenticationType.CLIENT_CREDENTIAL,
text: "emailProviders:fields.authentication.types.client_credential.name",
value: AuthenticationType.CLIENT_CREDENTIAL
},
];
}
194 changes: 190 additions & 4 deletions features/admin.email-providers.v1/pages/email-providers.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -17,6 +17,8 @@
*/

import { useRequiredScopes } from "@wso2is/access-control";
import Box from "@oxygen-ui/react/Box";
import InputAdornment from "@oxygen-ui/react/InputAdornment";
import { AppConstants } from "@wso2is/admin.core.v1/constants/app-constants";
import { history } from "@wso2is/admin.core.v1/helpers/history";
import { FeatureConfigInterface } from "@wso2is/admin.core.v1/models/config";
Expand All @@ -32,6 +34,7 @@ import {
DangerZoneGroup,
DocumentationLink,
EmphasizedSegment,
Heading,
PageLayout,
PrimaryButton,
SecondaryButton,
Expand All @@ -49,9 +52,13 @@ import React, {
import { Trans, useTranslation } from "react-i18next";
import { useDispatch, useSelector } from "react-redux";
import { Dispatch } from "redux";
import { Divider, Grid, Placeholder, Ref } from "semantic-ui-react";
import { Divider, DropdownProps, Grid, Placeholder, Ref, Icon } from "semantic-ui-react";
import { deleteEmailProviderConfigurations, updateEmailProviderConfigurations, useEmailProviderConfig } from "../api";
import { EmailProviderConstants } from "../constants";
import {
AuthenticationTypeDropdownOption,
AuthenticationType,
EmailProviderConstants
} from "../constants";
import {
EmailProviderConfigAPIResponseInterface,
EmailProviderConfigFormErrorValidationsInterface,
Expand Down Expand Up @@ -101,6 +108,10 @@ const EmailProvidersPage: FunctionComponent<EmailProvidersPageInterface> = (
error: emailProviderConfigFetchRequestError
} = useEmailProviderConfig();

const [ endpointAuthType, setEndpointAuthType ] = useState<AuthenticationType>(null);
const [ showPrimarySecret, setShowPrimarySecret ] = useState<boolean>(false);
const [ showSecondarySecret, setShowSecondarySecret ] = useState<boolean>(false);

useEffect(() => {
if (originalEmailProviderConfig instanceof IdentityAppsApiException || emailProviderConfigFetchRequestError) {
handleRetrieveError();
Expand Down Expand Up @@ -138,6 +149,148 @@ const EmailProvidersPage: FunctionComponent<EmailProvidersPageInterface> = (
}
}, [ originalEmailProviderConfig ]);


const renderInputAdornmentOfSecret = (showSecret: boolean, onClick: () => void): ReactElement => (
<InputAdornment position="end">
<Icon
link={ true }
className="list-icon reset-field-to-default-adornment"
size="small"
color="grey"
name={ !showSecret ? "eye" : "eye slash" }
data-componentid={ `${componentId}-endpoint-authentication-property-secret-view-button` }
onClick={ onClick }
/>
</InputAdornment>
);

/**
* This method renders property fields of each endpoint authentication type.
*
* @returns property fields of the selected authentication type.
*/
const renderEndpointAuthPropertyFields = (): ReactElement => {
switch (endpointAuthType) {
case AuthenticationType.BASIC:
return (
<>
<Field.Input
ariaLabel="username"
className="addon-field-wrapper"
name="usernameAuthProperty"
label={ t(
"customAuthenticator:fields.createWizard.configurationsStep." +
"authenticationTypeDropdown.authProperties.username.label"
) }
placeholder={ t(
"customAuthenticator:fields.createWizard.configurationsStep." +
"authenticationTypeDropdown.authProperties.username.placeholder"
) }
inputType="password"
type={ showPrimarySecret ? "text" : "password" }
InputProps={ {
endAdornment: renderInputAdornmentOfSecret(showPrimarySecret, () =>
setShowPrimarySecret(!showPrimarySecret)
)
} }
required={ true }
maxLength={ 100 }
minLength={ 0 }
data-componentid={ `${componentId}-endpoint-authentication-property-username` }
width={ 15 }
/>
<Field.Input
ariaLabel="password"
className="addon-field-wrapper"
label={ t(
"customAuthenticator:fields.createWizard.configurationsStep." +
"authenticationTypeDropdown.authProperties.password.label"
) }
placeholder={ t(
"customAuthenticator:fields.createWizard.configurationsStep." +
"authenticationTypeDropdown.authProperties.password.placeholder"
) }
name="passwordAuthProperty"
inputType="password"
type={ showSecondarySecret ? "text" : "password" }
InputProps={ {
endAdornment: renderInputAdornmentOfSecret(showSecondarySecret, () =>
setShowSecondarySecret(!showSecondarySecret)
)
} }
required={ true }
maxLength={ 100 }
minLength={ 0 }
data-componentid={ `${componentId}-endpoint-authentication-property-password` }
width={ 15 }
/>
</>
);
case AuthenticationType.CLIENT_CREDENTIAL:
return (
<>
<Field.Input
ariaLabel="clientID"
className="addon-field-wrapper"
name="clientIDAuthProperty"
inputType="text"
type={ "text" }
label={ t(
"customAuthenticator:fields.createWizard.configurationsStep." +
"authenticationTypeDropdown.authProperties.header.label"
) }
placeholder={ t(
"customAuthenticator:fields.createWizard.configurationsStep." +
"authenticationTypeDropdown.authProperties.header.placeholder"
) }
required={ true }
maxLength={ 100 }
minLength={ 0 }
data-componentid={ `${componentId}-endpoint-authentication-property-header` }
width={ 15 }
/>
<Field.Input
ariaLabel="value"
className="addon-field-wrapper"
name="valueAuthProperty"
inputType="password"
type={ showSecondarySecret ? "text" : "password" }
InputProps={ {
endAdornment: renderInputAdornmentOfSecret(showSecondarySecret, () =>
setShowSecondarySecret(!showSecondarySecret)
)
} }
label={ t(
"customAuthenticator:fields.createWizard.configurationsStep." +
"authenticationTypeDropdown.authProperties.value.label"
) }
placeholder={ t(
"customAuthenticator:fields.createWizard.configurationsStep." +
"authenticationTypeDropdown.authProperties.value.placeholder"
) }
required={ true }
maxLength={ 100 }
minLength={ 0 }
data-componentid={ `${componentId}-endpoint-authentication-property-value` }
width={ 15 }
/>
</>
);
default:
break;
}
};

/**
* This method handles authentication type dropdown changes.
*
* @param event - event associated with the dropdown change.
* @param data - data changed by the event
*/
const handleDropdownChange = (event: React.MouseEvent<HTMLAnchorElement>, data: DropdownProps) => {
setEndpointAuthType(data.value as AuthenticationType);
};

/**
* Displays the error banner when unable to fetch email provider configuration.
*/
Expand Down Expand Up @@ -635,7 +788,7 @@ const EmailProvidersPage: FunctionComponent<EmailProvidersPageInterface> = (
/>
</Grid.Column>
</Grid.Row>
<Grid.Row columns={ 2 } key={ 3 }>
{/* <Grid.Row columns={ 2 } key={ 3 }>
<Grid.Column key="userName">
<Field.Input
ariaLabel="Username Field"
Expand Down Expand Up @@ -687,7 +840,7 @@ const EmailProvidersPage: FunctionComponent<EmailProvidersPageInterface> = (
autoComplete="new-password"
/>
</Grid.Column>
</Grid.Row>
</Grid.Row> */}
<Grid.Row columns={ 2 } key={ 3 }>
<Grid.Column key="displayName">
<Field.Input
Expand Down Expand Up @@ -716,6 +869,39 @@ const EmailProvidersPage: FunctionComponent<EmailProvidersPageInterface> = (
</Grid.Column>
</Grid.Row>
</Grid>

<Divider className="divider-container" />
<Heading className="heading-container" as="h5">
{ t("emailProviders:fields.authenticationTypeDropdown.title") }
</Heading>
<Box className="box-container">
<Field.Dropdown
ariaLabel="authenticationType"
name="authenticationType"
label={ t(
"emailProviders:fields.authenticationTypeDropdown.label"
) }
placeholder={ t(
"emailProviders:fields.authenticationTypeDropdown.placeholder"
) }
hint={ t(
"emailProviders:fields.authenticationTypeDropdown.hint"
) }
required={ true }
value={ endpointAuthType }
options={ [
...EmailProviderConstants.AUTH_TYPES.map((option: AuthenticationTypeDropdownOption) => ({
text: t(option.text),
value: option.value.toString()
}))
] }
onChange={ handleDropdownChange }
enableReinitialize={ true }
data-componentid={ `${componentId}-create-wizard-endpoint-authentication-dropdown` }
width={ 15 }
/>
<div className="box-field">{ renderEndpointAuthPropertyFields() }</div>
</Box>
</Form>
{
hasEmailProviderUpdatePermissions && (
Expand Down
5 changes: 5 additions & 0 deletions modules/i18n/src/constants.ts
Original file line number Diff line number Diff line change
Expand Up @@ -369,4 +369,9 @@ export class I18nModuleConstants {
* This key is used to store the user's language preference in cookies or local storage.
*/
public static readonly PREFERENCE_STORAGE_KEY: string = "ui_lang";

/**
* Email Provider namespace.
*/
public static readonly EMAIL_PROVIDERS_NAMESPACE: string = "emailProviders";
}
Loading
Loading