@@ -22,6 +22,7 @@ function FluxState(props) {
22
22
23
23
export function Kustomizations ( props ) {
24
24
const { capacitorClient, fluxState, targetReference, handleNavigationSelect } = props
25
+ const [ filter , setFilter ] = useState ( false )
25
26
const kustomizations = fluxState . kustomizations ;
26
27
27
28
const sortedKustomizations = useMemo ( ( ) => {
@@ -32,10 +33,17 @@ export function Kustomizations(props){
32
33
return [ ...kustomizations ] . sort ( ( a , b ) => a . metadata . name . localeCompare ( b . metadata . name ) ) ;
33
34
} , [ kustomizations ] ) ;
34
35
36
+ const filteredKustomizations = filterResources ( sortedKustomizations , filter )
37
+
35
38
return (
36
39
< div className = "space-y-4" >
40
+ < button className = { ( filter ? "text-blue-50 bg-blue-600" : "bg-gray-50 text-gray-600" ) + " rounded-full px-3" }
41
+ onClick = { ( ) => setFilter ( ! filter ) }
42
+ >
43
+ Filter errors
44
+ </ button >
37
45
{
38
- sortedKustomizations ?. map ( kustomization =>
46
+ filteredKustomizations ?. map ( kustomization =>
39
47
< Kustomization
40
48
key = { kustomization . metadata . namespace + kustomization . metadata . name }
41
49
capacitorClient = { capacitorClient }
@@ -52,7 +60,7 @@ export function Kustomizations(props){
52
60
53
61
export function HelmReleases ( props ) {
54
62
const { capacitorClient, helmReleases, targetReference, handleNavigationSelect } = props
55
-
63
+ const [ filter , setFilter ] = useState ( false )
56
64
const sortedHelmReleases = useMemo ( ( ) => {
57
65
if ( ! helmReleases ) {
58
66
return null ;
@@ -61,10 +69,17 @@ export function HelmReleases(props) {
61
69
return [ ...helmReleases ] . sort ( ( a , b ) => a . metadata . name . localeCompare ( b . metadata . name ) ) ;
62
70
} , [ helmReleases ] ) ;
63
71
72
+ const filteredHelmReleases = filterResources ( sortedHelmReleases , filter )
73
+
64
74
return (
65
75
< div className = "space-y-4" >
76
+ < button className = { ( filter ? "text-blue-50 bg-blue-600" : "bg-gray-50 text-gray-600" ) + " rounded-full px-3" }
77
+ onClick = { ( ) => setFilter ( ! filter ) }
78
+ >
79
+ Filter errors
80
+ </ button >
66
81
{
67
- sortedHelmReleases ?. map ( helmRelease =>
82
+ filteredHelmReleases ?. map ( helmRelease =>
68
83
< HelmRelease
69
84
key = { "hr-" + helmRelease . metadata . namespace + helmRelease . metadata . name }
70
85
capacitorClient = { capacitorClient }
@@ -122,7 +137,7 @@ function HelmRelease(props) {
122
137
123
138
export function Sources ( props ) {
124
139
const { capacitorClient, fluxState, targetReference } = props
125
-
140
+ const [ filter , setFilter ] = useState ( false )
126
141
const sortedSources = useMemo ( ( ) => {
127
142
const sources = [ ] ;
128
143
if ( fluxState . ociRepositories ) {
@@ -133,10 +148,17 @@ export function Sources(props){
133
148
return [ ...sources ] . sort ( ( a , b ) => a . metadata . name . localeCompare ( b . metadata . name ) ) ;
134
149
} , [ fluxState ] ) ;
135
150
151
+ const filteredSources = filterResources ( sortedSources , filter )
152
+
136
153
return (
137
154
< div className = "space-y-4" >
155
+ < button className = { ( filter ? "text-blue-50 bg-blue-600" : "bg-gray-50 text-gray-600" ) + " rounded-full px-3" }
156
+ onClick = { ( ) => setFilter ( ! filter ) }
157
+ >
158
+ Filter errors
159
+ </ button >
138
160
{
139
- sortedSources ?. map ( source =>
161
+ filteredSources ?. map ( source =>
140
162
< Source
141
163
key = { "source-" + source . metadata . namespace + source . metadata . name }
142
164
capacitorClient = { capacitorClient }
@@ -149,6 +171,45 @@ export function Sources(props){
149
171
)
150
172
}
151
173
174
+ function filterResources ( resources , filterErrors ) {
175
+ let filteredResources = resources ;
176
+ if ( filterErrors ) {
177
+ filteredResources = filteredResources . filter ( resource => {
178
+ const readyConditions = jp . query ( resource . status , '$..conditions[?(@.type=="Ready")]' ) ;
179
+ const readyCondition = readyConditions . length === 1 ? readyConditions [ 0 ] : undefined
180
+ const ready = readyCondition && readyConditions [ 0 ] . status === "True"
181
+
182
+ const dependencyNotReady = readyCondition && readyCondition . reason === "DependencyNotReady"
183
+
184
+ const readyTransitionTime = readyCondition ? readyCondition . lastTransitionTime : undefined
185
+ const parsed = Date . parse ( readyTransitionTime , "yyyy-MM-dd'T'HH:mm:ss" ) ;
186
+ const fiveMinutesAgo = new Date ( ) ;
187
+ fiveMinutesAgo . setMinutes ( fiveMinutesAgo . getMinutes ( ) - 5 ) ;
188
+ const stalled = fiveMinutesAgo > parsed
189
+
190
+ const reconcilingConditions = jp . query ( resource . status , '$..conditions[?(@.type=="Reconciling")]' ) ;
191
+ const reconcilingCondition = reconcilingConditions . length === 1 ? reconcilingConditions [ 0 ] : undefined
192
+ const reconciling = reconcilingCondition && reconcilingCondition . status === "True"
193
+
194
+ const fetchFailedConditions = jp . query ( resource . status , '$..conditions[?(@.type=="FetchFailed")]' ) ;
195
+ const fetchFailedCondition = fetchFailedConditions . length === 1 ? fetchFailedConditions [ 0 ] : undefined
196
+ const fetchFailed = fetchFailedCondition && fetchFailedCondition . status === "True"
197
+
198
+ if ( resource . kind === 'GitRepository' || resource . kind === "OCIRepository" || resource . kind === "Bucket" ) {
199
+ return fetchFailed
200
+ }
201
+
202
+ if ( ready || ( ( reconciling || dependencyNotReady ) && ! stalled ) ) {
203
+ return false ;
204
+ } else {
205
+ return true ;
206
+ }
207
+ } )
208
+ }
209
+
210
+ return filteredResources ;
211
+ }
212
+
152
213
function Source ( props ) {
153
214
const { capacitorClient, source, targetReference } = props ;
154
215
const ref = useRef ( null ) ;
0 commit comments