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

feat: extend the time we resume the session without showing OTP #7212

Merged
merged 33 commits into from
Sep 27, 2023
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
Show all changes
33 commits
Select commit Hold shift + click to select a range
c5f5b88
feat: extend the time we resume the session without showing OTP
omridan159 Sep 13, 2023
2ca8e3d
feat: cleaning
omridan159 Sep 14, 2023
72b423c
feat: cleaning
omridan159 Sep 14, 2023
58e308b
Merge remote-tracking branch 'origin/main' into feat_updating-OTP-flow
omridan159 Sep 14, 2023
c222022
feat: cleaning
omridan159 Sep 14, 2023
79c9ba3
Merge remote-tracking branch 'origin/main' into feat_updating-OTP-flow
omridan159 Sep 18, 2023
61b0c8b
feat: extend the time we resume the session without showing OTP
omridan159 Sep 13, 2023
33fb7e6
feat: cleaning
omridan159 Sep 14, 2023
510554a
feat: cleaning
omridan159 Sep 14, 2023
2baf8c9
feat: cleaning
omridan159 Sep 14, 2023
c7045ac
feat: improve sdk init process
abretonc7s Sep 18, 2023
47b476a
feat: remove webrtc references
abretonc7s Sep 18, 2023
0437636
fix: do not remember me setting
abretonc7s Sep 19, 2023
b992025
feat: always update public key from deeplink
abretonc7s Sep 20, 2023
2a5eb2e
feat: disable logs
abretonc7s Sep 20, 2023
e13eaea
feat: update comm layer to 0.7.0
abretonc7s Sep 21, 2023
4bf0e48
Merge remote-tracking branch 'origin/main' into feat_updating-OTP-flow
omridan159 Sep 22, 2023
f9657c0
Merge remote-tracking branch 'origin/feat_updating-OTP-flow' into fea…
omridan159 Sep 22, 2023
f33c229
feat: extend the time we resume the session without showing OTP
omridan159 Sep 13, 2023
0538d66
feat: cleaning
omridan159 Sep 14, 2023
774ce2c
feat: cleaning
omridan159 Sep 14, 2023
e7bd536
feat: cleaning
omridan159 Sep 14, 2023
99ee6a6
feat: improve sdk init process
abretonc7s Sep 18, 2023
bc93b37
feat: remove webrtc references
abretonc7s Sep 18, 2023
306f55d
fix: do not remember me setting
abretonc7s Sep 19, 2023
29c5ac3
feat: always update public key from deeplink
abretonc7s Sep 20, 2023
8271394
feat: disable logs
abretonc7s Sep 20, 2023
d3e36e3
feat: update comm layer to 0.7.0
abretonc7s Sep 21, 2023
14b27e9
Merge remote-tracking branch 'origin/feat_updating-OTP-flow' into fea…
omridan159 Sep 22, 2023
54f645c
chore: reset the state 'lastAuthorized' after OTP approved
omridan159 Sep 22, 2023
81abc46
Merge remote-tracking branch 'origin/main' into feat_updating-OTP-flow
omridan159 Sep 22, 2023
8a4e538
building bitrise
christopherferreira9 Sep 26, 2023
a58d6bd
Merge branch 'main' into feat_updating-OTP-flow
christopherferreira9 Sep 26, 2023
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
4 changes: 2 additions & 2 deletions android/app/build.gradle
Original file line number Diff line number Diff line change
Expand Up @@ -137,8 +137,8 @@ android {
applicationId "io.metamask"
minSdkVersion rootProject.ext.minSdkVersion
targetSdkVersion rootProject.ext.targetSdkVersion
versionCode 1171
versionName "7.7.0"
versionCode 1173
versionName "7.8.0"
testBuildType System.getProperty('testBuildType', 'debug')
missingDimensionStrategy 'react-native-camera', 'general'
testInstrumentationRunner "androidx.test.runner.AndroidJUnitRunner"
Expand Down
12 changes: 8 additions & 4 deletions app/components/Nav/App/index.js
Original file line number Diff line number Diff line change
Expand Up @@ -231,6 +231,8 @@ const App = ({ userLoggedIn }) => {
const { colors } = useTheme();
const { toastRef } = useContext(ToastContext);
const dispatch = useDispatch();
const sdkInit = useRef(false);
const [onboarded, setOnboarded] = useState(false);
const triggerSetCurrentRoute = (route) => {
dispatch(setCurrentRoute(route));
if (route === 'Wallet' || route === 'BrowserView') {
Expand Down Expand Up @@ -349,17 +351,18 @@ const App = ({ userLoggedIn }) => {
initAnalytics();
}, []);

const sdkInit = useRef(false);
useEffect(() => {
if (navigator && !sdkInit.current) {
sdkInit.current = true;
if (navigator?.getCurrentRoute && !sdkInit.current && onboarded) {
SDKConnect.getInstance()
.init({ navigation: navigator })
.then(() => {
sdkInit.current = true;
})
.catch((err) => {
console.error(`Cannot initialize SDKConnect`, err);
});
}
}, [navigator]);
}, [navigator, onboarded]);

useEffect(() => {
if (isWC2Enabled) {
Expand All @@ -372,6 +375,7 @@ const App = ({ userLoggedIn }) => {
useEffect(() => {
async function checkExisting() {
const existingUser = await AsyncStorage.getItem(EXISTING_USER);
setOnboarded(!!existingUser);
const route = !existingUser
? Routes.ONBOARDING.ROOT_NAV
: Routes.ONBOARDING.LOGIN;
Expand Down
2 changes: 2 additions & 0 deletions app/core/DeeplinkManager.js
Original file line number Diff line number Diff line change
Expand Up @@ -257,6 +257,7 @@ class DeeplinkManager {
}
SDKConnect.getInstance().reconnect({
channelId: params.channelId,
otherPublicKey: params.pubkey,
context: 'deeplink (universal)',
});
} else {
Expand Down Expand Up @@ -387,6 +388,7 @@ class DeeplinkManager {
}
SDKConnect.getInstance().reconnect({
channelId: params.channelId,
otherPublicKey: params.pubkey,
context: 'deeplink (metamask)',
});
} else {
Expand Down
139 changes: 86 additions & 53 deletions app/core/SDKConnect/SDKConnect.ts
Original file line number Diff line number Diff line change
Expand Up @@ -2,7 +2,6 @@ import { StackNavigationProp } from '@react-navigation/stack';
import BackgroundTimer from 'react-native-background-timer';
import DefaultPreference from 'react-native-default-preference';
import AppConstants from '../AppConstants';

import {
TransactionController,
WalletDevice,
Expand Down Expand Up @@ -35,16 +34,6 @@ import {
} from '@metamask/sdk-communication-layer';
import { ethErrors } from 'eth-rpc-errors';
import { EventEmitter2 } from 'eventemitter2';
import {
MediaStream,
MediaStreamTrack,
RTCIceCandidate,
RTCPeerConnection,
RTCSessionDescription,
RTCView,
mediaDevices,
registerGlobals,
} from 'react-native-webrtc';
import Routes from '../../../app/constants/navigation/Routes';
import generateOTP from './utils/generateOTP.util';
import {
Expand All @@ -53,7 +42,6 @@ import {
waitForEmptyRPCQueue,
waitForKeychainUnlocked,
} from './utils/wait.util';

import { Json } from '@metamask/controller-utils';
import { PROTOCOLS } from '../../constants/deeplinks';
import { Minimizer } from '../NativeModules';
Expand All @@ -65,17 +53,6 @@ export const HOUR_IN_MS = MIN_IN_MS * 60;
export const DAY_IN_MS = HOUR_IN_MS * 24;
export const DEFAULT_SESSION_TIMEOUT_MS = 7 * DAY_IN_MS;

const webrtc = {
RTCPeerConnection,
RTCIceCandidate,
RTCSessionDescription,
RTCView,
MediaStream,
MediaStreamTrack,
mediaDevices,
registerGlobals,
};

export interface ConnectionProps {
id: string;
otherPublicKey: string;
Expand All @@ -84,6 +61,7 @@ export interface ConnectionProps {
initialConnection?: boolean;
originatorInfo?: OriginatorInfo;
validUntil: number;
lastAuthorized: number; // timestamp of last received activity
}
export interface ConnectedSessions {
[id: string]: Connection;
Expand Down Expand Up @@ -152,6 +130,11 @@ export class Connection extends EventEmitter2 {
isResumed = false;
initialConnection: boolean;

/*
* Timestamp of last activity, used to check if channel is still active and to prevent showing OTP approval modal too often.
*/
lastAuthorized: number;

/**
* Prevent double sending 'authorized' message.
*/
Expand Down Expand Up @@ -191,6 +174,7 @@ export class Connection extends EventEmitter2 {
rpcQueueManager,
originatorInfo,
approveHost,
lastAuthorized,
getApprovedHosts,
disapprove,
revalidate,
Expand All @@ -213,6 +197,7 @@ export class Connection extends EventEmitter2 {
super();
this.origin = origin;
this.channelId = id;
this.lastAuthorized = lastAuthorized;
this.reconnect = reconnect || false;
this.isResumed = false;
this.originatorInfo = originatorInfo;
Expand All @@ -233,7 +218,6 @@ export class Connection extends EventEmitter2 {
communicationServerUrl: AppConstants.MM_SDK.SERVER_URL,
communicationLayerPreference: CommunicationLayerPreference.SOCKET,
otherPublicKey,
webRTCLib: webrtc,
reconnect,
walletInfo: {
type: 'MetaMask Mobile',
Expand Down Expand Up @@ -324,30 +308,50 @@ export class Connection extends EventEmitter2 {
!this.initialConnection &&
this.origin === AppConstants.DEEPLINKS.ORIGIN_QR_CODE
) {
if (approvalController.get(this.channelId)) {
// cleaning previous pending approval
approvalController.reject(
this.channelId,
ethErrors.provider.userRejectedRequest(),
);
}
this.approvalPromise = undefined;
const currentTime = Date.now();
const channelWasActiveRecently =
!!this.lastAuthorized &&
currentTime - this.lastAuthorized < HOUR_IN_MS;

if (!this.otps) {
this.otps = generateOTP();
}
this.sendMessage({
type: MessageType.OTP,
otpAnswer: this.otps?.[0],
}).catch((err) => {
Logger.log(err, `SDKConnect:: Connection failed to send otp`);
});
// Prevent auto approval if metamask is killed and restarted
disapprove(this.channelId);
if (channelWasActiveRecently) {
this.approvalPromise = undefined;

// Always need to re-approve connection first.
await this.checkPermissions();
this.sendAuthorized(true);
// Prevent auto approval if metamask is killed and restarted
disapprove(this.channelId);

// Always need to re-approve connection first.
await this.checkPermissions({
lastAuthorized: this.lastAuthorized,
});

this.sendAuthorized(true);
} else {
if (approvalController.get(this.channelId)) {
// cleaning previous pending approval
approvalController.reject(
this.channelId,
ethErrors.provider.userRejectedRequest(),
);
}
this.approvalPromise = undefined;

if (!this.otps) {
this.otps = generateOTP();
}
this.sendMessage({
type: MessageType.OTP,
otpAnswer: this.otps?.[0],
}).catch((err) => {
Logger.log(err, `SDKConnect:: Connection failed to send otp`);
});
// Prevent auto approval if metamask is killed and restarted
disapprove(this.channelId);

// Always need to re-approve connection first.
await this.checkPermissions();
this.sendAuthorized(true);
this.lastAuthorized = Date.now();
}
} else if (
!this.initialConnection &&
this.origin === AppConstants.DEEPLINKS.ORIGIN_DEEPLINK
Expand Down Expand Up @@ -424,7 +428,7 @@ export class Connection extends EventEmitter2 {
// Wait for bridge to be ready before handling messages.
// It will wait until user accept/reject the connection request.
try {
await this.checkPermissions(message);
await this.checkPermissions({ message });
if (!this.receivedDisconnect) {
await waitForConnectionReadiness({ connection: this });
this.sendAuthorized();
Expand Down Expand Up @@ -630,9 +634,17 @@ export class Connection extends EventEmitter2 {
* @returns {boolean} true when host is approved or user approved the request.
* @throws error if the user reject approval request.
*/
private async checkPermissions(
_message?: CommunicationLayerMessage,
): Promise<boolean> {
private async checkPermissions({
// eslint-disable-next-line
message,
lastAuthorized,
}: {
message?: CommunicationLayerMessage;
lastAuthorized?: number;
} = {}): Promise<boolean> {
const channelWasActiveRecently =
!!lastAuthorized && Date.now() - lastAuthorized < HOUR_IN_MS;

// only ask approval if needed
const approved = this.isApproved({
channelId: this.channelId,
Expand Down Expand Up @@ -663,6 +675,10 @@ export class Connection extends EventEmitter2 {
this.revalidate({ channelId: this.channelId });
}

if (channelWasActiveRecently) {
return true;
}

const approvalRequest = {
origin: this.origin,
type: ApprovalTypes.CONNECT_ACCOUNTS,
Expand Down Expand Up @@ -812,6 +828,8 @@ export class SDKConnect extends EventEmitter2 {
await this.reconnect({
channelId: id,
initialConnection: false,
otherPublicKey:
this.connected[id].remote.getKeyInfo()?.ecies.otherPubKey ?? '',
context: 'connectToChannel',
});
return;
Expand All @@ -825,6 +843,7 @@ export class SDKConnect extends EventEmitter2 {
otherPublicKey,
origin,
validUntil: Date.now() + DEFAULT_SESSION_TIMEOUT_MS,
lastAuthorized: Date.now(),
};

const initialConnection = this.approvedHosts[id] === undefined;
Expand Down Expand Up @@ -876,7 +895,8 @@ export class SDKConnect extends EventEmitter2 {
connection.remote.on(EventType.CLIENTS_DISCONNECTED, () => {
const host = AppConstants.MM_SDK.SDK_REMOTE_ORIGIN + connection.channelId;
// Prevent disabled connection ( if user chose do not remember session )
if (this.disabledHosts[host] !== undefined) {
const isDisabled = this.disabledHosts[host]; // should be 0 when disabled.
if (isDisabled !== undefined) {
this.updateSDKLoadingState({
channelId: connection.channelId,
loading: false,
Expand All @@ -886,6 +906,8 @@ export class SDKConnect extends EventEmitter2 {
`SDKConnect::watchConnection can't update SDK loading state`,
);
});
// Force terminate connection since it was disabled (do not remember)
this.removeChannel(connection.channelId, true);
}
});

Expand Down Expand Up @@ -971,10 +993,12 @@ export class SDKConnect extends EventEmitter2 {

async reconnect({
channelId,
otherPublicKey,
initialConnection,
context,
}: {
channelId: string;
otherPublicKey: string;
context?: string;
initialConnection: boolean;
}) {
Expand All @@ -988,6 +1012,7 @@ export class SDKConnect extends EventEmitter2 {
} connecting=${connecting} socketConnected=${socketConnected} existingConnection=${
existingConnection !== undefined
}`,
otherPublicKey,
);

let interruptReason = '';
Expand Down Expand Up @@ -1026,6 +1051,7 @@ export class SDKConnect extends EventEmitter2 {
this.connecting[channelId] = true;
this.connected[channelId] = new Connection({
...connection,
otherPublicKey,
reconnect: true,
initialConnection,
rpcQueueManager: this.rpcqueueManager,
Expand All @@ -1044,7 +1070,9 @@ export class SDKConnect extends EventEmitter2 {
withKeyExchange: true,
});
this.watchConnection(this.connected[channelId]);
this.connecting[channelId] = false;
const afterConnected =
this.connected[channelId].remote.isConnected() ?? false;
this.connecting[channelId] = !afterConnected; // If not connected, it means it's connecting.
this.emit('refresh');
}

Expand All @@ -1058,6 +1086,7 @@ export class SDKConnect extends EventEmitter2 {
if (channelId) {
this.reconnect({
channelId,
otherPublicKey: this.connections[channelId].otherPublicKey,
initialConnection: false,
context: 'reconnectAll',
}).catch((err) => {
Expand Down Expand Up @@ -1142,7 +1171,7 @@ export class SDKConnect extends EventEmitter2 {

/**
* Invalidate a channel/session by preventing future connection to be established.
* Instead of removing the channel, it creates sets the session to timeout on next
* Instead of removing the channel, it sets the session to timeout on next
* connection which will remove it while conitnuing current session.
*
* @param channelId
Expand Down Expand Up @@ -1251,11 +1280,15 @@ export class SDKConnect extends EventEmitter2 {
}

private _approveHost({ host }: approveHostProps) {
const channelId = host.replace(AppConstants.MM_SDK.SDK_REMOTE_ORIGIN, '');
if (this.disabledHosts[host]) {
// Might be useful for future feature.
} else {
// Host is approved for 24h.
this.approvedHosts[host] = Date.now() + DAY_IN_MS;
if (this.connections[channelId]) {
this.connections[channelId].lastAuthorized = Date.now();
}
// Prevent disabled hosts from being persisted.
DefaultPreference.set(
AppConstants.MM_SDK.SDK_APPROVEDHOSTS,
Expand Down
Loading