2 * ============LICENSE_START=======================================================
3 * Copyright (C) 2021 Nordix Foundation.
4 * Modifications Copyright (C) 2021 AT&T Intellectual Property. All rights reserved.
5 * ================================================================================
6 * Licensed under the Apache License, Version 2.0 (the "License");
7 * you may not use this file except in compliance with the License.
8 * You may obtain a copy of the License at
10 * http://www.apache.org/licenses/LICENSE-2.0
12 * Unless required by applicable law or agreed to in writing, software
13 * distributed under the License is distributed on an "AS IS" BASIS,
14 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
15 * See the License for the specific language governing permissions and
16 * limitations under the License.
18 * SPDX-License-Identifier: Apache-2.0
19 * ============LICENSE_END=========================================================
22 package org.onap.policy.clamp.controlloop.runtime.supervision;
24 import java.util.List;
27 import java.util.UUID;
28 import javax.ws.rs.core.Response;
29 import javax.ws.rs.core.Response.Status;
30 import lombok.AllArgsConstructor;
31 import org.apache.commons.collections4.CollectionUtils;
32 import org.onap.policy.clamp.controlloop.common.exception.ControlLoopException;
33 import org.onap.policy.clamp.controlloop.models.controlloop.concepts.ControlLoop;
34 import org.onap.policy.clamp.controlloop.models.controlloop.concepts.ControlLoopElementAck;
35 import org.onap.policy.clamp.controlloop.models.controlloop.concepts.ControlLoopInfo;
36 import org.onap.policy.clamp.controlloop.models.controlloop.concepts.ControlLoopState;
37 import org.onap.policy.clamp.controlloop.models.controlloop.concepts.Participant;
38 import org.onap.policy.clamp.controlloop.models.controlloop.concepts.ParticipantHealthStatus;
39 import org.onap.policy.clamp.controlloop.models.controlloop.concepts.ParticipantState;
40 import org.onap.policy.clamp.controlloop.models.controlloop.concepts.ParticipantUtils;
41 import org.onap.policy.clamp.controlloop.models.controlloop.persistence.provider.ControlLoopProvider;
42 import org.onap.policy.clamp.controlloop.models.controlloop.persistence.provider.ParticipantProvider;
43 import org.onap.policy.clamp.controlloop.models.controlloop.persistence.provider.ServiceTemplateProvider;
44 import org.onap.policy.clamp.controlloop.models.messages.dmaap.participant.ControlLoopAck;
45 import org.onap.policy.clamp.controlloop.models.messages.dmaap.participant.ParticipantDeregister;
46 import org.onap.policy.clamp.controlloop.models.messages.dmaap.participant.ParticipantMessage;
47 import org.onap.policy.clamp.controlloop.models.messages.dmaap.participant.ParticipantRegister;
48 import org.onap.policy.clamp.controlloop.models.messages.dmaap.participant.ParticipantStatus;
49 import org.onap.policy.clamp.controlloop.models.messages.dmaap.participant.ParticipantUpdateAck;
50 import org.onap.policy.clamp.controlloop.runtime.monitoring.MonitoringProvider;
51 import org.onap.policy.clamp.controlloop.runtime.supervision.comm.ControlLoopStateChangePublisher;
52 import org.onap.policy.clamp.controlloop.runtime.supervision.comm.ControlLoopUpdatePublisher;
53 import org.onap.policy.clamp.controlloop.runtime.supervision.comm.ParticipantDeregisterAckPublisher;
54 import org.onap.policy.clamp.controlloop.runtime.supervision.comm.ParticipantRegisterAckPublisher;
55 import org.onap.policy.clamp.controlloop.runtime.supervision.comm.ParticipantUpdatePublisher;
56 import org.onap.policy.models.base.PfModelException;
57 import org.onap.policy.models.base.PfModelRuntimeException;
58 import org.onap.policy.models.tosca.authorative.concepts.ToscaConceptIdentifier;
59 import org.onap.policy.models.tosca.authorative.concepts.ToscaServiceTemplate;
60 import org.slf4j.Logger;
61 import org.slf4j.LoggerFactory;
62 import org.springframework.stereotype.Component;
65 * This class handles supervision of control loop instances, so only one object of this type should be built at a time.
68 * It is effectively a singleton that is started at system start.
72 public class SupervisionHandler {
73 private static final Logger LOGGER = LoggerFactory.getLogger(SupervisionHandler.class);
75 private static final String CONTROL_LOOP_CANNOT_TRANSITION_FROM_STATE = "Control loop can't transition from state ";
76 private static final String CONTROL_LOOP_IS_ALREADY_IN_STATE = "Control loop is already in state ";
77 private static final String TO_STATE = " to state ";
78 private static final String AND_TRANSITIONING_TO_STATE = " and transitioning to state ";
80 private final ControlLoopProvider controlLoopProvider;
81 private final ParticipantProvider participantProvider;
82 private final MonitoringProvider monitoringProvider;
83 private final ServiceTemplateProvider serviceTemplateProvider;
85 // Publishers for participant communication
86 private final ControlLoopUpdatePublisher controlLoopUpdatePublisher;
87 private final ControlLoopStateChangePublisher controlLoopStateChangePublisher;
88 private final ParticipantRegisterAckPublisher participantRegisterAckPublisher;
89 private final ParticipantDeregisterAckPublisher participantDeregisterAckPublisher;
90 private final ParticipantUpdatePublisher participantUpdatePublisher;
93 * Supervision trigger called when a command is issued on control loops.
96 * Causes supervision to start or continue supervision on the control loops in question.
98 * @param controlLoopIdentifierList the control loops for which the supervision command has been issued
99 * @throws ControlLoopException on supervision triggering exceptions
101 public void triggerControlLoopSupervision(List<ToscaConceptIdentifier> controlLoopIdentifierList)
102 throws ControlLoopException {
104 LOGGER.debug("triggering control loop supervision on control loops {}", controlLoopIdentifierList);
106 if (CollectionUtils.isEmpty(controlLoopIdentifierList)) {
107 // This is just to force throwing of the exception in certain circumstances.
108 exceptionOccured(Response.Status.NOT_ACCEPTABLE, "The list of control loops for supervision is empty");
111 for (ToscaConceptIdentifier controlLoopId : controlLoopIdentifierList) {
113 var controlLoop = controlLoopProvider.getControlLoop(controlLoopId);
115 superviseControlLoop(controlLoop);
117 controlLoopProvider.updateControlLoop(controlLoop);
118 } catch (PfModelException pfme) {
119 throw new ControlLoopException(pfme.getErrorResponse().getResponseCode(), pfme.getMessage(), pfme);
125 * Handle a ParticipantStatus message from a participant.
127 * @param participantStatusMessage the ParticipantStatus message received from a participant
130 public void handleParticipantMessage(ParticipantStatus participantStatusMessage) {
131 LOGGER.debug("Participant Status received {}", participantStatusMessage);
133 superviseParticipant(participantStatusMessage);
134 } catch (PfModelException | ControlLoopException svExc) {
135 LOGGER.warn("error supervising participant {}", participantStatusMessage.getParticipantId(), svExc);
140 superviseControlLoops(participantStatusMessage);
141 } catch (PfModelException | ControlLoopException svExc) {
142 LOGGER.warn("error supervising participant {}", participantStatusMessage.getParticipantId(), svExc);
147 * Handle a ParticipantRegister message from a participant.
149 * @param participantRegisterMessage the ParticipantRegister message received from a participant
152 public boolean handleParticipantMessage(ParticipantRegister participantRegisterMessage) {
153 LOGGER.debug("Participant Register received {}", participantRegisterMessage);
155 checkParticipant(participantRegisterMessage, ParticipantState.UNKNOWN, ParticipantHealthStatus.UNKNOWN);
156 } catch (PfModelException | ControlLoopException svExc) {
157 LOGGER.warn("error saving participant {}", participantRegisterMessage.getParticipantId(), svExc);
160 var isCommissioning = participantUpdatePublisher.sendCommissioning(null, null,
161 participantRegisterMessage.getParticipantId(), participantRegisterMessage.getParticipantType());
163 participantRegisterAckPublisher.send(participantRegisterMessage.getMessageId(),
164 participantRegisterMessage.getParticipantId(), participantRegisterMessage.getParticipantType());
165 return isCommissioning;
169 * Handle a ParticipantDeregister message from a participant.
171 * @param participantDeregisterMessage the ParticipantDeregister message received from a participant
174 public void handleParticipantMessage(ParticipantDeregister participantDeregisterMessage) {
175 LOGGER.debug("Participant Deregister received {}", participantDeregisterMessage);
178 participantProvider.findParticipant(participantDeregisterMessage.getParticipantId().getName(),
179 participantDeregisterMessage.getParticipantId().getVersion());
181 if (participantOpt.isPresent()) {
182 var participant = participantOpt.get();
183 participant.setParticipantState(ParticipantState.TERMINATED);
184 participant.setHealthStatus(ParticipantHealthStatus.OFF_LINE);
185 participantProvider.saveParticipant(participant);
187 } catch (PfModelException pfme) {
188 LOGGER.warn("Model exception occured with participant id {}",
189 participantDeregisterMessage.getParticipantId());
192 participantDeregisterAckPublisher.send(participantDeregisterMessage.getMessageId());
196 * Handle a ParticipantUpdateAck message from a participant.
198 * @param participantUpdateAckMessage the ParticipantUpdateAck message received from a participant
201 public void handleParticipantMessage(ParticipantUpdateAck participantUpdateAckMessage) {
202 LOGGER.debug("Participant Update Ack received {}", participantUpdateAckMessage);
205 participantProvider.findParticipant(participantUpdateAckMessage.getParticipantId().getName(),
206 participantUpdateAckMessage.getParticipantId().getVersion());
208 if (participantOpt.isPresent()) {
209 var participant = participantOpt.get();
210 participant.setParticipantState(participantUpdateAckMessage.getState());
211 participantProvider.saveParticipant(participant);
213 LOGGER.warn("Participant not found in database {}", participantUpdateAckMessage.getParticipantId());
215 } catch (PfModelException pfme) {
216 LOGGER.warn("Model exception occured with participant id {}",
217 participantUpdateAckMessage.getParticipantId());
222 * Send commissioning update message to dmaap.
224 * @param name the ToscaServiceTemplate name
225 * @param version the ToscaServiceTemplate version
227 public void handleSendCommissionMessage(String name, String version) {
228 LOGGER.debug("Participant update message with serviveTemplate {} {} being sent to all participants", name,
230 participantUpdatePublisher.sendComissioningBroadcast(name, version);
234 * Send decommissioning update message to dmaap.
237 public void handleSendDeCommissionMessage() {
238 LOGGER.debug("Participant update message being sent");
239 participantUpdatePublisher.sendDecomisioning();
243 * Handle a ControlLoop update acknowledge message from a participant.
245 * @param controlLoopAckMessage the ControlLoopAck message received from a participant
248 public void handleControlLoopUpdateAckMessage(ControlLoopAck controlLoopAckMessage) {
249 LOGGER.debug("ControlLoop Update Ack message received {}", controlLoopAckMessage);
250 setClElementStateInDb(controlLoopAckMessage);
254 * Handle a ControlLoop statechange acknowledge message from a participant.
256 * @param controlLoopAckMessage the ControlLoopAck message received from a participant
259 public void handleControlLoopStateChangeAckMessage(ControlLoopAck controlLoopAckMessage) {
260 LOGGER.debug("ControlLoop StateChange Ack message received {}", controlLoopAckMessage);
261 setClElementStateInDb(controlLoopAckMessage);
264 private void setClElementStateInDb(ControlLoopAck controlLoopAckMessage) {
265 if (controlLoopAckMessage.getControlLoopResultMap() != null) {
267 var controlLoop = controlLoopProvider.getControlLoop(controlLoopAckMessage.getControlLoopId());
268 if (controlLoop != null) {
269 var updated = updateState(controlLoop, controlLoopAckMessage.getControlLoopResultMap().entrySet());
270 updated |= setPrimed(controlLoop);
272 controlLoopProvider.updateControlLoop(controlLoop);
275 LOGGER.warn("ControlLoop not found in database {}", controlLoopAckMessage.getControlLoopId());
277 } catch (PfModelException pfme) {
278 LOGGER.warn("Model exception occured with ControlLoop Id {}", controlLoopAckMessage.getControlLoopId());
283 private boolean updateState(ControlLoop controlLoop,
284 Set<Map.Entry<UUID, ControlLoopElementAck>> controlLoopResultSet) {
286 for (var clElementAck : controlLoopResultSet) {
287 var element = controlLoop.getElements().get(clElementAck.getKey());
288 if (element != null) {
289 element.setState(clElementAck.getValue().getState());
296 private boolean setPrimed(ControlLoop controlLoop) {
297 var clElements = controlLoop.getElements().values();
298 if (clElements != null) {
299 Boolean primedFlag = true;
300 var checkOpt = controlLoop.getElements().values().stream()
301 .filter(clElement -> (!clElement.getState().equals(ControlLoopState.PASSIVE)
302 || !clElement.getState().equals(ControlLoopState.RUNNING)))
304 if (checkOpt.isEmpty()) {
307 controlLoop.setPrimed(primedFlag);
315 * Supervise a control loop, performing whatever actions need to be performed on the control loop.
317 * @param controlLoop the control loop to supervises
318 * @throws ControlLoopException on supervision errors
320 private void superviseControlLoop(ControlLoop controlLoop) throws ControlLoopException {
321 switch (controlLoop.getOrderedState()) {
323 superviseControlLoopUninitialization(controlLoop);
327 superviseControlLoopPassivation(controlLoop);
331 superviseControlLoopActivation(controlLoop);
335 exceptionOccured(Response.Status.NOT_ACCEPTABLE,
336 "A control loop cannot be commanded to go into state " + controlLoop.getOrderedState().name());
341 * Supervise a control loop uninitialisation, performing whatever actions need to be performed on the control loop,
342 * control loop ordered state is UNINITIALIZED.
344 * @param controlLoop the control loop to supervises
345 * @throws ControlLoopException on supervision errors
347 private void superviseControlLoopUninitialization(ControlLoop controlLoop) throws ControlLoopException {
348 switch (controlLoop.getState()) {
350 exceptionOccured(Response.Status.NOT_ACCEPTABLE,
351 CONTROL_LOOP_IS_ALREADY_IN_STATE + controlLoop.getState().name());
354 case UNINITIALISED2PASSIVE:
356 controlLoop.setState(ControlLoopState.PASSIVE2UNINITIALISED);
357 controlLoopStateChangePublisher.send(controlLoop, getFirstStartPhase(controlLoop));
360 case PASSIVE2UNINITIALISED:
361 exceptionOccured(Response.Status.NOT_ACCEPTABLE, CONTROL_LOOP_IS_ALREADY_IN_STATE
362 + controlLoop.getState().name() + AND_TRANSITIONING_TO_STATE + controlLoop.getOrderedState());
366 exceptionOccured(Response.Status.NOT_ACCEPTABLE, CONTROL_LOOP_CANNOT_TRANSITION_FROM_STATE
367 + controlLoop.getState().name() + TO_STATE + controlLoop.getOrderedState());
372 private void superviseControlLoopPassivation(ControlLoop controlLoop) throws ControlLoopException {
373 switch (controlLoop.getState()) {
375 exceptionOccured(Response.Status.NOT_ACCEPTABLE,
376 CONTROL_LOOP_IS_ALREADY_IN_STATE + controlLoop.getState().name());
379 controlLoop.setState(ControlLoopState.UNINITIALISED2PASSIVE);
380 controlLoopUpdatePublisher.send(controlLoop);
383 case UNINITIALISED2PASSIVE:
384 case RUNNING2PASSIVE:
385 exceptionOccured(Response.Status.NOT_ACCEPTABLE, CONTROL_LOOP_IS_ALREADY_IN_STATE
386 + controlLoop.getState().name() + AND_TRANSITIONING_TO_STATE + controlLoop.getOrderedState());
390 controlLoop.setState(ControlLoopState.RUNNING2PASSIVE);
391 controlLoopStateChangePublisher.send(controlLoop, getFirstStartPhase(controlLoop));
395 exceptionOccured(Response.Status.NOT_ACCEPTABLE, CONTROL_LOOP_CANNOT_TRANSITION_FROM_STATE
396 + controlLoop.getState().name() + TO_STATE + controlLoop.getOrderedState());
401 private void superviseControlLoopActivation(ControlLoop controlLoop) throws ControlLoopException {
402 switch (controlLoop.getState()) {
404 exceptionOccured(Response.Status.NOT_ACCEPTABLE,
405 CONTROL_LOOP_IS_ALREADY_IN_STATE + controlLoop.getState().name());
408 case PASSIVE2RUNNING:
409 exceptionOccured(Response.Status.NOT_ACCEPTABLE, CONTROL_LOOP_IS_ALREADY_IN_STATE
410 + controlLoop.getState().name() + AND_TRANSITIONING_TO_STATE + controlLoop.getOrderedState());
414 controlLoop.setState(ControlLoopState.PASSIVE2RUNNING);
415 controlLoopStateChangePublisher.send(controlLoop, getFirstStartPhase(controlLoop));
419 exceptionOccured(Response.Status.NOT_ACCEPTABLE, CONTROL_LOOP_CANNOT_TRANSITION_FROM_STATE
420 + controlLoop.getState().name() + TO_STATE + controlLoop.getOrderedState());
425 private int getFirstStartPhase(ControlLoop controlLoop) {
426 ToscaServiceTemplate toscaServiceTemplate = null;
428 toscaServiceTemplate = serviceTemplateProvider.getServiceTemplateList(null, null).get(0);
429 } catch (PfModelException e) {
430 throw new PfModelRuntimeException(Status.BAD_REQUEST, "Canont load ToscaServiceTemplate from DB", e);
432 return ParticipantUtils.getFirstStartPhase(controlLoop, toscaServiceTemplate);
435 private void checkParticipant(ParticipantMessage participantMessage, ParticipantState participantState,
436 ParticipantHealthStatus healthStatus) throws ControlLoopException, PfModelException {
437 if (participantMessage.getParticipantId() == null) {
438 exceptionOccured(Response.Status.NOT_FOUND, "Participant ID on PARTICIPANT_STATUS message is null");
440 var participantOpt = participantProvider.findParticipant(participantMessage.getParticipantId().getName(),
441 participantMessage.getParticipantId().getVersion());
443 if (participantOpt.isEmpty()) {
444 var participant = new Participant();
445 participant.setName(participantMessage.getParticipantId().getName());
446 participant.setVersion(participantMessage.getParticipantId().getVersion());
447 participant.setDefinition(participantMessage.getParticipantId());
448 participant.setParticipantType(participantMessage.getParticipantType());
449 participant.setParticipantState(participantState);
450 participant.setHealthStatus(healthStatus);
452 participantProvider.saveParticipant(participant);
454 var participant = participantOpt.get();
455 participant.setParticipantState(participantState);
456 participant.setHealthStatus(healthStatus);
458 participantProvider.saveParticipant(participant);
462 private void superviseParticipant(ParticipantStatus participantStatusMessage)
463 throws PfModelException, ControlLoopException {
465 checkParticipant(participantStatusMessage, participantStatusMessage.getState(),
466 participantStatusMessage.getHealthStatus());
468 monitoringProvider.createParticipantStatistics(List.of(participantStatusMessage.getParticipantStatistics()));
471 private void superviseControlLoops(ParticipantStatus participantStatusMessage)
472 throws PfModelException, ControlLoopException {
473 if (participantStatusMessage.getControlLoopInfoList() != null) {
474 for (ControlLoopInfo clEntry : participantStatusMessage.getControlLoopInfoList()) {
476 controlLoopProvider.getControlLoop(new ToscaConceptIdentifier(clEntry.getControlLoopId()));
477 if (dbControlLoop == null) {
478 exceptionOccured(Response.Status.NOT_FOUND,
479 "PARTICIPANT_STATUS control loop not found in database: " + clEntry.getControlLoopId());
481 dbControlLoop.setState(clEntry.getState());
482 monitoringProvider.createClElementStatistics(
483 clEntry.getControlLoopStatistics().getClElementStatisticsList().getClElementStatistics());
488 private void exceptionOccured(Response.Status status, String reason) throws ControlLoopException {
489 throw new ControlLoopException(status, reason);