From 755bcf594372c97bd712faafe4c2a48baaf3aaea Mon Sep 17 00:00:00 2001 From: Max Cao Date: Fri, 5 Aug 2022 17:02:25 -0400 Subject: [PATCH 1/4] select target to none when notification LOST --- src/app/Shared/MatchExpressionEvaluator.tsx | 2 +- src/app/TargetSelect/TargetSelect.tsx | 34 +++++++++++++++++---- 2 files changed, 29 insertions(+), 7 deletions(-) diff --git a/src/app/Shared/MatchExpressionEvaluator.tsx b/src/app/Shared/MatchExpressionEvaluator.tsx index e8e1c85fb..c9761625b 100644 --- a/src/app/Shared/MatchExpressionEvaluator.tsx +++ b/src/app/Shared/MatchExpressionEvaluator.tsx @@ -60,7 +60,7 @@ export const MatchExpressionEvaluator: React.FunctionComponent { if (!props.matchExpression || !target?.connectUrl) { diff --git a/src/app/TargetSelect/TargetSelect.tsx b/src/app/TargetSelect/TargetSelect.tsx index bdd65a8a7..ee08cccf4 100644 --- a/src/app/TargetSelect/TargetSelect.tsx +++ b/src/app/TargetSelect/TargetSelect.tsx @@ -47,6 +47,7 @@ import { catchError, first } from 'rxjs/operators'; import { CreateTargetModal } from './CreateTargetModal'; import _ from 'lodash'; +import { NotificationCategory } from '@app/Shared/Services/NotificationChannel.service'; export interface TargetSelectProps { } @@ -63,13 +64,34 @@ export const TargetSelect: React.FunctionComponent = (props) const addSubscription = useSubscriptions(); React.useEffect(() => { - const sub = context.targets.targets().subscribe((targets) => { - setTargets(targets); - selectTargetFromCache(targets); - }); - return () => sub.unsubscribe(); + addSubscription( + context.targets.targets().subscribe((targets) => { + setTargets(targets); + selectTargetFromCache(targets); + }) + ); }, [context, context.targets, setTargets]); + React.useEffect(() => { + addSubscription( + context.notificationChannel.messages(NotificationCategory.TargetJvmDiscovery) + .subscribe(v => { + const evt: TargetDiscoveryEvent = v.message.event; + if (evt.kind === 'LOST') { + const target: Target = { + connectUrl: evt.serviceRef.connectUrl, + alias: evt.serviceRef.alias, + } + context.target.target().subscribe((currentTarget) => { + if (currentTarget.connectUrl == target.connectUrl && currentTarget.alias == target.alias) { + selectNone(); + } + }) + } + }) + ); + }, [addSubscription, context, context.notificationChannel, context.target]); + const refreshTargetList = React.useCallback(() => { setLoading(true); addSubscription( @@ -126,7 +148,7 @@ export const TargetSelect: React.FunctionComponent = (props) } } setExpanded(false); - }, [context, selected, notifications, setExpanded]); + }, [context, context.target, selected, notifications, setExpanded, setCachedTargetSelection, removeCachedTargetSelection]); const selectNone = React.useCallback(() => { onSelect(undefined, undefined, true); From a8b9ba9f2e7150199ded4e50a1e49636fc54ac55 Mon Sep 17 00:00:00 2001 From: Max Cao Date: Fri, 5 Aug 2022 17:29:48 -0400 Subject: [PATCH 2/4] fixed tests --- src/test/Rules/CreateRule.test.tsx | 14 ++++++++++++++ 1 file changed, 14 insertions(+) diff --git a/src/test/Rules/CreateRule.test.tsx b/src/test/Rules/CreateRule.test.tsx index 8d6876696..f4b3acb25 100644 --- a/src/test/Rules/CreateRule.test.tsx +++ b/src/test/Rules/CreateRule.test.tsx @@ -48,6 +48,7 @@ import { ServiceContext, defaultServices } from '@app/Shared/Services/Services'; import { CreateRule } from '@app/Rules/CreateRule'; import { EventTemplate } from '@app/CreateRecording/CreateRecording'; import { Target } from '@app/Shared/Services/Target.service'; +import { NotificationMessage } from '@app/Shared/Services/NotificationChannel.service'; const escapeKeyboardInput = (value: string) => { return value.replace(/[{[]/g, '$&$&'); @@ -82,6 +83,18 @@ const mockRule: Rule = { maxSizeBytes: 0 }; +const mockNewTarget: Target = { + connectUrl: 'service:jmx:rmi://someUrl1', + alias: 'someAlias', +} + +const mockTargetFoundNotification = { + message: { + event: { kind: 'FOUND', serviceRef: mockNewTarget } + } +} as NotificationMessage; + + const history = createMemoryHistory(); jest.mock('react-router-dom', () => ({ @@ -91,6 +104,7 @@ jest.mock('react-router-dom', () => ({ })); const createSpy = jest.spyOn(defaultServices.api, 'createRule').mockReturnValue(of(true)); +jest.spyOn(defaultServices.notificationChannel, 'messages').mockReturnValue(of(mockTargetFoundNotification)); jest.spyOn(defaultServices.api, 'doGet').mockReturnValue(of([mockEventTemplate])); jest.spyOn(defaultServices.target, 'target').mockReturnValue(of(mockTarget)); jest.spyOn(defaultServices.targets, 'targets').mockReturnValue(of([mockTarget])); From 0ebc0f80fae926d3886f1ef0d1858472f66b7ed3 Mon Sep 17 00:00:00 2001 From: Max Cao Date: Fri, 5 Aug 2022 17:39:26 -0400 Subject: [PATCH 3/4] fixed callback dependencies: --- src/app/TargetSelect/TargetSelect.tsx | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/src/app/TargetSelect/TargetSelect.tsx b/src/app/TargetSelect/TargetSelect.tsx index ee08cccf4..2007a12eb 100644 --- a/src/app/TargetSelect/TargetSelect.tsx +++ b/src/app/TargetSelect/TargetSelect.tsx @@ -148,7 +148,7 @@ export const TargetSelect: React.FunctionComponent = (props) } } setExpanded(false); - }, [context, context.target, selected, notifications, setExpanded, setCachedTargetSelection, removeCachedTargetSelection]); + }, [context, context.target, selected, notifications, setExpanded]); const selectNone = React.useCallback(() => { onSelect(undefined, undefined, true); From 132730382a252d70156debaebb44821868fae6ad Mon Sep 17 00:00:00 2001 From: Max Cao Date: Fri, 5 Aug 2022 17:48:41 -0400 Subject: [PATCH 4/4] fixed unclosed subscription --- src/app/TargetSelect/TargetSelect.tsx | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/src/app/TargetSelect/TargetSelect.tsx b/src/app/TargetSelect/TargetSelect.tsx index 2007a12eb..0212a2e33 100644 --- a/src/app/TargetSelect/TargetSelect.tsx +++ b/src/app/TargetSelect/TargetSelect.tsx @@ -82,8 +82,8 @@ export const TargetSelect: React.FunctionComponent = (props) connectUrl: evt.serviceRef.connectUrl, alias: evt.serviceRef.alias, } - context.target.target().subscribe((currentTarget) => { - if (currentTarget.connectUrl == target.connectUrl && currentTarget.alias == target.alias) { + context.target.target().pipe(first()).subscribe((currentTarget) => { + if (currentTarget.connectUrl === target.connectUrl && currentTarget.alias === target.alias) { selectNone(); } })