mirror of https://github.com/apache/nifi.git
[NIFI-12761] refresh RPG (#8401)
* [NIFI-12761] refresh RPG * address feedback * stop polling on error * fix linting errors introduced in this PR * reload the connections * refresh rpg, if first call does not have updated timestamp start polling * rename action * final touches This closes #8401
This commit is contained in:
parent
6f6ddf8960
commit
3c5b997ce7
|
@ -37,6 +37,7 @@ import {
|
||||||
navigateToViewStatusHistoryForComponent,
|
navigateToViewStatusHistoryForComponent,
|
||||||
reloadFlow,
|
reloadFlow,
|
||||||
replayLastProvenanceEvent,
|
replayLastProvenanceEvent,
|
||||||
|
requestRefreshRemoteProcessGroup,
|
||||||
runOnce,
|
runOnce,
|
||||||
startComponents,
|
startComponents,
|
||||||
startCurrentProcessGroup,
|
startCurrentProcessGroup,
|
||||||
|
@ -750,12 +751,19 @@ export class CanvasContextMenu implements ContextMenuDefinitionProvider {
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
condition: (selection: any) => {
|
condition: (selection: any) => {
|
||||||
return this.canvasUtils.isRemoteProcessGroup(selection);
|
return this.canvasUtils.canRead(selection) && this.canvasUtils.isRemoteProcessGroup(selection);
|
||||||
},
|
},
|
||||||
clazz: 'fa fa-refresh',
|
clazz: 'fa fa-refresh',
|
||||||
text: 'Refresh remote',
|
text: 'Refresh remote',
|
||||||
action: () => {
|
action: (selection: any) => {
|
||||||
// TODO - refreshRemoteFlow
|
const d = selection.datum();
|
||||||
|
const id = d.id;
|
||||||
|
const refreshTimestamp = d.component.flowRefreshed;
|
||||||
|
const request = {
|
||||||
|
id,
|
||||||
|
refreshTimestamp
|
||||||
|
};
|
||||||
|
this.store.dispatch(requestRefreshRemoteProcessGroup({ request }));
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
|
@ -964,17 +972,15 @@ export class CanvasContextMenu implements ContextMenuDefinitionProvider {
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
condition: (selection: any) => {
|
condition: (selection: any) => {
|
||||||
return this.canvasUtils.isRemoteProcessGroup(selection);
|
return this.canvasUtils.canRead(selection) && this.canvasUtils.isRemoteProcessGroup(selection);
|
||||||
},
|
},
|
||||||
clazz: 'fa fa-external-link',
|
clazz: 'fa fa-external-link',
|
||||||
text: 'Go to',
|
text: 'Go to',
|
||||||
action: (selection: any) => {
|
action: (selection: any) => {
|
||||||
if (selection.size() === 1 && this.canvasUtils.isRemoteProcessGroup(selection)) {
|
const selectionData = selection.datum();
|
||||||
const selectionData = selection.datum();
|
const uri = selectionData.component.targetUri;
|
||||||
const uri = selectionData.component.targetUri;
|
|
||||||
|
|
||||||
this.store.dispatch(goToRemoteProcessGroup({ request: { uri } }));
|
this.store.dispatch(goToRemoteProcessGroup({ request: { uri } }));
|
||||||
}
|
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
|
|
|
@ -75,7 +75,8 @@ import {
|
||||||
CenterComponentRequest,
|
CenterComponentRequest,
|
||||||
ImportFromRegistryDialogRequest,
|
ImportFromRegistryDialogRequest,
|
||||||
ImportFromRegistryRequest,
|
ImportFromRegistryRequest,
|
||||||
GoToRemoteProcessGroupRequest
|
GoToRemoteProcessGroupRequest,
|
||||||
|
RefreshRemoteProcessGroupRequest
|
||||||
} from './index';
|
} from './index';
|
||||||
import { StatusHistoryRequest } from '../../../../state/status-history';
|
import { StatusHistoryRequest } from '../../../../state/status-history';
|
||||||
|
|
||||||
|
@ -232,6 +233,17 @@ export const goToRemoteProcessGroup = createAction(
|
||||||
props<{ request: GoToRemoteProcessGroupRequest }>()
|
props<{ request: GoToRemoteProcessGroupRequest }>()
|
||||||
);
|
);
|
||||||
|
|
||||||
|
export const refreshRemoteProcessGroup = createAction(`${CANVAS_PREFIX} Refresh Remote Process Group`);
|
||||||
|
|
||||||
|
export const requestRefreshRemoteProcessGroup = createAction(
|
||||||
|
`${CANVAS_PREFIX} Request Refresh Remote Process Group Polling`,
|
||||||
|
props<{ request: RefreshRemoteProcessGroupRequest }>()
|
||||||
|
);
|
||||||
|
|
||||||
|
export const startRemoteProcessGroupPolling = createAction(`${CANVAS_PREFIX} Start Remote Process Group Polling`);
|
||||||
|
|
||||||
|
export const stopRemoteProcessGroupPolling = createAction(`${CANVAS_PREFIX} Stop Remote Process Group Polling`);
|
||||||
|
|
||||||
export const createProcessGroup = createAction(
|
export const createProcessGroup = createAction(
|
||||||
`${CANVAS_PREFIX} Create Process Group`,
|
`${CANVAS_PREFIX} Create Process Group`,
|
||||||
props<{ request: CreateProcessGroupRequest }>()
|
props<{ request: CreateProcessGroupRequest }>()
|
||||||
|
|
|
@ -58,6 +58,7 @@ import {
|
||||||
selectProcessGroup,
|
selectProcessGroup,
|
||||||
selectProcessor,
|
selectProcessor,
|
||||||
selectRemoteProcessGroup,
|
selectRemoteProcessGroup,
|
||||||
|
selectRefreshRpgDetails,
|
||||||
selectSaving
|
selectSaving
|
||||||
} from './flow.selectors';
|
} from './flow.selectors';
|
||||||
import { ConnectionManager } from '../../service/manager/connection-manager.service';
|
import { ConnectionManager } from '../../service/manager/connection-manager.service';
|
||||||
|
@ -67,6 +68,7 @@ import { EditPort } from '../../ui/canvas/items/port/edit-port/edit-port.compone
|
||||||
import {
|
import {
|
||||||
BucketEntity,
|
BucketEntity,
|
||||||
ComponentType,
|
ComponentType,
|
||||||
|
isDefinedAndNotNull,
|
||||||
RegistryClientEntity,
|
RegistryClientEntity,
|
||||||
VersionedFlowEntity,
|
VersionedFlowEntity,
|
||||||
VersionedFlowSnapshotMetadataEntity
|
VersionedFlowSnapshotMetadataEntity
|
||||||
|
@ -96,6 +98,7 @@ import { ImportFromRegistry } from '../../ui/canvas/items/flow/import-from-regis
|
||||||
import { selectCurrentUser } from '../../../../state/current-user/current-user.selectors';
|
import { selectCurrentUser } from '../../../../state/current-user/current-user.selectors';
|
||||||
import { NoRegistryClientsDialog } from '../../ui/common/no-registry-clients-dialog/no-registry-clients-dialog.component';
|
import { NoRegistryClientsDialog } from '../../ui/common/no-registry-clients-dialog/no-registry-clients-dialog.component';
|
||||||
import { EditRemoteProcessGroup } from '../../ui/canvas/items/remote-process-group/edit-remote-process-group/edit-remote-process-group.component';
|
import { EditRemoteProcessGroup } from '../../ui/canvas/items/remote-process-group/edit-remote-process-group/edit-remote-process-group.component';
|
||||||
|
import { ErrorHelper } from '../../../../service/error-helper.service';
|
||||||
|
|
||||||
@Injectable()
|
@Injectable()
|
||||||
export class FlowEffects {
|
export class FlowEffects {
|
||||||
|
@ -359,6 +362,70 @@ export class FlowEffects {
|
||||||
{ dispatch: false }
|
{ dispatch: false }
|
||||||
);
|
);
|
||||||
|
|
||||||
|
startRemoteProcessGroupPolling$ = createEffect(() =>
|
||||||
|
this.actions$.pipe(
|
||||||
|
ofType(FlowActions.startRemoteProcessGroupPolling),
|
||||||
|
switchMap(() => {
|
||||||
|
return interval(3000, asyncScheduler).pipe(
|
||||||
|
takeUntil(this.actions$.pipe(ofType(FlowActions.stopRemoteProcessGroupPolling)))
|
||||||
|
);
|
||||||
|
}),
|
||||||
|
switchMap(() => {
|
||||||
|
return of(FlowActions.refreshRemoteProcessGroup());
|
||||||
|
})
|
||||||
|
)
|
||||||
|
);
|
||||||
|
|
||||||
|
requestRefreshRemoteProcessGroup$ = createEffect(() =>
|
||||||
|
this.actions$.pipe(
|
||||||
|
ofType(FlowActions.requestRefreshRemoteProcessGroup),
|
||||||
|
switchMap(() => {
|
||||||
|
return of(FlowActions.refreshRemoteProcessGroup());
|
||||||
|
})
|
||||||
|
)
|
||||||
|
);
|
||||||
|
|
||||||
|
refreshRemoteProcessGroup$ = createEffect(() =>
|
||||||
|
this.actions$.pipe(
|
||||||
|
ofType(FlowActions.refreshRemoteProcessGroup),
|
||||||
|
concatLatestFrom(() => this.store.select(selectRefreshRpgDetails).pipe(isDefinedAndNotNull())),
|
||||||
|
switchMap(([, refreshRpgDetails]) =>
|
||||||
|
from(
|
||||||
|
this.flowService.getRemoteProcessGroup(refreshRpgDetails.request.id).pipe(
|
||||||
|
map((response: any) => {
|
||||||
|
const entity = response;
|
||||||
|
|
||||||
|
if (refreshRpgDetails.request.refreshTimestamp !== entity.component.flowRefreshed) {
|
||||||
|
this.store.dispatch(FlowActions.stopRemoteProcessGroupPolling());
|
||||||
|
|
||||||
|
// reload the group's connections
|
||||||
|
this.store.dispatch(FlowActions.loadConnectionsForComponent({ id: entity.id }));
|
||||||
|
} else {
|
||||||
|
if (!refreshRpgDetails.polling) {
|
||||||
|
this.store.dispatch(FlowActions.startRemoteProcessGroupPolling());
|
||||||
|
}
|
||||||
|
|
||||||
|
entity.component.flowRefreshed = 'Refreshing...';
|
||||||
|
}
|
||||||
|
|
||||||
|
return FlowActions.loadRemoteProcessGroupSuccess({
|
||||||
|
response: {
|
||||||
|
id: entity.id,
|
||||||
|
remoteProcessGroup: entity
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}),
|
||||||
|
catchError((error) => {
|
||||||
|
this.store.dispatch(FlowActions.stopRemoteProcessGroupPolling());
|
||||||
|
|
||||||
|
return of(FlowActions.flowApiError({ error: error.error }));
|
||||||
|
})
|
||||||
|
)
|
||||||
|
)
|
||||||
|
)
|
||||||
|
)
|
||||||
|
);
|
||||||
|
|
||||||
openNewProcessGroupDialog$ = createEffect(
|
openNewProcessGroupDialog$ = createEffect(
|
||||||
() =>
|
() =>
|
||||||
this.actions$.pipe(
|
this.actions$.pipe(
|
||||||
|
|
|
@ -38,6 +38,7 @@ import {
|
||||||
loadProcessorSuccess,
|
loadProcessorSuccess,
|
||||||
loadRemoteProcessGroupSuccess,
|
loadRemoteProcessGroupSuccess,
|
||||||
navigateWithoutTransform,
|
navigateWithoutTransform,
|
||||||
|
requestRefreshRemoteProcessGroup,
|
||||||
resetFlowState,
|
resetFlowState,
|
||||||
runOnce,
|
runOnce,
|
||||||
runOnceSuccess,
|
runOnceSuccess,
|
||||||
|
@ -49,7 +50,9 @@ import {
|
||||||
setTransitionRequired,
|
setTransitionRequired,
|
||||||
startComponent,
|
startComponent,
|
||||||
startComponentSuccess,
|
startComponentSuccess,
|
||||||
|
startRemoteProcessGroupPolling,
|
||||||
stopComponentSuccess,
|
stopComponentSuccess,
|
||||||
|
stopRemoteProcessGroupPolling,
|
||||||
updateComponent,
|
updateComponent,
|
||||||
updateComponentFailure,
|
updateComponentFailure,
|
||||||
updateComponentSuccess,
|
updateComponentSuccess,
|
||||||
|
@ -127,6 +130,7 @@ export const initialState: FlowState = {
|
||||||
connectedNodeCount: 0,
|
connectedNodeCount: 0,
|
||||||
totalNodeCount: 0
|
totalNodeCount: 0
|
||||||
},
|
},
|
||||||
|
refreshRpgDetails: null,
|
||||||
controllerBulletins: {
|
controllerBulletins: {
|
||||||
bulletins: [],
|
bulletins: [],
|
||||||
controllerServiceBulletins: [],
|
controllerServiceBulletins: [],
|
||||||
|
@ -150,6 +154,24 @@ export const flowReducer = createReducer(
|
||||||
on(resetFlowState, () => ({
|
on(resetFlowState, () => ({
|
||||||
...initialState
|
...initialState
|
||||||
})),
|
})),
|
||||||
|
on(requestRefreshRemoteProcessGroup, (state, { request }) => ({
|
||||||
|
...state,
|
||||||
|
refreshRpgDetails: {
|
||||||
|
request,
|
||||||
|
polling: false
|
||||||
|
}
|
||||||
|
})),
|
||||||
|
on(startRemoteProcessGroupPolling, (state) => {
|
||||||
|
return produce(state, (draftState) => {
|
||||||
|
if (draftState.refreshRpgDetails) {
|
||||||
|
draftState.refreshRpgDetails.polling = true;
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}),
|
||||||
|
on(stopRemoteProcessGroupPolling, (state) => ({
|
||||||
|
...state,
|
||||||
|
refreshRpgDetails: null
|
||||||
|
})),
|
||||||
on(loadProcessGroup, (state, { request }) => ({
|
on(loadProcessGroup, (state, { request }) => ({
|
||||||
...state,
|
...state,
|
||||||
transitionRequired: request.transitionRequired,
|
transitionRequired: request.transitionRequired,
|
||||||
|
|
|
@ -32,6 +32,8 @@ export const selectSaving = createSelector(selectFlowState, (state: FlowState) =
|
||||||
|
|
||||||
export const selectCurrentProcessGroupId = createSelector(selectFlowState, (state: FlowState) => state.id);
|
export const selectCurrentProcessGroupId = createSelector(selectFlowState, (state: FlowState) => state.id);
|
||||||
|
|
||||||
|
export const selectRefreshRpgDetails = createSelector(selectFlowState, (state: FlowState) => state.refreshRpgDetails);
|
||||||
|
|
||||||
export const selectCurrentParameterContext = createSelector(
|
export const selectCurrentParameterContext = createSelector(
|
||||||
selectFlowState,
|
selectFlowState,
|
||||||
(state: FlowState) => state.flow.processGroupFlow.parameterContext
|
(state: FlowState) => state.flow.processGroupFlow.parameterContext
|
||||||
|
|
|
@ -208,6 +208,16 @@ export interface GoToRemoteProcessGroupRequest {
|
||||||
uri: string;
|
uri: string;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
export interface RefreshRemoteProcessGroupRequest {
|
||||||
|
id: string;
|
||||||
|
refreshTimestamp: string;
|
||||||
|
}
|
||||||
|
|
||||||
|
export interface RefreshRemoteProcessGroupPollingDetailsRequest {
|
||||||
|
request: RefreshRemoteProcessGroupRequest;
|
||||||
|
polling: boolean;
|
||||||
|
}
|
||||||
|
|
||||||
export interface CreateProcessorRequest extends CreateComponentRequest {
|
export interface CreateProcessorRequest extends CreateComponentRequest {
|
||||||
processorType: string;
|
processorType: string;
|
||||||
processorBundle: Bundle;
|
processorBundle: Bundle;
|
||||||
|
@ -503,6 +513,7 @@ export interface FlowState {
|
||||||
id: string;
|
id: string;
|
||||||
flow: ProcessGroupFlowEntity;
|
flow: ProcessGroupFlowEntity;
|
||||||
flowStatus: ControllerStatusEntity;
|
flowStatus: ControllerStatusEntity;
|
||||||
|
refreshRpgDetails: RefreshRemoteProcessGroupPollingDetailsRequest | null;
|
||||||
clusterSummary: ClusterSummary;
|
clusterSummary: ClusterSummary;
|
||||||
controllerBulletins: ControllerBulletinsEntity;
|
controllerBulletins: ControllerBulletinsEntity;
|
||||||
dragging: boolean;
|
dragging: boolean;
|
||||||
|
|
Loading…
Reference in New Issue