122 lines
3.7 KiB
Swift
122 lines
3.7 KiB
Swift
//
|
|
// LoginState.swift
|
|
// punchnet
|
|
//
|
|
// Created by 安礼成 on 2026/1/16.
|
|
//
|
|
|
|
import Foundation
|
|
import Observation
|
|
|
|
@Observable
|
|
class AppContext {
|
|
var noticePort: Int
|
|
|
|
// 调用 "/connect" 之后的网络信息
|
|
var networkContext: SDLAPIClient.NetworkContext = .default()
|
|
|
|
// 保存当前登陆vpn使用的配置项
|
|
var vpnOptions: [String: NSObject]? = nil
|
|
|
|
// 当前app所处的场景
|
|
var loginScene: LoginScene = .login(username: nil)
|
|
|
|
// 当前的场景
|
|
enum LoginScene: Equatable {
|
|
case login(username: String?)
|
|
case register
|
|
case resetPassword
|
|
}
|
|
|
|
// 登陆凭证
|
|
var loginCredit: Credit?
|
|
|
|
enum Credit {
|
|
case token(token: String, session: SDLAPIClient.NetworkSession)
|
|
case accountAndPasword(account: String, password: String, session: SDLAPIClient.NetworkSession)
|
|
}
|
|
|
|
@ObservationIgnored
|
|
var networkSession: SDLAPIClient.NetworkSession? {
|
|
guard let loginCredit = self.loginCredit else {
|
|
return nil
|
|
}
|
|
|
|
switch loginCredit {
|
|
case .token(_, let session):
|
|
return session
|
|
case .accountAndPasword(_, _, let session):
|
|
return session
|
|
}
|
|
}
|
|
|
|
init(noticePort: Int) {
|
|
self.noticePort = noticePort
|
|
}
|
|
|
|
func loginWith(token: String) async throws {
|
|
let networkSession = try await SDLAPIClient.loginWithToken(token: token)
|
|
self.loginCredit = .token(token: token, session: networkSession)
|
|
// 将数据缓存到keychain
|
|
if let data = token.data(using: .utf8) {
|
|
try KeychainStore.shared.save(data, account: "token")
|
|
}
|
|
}
|
|
|
|
func loginWith(username: String, password: String) async throws {
|
|
let networkSession = try await SDLAPIClient.loginWithAccountAndPassword(username: username, password: password)
|
|
self.loginCredit = .accountAndPasword(account: username, password: password, session: networkSession)
|
|
// 将数据缓存到keychain
|
|
if let data = "\(username):\(password)".data(using: .utf8) {
|
|
try KeychainStore.shared.save(data, account: "accountAndPasword")
|
|
}
|
|
}
|
|
|
|
// 连接到对应的网络
|
|
func connectNetwork() async throws {
|
|
guard let session = self.networkSession else {
|
|
return
|
|
}
|
|
|
|
let context = try await SDLAPIClient.connectNetwork(accesToken: session.accessToken)
|
|
if let options = SystemConfig.getOptions(
|
|
networkId: UInt32(session.networkId),
|
|
networkDomain: session.networkDomain,
|
|
ip: context.ip,
|
|
maskLen: context.maskLen,
|
|
accessToken: session.accessToken,
|
|
identityId: context.identityId,
|
|
hostname: context.hostname,
|
|
noticePort: noticePort
|
|
) {
|
|
try await VPNManager.shared.enableVpn(options: options)
|
|
self.networkContext = context
|
|
self.vpnOptions = options
|
|
}
|
|
}
|
|
|
|
func loadCacheToken() -> String? {
|
|
if let data = try? KeychainStore.shared.load(account: "token") {
|
|
return String(data: data, encoding: .utf8)
|
|
}
|
|
return nil
|
|
}
|
|
|
|
func loadCacheUsernameAndPassword() -> (String, String)? {
|
|
if let data = try? KeychainStore.shared.load(account: "accountAndPasword"),
|
|
let str = String(data: data, encoding: .utf8) {
|
|
let parts = str.split(separator: ":")
|
|
if parts.count == 2 {
|
|
return (String(parts[0]), String(parts[1]))
|
|
}
|
|
}
|
|
return nil
|
|
}
|
|
|
|
// 退出登陆
|
|
func logout() async throws {
|
|
try await VPNManager.shared.disableVpn()
|
|
}
|
|
|
|
}
|