/* * Copyright (c) 2021 PANTHEON.tech, s.r.o. and others. All rights reserved. * * This program and the accompanying materials are made available under the * terms of the Eclipse Public License v1.0 which accompanies this distribution, * and is available at http://www.eclipse.org/legal/epl-v10.html */ package org.opendaylight.controller.eos.akka.owner.supervisor; import static java.util.Objects.requireNonNull; import akka.actor.typed.Behavior; import akka.actor.typed.javadsl.ActorContext; import akka.actor.typed.javadsl.Behaviors; import akka.actor.typed.javadsl.Receive; import akka.cluster.Member; import akka.cluster.typed.Cluster; import akka.pattern.StatusReply; import org.opendaylight.controller.eos.akka.owner.supervisor.command.ActivateDataCenter; import org.opendaylight.controller.eos.akka.owner.supervisor.command.ClearCandidates; import org.opendaylight.controller.eos.akka.owner.supervisor.command.ClearCandidatesForMember; import org.opendaylight.controller.eos.akka.owner.supervisor.command.GetEntitiesBackendRequest; import org.opendaylight.controller.eos.akka.owner.supervisor.command.GetEntityBackendRequest; import org.opendaylight.controller.eos.akka.owner.supervisor.command.GetEntityOwnerBackendRequest; import org.opendaylight.controller.eos.akka.owner.supervisor.command.OwnerSupervisorCommand; import org.opendaylight.controller.eos.akka.owner.supervisor.command.OwnerSupervisorRequest; import org.opendaylight.mdsal.binding.dom.codec.api.BindingInstanceIdentifierCodec; import org.slf4j.Logger; import org.slf4j.LoggerFactory; /** * Initial Supervisor behavior that stays idle and only switches itself to the active behavior when its running * in the primary datacenter, or is activated on demand. Once the supervisor instance is no longer needed in the * secondary datacenter it needs to be deactivated manually. */ public final class IdleSupervisor extends AbstractSupervisor { private static final Logger LOG = LoggerFactory.getLogger(IdleSupervisor.class); private static final String DATACENTER_PREFIX = "dc-"; private static final String DEFAULT_DATACENTER = "dc-default"; private final BindingInstanceIdentifierCodec iidCodec; private IdleSupervisor(final ActorContext context, final BindingInstanceIdentifierCodec iidCodec) { super(context); this.iidCodec = requireNonNull(iidCodec); final Cluster cluster = Cluster.get(context.getSystem()); final String datacenterRole = extractDatacenterRole(cluster.selfMember()); if (datacenterRole.equals(DEFAULT_DATACENTER)) { LOG.debug("No datacenter configured, activating default data center"); context.getSelf().tell(new ActivateDataCenter(null)); } LOG.debug("Idle supervisor started on {}.", cluster.selfMember()); } public static Behavior create(final BindingInstanceIdentifierCodec iidCodec) { return Behaviors.setup(context -> new IdleSupervisor(context, iidCodec)); } @Override public Receive createReceive() { return newReceiveBuilder() .onMessage(ActivateDataCenter.class, this::onActivateDataCenter) .onMessage(GetEntitiesBackendRequest.class, this::onFailEntityRpc) .onMessage(GetEntityBackendRequest.class, this::onFailEntityRpc) .onMessage(GetEntityOwnerBackendRequest.class, this::onFailEntityRpc) .onMessage(ClearCandidatesForMember.class, this::onClearCandidatesForMember) .onMessage(ClearCandidates.class, this::finishClearCandidates) .build(); } private Behavior onFailEntityRpc(final OwnerSupervisorRequest message) { LOG.debug("Failing rpc request. {}", message); message.getReplyTo().tell(StatusReply.error("OwnerSupervisor is inactive so it" + " cannot handle entity rpc requests.")); return this; } private Behavior onActivateDataCenter(final ActivateDataCenter message) { LOG.debug("Received ActivateDataCenter command switching to syncer behavior,"); return OwnerSyncer.create(message.getReplyTo(), iidCodec); } private static String extractDatacenterRole(final Member selfMember) { return selfMember.getRoles().stream() .filter(role -> role.startsWith(DATACENTER_PREFIX)) .findFirst() .orElseThrow(() -> new IllegalArgumentException(selfMember + " does not have a valid role")); } @Override Logger getLogger() { return LOG; } }