From 45450f18c4b7df04b60d7000809da6e183d1c747 Mon Sep 17 00:00:00 2001
From: IZUMI-Zu <274620705z@gmail.com>
Date: Tue, 24 Sep 2024 19:57:54 +0800
Subject: [PATCH] feat: improve login logic
---
App.js | 15 ++-
CasdoorLoginPage.js | 180 ++++++++++++++++-----------
EnterCasdoorSdkConfig.js | 258 ++++++++++++---------------------------
Header.js | 25 +++-
LoginMethodSelector.js | 62 ++++++++++
SettingPage.js | 110 ++++++++++-------
package-lock.json | 109 ++++++++++-------
package.json | 11 +-
syncLogic.js | 74 ++++++-----
9 files changed, 466 insertions(+), 378 deletions(-)
create mode 100644 LoginMethodSelector.js
diff --git a/App.js b/App.js
index d8cb384..6629a30 100644
--- a/App.js
+++ b/App.js
@@ -20,6 +20,7 @@ import ContentLoader, {Circle, Rect} from "react-content-loader/native";
import {ZoomInDownZoomOutUp, createNotifications} from "react-native-notificated";
import {GestureHandlerRootView} from "react-native-gesture-handler";
import {useMigrations} from "drizzle-orm/expo-sqlite/migrator";
+import {ActionSheetProvider} from "@expo/react-native-action-sheet";
import Header from "./Header";
import NavigationBar from "./NavigationBar";
@@ -77,12 +78,14 @@ const App = () => {
return (
-
-
-
-
-
-
+
+
+
+
+
+
+
+
);
diff --git a/CasdoorLoginPage.js b/CasdoorLoginPage.js
index 8094681..22a5c6d 100644
--- a/CasdoorLoginPage.js
+++ b/CasdoorLoginPage.js
@@ -20,19 +20,21 @@ import {useNotifications} from "react-native-notificated";
import SDK from "casdoor-react-native-sdk";
import PropTypes from "prop-types";
import EnterCasdoorSdkConfig from "./EnterCasdoorSdkConfig";
+import ScanQRCodeForLogin from "./ScanLogin";
import useStore from "./useStorage";
-// import {LogBox} from "react-native";
-// LogBox.ignoreAllLogs();
+import DefaultCasdoorSdkConfig from "./DefaultCasdoorSdkConfig";
let sdk = null;
-const CasdoorLoginPage = ({onWebviewClose}) => {
+
+function CasdoorLoginPage({onWebviewClose, initialMethod}) {
CasdoorLoginPage.propTypes = {
onWebviewClose: PropTypes.func.isRequired,
+ initialMethod: PropTypes.oneOf(["manual", "scan", "demo"]).isRequired,
};
const {notify} = useNotifications();
const [casdoorLoginURL, setCasdoorLoginURL] = useState("");
- const [showConfigPage, setShowConfigPage] = useState(true);
+ const [currentView, setCurrentView] = useState(initialMethod === "scan" ? "scanner" : "config");
const {
serverUrl,
@@ -42,38 +44,55 @@ const CasdoorLoginPage = ({onWebviewClose}) => {
organizationName,
token,
getCasdoorConfig,
+ setCasdoorConfig,
+ setServerUrl,
+ setClientId,
+ setAppName,
+ setOrganizationName,
setUserInfo,
setToken,
} = useStore();
- const handleHideConfigPage = () => {
- setShowConfigPage(false);
- };
+ useEffect(() => {
+ if (initialMethod === "demo") {
+ setCasdoorConfig(DefaultCasdoorSdkConfig);
+ }
+ }, [initialMethod, setCasdoorConfig]);
- const handleShowConfigPage = () => {
- setShowConfigPage(true);
+ const initSdk = () => {
+ const configs = {
+ demo: DefaultCasdoorSdkConfig,
+ scan: getCasdoorConfig(),
+ manual: serverUrl && clientId && redirectPath && appName && organizationName ? getCasdoorConfig() : null,
+ };
+ sdk = configs[initialMethod] ? new SDK(configs[initialMethod]) : null;
};
const getCasdoorSignInUrl = async() => {
- const signinUrl = await sdk.getSigninUrl();
- setCasdoorLoginURL(signinUrl);
+ initSdk();
+ if (sdk) {
+ const signinUrl = await sdk.getSigninUrl();
+ setCasdoorLoginURL(signinUrl);
+ }
};
- useEffect(() => {
- if (serverUrl && clientId && redirectPath && appName && organizationName) {
- sdk = new SDK(getCasdoorConfig());
- getCasdoorSignInUrl();
- }
- }, [serverUrl, clientId, redirectPath, appName, organizationName]);
+ const handleLogin = (method) => {
+ const actions = {
+ manual: () => {
+ getCasdoorSignInUrl();
+ setCurrentView("webview");
+ },
+ demo: () => {
+ getCasdoorSignInUrl();
+ setCurrentView("webview");
+ },
+ scan: () => setCurrentView("scanner"),
+ };
- useEffect(() => {
- if (token) {
- onWebviewClose();
- }
- }, [token]);
+ actions[method]?.();
+ };
const onNavigationStateChange = async(navState) => {
- const {redirectPath} = getCasdoorConfig();
if (navState.url.startsWith(redirectPath)) {
onWebviewClose();
const token = await sdk.getAccessToken(navState.url);
@@ -83,57 +102,70 @@ const CasdoorLoginPage = ({onWebviewClose}) => {
}
};
- const handleErrorResponse = (error) => {
- notify("error", {
- params: {
- text1: "Error",
- text2: error.description,
- },
- });
- setShowConfigPage(true);
+ const handleQRLogin = (loginInfo) => {
+ setServerUrl(loginInfo.serverUrl);
+ setClientId("");
+ setAppName("");
+ setOrganizationName("");
+ initSdk();
+ try {
+ const accessToken = loginInfo.accessToken;
+ const userInfo = sdk.JwtDecode(accessToken);
+ setToken(accessToken);
+ setUserInfo(userInfo);
+ notify("success", {params: {title: "Success", description: "Logged in successfully!"}});
+ setCurrentView("config");
+ onWebviewClose();
+ } catch (error) {
+ notify("error", {params: {title: "Error in login", description: error.message}});
+ }
};
- return (
-
-
- {showConfigPage && (
- {
+ const views = {
+ config: (
+ handleLogin(initialMethod)}
+ onWebviewClose={onWebviewClose}
+ />
+ ),
+ scanner: (
+ {
+ setCurrentView("config");
+ onWebviewClose();
+ }}
+ onLogin={handleQRLogin}
+ />
+ ),
+ webview: casdoorLoginURL && !token && (
+
+ setCurrentView("config")}>
+ Back to Config
+
+ {
+ notify("error", {params: {title: "Error", description: nativeEvent.description}});
+ setCurrentView("config");
+ }}
+ style={styles.webview}
+ mixedContentMode="always"
+ javaScriptEnabled={true}
/>
- )}
- {!showConfigPage && casdoorLoginURL !== "" && !token && (
- <>
-
- Back to Config
-
- {
- const {nativeEvent} = syntheticEvent;
- handleErrorResponse(nativeEvent);
- }}
- style={styles.webview}
- mixedContentMode="always"
- javaScriptEnabled={true}
- />
- >
- )}
-
-
- );
-};
+
+ ),
+ };
+
+ return views[currentView] || null;
+ };
+
+ return {renderContent()};
+}
const styles = StyleSheet.create({
- container: {
- flex: 1,
- backgroundColor: "white",
- paddingTop: Platform.OS === "android" ? StatusBar.currentHeight : 0,
- },
webview: {
flex: 1,
},
@@ -146,10 +178,18 @@ const styles = StyleSheet.create({
color: "white",
fontWeight: "bold",
},
+ safeArea: {
+ flex: 1,
+ backgroundColor: "white",
+ paddingTop: Platform.OS === "android" ? StatusBar.currentHeight : 0,
+
+ },
});
export const CasdoorLogout = () => {
- if (sdk) {sdk.clearState();}
+ if (sdk) {
+ sdk.clearState();
+ }
};
export default CasdoorLoginPage;
diff --git a/EnterCasdoorSdkConfig.js b/EnterCasdoorSdkConfig.js
index 6f52551..7458d10 100644
--- a/EnterCasdoorSdkConfig.js
+++ b/EnterCasdoorSdkConfig.js
@@ -12,22 +12,14 @@
// See the License for the specific language governing permissions and
// limitations under the License.
-import React, {useState} from "react";
-import {ScrollView, Text, View} from "react-native";
-import {Button, IconButton, Portal, TextInput} from "react-native-paper";
+import React from "react";
+import {StyleSheet, Text, View} from "react-native";
+import {Button, Portal, TextInput} from "react-native-paper";
import {useNotifications} from "react-native-notificated";
-import SDK from "casdoor-react-native-sdk";
-import DefaultCasdoorSdkConfig from "./DefaultCasdoorSdkConfig";
import PropTypes from "prop-types";
-import ScanQRCodeForLogin from "./ScanLogin";
import useStore from "./useStorage";
-const EnterCasdoorSdkConfig = ({onClose, onWebviewClose}) => {
- EnterCasdoorSdkConfig.propTypes = {
- onClose: PropTypes.func.isRequired,
- onWebviewClose: PropTypes.func.isRequired,
- };
-
+function EnterCasdoorSdkConfig({onClose, onWebviewClose}) {
const {
serverUrl,
clientId,
@@ -38,16 +30,10 @@ const EnterCasdoorSdkConfig = ({onClose, onWebviewClose}) => {
setClientId,
setAppName,
setOrganizationName,
- setCasdoorConfig,
- getCasdoorConfig,
- setToken,
- setUserInfo,
} = useStore();
const {notify} = useNotifications();
- const [showScanner, setShowScanner] = useState(false);
-
const closeConfigPage = () => {
onClose();
onWebviewClose();
@@ -66,207 +52,121 @@ const EnterCasdoorSdkConfig = ({onClose, onWebviewClose}) => {
onClose();
};
- const handleScanToLogin = () => {
- setShowScanner(true);
- };
-
- const handleLogin = (loginInfo) => {
- setServerUrl(loginInfo.serverUrl);
- setClientId("");
- setAppName("");
- setOrganizationName("");
-
- const sdk = new SDK(getCasdoorConfig());
-
- try {
- const accessToken = loginInfo.accessToken;
- const userInfo = sdk.JwtDecode(accessToken);
- setToken(accessToken);
- setUserInfo(userInfo);
-
- notify("success", {
- params: {
- title: "Success",
- description: "Logged in successfully!",
- },
- });
-
- setShowScanner(false);
- onClose();
- onWebviewClose();
- } catch (error) {
- notify("error", {
- params: {
- title: "Error in login",
- description: error,
- },
- });
- }
- };
-
- const handleUseDefault = () => {
- setCasdoorConfig(DefaultCasdoorSdkConfig);
- onClose();
- };
-
return (
-
+
-
- Casdoor server
- Casdoor Configuration
+
+
+
+
+
-
-
-
-
-
+
-
-
- {showScanner && (
- setShowScanner(false)}
- onLogin={handleLogin}
- />
- )}
+
);
+}
+
+EnterCasdoorSdkConfig.propTypes = {
+ onClose: PropTypes.func.isRequired,
+ onWebviewClose: PropTypes.func.isRequired,
};
-const styles = {
- scrollContainer: {
- flexGrow: 1,
- width: "100%",
+const styles = StyleSheet.create({
+ container: {
+ flex: 1,
justifyContent: "center",
alignItems: "center",
- backgroundColor: "rgba(255, 255, 255, 0.5)",
+ backgroundColor: "rgba(0, 0, 0, 0.5)",
},
content: {
- width: "95%",
- borderRadius: 10,
- padding: 20,
- backgroundColor: "#F5F5F5",
+ width: "90%",
+ maxWidth: 400,
+ borderRadius: 28,
+ padding: 24,
+ backgroundColor: "#FFFFFF",
shadowColor: "#000",
shadowOffset: {width: 0, height: 2},
shadowOpacity: 0.1,
- shadowRadius: 10,
+ shadowRadius: 8,
elevation: 5,
},
- input: {
- marginVertical: 10,
- fontSize: 16,
- backgroundColor: "white",
+ title: {
+ fontSize: 20,
+ fontWeight: "bold",
+ fontFamily: "Lato-Bold",
+ color: "#212121",
+ textAlign: "center",
+ marginBottom: 16,
},
- buttonRow: {
+ buttonContainer: {
flexDirection: "row",
justifyContent: "space-between",
- marginTop: 14,
- marginBottom: 12,
},
button: {
- borderRadius: 5,
- paddingVertical: 8,
- },
- confirmButton: {
- backgroundColor: "#6200EE",
flex: 1,
- marginRight: 5,
- },
- scanButton: {
- backgroundColor: "#03DAC6",
- flex: 1,
- marginLeft: 5,
+ marginHorizontal: 8,
+ borderRadius: 100,
},
buttonLabel: {
+ paddingVertical: 4,
fontSize: 16,
- color: "white",
- },
- outlinedButton: {
- borderColor: "#6200EE",
- borderWidth: 1,
- width: "100%",
- },
- outlinedButtonLabel: {
- color: "#6200EE",
- fontSize: 16,
- textAlign: "center",
- },
- header: {
- position: "relative",
- alignItems: "center",
- marginBottom: 20,
- },
- title: {
- fontSize: 24,
fontWeight: "bold",
- color: "#333",
- textAlign: "center",
},
- closeButton: {
- position: "absolute",
- right: 0,
- top: -8,
+ formContainer: {
+ marginBottom: 8,
},
-};
+ input: {
+ marginBottom: 16,
+ },
+});
export default EnterCasdoorSdkConfig;
diff --git a/Header.js b/Header.js
index d2300e8..bc94279 100644
--- a/Header.js
+++ b/Header.js
@@ -20,6 +20,7 @@ import Icon from "react-native-vector-icons/MaterialCommunityIcons";
import CasdoorLoginPage, {CasdoorLogout} from "./CasdoorLoginPage";
import useStore from "./useStorage";
import {useAccountSync} from "./useAccountStore";
+import LoginMethodSelector from "./LoginMethodSelector";
const {width} = Dimensions.get("window");
@@ -28,6 +29,7 @@ const Header = () => {
const {isSyncing, syncError, clearSyncError} = useAccountSync();
const [showLoginPage, setShowLoginPage] = React.useState(false);
const [menuVisible, setMenuVisible] = React.useState(false);
+ const [loginMethod, setLoginMethod] = React.useState(null);
const {notify} = useNotifications();
const openMenu = () => setMenuVisible(true);
@@ -38,8 +40,20 @@ const Header = () => {
closeMenu();
};
- const handleCasdoorLogin = () => setShowLoginPage(true);
- const handleHideLoginPage = () => setShowLoginPage(false);
+ const {openActionSheet} = LoginMethodSelector({
+ onSelectMethod: (method) => {
+ setLoginMethod(method);
+ setShowLoginPage(true);
+ },
+ });
+
+ const handleCasdoorLogin = () => {
+ openActionSheet();
+ };
+
+ const handleHideLoginPage = () => {
+ setShowLoginPage(false);
+ };
const handleCasdoorLogout = () => {
CasdoorLogout();
@@ -114,7 +128,12 @@ const Header = () => {
- {showLoginPage && }
+ {showLoginPage && (
+
+ )}
);
};
diff --git a/LoginMethodSelector.js b/LoginMethodSelector.js
new file mode 100644
index 0000000..4106362
--- /dev/null
+++ b/LoginMethodSelector.js
@@ -0,0 +1,62 @@
+// Copyright 2023 The Casdoor Authors. All Rights Reserved.
+//
+// Licensed under the Apache License, Version 2.0 (the "License");
+// you may not use this file except in compliance with the License.
+// You may obtain a copy of the License at
+//
+// http://www.apache.org/licenses/LICENSE-2.0
+//
+// Unless required by applicable law or agreed to in writing, software
+// distributed under the License is distributed on an "AS IS" BASIS,
+// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+// See the License for the specific language governing permissions and
+// limitations under the License.
+
+import {useActionSheet} from "@expo/react-native-action-sheet";
+
+const LoginMethodSelector = ({onSelectMethod}) => {
+ const {showActionSheetWithOptions} = useActionSheet();
+
+ const openActionSheet = () => {
+ const options = [
+ "Manual Server Setup",
+ "Login Using QR Code",
+ "Try Casdoor Demo Site",
+ "Cancel",
+ ];
+ const cancelButtonIndex = 3;
+
+ showActionSheetWithOptions(
+ {
+ title: "Select Login Method",
+ cancelButtonTintColor: "red",
+ options,
+ cancelButtonIndex,
+ },
+ (buttonIndex) => {
+ handleSelection(buttonIndex);
+ }
+ );
+ };
+
+ const handleSelection = (buttonIndex) => {
+ switch (buttonIndex) {
+ case 0:
+ onSelectMethod("manual");
+ break;
+ case 1:
+ onSelectMethod("scan");
+ break;
+ case 2:
+ onSelectMethod("demo");
+ break;
+ default:
+ // Cancel was pressed
+ break;
+ }
+ };
+
+ return {openActionSheet};
+};
+
+export default LoginMethodSelector;
diff --git a/SettingPage.js b/SettingPage.js
index 69503e8..f7f999f 100644
--- a/SettingPage.js
+++ b/SettingPage.js
@@ -12,64 +12,88 @@
// See the License for the specific language governing permissions and
// limitations under the License.
-import * as React from "react";
-import {StyleSheet, View, useWindowDimensions} from "react-native";
+import React, {useState} from "react";
+import {Dimensions, StyleSheet, View} from "react-native";
import {Button, Surface, Text} from "react-native-paper";
+import {ActionSheetProvider} from "@expo/react-native-action-sheet";
import CasdoorLoginPage, {CasdoorLogout} from "./CasdoorLoginPage";
+import LoginMethodSelector from "./LoginMethodSelector";
import useStore from "./useStorage";
+const {width} = Dimensions.get("window");
+
const SettingPage = () => {
- const [showLoginPage, setShowLoginPage] = React.useState(false);
+ const [showLoginPage, setShowLoginPage] = useState(false);
+ const [loginMethod, setLoginMethod] = useState(null);
const {userInfo, clearAll} = useStore();
- const {width} = useWindowDimensions();
- const handleCasdoorLogin = () => setShowLoginPage(true);
- const handleHideLoginPage = () => setShowLoginPage(false);
+ const {openActionSheet} = LoginMethodSelector({
+ onSelectMethod: (method) => {
+ setLoginMethod(method);
+ setShowLoginPage(true);
+ },
+ });
+
+ const handleCasdoorLogin = () => {
+ openActionSheet();
+ };
+
+ const handleHideLoginPage = () => {
+ setShowLoginPage(false);
+ setLoginMethod(null);
+ };
const handleCasdoorLogout = () => {
CasdoorLogout();
clearAll();
};
- const styles = StyleSheet.create({
- container: {
- flex: 1,
- justifyContent: "center",
- alignItems: "center",
- padding: 16,
- },
- surface: {
- padding: 16,
- width: width > 600 ? 400 : "100%",
- maxWidth: 400,
- alignItems: "center",
- },
- title: {
- fontSize: 24,
- marginBottom: 24,
- },
- button: {
- marginTop: 16,
- width: "100%",
- },
- });
-
return (
-
-
- Account Settings
-
-
- {showLoginPage && }
-
+
+
+
+ Account Settings
+
+
+ {showLoginPage && (
+
+ )}
+
+
);
};
+const styles = StyleSheet.create({
+ container: {
+ flex: 1,
+ justifyContent: "center",
+ alignItems: "center",
+ padding: 16,
+ },
+ surface: {
+ padding: 16,
+ width: width > 600 ? 400 : "100%",
+ maxWidth: 400,
+ alignItems: "center",
+ },
+ title: {
+ fontSize: 24,
+ marginBottom: 24,
+ },
+ button: {
+ marginTop: 16,
+ width: "100%",
+ },
+});
+
export default SettingPage;
diff --git a/package-lock.json b/package-lock.json
index 7aac8cd..bc6ba9d 100644
--- a/package-lock.json
+++ b/package-lock.json
@@ -8,6 +8,7 @@
"name": "casdoor-app",
"version": "1.0.0",
"dependencies": {
+ "@expo/react-native-action-sheet": "^4.1.0",
"@react-native-async-storage/async-storage": "1.23.1",
"@react-native-community/masked-view": "^0.1.11",
"@react-native-community/netinfo": "11.3.1",
@@ -18,18 +19,18 @@
"casdoor-react-native-sdk": "1.1.0",
"drizzle-orm": "^0.33.0",
"eslint-plugin-import": "^2.28.1",
- "expo": "~51.0.31",
+ "expo": "~51.0.34",
"expo-asset": "~10.0.10",
- "expo-camera": "~15.0.15",
+ "expo-camera": "~15.0.16",
"expo-crypto": "~13.0.2",
- "expo-dev-client": "~4.0.26",
+ "expo-dev-client": "~4.0.27",
"expo-drizzle-studio-plugin": "^0.0.2",
- "expo-image": "~1.12.15",
+ "expo-image": "~1.13.0",
"expo-image-picker": "~15.0.7",
"expo-sqlite": "^14.0.6",
"expo-status-bar": "~1.12.1",
"expo-system-ui": "~3.0.7",
- "expo-updates": "~0.25.24",
+ "expo-updates": "~0.25.25",
"hi-base32": "^0.5.1",
"hotp-totp": "^1.0.6",
"prop-types": "^15.8.1",
@@ -3505,9 +3506,10 @@
}
},
"node_modules/@expo/config-plugins": {
- "version": "8.0.8",
- "resolved": "https://registry.npmjs.org/@expo/config-plugins/-/config-plugins-8.0.8.tgz",
- "integrity": "sha512-Fvu6IO13EUw0R9WeqxUO37FkM62YJBNcZb9DyJAOgMz7Ez/vaKQGEjKt9cwT+Q6uirtCATMgaq6VWAW7YW8xXw==",
+ "version": "8.0.9",
+ "resolved": "https://registry.npmjs.org/@expo/config-plugins/-/config-plugins-8.0.9.tgz",
+ "integrity": "sha512-dNCG45C7BbDPV9MdWvCbsFtJtVn4w/TJbb5b7Yr6FA8HYIlaaVM0wqUMzTPmGj54iYXw8X/Vge8uCPxg7RWgeA==",
+ "license": "MIT",
"dependencies": {
"@expo/config-types": "^51.0.0-unreleased",
"@expo/json-file": "~8.3.0",
@@ -4487,6 +4489,19 @@
"node": ">= 10.0.0"
}
},
+ "node_modules/@expo/react-native-action-sheet": {
+ "version": "4.1.0",
+ "resolved": "https://registry.npmjs.org/@expo/react-native-action-sheet/-/react-native-action-sheet-4.1.0.tgz",
+ "integrity": "sha512-RILoWhREgjMdr1NUSmZa/cHg8onV2YPDAMOy0iIP1c3H7nT9QQZf5dQNHK8ehcLM82sarVxriBJyYSSHAx7j6w==",
+ "license": "MIT",
+ "dependencies": {
+ "@types/hoist-non-react-statics": "^3.3.1",
+ "hoist-non-react-statics": "^3.3.0"
+ },
+ "peerDependencies": {
+ "react": ">=18.0.0"
+ }
+ },
"node_modules/@expo/rudder-sdk-node": {
"version": "1.1.1",
"resolved": "https://registry.npmjs.org/@expo/rudder-sdk-node/-/rudder-sdk-node-1.1.1.tgz",
@@ -4522,9 +4537,10 @@
}
},
"node_modules/@expo/vector-icons": {
- "version": "14.0.2",
- "resolved": "https://registry.npmjs.org/@expo/vector-icons/-/vector-icons-14.0.2.tgz",
- "integrity": "sha512-70LpmXQu4xa8cMxjp1fydgRPsalefnHaXLzIwaHMEzcZhnyjw2acZz8azRrZOslPVAWlxItOa2Dd7WtD/kI+CA==",
+ "version": "14.0.3",
+ "resolved": "https://registry.npmjs.org/@expo/vector-icons/-/vector-icons-14.0.3.tgz",
+ "integrity": "sha512-UJAKOXPPi6ez/1QZfoFVopCH3+c12Sw+T+IIVkvONCEN7zjN1fLxxWHkZ7Spz4WO5EH2ObtaJfCe/k4rw+ftxA==",
+ "license": "MIT",
"dependencies": {
"prop-types": "^15.8.1"
}
@@ -7281,6 +7297,16 @@
"resolved": "https://registry.npmjs.org/@types/hammerjs/-/hammerjs-2.0.42.tgz",
"integrity": "sha512-Xxk14BrwHnGi0xlURPRb+Y0UNn2w3cTkeFm7pKMsYOaNgH/kabbJLhcBoNIodwsbTz7Z8KcWjtDvlGH0nc0U9w=="
},
+ "node_modules/@types/hoist-non-react-statics": {
+ "version": "3.3.5",
+ "resolved": "https://registry.npmjs.org/@types/hoist-non-react-statics/-/hoist-non-react-statics-3.3.5.tgz",
+ "integrity": "sha512-SbcrWzkKBw2cdwRTwQAswfpB9g9LJWfjtUeW/jvNwbhC8cpmmNYVePa+ncbUe0rGTQ7G3Ff6mYUN2VMfLVr+Sg==",
+ "license": "MIT",
+ "dependencies": {
+ "@types/react": "*",
+ "hoist-non-react-statics": "^3.3.0"
+ }
+ },
"node_modules/@types/istanbul-lib-coverage": {
"version": "2.0.4",
"resolved": "https://registry.npmjs.org/@types/istanbul-lib-coverage/-/istanbul-lib-coverage-2.0.4.tgz",
@@ -7329,14 +7355,12 @@
"node_modules/@types/prop-types": {
"version": "15.7.7",
"resolved": "https://registry.npmjs.org/@types/prop-types/-/prop-types-15.7.7.tgz",
- "integrity": "sha512-FbtmBWCcSa2J4zL781Zf1p5YUBXQomPEcep9QZCfRfQgTxz3pJWiDFLebohZ9fFntX5ibzOkSsrJ0TEew8cAog==",
- "devOptional": true
+ "integrity": "sha512-FbtmBWCcSa2J4zL781Zf1p5YUBXQomPEcep9QZCfRfQgTxz3pJWiDFLebohZ9fFntX5ibzOkSsrJ0TEew8cAog=="
},
"node_modules/@types/react": {
"version": "18.2.79",
"resolved": "https://registry.npmjs.org/@types/react/-/react-18.2.79.tgz",
"integrity": "sha512-RwGAGXPl9kSXwdNTafkOEuFrTBD5SA2B3iEB96xi8+xu5ddUa/cpvyVCSNn+asgLCTHkb5ZxN8gbuibYJi4s1w==",
- "devOptional": true,
"dependencies": {
"@types/prop-types": "*",
"csstype": "^3.0.2"
@@ -9056,8 +9080,7 @@
"node_modules/csstype": {
"version": "3.1.2",
"resolved": "https://registry.npmjs.org/csstype/-/csstype-3.1.2.tgz",
- "integrity": "sha512-I7K1Uu0MBPzaFKg4nI5Q7Vs2t+3gWWW648spaF+Rg7pI9ds18Ugn+lvg4SHczUdKlHI5LWBXyqfS8+DufyBsgQ==",
- "devOptional": true
+ "integrity": "sha512-I7K1Uu0MBPzaFKg4nI5Q7Vs2t+3gWWW648spaF+Rg7pI9ds18Ugn+lvg4SHczUdKlHI5LWBXyqfS8+DufyBsgQ=="
},
"node_modules/dag-map": {
"version": "1.0.2",
@@ -10420,24 +10443,24 @@
}
},
"node_modules/expo": {
- "version": "51.0.31",
- "resolved": "https://registry.npmjs.org/expo/-/expo-51.0.31.tgz",
- "integrity": "sha512-YiUNcxzSkQ0jlKW+e8F81KnZfAhCugEZI9VYmuIsFONHivtiYIADHdcFvUWnexUEdgPQDkgWw85XBnIbzIZ39Q==",
+ "version": "51.0.34",
+ "resolved": "https://registry.npmjs.org/expo/-/expo-51.0.34.tgz",
+ "integrity": "sha512-l2oi+hIj/ph3qGcvM54Nyd2uF3Zq5caVmSg7AXfBUgtvcdv5Pj1EI/2xCXP9tfMNQo351CWyOwBkTGjv+GdrLg==",
"license": "MIT",
"dependencies": {
"@babel/runtime": "^7.20.0",
"@expo/cli": "0.18.29",
"@expo/config": "9.0.3",
- "@expo/config-plugins": "8.0.8",
+ "@expo/config-plugins": "8.0.9",
"@expo/metro-config": "0.18.11",
- "@expo/vector-icons": "^14.0.0",
+ "@expo/vector-icons": "^14.0.3",
"babel-preset-expo": "~11.0.14",
"expo-asset": "~10.0.10",
"expo-file-system": "~17.0.1",
- "expo-font": "~12.0.9",
+ "expo-font": "~12.0.10",
"expo-keep-awake": "~13.0.2",
"expo-modules-autolinking": "1.11.2",
- "expo-modules-core": "1.12.23",
+ "expo-modules-core": "1.12.24",
"fbemitter": "^3.0.0",
"whatwg-url-without-unicode": "8.0.0-3"
},
@@ -10459,9 +10482,9 @@
}
},
"node_modules/expo-camera": {
- "version": "15.0.15",
- "resolved": "https://registry.npmjs.org/expo-camera/-/expo-camera-15.0.15.tgz",
- "integrity": "sha512-zJS0rfOwGfyDrccMsFaLH9s0mW0f6czw+W+RHvbyAdAmoAh6ZtzZRGbosKIYoRsn/8L7ajNp01seJUjsT0FtzA==",
+ "version": "15.0.16",
+ "resolved": "https://registry.npmjs.org/expo-camera/-/expo-camera-15.0.16.tgz",
+ "integrity": "sha512-FLE02DMqkjwsb7IugKAqQvBe6s+TCQeb5LupO1+r//wAhBwmHncOrc6zV95ZEC2f9PTPK34nFH/s8CDGiVzIAA==",
"license": "MIT",
"dependencies": {
"invariant": "^2.2.4"
@@ -10495,9 +10518,9 @@
}
},
"node_modules/expo-dev-client": {
- "version": "4.0.26",
- "resolved": "https://registry.npmjs.org/expo-dev-client/-/expo-dev-client-4.0.26.tgz",
- "integrity": "sha512-GM+X7bngAK2vr0YMkPnQFUFVW22eG3CjoxTJ0yUwW3RgCqFdMkTeAIS/1sEXjyNYjGkigtgtch+bdYtJxfqpuw==",
+ "version": "4.0.27",
+ "resolved": "https://registry.npmjs.org/expo-dev-client/-/expo-dev-client-4.0.27.tgz",
+ "integrity": "sha512-4f0eO7GTdGzYYg3qABR98Vc2iiCBA2HICh8namVAvqkcVCuh44I9lOctaAEe/932+lLugEW4+Mv29pdEHq3/FA==",
"license": "MIT",
"dependencies": {
"expo-dev-launcher": "4.0.27",
@@ -10620,9 +10643,10 @@
}
},
"node_modules/expo-font": {
- "version": "12.0.9",
- "resolved": "https://registry.npmjs.org/expo-font/-/expo-font-12.0.9.tgz",
- "integrity": "sha512-seTCyf0tbgkAnp3ZI9ZfK9QVtURQUgFnuj+GuJ5TSnN0XsOtVe1s2RxTvmMgkfuvfkzcjJ69gyRpsZS1cC8hjw==",
+ "version": "12.0.10",
+ "resolved": "https://registry.npmjs.org/expo-font/-/expo-font-12.0.10.tgz",
+ "integrity": "sha512-Q1i2NuYri3jy32zdnBaHHCya1wH1yMAsI+3CCmj9zlQzlhsS9Bdwcj2W3c5eU5FvH2hsNQy4O+O1NnM6o/pDaQ==",
+ "license": "MIT",
"dependencies": {
"fontfaceobserver": "^2.1.0"
},
@@ -10631,9 +10655,9 @@
}
},
"node_modules/expo-image": {
- "version": "1.12.15",
- "resolved": "https://registry.npmjs.org/expo-image/-/expo-image-1.12.15.tgz",
- "integrity": "sha512-rjvnNSaFnrmlugDESTaYJhgdqRLn+M5vu0lD5NGNd2LkxGG5HrRV3gSzeyQQ68XRhrDN8eJvkcKujPKJUTMraw==",
+ "version": "1.13.0",
+ "resolved": "https://registry.npmjs.org/expo-image/-/expo-image-1.13.0.tgz",
+ "integrity": "sha512-0NLDcFmEn4Nh1sXeRvNzDHT+Fl6FXtTol6ki6kYYH0/iDeSFWyIy/Fek6kzDDYAmhipSMR7buPf7VVoHseTbAA==",
"license": "MIT",
"peerDependencies": {
"expo": "*"
@@ -10810,9 +10834,9 @@
}
},
"node_modules/expo-modules-core": {
- "version": "1.12.23",
- "resolved": "https://registry.npmjs.org/expo-modules-core/-/expo-modules-core-1.12.23.tgz",
- "integrity": "sha512-NYp/rWhKW6zlqNdC8/r+FckzlAGWX0IJEjOxwYHuYeRUn/vnKksb43G4E3jcaQEZgmWlKxK4LpxL3gr7m0RJFA==",
+ "version": "1.12.24",
+ "resolved": "https://registry.npmjs.org/expo-modules-core/-/expo-modules-core-1.12.24.tgz",
+ "integrity": "sha512-3geIe2ecizlp7l26iY8Nmc59z2d1RUC5nQZtI9iJoi5uHEUV/zut8e4zRLFVnZb8KOcMcEDsrvaBL5DPnqdfpg==",
"license": "MIT",
"dependencies": {
"invariant": "^2.2.4"
@@ -10852,9 +10876,9 @@
}
},
"node_modules/expo-updates": {
- "version": "0.25.24",
- "resolved": "https://registry.npmjs.org/expo-updates/-/expo-updates-0.25.24.tgz",
- "integrity": "sha512-juqdOUvaMfu6zeUg3fTk6ciLw4QK+0HXNR0+X41BVOFilNmlTFQZ6LyRGJAZJP7HQs2bHR5d/btAXkejtIqVXw==",
+ "version": "0.25.25",
+ "resolved": "https://registry.npmjs.org/expo-updates/-/expo-updates-0.25.25.tgz",
+ "integrity": "sha512-Z9sCf6w3876JLlj6DGRsXFI/NnRhXM0gfXT2dusniagt4qvwThGKxS/zEcpo9JUyO411yVL/XGv411Czeaw9xA==",
"license": "MIT",
"dependencies": {
"@expo/code-signing-certificates": "0.0.5",
@@ -11216,7 +11240,8 @@
"node_modules/fontfaceobserver": {
"version": "2.3.0",
"resolved": "https://registry.npmjs.org/fontfaceobserver/-/fontfaceobserver-2.3.0.tgz",
- "integrity": "sha512-6FPvD/IVyT4ZlNe7Wcn5Fb/4ChigpucKYSvD6a+0iMoLn2inpo711eyIcKjmDtE5XNcgAkSH9uN/nfAeZzHEfg=="
+ "integrity": "sha512-6FPvD/IVyT4ZlNe7Wcn5Fb/4ChigpucKYSvD6a+0iMoLn2inpo711eyIcKjmDtE5XNcgAkSH9uN/nfAeZzHEfg==",
+ "license": "BSD-2-Clause"
},
"node_modules/for-each": {
"version": "0.3.3",
diff --git a/package.json b/package.json
index 0fcc210..9bc59e9 100644
--- a/package.json
+++ b/package.json
@@ -10,6 +10,7 @@
"release": "npx -p semantic-release-expo -p semantic-release -p @semantic-release/git -p @semantic-release/changelog -p @semantic-release/exec semantic-release"
},
"dependencies": {
+ "@expo/react-native-action-sheet": "^4.1.0",
"@react-native-async-storage/async-storage": "1.23.1",
"@react-native-community/masked-view": "^0.1.11",
"@react-native-community/netinfo": "11.3.1",
@@ -20,18 +21,18 @@
"casdoor-react-native-sdk": "1.1.0",
"drizzle-orm": "^0.33.0",
"eslint-plugin-import": "^2.28.1",
- "expo": "~51.0.31",
+ "expo": "~51.0.34",
"expo-asset": "~10.0.10",
- "expo-camera": "~15.0.15",
+ "expo-camera": "~15.0.16",
"expo-crypto": "~13.0.2",
- "expo-dev-client": "~4.0.26",
+ "expo-dev-client": "~4.0.27",
"expo-drizzle-studio-plugin": "^0.0.2",
- "expo-image": "~1.12.15",
+ "expo-image": "~1.13.0",
"expo-image-picker": "~15.0.7",
"expo-sqlite": "^14.0.6",
"expo-status-bar": "~1.12.1",
"expo-system-ui": "~3.0.7",
- "expo-updates": "~0.25.24",
+ "expo-updates": "~0.25.25",
"hi-base32": "^0.5.1",
"hotp-totp": "^1.0.6",
"prop-types": "^15.8.1",
diff --git a/syncLogic.js b/syncLogic.js
index dfef243..a1df988 100644
--- a/syncLogic.js
+++ b/syncLogic.js
@@ -16,6 +16,12 @@ import {eq} from "drizzle-orm";
import * as schema from "./db/schema";
import * as api from "./api";
import {generateToken} from "./totpUtil";
+import useStore from "./useStorage";
+
+function handleTokenExpiration() {
+ const {clearAll} = useStore.getState();
+ clearAll();
+}
function getLocalAccounts(db) {
return db.select().from(schema.accounts).all();
@@ -134,48 +140,56 @@ function mergeAccounts(localAccounts, serverAccounts, serverTimestamp) {
}
export async function syncWithCloud(db, userInfo, serverUrl, token) {
- // db.delete(schema.accounts).run();
- const localAccounts = await getLocalAccounts(db);
+ try {
+ const localAccounts = await getLocalAccounts(db);
- const {updatedTime, mfaAccounts: serverAccounts} = await api.getMfaAccounts(
- serverUrl,
- userInfo.owner,
- userInfo.name,
- token
- );
+ const {updatedTime, mfaAccounts: serverAccounts} = await api.getMfaAccounts(
+ serverUrl,
+ userInfo.owner,
+ userInfo.name,
+ token
+ );
- const mergedAccounts = mergeAccounts(localAccounts, serverAccounts, updatedTime);
+ const mergedAccounts = mergeAccounts(localAccounts, serverAccounts, updatedTime);
- await updateLocalDatabase(db, mergedAccounts);
+ await updateLocalDatabase(db, mergedAccounts);
- const accountsToSync = mergedAccounts.filter(account => account.deletedAt === null || account.deletedAt === undefined)
- .map(account => ({
+ const accountsToSync = mergedAccounts.filter(account => account.deletedAt === null || account.deletedAt === undefined)
+ .map(account => ({
+ issuer: account.issuer,
+ accountName: account.accountName,
+ secretKey: account.secretKey,
+ }));
+
+ const serverAccountsStringified = serverAccounts.map(account => JSON.stringify({
issuer: account.issuer,
accountName: account.accountName,
secretKey: account.secretKey,
}));
- const serverAccountsStringified = serverAccounts.map(account => JSON.stringify({
- issuer: account.issuer,
- accountName: account.accountName,
- secretKey: account.secretKey,
- }));
+ const accountsToSyncStringified = accountsToSync.map(account => JSON.stringify(account));
- const accountsToSyncStringified = accountsToSync.map(account => JSON.stringify(account));
+ if (JSON.stringify(accountsToSyncStringified.sort()) !== JSON.stringify(serverAccountsStringified.sort())) {
+ const {status} = await api.updateMfaAccounts(
+ serverUrl,
+ userInfo.owner,
+ userInfo.name,
+ accountsToSync,
+ token
+ );
- if (JSON.stringify(accountsToSyncStringified.sort()) !== JSON.stringify(serverAccountsStringified.sort())) {
- const {status} = await api.updateMfaAccounts(
- serverUrl,
- userInfo.owner,
- userInfo.name,
- accountsToSync,
- token
- );
+ if (status !== "ok") {
+ throw new Error("Sync failed");
+ }
+ }
- if (status !== "ok") {
- throw new Error("Sync failed");
+ await db.update(schema.accounts).set({syncAt: new Date()}).run();
+
+ } catch (error) {
+ if (error.message.includes("Access token has expired")) {
+ handleTokenExpiration();
+ throw new Error("Access token has expired, please login again.");
}
+ throw error;
}
-
- await db.update(schema.accounts).set({syncAt: new Date()}).run();
}