2021-03-10 09:38:14 +01:00
//
// A P I S e r v i c e + P e r s i s t + P e r s i s t M e m o . s w i f t
// M a s t o d o n
//
// C r e a t e d b y M a i n a s u K C i r n o o n 2 0 2 1 - 3 - 1 0 .
//
import os . log
import Foundation
import CoreData
import CoreDataStack
import MastodonSDK
extension APIService . Persist {
class PersistMemo < T , U > {
let status : T
let children : [ PersistMemo < T , U > ]
let memoType : MemoType
let statusProcessType : ProcessType
let authorProcessType : ProcessType
enum MemoType {
case homeTimeline
case mentionTimeline
case userTimeline
case publicTimeline
case likeList
case searchList
case lookUp
case reblog
var flag : String {
switch self {
case . homeTimeline : return " H "
case . mentionTimeline : return " M "
case . userTimeline : return " U "
case . publicTimeline : return " P "
case . likeList : return " L "
case . searchList : return " S "
case . lookUp : return " LU "
case . reblog : return " R "
}
}
}
enum ProcessType {
case create
case merge
var flag : String {
switch self {
case . create : return " + "
case . merge : return " ~ "
}
}
}
init (
status : T ,
children : [ PersistMemo < T , U > ] ,
memoType : MemoType ,
statusProcessType : ProcessType ,
authorProcessType : ProcessType
) {
self . status = status
self . children = children
self . memoType = memoType
self . statusProcessType = statusProcessType
self . authorProcessType = authorProcessType
}
}
}
extension APIService . Persist . PersistMemo {
struct Counting {
var status = Counter ( )
var user = Counter ( )
static func + ( left : Counting , right : Counting ) -> Counting {
return Counting (
status : left . status + right . status ,
user : left . user + right . user
)
}
struct Counter {
var create = 0
var merge = 0
static func + ( left : Counter , right : Counter ) -> Counter {
return Counter (
create : left . create + right . create ,
merge : left . merge + right . merge
)
}
}
}
func count ( ) -> Counting {
var counting = Counting ( )
switch statusProcessType {
case . create : counting . status . create += 1
case . merge : counting . status . merge += 1
}
switch authorProcessType {
case . create : counting . user . create += 1
case . merge : counting . user . merge += 1
}
for child in children {
let childCounting = child . count ( )
counting = counting + childCounting
}
return counting
}
}
2021-04-01 08:39:15 +02:00
extension APIService . Persist . PersistMemo where T = = Status , U = = MastodonUser {
2021-03-10 09:38:14 +01:00
2021-04-01 08:39:15 +02:00
static func createOrMergeStatus (
2021-03-10 09:38:14 +01:00
into managedObjectContext : NSManagedObjectContext ,
for requestMastodonUser : MastodonUser ? ,
requestMastodonUserID : MastodonUser . ID ? ,
domain : String ,
entity : Mastodon . Entity . Status ,
memoType : MemoType ,
2021-04-01 08:39:15 +02:00
statusCache : APIService . Persist . PersistCache < T > ? ,
2021-03-10 09:38:14 +01:00
userCache : APIService . Persist . PersistCache < U > ? ,
networkDate : Date ,
log : OSLog
) -> APIService . Persist . PersistMemo < T , U > {
let processEntityTaskSignpostID = OSSignpostID ( log : log )
2021-04-01 08:39:15 +02:00
os_signpost ( . begin , log : log , name : " update database - process entity: createOrMergeStatus " , signpostID : processEntityTaskSignpostID , " process status %{public}s " , entity . id )
2021-03-10 09:38:14 +01:00
defer {
2021-04-01 08:39:15 +02:00
os_signpost ( . end , log : log , name : " update database - process entity: createOrMergeStatus " , signpostID : processEntityTaskSignpostID , " finish process status %{public}s " , entity . id )
2021-03-10 09:38:14 +01:00
}
// b u i l d t r e e
let reblogMemo = entity . reblog . flatMap { entity -> APIService . Persist . PersistMemo < T , U > in
2021-04-01 08:39:15 +02:00
createOrMergeStatus (
2021-03-10 09:38:14 +01:00
into : managedObjectContext ,
for : requestMastodonUser ,
requestMastodonUserID : requestMastodonUserID ,
domain : domain ,
entity : entity ,
memoType : . reblog ,
2021-04-01 08:39:15 +02:00
statusCache : statusCache ,
2021-03-10 09:38:14 +01:00
userCache : userCache ,
networkDate : networkDate ,
log : log
)
}
let children = [ reblogMemo ] . compactMap { $0 }
2021-04-01 08:39:15 +02:00
let ( status , isStatusCreated , isMastodonUserCreated ) = APIService . CoreData . createOrMergeStatus (
2021-03-10 09:38:14 +01:00
into : managedObjectContext ,
for : requestMastodonUser ,
domain : domain ,
entity : entity ,
2021-04-01 08:39:15 +02:00
statusCache : statusCache ,
2021-03-10 09:38:14 +01:00
userCache : userCache ,
networkDate : networkDate ,
log : log
)
let memo = APIService . Persist . PersistMemo < T , U > (
2021-04-01 08:39:15 +02:00
status : status ,
2021-03-10 09:38:14 +01:00
children : children ,
memoType : memoType ,
2021-04-01 08:39:15 +02:00
statusProcessType : isStatusCreated ? . create : . merge ,
2021-03-10 09:38:14 +01:00
authorProcessType : isMastodonUserCreated ? . create : . merge
)
switch ( memo . statusProcessType , memoType ) {
case ( . create , . homeTimeline ) , ( . merge , . homeTimeline ) :
2021-04-01 08:39:15 +02:00
let timelineIndex = status . homeTimelineIndexes ?
2021-03-10 09:38:14 +01:00
. first { $0 . userID = = requestMastodonUserID }
guard let requestMastodonUserID = requestMastodonUserID else {
assertionFailure ( )
break
}
if timelineIndex = = nil {
// m a k e i t i n d e x e d
let timelineIndexProperty = HomeTimelineIndex . Property ( domain : domain , userID : requestMastodonUserID )
2021-04-01 08:39:15 +02:00
let _ = HomeTimelineIndex . insert ( into : managedObjectContext , property : timelineIndexProperty , status : status )
2021-03-10 09:38:14 +01:00
} else {
// e n i t y a l r e a d y i n h o m e t i m e l i n e
}
case ( . create , . mentionTimeline ) , ( . merge , . mentionTimeline ) :
break
// TODO:
default :
break
}
return memo
}
func log ( indentLevel : Int = 0 ) -> String {
let indent = Array ( repeating : " " , count : indentLevel ) . joined ( )
let preview = status . content . prefix ( 32 ) . replacingOccurrences ( of : " \n " , with : " " )
let message = " \( indent ) [ \( statusProcessType . flag ) \( memoType . flag ) ]( \( status . id ) ) [ \( authorProcessType . flag ) ]( \( status . author . id ) )@ \( status . author . username ) ~> \( preview ) "
var childrenMessages : [ String ] = [ ]
for child in children {
childrenMessages . append ( child . log ( indentLevel : indentLevel + 1 ) )
}
let result = [ [ message ] + childrenMessages ]
. flatMap { $0 }
. joined ( separator : " \n " )
return result
}
}