2020-02-19 23:00:23 +00:00
|
|
|
//
|
|
|
|
// LiveUser.swift
|
2022-08-07 13:31:15 +01:00
|
|
|
// Mixonomer
|
2020-02-19 23:00:23 +00:00
|
|
|
//
|
|
|
|
// Created by Andy Pack on 19/02/2020.
|
|
|
|
// Copyright © 2020 Sarsoo. All rights reserved.
|
|
|
|
//
|
|
|
|
|
|
|
|
import Foundation
|
2020-03-06 23:36:51 +00:00
|
|
|
import Alamofire
|
|
|
|
import SwiftyJSON
|
2022-08-11 20:53:09 +01:00
|
|
|
import KeychainAccess
|
2020-02-19 23:00:23 +00:00
|
|
|
|
|
|
|
class LiveUser: ObservableObject {
|
2020-02-22 13:28:15 +00:00
|
|
|
|
|
|
|
@Published var playlists: [Playlist]
|
|
|
|
@Published var tags: [Tag]
|
2020-03-03 00:04:20 +00:00
|
|
|
@Published var username: String
|
2022-08-11 20:15:21 +01:00
|
|
|
@Published var user: User?
|
2020-02-19 23:00:23 +00:00
|
|
|
|
2020-04-25 15:44:54 +01:00
|
|
|
@Published var loggedIn: Bool {
|
|
|
|
didSet {
|
|
|
|
UserDefaults.standard.set(loggedIn, forKey: "loggedIn")
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-08-11 20:15:21 +01:00
|
|
|
@Published var isRefreshingUser = false
|
2020-03-07 18:51:52 +00:00
|
|
|
@Published var isRefreshingPlaylists = false
|
|
|
|
@Published var isRefreshingTags = false
|
|
|
|
|
2020-04-25 15:44:54 +01:00
|
|
|
init(playlists: [Playlist], tags: [Tag], username: String, loggedIn: Bool) {
|
2020-02-19 23:00:23 +00:00
|
|
|
self.playlists = playlists
|
2020-02-20 18:25:28 +00:00
|
|
|
self.tags = tags
|
2020-03-03 00:04:20 +00:00
|
|
|
self.username = username
|
2020-04-25 15:44:54 +01:00
|
|
|
self.loggedIn = loggedIn
|
2020-02-19 23:00:23 +00:00
|
|
|
}
|
2020-02-22 13:28:15 +00:00
|
|
|
|
2022-08-11 20:53:09 +01:00
|
|
|
init(playlists: [Playlist], tags: [Tag], username: String, loggedIn: Bool, user: User) {
|
|
|
|
self.playlists = playlists
|
|
|
|
self.tags = tags
|
|
|
|
self.username = username
|
|
|
|
self.loggedIn = loggedIn
|
|
|
|
self.user = user
|
|
|
|
}
|
|
|
|
|
|
|
|
func logout() {
|
|
|
|
let keychain = Keychain(service: "xyz.sarsoo.music.login")
|
|
|
|
|
|
|
|
do {
|
|
|
|
try keychain.remove("username")
|
|
|
|
try keychain.remove("jwt")
|
|
|
|
|
2022-08-11 23:13:51 +01:00
|
|
|
playlists.removeAll()
|
|
|
|
tags.removeAll()
|
|
|
|
username = ""
|
|
|
|
user = nil
|
|
|
|
|
2022-08-11 20:53:09 +01:00
|
|
|
UserDefaults.standard.removeObject(forKey: "playlists")
|
|
|
|
UserDefaults.standard.removeObject(forKey: "tags")
|
|
|
|
|
|
|
|
self.loggedIn = false
|
|
|
|
|
|
|
|
} catch let error {
|
|
|
|
debugPrint("Could not clear keychain, \(error)")
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-02-22 13:28:15 +00:00
|
|
|
func updatePlaylist(playlistIn: Playlist) {
|
|
|
|
guard let index = self.playlists.firstIndex(of: playlistIn) else {
|
|
|
|
fatalError("\(playlistIn) not found")
|
|
|
|
}
|
|
|
|
self.playlists[index] = playlistIn
|
|
|
|
}
|
2020-03-06 23:36:51 +00:00
|
|
|
|
2022-08-11 20:15:21 +01:00
|
|
|
func refreshUser(onSuccess: (() -> Void)? = nil, onFailure: (() -> Void)? = nil) {
|
|
|
|
self.isRefreshingUser = true
|
|
|
|
|
|
|
|
let api = UserApi.getUser
|
|
|
|
RequestBuilder.buildRequest(apiRequest: api).responseJSON{ response in
|
|
|
|
|
|
|
|
if self.checkNetworkResponse(response: response) {
|
|
|
|
|
|
|
|
guard let data = response.data else {
|
|
|
|
fatalError("error getting user")
|
|
|
|
}
|
|
|
|
|
|
|
|
guard let json = try? JSON(data: data) else {
|
|
|
|
fatalError("error parsing user")
|
|
|
|
}
|
|
|
|
|
|
|
|
// update state
|
|
|
|
self.user = UserApi.fromJSON(user: json)
|
|
|
|
|
|
|
|
self.isRefreshingUser = false
|
|
|
|
|
|
|
|
if let success = onSuccess {
|
|
|
|
success()
|
|
|
|
}
|
|
|
|
|
|
|
|
} else {
|
|
|
|
|
|
|
|
if let failure = onFailure {
|
|
|
|
failure()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func refreshPlaylists(onSuccess: (() -> Void)? = nil, onFailure: (() -> Void)? = nil) {
|
2020-03-07 18:51:52 +00:00
|
|
|
self.isRefreshingPlaylists = true
|
|
|
|
|
2020-03-06 23:36:51 +00:00
|
|
|
let api = PlaylistApi.getPlaylists
|
|
|
|
RequestBuilder.buildRequest(apiRequest: api).responseJSON{ response in
|
2022-08-08 21:59:28 +01:00
|
|
|
|
2022-08-11 20:15:21 +01:00
|
|
|
if self.checkNetworkResponse(response: response) {
|
|
|
|
|
2022-08-08 21:59:28 +01:00
|
|
|
guard let data = response.data else {
|
|
|
|
fatalError("error getting playlists")
|
|
|
|
}
|
2020-03-06 23:36:51 +00:00
|
|
|
|
2022-08-08 21:59:28 +01:00
|
|
|
guard let json = try? JSON(data: data) else {
|
|
|
|
fatalError("error parsing reponse")
|
|
|
|
}
|
|
|
|
|
|
|
|
let playlists = json["playlists"].arrayValue
|
2020-03-06 23:36:51 +00:00
|
|
|
|
2022-08-08 21:59:28 +01:00
|
|
|
// update state
|
|
|
|
self.playlists = PlaylistApi.fromJSON(playlist: playlists).sorted(by: { $0.name.lowercased() < $1.name.lowercased() })
|
|
|
|
|
|
|
|
self.isRefreshingPlaylists = false
|
|
|
|
|
2022-08-11 20:15:21 +01:00
|
|
|
if let success = onSuccess {
|
|
|
|
success()
|
|
|
|
}
|
|
|
|
|
2022-08-08 21:59:28 +01:00
|
|
|
let encoder = JSONEncoder()
|
|
|
|
do {
|
|
|
|
UserDefaults.standard.set(String(data: try encoder.encode(playlists), encoding: .utf8), forKey: "playlists")
|
|
|
|
} catch {
|
|
|
|
print("error encoding playlists: \(error)")
|
|
|
|
}
|
|
|
|
|
2022-08-11 20:15:21 +01:00
|
|
|
} else {
|
|
|
|
|
|
|
|
if let failure = onFailure {
|
|
|
|
failure()
|
|
|
|
}
|
2020-03-06 23:36:51 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-08-11 20:15:21 +01:00
|
|
|
func refreshTags(onSuccess: (() -> Void)? = nil, onFailure: (() -> Void)? = nil) {
|
2020-03-07 18:51:52 +00:00
|
|
|
self.isRefreshingTags = true
|
|
|
|
|
2020-03-06 23:36:51 +00:00
|
|
|
let api = TagApi.getTags
|
|
|
|
RequestBuilder.buildRequest(apiRequest: api).responseJSON{ response in
|
2022-08-09 17:42:01 +01:00
|
|
|
|
2022-08-11 20:15:21 +01:00
|
|
|
if self.checkNetworkResponse(response: response) {
|
|
|
|
|
2022-08-09 17:42:01 +01:00
|
|
|
guard let data = response.data else {
|
|
|
|
fatalError("error getting tags")
|
|
|
|
}
|
2020-03-06 23:36:51 +00:00
|
|
|
|
2022-08-09 17:42:01 +01:00
|
|
|
guard let json = try? JSON(data: data) else {
|
|
|
|
fatalError("error parsing reponse")
|
|
|
|
}
|
|
|
|
|
|
|
|
let tags = json["tags"].arrayValue
|
2020-03-06 23:36:51 +00:00
|
|
|
|
2022-08-09 17:42:01 +01:00
|
|
|
// update state
|
|
|
|
self.tags = TagApi.fromJSON(tag: tags).sorted(by: { $0.name.lowercased() < $1.name.lowercased() })
|
|
|
|
|
|
|
|
self.isRefreshingTags = false
|
|
|
|
|
2022-08-11 20:15:21 +01:00
|
|
|
if let success = onSuccess {
|
|
|
|
success()
|
|
|
|
}
|
|
|
|
|
2022-08-09 17:42:01 +01:00
|
|
|
let encoder = JSONEncoder()
|
|
|
|
do {
|
|
|
|
UserDefaults.standard.set(String(data: try encoder.encode(tags), encoding: .utf8), forKey: "tags")
|
|
|
|
} catch {
|
|
|
|
print("error encoding tags: \(error)")
|
|
|
|
}
|
|
|
|
|
2022-08-11 20:15:21 +01:00
|
|
|
} else {
|
|
|
|
|
|
|
|
if let failure = onFailure {
|
|
|
|
failure()
|
|
|
|
}
|
2020-03-06 23:36:51 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-08-11 20:15:21 +01:00
|
|
|
func checkNetworkResponse(response: AFDataResponse<Any>) -> Bool {
|
2022-08-09 17:42:01 +01:00
|
|
|
|
2022-08-11 20:15:21 +01:00
|
|
|
if let statusCode = response.response?.statusCode {
|
|
|
|
switch statusCode {
|
|
|
|
case 401: // token has expired
|
2022-08-11 20:53:09 +01:00
|
|
|
self.logout()
|
2022-08-11 20:15:21 +01:00
|
|
|
return false
|
|
|
|
case 400..<500:
|
|
|
|
return false
|
|
|
|
case 500..<600:
|
|
|
|
return false
|
|
|
|
case _: // 200 -> Success
|
|
|
|
return true
|
|
|
|
}
|
2022-08-09 17:42:01 +01:00
|
|
|
}
|
2022-08-11 20:15:21 +01:00
|
|
|
|
|
|
|
return false
|
2022-08-09 17:42:01 +01:00
|
|
|
}
|
|
|
|
|
2020-03-06 23:36:51 +00:00
|
|
|
func loadUserDefaults() -> LiveUser {
|
|
|
|
let defaults = UserDefaults.standard
|
|
|
|
let decoder = JSONDecoder()
|
|
|
|
|
|
|
|
let _strPlaylists = defaults.string(forKey: "playlists")
|
|
|
|
let _strTags = defaults.string(forKey: "tags")
|
2020-04-25 15:44:54 +01:00
|
|
|
loggedIn = defaults.bool(forKey: "loggedIn")
|
2020-03-06 23:36:51 +00:00
|
|
|
|
|
|
|
do {
|
|
|
|
if let _strPlaylists = _strPlaylists {
|
2020-03-20 00:08:18 +00:00
|
|
|
if _strPlaylists.count > 0 {
|
|
|
|
self.playlists = (try decoder.decode([Playlist].self, from: _strPlaylists.data(using: .utf8)!)).sorted(by: { $0.name.lowercased() < $1.name.lowercased() })
|
|
|
|
}
|
2020-03-06 23:36:51 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if let _strTags = _strTags {
|
2020-03-20 00:08:18 +00:00
|
|
|
if _strTags.count > 0 {
|
|
|
|
self.tags = (try decoder.decode([Tag].self, from: _strTags.data(using: .utf8)!)).sorted(by: { $0.name.lowercased() < $1.name.lowercased() })
|
|
|
|
}
|
2020-03-06 23:36:51 +00:00
|
|
|
}
|
|
|
|
} catch {
|
|
|
|
print("error decoding: \(error)")
|
|
|
|
}
|
|
|
|
|
|
|
|
return self
|
|
|
|
}
|
2020-02-19 23:00:23 +00:00
|
|
|
}
|