account.swift 8.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260
  1. //
  2. // account.swift
  3. // 74 桌
  4. //
  5. // Created by yunli on 2023/5/20.
  6. //
  7. import Alamofire
  8. import SwiftUI
  9. struct Constant {
  10. var cookiesDefaultsKey: String
  11. }
  12. let Constants: Constant = .init(cookiesDefaultsKey: "JSESSIONID")
  13. class CookieHandler {
  14. static let shared: CookieHandler = .init()
  15. let defaults = UserDefaults.standard
  16. let cookieStorage = HTTPCookieStorage.shared
  17. func getCookie(forURL url: String) -> [HTTPCookie] {
  18. let computedUrl = URL(string: url)
  19. let cookies = cookieStorage.cookies(for: computedUrl!) ?? []
  20. return cookies
  21. }
  22. func backupCookies(forURL url: String) {
  23. var cookieDict = [String: AnyObject]()
  24. for cookie in getCookie(forURL: url) {
  25. cookieDict[cookie.name] = cookie.properties as AnyObject?
  26. }
  27. defaults.set(cookieDict, forKey: Constants.cookiesDefaultsKey)
  28. }
  29. func restoreCookies() {
  30. if let cookieDictionary = defaults.dictionary(forKey: Constants.cookiesDefaultsKey) {
  31. for (_, cookieProperties) in cookieDictionary {
  32. if let cookie = HTTPCookie(properties: cookieProperties as! [HTTPCookiePropertyKey: Any]) {
  33. cookieStorage.setCookie(cookie)
  34. }
  35. }
  36. }
  37. }
  38. }
  39. struct Login: Encodable {
  40. let user: String
  41. let pwd: String
  42. let cook: String
  43. }
  44. struct UserInfo: Encodable {
  45. var name: String
  46. var o: String
  47. }
  48. class LoginHandler {
  49. static let sharedInstance = LoginHandler()
  50. func getCookie(name: String) -> HTTPCookie? {
  51. if let cookie = CookieHandler().getCookie(forURL: "https://debug.sdsz.icu:81/").filter({ cookie -> Bool in
  52. cookie.name == name
  53. }).first {
  54. return cookie
  55. }
  56. return nil
  57. }
  58. func fetchLoginCookie(action: @escaping (_: String) -> Void) {
  59. if let cookie = getCookie(name: "JSESSIONID") {
  60. HTTPCookieStorage.shared.deleteCookie(cookie)
  61. print(cookie)
  62. }
  63. let url = "https://debug.sdsz.icu:81/sso/login"
  64. AF.request(url, method: .get, parameters: nil, encoding: URLEncoding.default, headers: nil).response { _ in
  65. if let cookie = self.getCookie(name: "JSESSIONID") {
  66. action(cookie.value)
  67. }
  68. }
  69. }
  70. }
  71. class FetchHandler {
  72. static let sharedInstance = FetchHandler()
  73. func matches(url: String) -> Bool {
  74. do {
  75. return try url.matches(of: Regex(#"(^\[\]$|param":"|"workRests"|<tr>|DOCTYPE|in\?username=|requestParams|\"newmessage\"|Sorry, Page Not Found|北师大实验中学--登录|404 Not Found|502 Bad Gateway)"#)).count != 0
  76. } catch {
  77. return true
  78. }
  79. }
  80. func getUrl(url: String) -> String {
  81. var ret: String = url
  82. print("PRE: \(ret)")
  83. ret = ret
  84. .replacing("dd.sdsz.com.cn", with: "debug.sdsz.icu:81")
  85. .replacing(/^http:/, with: "https:")
  86. .replacing("service=http%3A%2F%2Fdebug.sdsz.icu:81", with: "service=http%3A%2F%2Fdd.sdsz.com.cn")
  87. print("AFTER: \(ret)")
  88. return ret
  89. }
  90. func workFetch(url: String, action: @escaping (_: String) -> Void) {
  91. if matches(url: url) {
  92. action(url)
  93. } else {
  94. AF.request(getUrl(url: url), method: .get, parameters: nil, encoding: URLEncoding.default, headers: nil).responseString { res in
  95. print(res)
  96. if let resv = res.value {
  97. self.workFetch(url: resv, action: action)
  98. }
  99. }
  100. }
  101. }
  102. func fetchAny(url: String, action: @escaping (_: String) -> Void) {
  103. let urlFull = "https://debug.sdsz.icu:81/\(url)"
  104. workFetch(url: urlFull, action: action)
  105. }
  106. }
  107. func doLogin(user: String, password: String, action: @escaping (_: Int) -> Void) {
  108. LoginHandler().fetchLoginCookie { (cookie: String) in
  109. let login = Login(user: user, pwd: password, cook: cookie)
  110. print(login)
  111. AF.request("https://debug.sdsz.icu/andlogin", method: .post, parameters: login, encoder: JSONParameterEncoder.default, headers: nil).responseString { res in
  112. print("\(res)")
  113. if res.value == "success" {
  114. action(1)
  115. } else {
  116. action(0)
  117. }
  118. }
  119. }
  120. }
  121. struct DecodableType: Decodable { let url: String }
  122. func getUserInfo(userInfo: inout UserInfo) {
  123. userInfo.name = "向量"
  124. FetchHandler().fetchAny(url: "bxn-portal/portal/osforstudent/index") { res in
  125. print(res)
  126. }
  127. }
  128. struct loginView: View {
  129. @State var username: String = ""
  130. @State var password: String = ""
  131. @State var btnText: String = "登录"
  132. @State var btnColor: Color = .blue
  133. @State var inProgress: Bool = false
  134. @FocusState private var isFocused: Bool
  135. @Binding var isLoggedIn: Int
  136. @Binding var userInfo: UserInfo
  137. var body: some View {
  138. VStack {
  139. Form {
  140. List {
  141. HStack {
  142. Image(systemName: "person.fill").foregroundColor(Color(red: 0.7, green: 0.7, blue: 0.7))
  143. TextField(text: $username, prompt: Text("数字校园号")) {
  144. Text("数字校园号")
  145. }.focused($isFocused)
  146. .keyboardType(.numberPad)
  147. }
  148. HStack {
  149. Image(systemName: "key.fill").foregroundColor(Color(red: 0.7, green: 0.7, blue: 0.7))
  150. SecureField(text: $password, prompt: Text("密码")) {
  151. Text("密码")
  152. }.focused($isFocused)
  153. }
  154. }.onTapGesture {
  155. isFocused = false
  156. }
  157. HStack {
  158. Spacer()
  159. Button(action: {
  160. inProgress = true
  161. doLogin(user: username, password: password) { (ret: Int) in
  162. isLoggedIn = ret
  163. inProgress = false
  164. if ret == 0 {
  165. btnText = "登录失败"
  166. btnColor = .red
  167. } else {
  168. getUserInfo(userInfo: &userInfo)
  169. }
  170. }
  171. }) {
  172. HStack {
  173. Text(btnText)
  174. if inProgress {
  175. ProgressView()
  176. } else {
  177. Image(systemName: "chevron.right")
  178. }
  179. }
  180. }.foregroundColor(btnColor).buttonStyle(.bordered)
  181. }
  182. }
  183. }
  184. }
  185. }
  186. struct accountView: View {
  187. @State var username: String = ""
  188. @State var password: String = ""
  189. @FocusState private var isFocused: Bool
  190. @Binding var isLoggedIn: Int
  191. @Binding var userInfo: UserInfo
  192. var body: some View {
  193. VStack {
  194. Form {
  195. Section(header: Text("基本信息")) {
  196. HStack {
  197. Text("姓名")
  198. Spacer()
  199. Text(userInfo.name)
  200. }
  201. HStack {
  202. Text("所属班级")
  203. Spacer()
  204. Text(userInfo.name)
  205. }
  206. }
  207. Section {
  208. Button(action: {
  209. isLoggedIn = 0
  210. if let cookie = LoginHandler().getCookie(name: "CASTGC") {
  211. HTTPCookieStorage.shared.deleteCookie(cookie)
  212. print(cookie)
  213. }
  214. }) {
  215. VStack {
  216. Text("退出").foregroundColor(.red)
  217. }
  218. }
  219. }
  220. }
  221. Spacer()
  222. }
  223. }
  224. }
  225. struct accountView_Previews: PreviewProvider {
  226. @State static var isLoggedIn = 0
  227. @State static var userInfo: UserInfo = .init(name: "", o: "")
  228. static var previews: some View {
  229. loginView(isLoggedIn: $isLoggedIn, userInfo: $userInfo)
  230. }
  231. }