forked from Ivasoft/mattermost-mobile
* MM_35115: ADDED select_server screen
* MM_35115: ADDED select_server screen
* MM_35115: ADDED files on which select_server is dependent
* MM_35115: ADDED react-native-button
* MM_35115: Fixing TS issues [IN PROGRESS]
* MM_35115: Started withObservables [IN PROGRESS]
* MM_35115: Started withObservables [IN PROGRESS]
* MM_35115: withObservables - defaulting when no connection is available [IN PROGRESS]
* MM_35115: withObservables - some code clean up [IN PROGRESS]
* MM_35115: withObservables - some code clean up [IN PROGRESS]
* MM_35115: withObservables - some code clean up [IN PROGRESS]
* MM_35115: Substituting mapDispatchToProps [IN PROGRESS]
* MM_35115: Substituting mapDispatchToProps [IN PROGRESS]
* MM_35115: Substituting mapDispatchToProps [IN PROGRESS]
* MM_35115: Removed resetPing action [IN PROGRESS]
* MM_35115: ADDED app/client
* MM_35115: Preparing scheduleExpiredNotification
* MM_35115: Adding some todos
* Server & LoginOptions
* Use default server if available and autoconnect if configured
* Fix login header & manual server url
* MM_35115: Login Options[IN PROGRESS]
* MM_35115: Login screen - email [IN PROGRESS]
* MM_35115: Login screen - email [IN PROGRESS]
* MM_35115: Login screen - email - login api call [IN PROGRESS]
* MM_35115: Login screen - email - login api call [IN PROGRESS]
* MM_35115: Login screen - email - saving to server db [IN PROGRESS]
* MM_35115: Login screen - email - saving to System, Preferences to db [IN PROGRESS]
* MM_35115: Login screen - enforcing unique check on System entity [IN PROGRESS]
* MM_35115: Login screen - writing TeamMembership [IN PROGRESS]
* MM_35115: Login screen - writing Teams [IN PROGRESS]
* MM_35115: Login screen [IN PROGRESS]
* MM_35115: Login screen- Refactored DataOperator handlers [IN PROGRESS]
* MM_35115: Login screen - Proper clean up [IN PROGRESS]
* MM_35115: Login screen - completeLogin [IN PROGRESS]
* MM_35115: Improving DataOperator
* MM_35115: Improving DataOperator
* MM_35115: 80% DONE - login with email and password - some todos
* MM_35115: 80% DONE - login with email and password - some todos
* MM_35115: 80% DONE - login with email and password - some todos
* MM_35115: Removing unused app/queries folder
* MM_35115: Clean up
* MM_35115: Clean up
* MM_35115: Clean up
* MM_35115: Clean up
* MM_35115: Clean up
* MM_35115: Adding roles for MYTEAM
* MM_35115: Code clean up
* MM_35115: Code clean up
* MM_35115: Code clean up
* MM_35115: Adding rn-fetch-blob for Android
* MM_35115: Code clean up
* MM_35115: Code clean up
* MM_35115: Added test setup
* MM_35115: Fix database utils
* MM_35115: ADDED loadRolesIfNeeded
* MM_35115: Fix TS issue
* MM_35115: ADDED Tests setup
* MM_35115: Fix TS issues
* MM_35115: Fix TS issues
* MM_35115: Fix TS issues
* MM_35115: Added alternative to site name
* MM_35115: Added alternative to site name
* MM_35115: Removed hardcoded values
* MM_35115: Clean up
* MM_35115 - Fixed Android platform check instead of hermes
* MM_35115 - Replaced emptyErrorHandlingFunction with emptyFunction
* MM_35115 : Implemented TS fixes
* Update index.ts
* MM-35115 - Fix react-test-renderer issue
* MM_35115 - Optimizing DatabaseManager
* MM_35115 : Implemented getDatabaseConnection
* MM_35115 : Refactoring set/getActiveDatabase to use flag record
* MM_35115 : Refactored active database to use flag in Global entity
* MM_35115 : Updated manual database manager test
* MM_35115 : Fix operator/utils/test
* MM_35115 : Fix for base_handler
* MM_35115 : Fix test issues with Handlers
* MM_35115 : Fix test issues with prepareRecords
* MM_35115 : Fix wrapper test issue
* MM_35115 : Updated getMostRecentServerConnection to return the serverUrl as well as the connection
* MM_35115 : Refactored the way we call DataOperator
* MM_35115 : Updated database manager mock
* Add getMostRecentServerUrl function (#5440)
* fix: add getMostRecentServerUrl func
* fix: add ts and tsx to editorconfig
* fix: rename functions
* fix: return type
* Fix unit test setup
* fix login screen unit tests
* MM-36205 [GEKIDOU] Login Flow SSO (#5454)
* MM_35115: Starting LoginOptions SSO
* MM_36205: SSO [IN PROGRESS]
* MM_36205 : SSO [ IN PROGRESS ]
* Update sso_with_redirect_url.tsx
* MM_36205 : SSO Tests [ IN PROGRESS ]
* MM_36205 : Passing serverUrl to SSO screen
* Update sso.test.tsx
* Fix ViewTypes imports and keyMirror method
* MM_36205 : Code clean up
* Fix : Clean up imports
* Update: Aligning with PR 5452
* Fix: AndroidManifest file to include redirection ofr scheme mmauthbeta
* refactor: SSO Login method via Gitlab now navigates to Channel screen
* refactor: SSO Login without redirectURL is also working
* feat: SSO - main test completed
* feat: ADDED test for sso_with_redirect_url
* fix : eslint correction
* fix: Updated Loading component name
* fix : code clean up from reviews
* fix: reviews check
* fix: Added mmauthbeta into info.plist
* Revert "fix: Added mmauthbeta into info.plist"
This reverts commit d87cc23f5b.
* Update Info.plist
* Update AppDelegate.m
* feat: ADDED Forgot Password - Test [ IN PROGRESS ]
* feat: Forgot Password - Completed & Tested
* fix: Including MFA screen [ IN PROGRESS ]
* MFA - Properly tested
* Properly testing forgot_password screen
* Fix login.test.tsx
* Fix SSO method calls chain
* Update index.tsx
* Sort imports for sceen/navigation
* fix: Reviews
* Update signing + act in test
* Removed todo comment on MFA
* feedback review
* fix login tests
Co-authored-by: Avinash Lingaloo <>
Co-authored-by: Elias Nahum <nahumhbl@gmail.com>
* App initialization refactor (#5430)
* fix: initial init refactor
* fix: await isServerPresent
* fix: more refactor
* fix: move out launch functions
* fix: remove comment
* fix: update credential functions
* fix: refactor launch functions
* fix: deep link parsing
* fix: lint change
* fix: update deeplink and notification handlers
* fix: indentation
* fix: add relaunchApp
* fix eslint
* refactor launchProps and autoconnect server for deeplink
* fix: use undefined
* fix: define OptionalLaunchProps
* fix: Android - handle server URL in push notification
* fix: rename func
* fix: use boolean launchError instead
* fix: use DatabaseModule
* fix: use DatabaseHelper instead
* fix: remove unnecessary null check
* fix: iOS - support for serverUrl
* fix: iOS - extract serverUrl in reply action
* fix: iOS - expose objc specific func
* fix: remove unnecessary deviceToken param
* fix: return if device is untrusted
Co-authored-by: Elias Nahum <nahumhbl@gmail.com>
* fix: bye bye modulePaths
* fix: ios build
* chore: remove unused aliases from babel.config
* chore: fix dependency format in package-lock.json
* chore: remove transparent window background color for android AppTheme
* chore: remove mattermost.js and use index.ts as app entry
* fix: login flow screens theme
* fix: Launch types
* chore: remove OptionalLaunchProps type
* fix: url utils unit tests
* chore: update en.json
Co-authored-by: Elias Nahum <nahumhbl@gmail.com>
Co-authored-by: Avinash Lingaloo <>
Co-authored-by: Elias Nahum <nahumhbl@gmail.com>
Co-authored-by: Miguel Alatzar <migbot@users.noreply.github.com>
191 lines
6.1 KiB
TypeScript
191 lines
6.1 KiB
TypeScript
// Copyright (c) 2015-present Mattermost, Inc. All Rights Reserved.
|
|
// See LICENSE.txt for license information.
|
|
|
|
import Emm, {ManagedConfig} from '@mattermost/react-native-emm';
|
|
import {Alert, AlertButton, AppState, AppStateStatus, Platform} from 'react-native';
|
|
import JailMonkey from 'jail-monkey';
|
|
|
|
import {DEFAULT_LOCALE, getTranslations, t} from '@i18n';
|
|
import {getIOSAppGroupDetails} from '@utils/mattermost_managed';
|
|
|
|
const PROMPT_IN_APP_PIN_CODE_AFTER = 5 * 1000;
|
|
|
|
class ManagedApp {
|
|
backgroundSince = 0;
|
|
enabled = false;
|
|
inAppPinCode = false;
|
|
performingAuthentication = false;
|
|
previousAppState?: AppStateStatus;
|
|
processConfigTimeout?: NodeJS.Timeout;
|
|
vendor = 'Mattermost';
|
|
|
|
constructor() {
|
|
Emm.addListener(this.processConfig);
|
|
|
|
this.setIOSAppGroupIdentifier();
|
|
|
|
AppState.addEventListener('change', this.onAppStateChange);
|
|
}
|
|
|
|
init() {
|
|
Emm.getManagedConfig().then(this.processConfig);
|
|
}
|
|
|
|
setIOSAppGroupIdentifier = () => {
|
|
if (Platform.OS === 'ios') {
|
|
const {appGroupIdentifier} = getIOSAppGroupDetails();
|
|
|
|
if (appGroupIdentifier) {
|
|
Emm.setAppGroupId(appGroupIdentifier);
|
|
}
|
|
}
|
|
}
|
|
|
|
processConfig = async (config?: ManagedConfig) => {
|
|
// If the managed configuration changed while authentication was
|
|
// being performed, delay the processing of this new configuration
|
|
// until authentication is complete.
|
|
if (this.performingAuthentication) {
|
|
if (this.processConfigTimeout) {
|
|
clearTimeout(this.processConfigTimeout);
|
|
}
|
|
|
|
this.processConfigTimeout = setTimeout(() => this.processConfig(config), 500);
|
|
}
|
|
|
|
this.enabled = Boolean(config && Object.keys(config).length);
|
|
if (!this.enabled) {
|
|
return;
|
|
}
|
|
|
|
const blurScreen = config!.blurApplicationScreen === 'true';
|
|
Emm.enableBlurScreen(blurScreen);
|
|
|
|
const vendor = config!.vendor;
|
|
if (vendor) {
|
|
this.vendor = vendor;
|
|
}
|
|
|
|
const jailbreakProtection = config!.jailbreakProtection === 'true';
|
|
if (jailbreakProtection && !this.isTrustedDevice()) {
|
|
this.alertDeviceIsUntrusted();
|
|
return;
|
|
}
|
|
|
|
const inAppPinCode = config!.inAppPinCode === 'true';
|
|
if (inAppPinCode) {
|
|
this.handleDeviceAuthentication();
|
|
}
|
|
};
|
|
|
|
alertDeviceIsUntrusted = () => {
|
|
const locale = DEFAULT_LOCALE; // TODO: Get current user or system locale
|
|
const translations = getTranslations(locale);
|
|
Alert.alert(
|
|
translations[t('mobile.managed.blocked_by')].replace('{vendor}', this.vendor),
|
|
translations[t('mobile.managed.jailbreak')].replace('{vendor}', this.vendor),
|
|
[{
|
|
text: translations[t('mobile.managed.exit')],
|
|
style: 'destructive',
|
|
onPress: () => {
|
|
Emm.exitApp();
|
|
},
|
|
}],
|
|
{cancelable: false},
|
|
);
|
|
};
|
|
|
|
handleDeviceAuthentication = async (authExpired = true) => {
|
|
this.performingAuthentication = true;
|
|
const isSecured = await Emm.isDeviceSecured();
|
|
const locale = DEFAULT_LOCALE;
|
|
const translations = getTranslations(locale);
|
|
|
|
if (!isSecured) {
|
|
await this.showNotSecuredAlert(translations);
|
|
Emm.exitApp();
|
|
return;
|
|
}
|
|
|
|
if (authExpired) {
|
|
try {
|
|
await Emm.authenticate({
|
|
reason: translations[t('mobile.managed.secured_by')].replace('{vendor}', this.vendor),
|
|
fallback: true,
|
|
supressEnterPassword: true,
|
|
});
|
|
} catch (err) {
|
|
Emm.exitApp();
|
|
return;
|
|
}
|
|
}
|
|
|
|
this.performingAuthentication = false;
|
|
};
|
|
|
|
isTrustedDevice = () => {
|
|
return __DEV__ || JailMonkey.trustFall();
|
|
};
|
|
|
|
onAppStateChange = async (appState: AppStateStatus) => {
|
|
const isActive = appState === 'active';
|
|
const isBackground = appState === 'background';
|
|
|
|
if (isActive && this.previousAppState === 'background') {
|
|
if (this.enabled && this.inAppPinCode) {
|
|
const authExpired = this.backgroundSince > 0 && (Date.now() - this.backgroundSince) >= PROMPT_IN_APP_PIN_CODE_AFTER;
|
|
await this.handleDeviceAuthentication(authExpired);
|
|
}
|
|
|
|
this.backgroundSince = 0;
|
|
} else if (isBackground) {
|
|
this.backgroundSince = Date.now();
|
|
}
|
|
|
|
this.previousAppState = appState;
|
|
};
|
|
|
|
showNotSecuredAlert = (translations: Record<string, string>) => {
|
|
return new Promise(async (resolve) => { /* eslint-disable-line no-async-promise-executor */
|
|
const buttons: AlertButton[] = [];
|
|
|
|
if (Platform.OS === 'android') {
|
|
buttons.push({
|
|
text: translations[t('mobile.managed.settings')],
|
|
onPress: () => {
|
|
Emm.openSecuritySettings();
|
|
},
|
|
});
|
|
}
|
|
|
|
buttons.push({
|
|
text: translations[t('mobile.managed.exit')],
|
|
onPress: resolve,
|
|
style: 'cancel',
|
|
});
|
|
|
|
let message;
|
|
if (Platform.OS === 'ios') {
|
|
const {face} = await Emm.deviceSecureWith();
|
|
|
|
if (face) {
|
|
message = translations[t('mobile.managed.not_secured.ios')];
|
|
} else {
|
|
message = translations[t('mobile.managed.not_secured.ios.touchId')];
|
|
}
|
|
} else {
|
|
message = translations[t('mobile.managed.not_secured.android')];
|
|
}
|
|
|
|
Alert.alert(
|
|
translations[t('mobile.managed.blocked_by')].replace('{vendor}', this.vendor),
|
|
message,
|
|
buttons,
|
|
{cancelable: false, onDismiss: () => resolve},
|
|
);
|
|
});
|
|
};
|
|
}
|
|
|
|
export default new ManagedApp();
|