@@ -2,20 +2,14 @@ import Relator from '../classes/relator';
2
2
import { SerializerOptions } from '../interfaces/serializer.interface' ;
3
3
import { Dictionary } from '../types/global.types' ;
4
4
5
- export async function recurseRelators (
5
+ async function recurseRelatorsDepth (
6
6
data : any [ ] ,
7
7
relators : Record < string , Relator < any > > ,
8
- include : number | string [ ] | undefined ,
8
+ depth : number ,
9
9
keys : string [ ] ,
10
10
relatorDataCache ?: Map < Relator < any > , Dictionary < any > [ ] >
11
11
) {
12
12
const included : any [ ] = [ ] ;
13
- let depth =
14
- typeof include === 'number'
15
- ? include
16
- : Array . isArray ( include )
17
- ? Math . max ( ...include . map ( ( i ) => i . split ( '.' ) . length ) )
18
- : 0 ;
19
13
20
14
let curRelatorDataCache = relatorDataCache || new Map < Relator < any > , Dictionary < any > [ ] > ( ) ;
21
15
@@ -34,40 +28,103 @@ export async function recurseRelators(
34
28
}
35
29
}
36
30
37
- let currentDepth = 0 ;
38
31
while ( depth -- > 0 && curRelatorDataCache . size > 0 ) {
39
32
const newRelatorDataCache = new Map < Relator < any > , Dictionary < any > [ ] > ( ) ;
40
- const includeFields : { field : string | undefined ; hasMore : boolean } [ ] | undefined =
41
- Array . isArray ( include )
42
- ? include
43
- . map ( ( i ) => i . split ( '.' ) )
44
- . filter ( ( i ) => i [ currentDepth ] )
45
- . map ( ( i ) => ( { field : i [ currentDepth ] , hasMore : i . length > currentDepth + 1 } ) )
46
- : undefined ;
47
33
48
34
for ( const [ relator , cache ] of curRelatorDataCache ) {
49
35
for ( let i = 0 ; i < cache . length ; i ++ ) {
50
- const shouldBuildRelatedCache : boolean =
51
- ( ! includeFields ||
52
- includeFields
53
- ?. filter ( ( i ) => i . field === relator . relatedName )
54
- ?. some ( ( i ) => i . hasMore ) ) ??
55
- false ;
56
-
57
36
const resource = await relator . getRelatedResource (
58
37
cache [ i ] ,
59
38
undefined ,
60
39
undefined ,
61
- // Only build the cache for the next iteration if needed.
62
- shouldBuildRelatedCache ? newRelatorDataCache : undefined
40
+ newRelatorDataCache
63
41
) ;
64
42
43
+ const key = resource . getKey ( ) ;
44
+ if ( ! keys . includes ( key ) ) {
45
+ keys . push ( key ) ;
46
+ included . push ( resource ) ;
47
+ }
48
+ }
49
+ }
50
+
51
+ curRelatorDataCache = newRelatorDataCache ;
52
+ }
53
+
54
+ return included ;
55
+ }
56
+
57
+ export async function recurseRelators (
58
+ data : any [ ] ,
59
+ relators : Record < string , Relator < any > > ,
60
+ include : number | string [ ] | undefined ,
61
+ keys : string [ ] ,
62
+ relatorDataCache ?: Map < Relator < any > , Dictionary < any > [ ] >
63
+ ) {
64
+ if ( include === undefined || typeof include === 'number' ) {
65
+ return recurseRelatorsDepth ( data , relators , include ?? 0 , keys , relatorDataCache ) ;
66
+ }
67
+
68
+ const included : any [ ] = [ ] ;
69
+
70
+ let curRelatorDataCache = relatorDataCache || new Map < Relator < any > , Dictionary < any > [ ] > ( ) ;
71
+
72
+ // Required to support backwards compatability where the first dataCache may
73
+ // not be passed in. All subsequent iterations will contain a dataCache
74
+ if ( ! relatorDataCache && include . length > 0 ) {
75
+ for ( const name in relators ) {
76
+ const cache = curRelatorDataCache . get ( relators [ name ] ) || [ ] ;
77
+ curRelatorDataCache . set ( relators [ name ] , cache ) ;
78
+ for ( const datum of data ) {
79
+ const relatedData = await relators [ name ] . getRelatedData ( datum ) ;
80
+ if ( relatedData !== null ) {
81
+ cache . push ( ...( Array . isArray ( relatedData ) ? relatedData : [ relatedData ] ) ) ;
82
+ }
83
+ }
84
+ }
85
+ }
86
+
87
+ const maxDepth = Math . max ( ...include . map ( ( i ) => i . split ( '.' ) . length ) ) ;
88
+
89
+ let currentDepth = 0 ;
90
+ while ( currentDepth < maxDepth ) {
91
+ const newRelatorDataCache = new Map < Relator < any > , Dictionary < any > [ ] > ( ) ;
92
+
93
+ const includeFields : { field : string | undefined ; hasMore : boolean } [ ] | undefined = include
94
+ . map ( ( i ) => i . split ( '.' ) )
95
+ . filter ( ( i ) => i [ currentDepth ] )
96
+ . map ( ( i ) => ( { field : i [ currentDepth ] , hasMore : i . length > currentDepth + 1 } ) )
97
+ . reduce ( ( acc , i ) => {
98
+ const match = acc . find ( ( j ) => j . field === i . field ) ;
99
+ if ( match ) {
100
+ match . hasMore = match . hasMore || i . hasMore ;
101
+ } else {
102
+ acc . push ( i ) ;
103
+ }
104
+ return acc ;
105
+ } , [ ] as { field : string ; hasMore : boolean } [ ] ) ;
106
+
107
+ for ( const [ relator , cache ] of curRelatorDataCache ) {
108
+ const shouldBuildRelatedCache : boolean =
109
+ ( ! includeFields ||
110
+ includeFields ?. filter ( ( i ) => i . field === relator . relatedName ) ?. some ( ( i ) => i . hasMore ) ) ??
111
+ false ;
112
+
113
+ for ( let i = 0 ; i < cache . length ; i ++ ) {
65
114
// Include if,
66
115
// - includeFields !== undefined
67
116
// - includeFields has entry where field = relatedName
68
117
if ( ! includeFields || includeFields . map ( ( i ) => i . field ) . includes ( relator . relatedName ) ) {
69
- const key = resource . getKey ( ) ;
118
+ const key = ` ${ relator . serializer . collectionName } : ${ cache [ i ] . id as string } ` ;
70
119
if ( ! keys . includes ( key ) ) {
120
+ // const key = resource.getKey();
121
+ const resource = await relator . getRelatedResource (
122
+ cache [ i ] ,
123
+ undefined ,
124
+ undefined ,
125
+ // Only build the cache for the next iteration if needed.
126
+ shouldBuildRelatedCache ? newRelatorDataCache : undefined
127
+ ) ;
71
128
keys . push ( key ) ;
72
129
included . push ( resource ) ;
73
130
}
0 commit comments