ONOS-2003 Fixing intent reroute after cluster change
Objective trackers update when partitions are shuffled to track "local" intents. Change-Id: I7cd9e4a935ddbc94813d5067d4febc084a89f508
Showing
8 changed files
with
194 additions
and
2 deletions
| 1 | +/* | ||
| 2 | + * Copyright 2015 Open Networking Laboratory | ||
| 3 | + * | ||
| 4 | + * Licensed under the Apache License, Version 2.0 (the "License"); | ||
| 5 | + * you may not use this file except in compliance with the License. | ||
| 6 | + * You may obtain a copy of the License at | ||
| 7 | + * | ||
| 8 | + * http://www.apache.org/licenses/LICENSE-2.0 | ||
| 9 | + * | ||
| 10 | + * Unless required by applicable law or agreed to in writing, software | ||
| 11 | + * distributed under the License is distributed on an "AS IS" BASIS, | ||
| 12 | + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. | ||
| 13 | + * See the License for the specific language governing permissions and | ||
| 14 | + * limitations under the License. | ||
| 15 | + */ | ||
| 16 | +package org.onosproject.net.intent; | ||
| 17 | + | ||
| 18 | +import org.onosproject.event.AbstractEvent; | ||
| 19 | + | ||
| 20 | +/** | ||
| 21 | + * Partition event. | ||
| 22 | + */ | ||
| 23 | +//TODO change String into a proper object type | ||
| 24 | +public class PartitionEvent extends AbstractEvent<PartitionEvent.Type, String> { | ||
| 25 | + | ||
| 26 | + public enum Type { | ||
| 27 | + LEADER_CHANGED | ||
| 28 | + } | ||
| 29 | + | ||
| 30 | + public PartitionEvent(Type type, String partition) { | ||
| 31 | + super(type, partition); | ||
| 32 | + } | ||
| 33 | +} |
| 1 | +/* | ||
| 2 | + * Copyright 2015 Open Networking Laboratory | ||
| 3 | + * | ||
| 4 | + * Licensed under the Apache License, Version 2.0 (the "License"); | ||
| 5 | + * you may not use this file except in compliance with the License. | ||
| 6 | + * You may obtain a copy of the License at | ||
| 7 | + * | ||
| 8 | + * http://www.apache.org/licenses/LICENSE-2.0 | ||
| 9 | + * | ||
| 10 | + * Unless required by applicable law or agreed to in writing, software | ||
| 11 | + * distributed under the License is distributed on an "AS IS" BASIS, | ||
| 12 | + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. | ||
| 13 | + * See the License for the specific language governing permissions and | ||
| 14 | + * limitations under the License. | ||
| 15 | + */ | ||
| 16 | +package org.onosproject.net.intent; | ||
| 17 | + | ||
| 18 | +import org.onosproject.event.EventListener; | ||
| 19 | + | ||
| 20 | +/** | ||
| 21 | + * Entity capable of receiving device partition-related events. | ||
| 22 | + */ | ||
| 23 | +public interface PartitionEventListener extends EventListener<PartitionEvent> { | ||
| 24 | +} |
| ... | @@ -40,4 +40,18 @@ public interface PartitionService { | ... | @@ -40,4 +40,18 @@ public interface PartitionService { |
| 40 | NodeId getLeader(Key intentKey); | 40 | NodeId getLeader(Key intentKey); |
| 41 | 41 | ||
| 42 | // TODO add API for rebalancing partitions | 42 | // TODO add API for rebalancing partitions |
| 43 | + | ||
| 44 | + /** | ||
| 45 | + * Registers a event listener to be notified of partition events. | ||
| 46 | + * | ||
| 47 | + * @param listener listener that will asynchronously notified of partition events. | ||
| 48 | + */ | ||
| 49 | + void addListener(PartitionEventListener listener); | ||
| 50 | + | ||
| 51 | + /** | ||
| 52 | + * Unregisters a event listener for partition events. | ||
| 53 | + * | ||
| 54 | + * @param listener listener to be removed. | ||
| 55 | + */ | ||
| 56 | + void removeListener(PartitionEventListener listener); | ||
| 43 | } | 57 | } | ... | ... |
| ... | @@ -23,6 +23,7 @@ import org.apache.felix.scr.annotations.Component; | ... | @@ -23,6 +23,7 @@ import org.apache.felix.scr.annotations.Component; |
| 23 | import org.apache.felix.scr.annotations.Deactivate; | 23 | import org.apache.felix.scr.annotations.Deactivate; |
| 24 | import org.apache.felix.scr.annotations.Reference; | 24 | import org.apache.felix.scr.annotations.Reference; |
| 25 | import org.apache.felix.scr.annotations.ReferenceCardinality; | 25 | import org.apache.felix.scr.annotations.ReferenceCardinality; |
| 26 | +import org.apache.felix.scr.annotations.ReferencePolicy; | ||
| 26 | import org.apache.felix.scr.annotations.Service; | 27 | import org.apache.felix.scr.annotations.Service; |
| 27 | import org.onosproject.core.ApplicationId; | 28 | import org.onosproject.core.ApplicationId; |
| 28 | import org.onosproject.event.Event; | 29 | import org.onosproject.event.Event; |
| ... | @@ -38,8 +39,12 @@ import org.onosproject.net.device.DeviceService; | ... | @@ -38,8 +39,12 @@ import org.onosproject.net.device.DeviceService; |
| 38 | import org.onosproject.net.host.HostEvent; | 39 | import org.onosproject.net.host.HostEvent; |
| 39 | import org.onosproject.net.host.HostListener; | 40 | import org.onosproject.net.host.HostListener; |
| 40 | import org.onosproject.net.host.HostService; | 41 | import org.onosproject.net.host.HostService; |
| 42 | +import org.onosproject.net.intent.Intent; | ||
| 41 | import org.onosproject.net.intent.IntentService; | 43 | import org.onosproject.net.intent.IntentService; |
| 42 | import org.onosproject.net.intent.Key; | 44 | import org.onosproject.net.intent.Key; |
| 45 | +import org.onosproject.net.intent.PartitionEvent; | ||
| 46 | +import org.onosproject.net.intent.PartitionEventListener; | ||
| 47 | +import org.onosproject.net.intent.PartitionService; | ||
| 43 | import org.onosproject.net.link.LinkEvent; | 48 | import org.onosproject.net.link.LinkEvent; |
| 44 | import org.onosproject.net.resource.link.LinkResourceEvent; | 49 | import org.onosproject.net.resource.link.LinkResourceEvent; |
| 45 | import org.onosproject.net.resource.link.LinkResourceListener; | 50 | import org.onosproject.net.resource.link.LinkResourceListener; |
| ... | @@ -54,6 +59,10 @@ import java.util.Collections; | ... | @@ -54,6 +59,10 @@ import java.util.Collections; |
| 54 | import java.util.HashSet; | 59 | import java.util.HashSet; |
| 55 | import java.util.Set; | 60 | import java.util.Set; |
| 56 | import java.util.concurrent.ExecutorService; | 61 | import java.util.concurrent.ExecutorService; |
| 62 | +import java.util.concurrent.Executors; | ||
| 63 | +import java.util.concurrent.ScheduledExecutorService; | ||
| 64 | +import java.util.concurrent.TimeUnit; | ||
| 65 | +import java.util.concurrent.atomic.AtomicBoolean; | ||
| 57 | 66 | ||
| 58 | import static com.google.common.base.Preconditions.checkArgument; | 67 | import static com.google.common.base.Preconditions.checkArgument; |
| 59 | import static com.google.common.base.Preconditions.checkNotNull; | 68 | import static com.google.common.base.Preconditions.checkNotNull; |
| ... | @@ -94,25 +103,35 @@ public class ObjectiveTracker implements ObjectiveTrackerService { | ... | @@ -94,25 +103,35 @@ public class ObjectiveTracker implements ObjectiveTrackerService { |
| 94 | @Reference(cardinality = ReferenceCardinality.MANDATORY_UNARY) | 103 | @Reference(cardinality = ReferenceCardinality.MANDATORY_UNARY) |
| 95 | protected HostService hostService; | 104 | protected HostService hostService; |
| 96 | 105 | ||
| 97 | - @Reference(cardinality = ReferenceCardinality.OPTIONAL_UNARY) | 106 | + @Reference(cardinality = ReferenceCardinality.OPTIONAL_UNARY, |
| 107 | + policy = ReferencePolicy.DYNAMIC) | ||
| 98 | protected IntentService intentService; | 108 | protected IntentService intentService; |
| 99 | 109 | ||
| 110 | + @Reference(cardinality = ReferenceCardinality.MANDATORY_UNARY) | ||
| 111 | + protected PartitionService partitionService; | ||
| 112 | + | ||
| 100 | private ExecutorService executorService = | 113 | private ExecutorService executorService = |
| 101 | newSingleThreadExecutor(groupedThreads("onos/intent", "objectivetracker")); | 114 | newSingleThreadExecutor(groupedThreads("onos/intent", "objectivetracker")); |
| 115 | + private ScheduledExecutorService executor = Executors | ||
| 116 | + .newScheduledThreadPool(1); | ||
| 102 | 117 | ||
| 103 | private TopologyListener listener = new InternalTopologyListener(); | 118 | private TopologyListener listener = new InternalTopologyListener(); |
| 104 | private LinkResourceListener linkResourceListener = | 119 | private LinkResourceListener linkResourceListener = |
| 105 | new InternalLinkResourceListener(); | 120 | new InternalLinkResourceListener(); |
| 106 | private DeviceListener deviceListener = new InternalDeviceListener(); | 121 | private DeviceListener deviceListener = new InternalDeviceListener(); |
| 107 | private HostListener hostListener = new InternalHostListener(); | 122 | private HostListener hostListener = new InternalHostListener(); |
| 123 | + private PartitionEventListener partitionListener = new InternalPartitionListener(); | ||
| 108 | private TopologyChangeDelegate delegate; | 124 | private TopologyChangeDelegate delegate; |
| 109 | 125 | ||
| 126 | + protected final AtomicBoolean updateScheduled = new AtomicBoolean(false); | ||
| 127 | + | ||
| 110 | @Activate | 128 | @Activate |
| 111 | public void activate() { | 129 | public void activate() { |
| 112 | topologyService.addListener(listener); | 130 | topologyService.addListener(listener); |
| 113 | resourceManager.addListener(linkResourceListener); | 131 | resourceManager.addListener(linkResourceListener); |
| 114 | deviceService.addListener(deviceListener); | 132 | deviceService.addListener(deviceListener); |
| 115 | hostService.addListener(hostListener); | 133 | hostService.addListener(hostListener); |
| 134 | + partitionService.addListener(partitionListener); | ||
| 116 | log.info("Started"); | 135 | log.info("Started"); |
| 117 | } | 136 | } |
| 118 | 137 | ||
| ... | @@ -122,6 +141,7 @@ public class ObjectiveTracker implements ObjectiveTrackerService { | ... | @@ -122,6 +141,7 @@ public class ObjectiveTracker implements ObjectiveTrackerService { |
| 122 | resourceManager.removeListener(linkResourceListener); | 141 | resourceManager.removeListener(linkResourceListener); |
| 123 | deviceService.removeListener(deviceListener); | 142 | deviceService.removeListener(deviceListener); |
| 124 | hostService.removeListener(hostListener); | 143 | hostService.removeListener(hostListener); |
| 144 | + partitionService.removeListener(partitionListener); | ||
| 125 | log.info("Stopped"); | 145 | log.info("Stopped"); |
| 126 | } | 146 | } |
| 127 | 147 | ||
| ... | @@ -268,7 +288,7 @@ public class ObjectiveTracker implements ObjectiveTrackerService { | ... | @@ -268,7 +288,7 @@ public class ObjectiveTracker implements ObjectiveTrackerService { |
| 268 | 288 | ||
| 269 | private void updateTrackedResources(ApplicationId appId, boolean track) { | 289 | private void updateTrackedResources(ApplicationId appId, boolean track) { |
| 270 | if (intentService == null) { | 290 | if (intentService == null) { |
| 271 | - log.debug("Intent service is not bound yet"); | 291 | + log.warn("Intent service is not bound yet"); |
| 272 | return; | 292 | return; |
| 273 | } | 293 | } |
| 274 | intentService.getIntents().forEach(intent -> { | 294 | intentService.getIntents().forEach(intent -> { |
| ... | @@ -342,4 +362,52 @@ public class ObjectiveTracker implements ObjectiveTrackerService { | ... | @@ -342,4 +362,52 @@ public class ObjectiveTracker implements ObjectiveTrackerService { |
| 342 | executorService.execute(new DeviceAvailabilityHandler(id, available)); | 362 | executorService.execute(new DeviceAvailabilityHandler(id, available)); |
| 343 | } | 363 | } |
| 344 | } | 364 | } |
| 365 | + | ||
| 366 | + protected void doIntentUpdate() { | ||
| 367 | + updateScheduled.set(false); | ||
| 368 | + if (intentService == null) { | ||
| 369 | + log.warn("Intent service is not bound yet"); | ||
| 370 | + return; | ||
| 371 | + } | ||
| 372 | + try { | ||
| 373 | + //FIXME very inefficient | ||
| 374 | + for (Intent intent : intentService.getIntents()) { | ||
| 375 | + try { | ||
| 376 | + if (intentService.isLocal(intent.key())) { | ||
| 377 | + log.warn("intent {}, old: {}, new: {}", | ||
| 378 | + intent.key(), intentsByDevice.values().contains(intent.key()), true); | ||
| 379 | + addTrackedResources(intent.key(), intent.resources()); | ||
| 380 | + intentService.getInstallableIntents(intent.key()).stream() | ||
| 381 | + .forEach(installable -> | ||
| 382 | + addTrackedResources(intent.key(), installable.resources())); | ||
| 383 | + } else { | ||
| 384 | + log.warn("intent {}, old: {}, new: {}", | ||
| 385 | + intent.key(), intentsByDevice.values().contains(intent.key()), false); | ||
| 386 | + removeTrackedResources(intent.key(), intent.resources()); | ||
| 387 | + intentService.getInstallableIntents(intent.key()).stream() | ||
| 388 | + .forEach(installable -> | ||
| 389 | + removeTrackedResources(intent.key(), installable.resources())); | ||
| 390 | + } | ||
| 391 | + } catch (NullPointerException npe) { | ||
| 392 | + log.warn("intent error {}", intent.key(), npe); | ||
| 393 | + } | ||
| 394 | + } | ||
| 395 | + } catch (Exception e) { | ||
| 396 | + log.warn("Exception caught during update task", e); | ||
| 397 | + } | ||
| 398 | + } | ||
| 399 | + | ||
| 400 | + private void scheduleIntentUpdate(int afterDelaySec) { | ||
| 401 | + if (updateScheduled.compareAndSet(false, true)) { | ||
| 402 | + executor.schedule(this::doIntentUpdate, afterDelaySec, TimeUnit.SECONDS); | ||
| 403 | + } | ||
| 404 | + } | ||
| 405 | + | ||
| 406 | + private final class InternalPartitionListener implements PartitionEventListener { | ||
| 407 | + @Override | ||
| 408 | + public void event(PartitionEvent event) { | ||
| 409 | + log.warn("got message {}", event.subject()); | ||
| 410 | + scheduleIntentUpdate(1); | ||
| 411 | + } | ||
| 412 | + } | ||
| 345 | } | 413 | } | ... | ... |
| ... | @@ -30,7 +30,11 @@ import org.onosproject.cluster.LeadershipEvent; | ... | @@ -30,7 +30,11 @@ import org.onosproject.cluster.LeadershipEvent; |
| 30 | import org.onosproject.cluster.LeadershipEventListener; | 30 | import org.onosproject.cluster.LeadershipEventListener; |
| 31 | import org.onosproject.cluster.LeadershipService; | 31 | import org.onosproject.cluster.LeadershipService; |
| 32 | import org.onosproject.cluster.NodeId; | 32 | import org.onosproject.cluster.NodeId; |
| 33 | +import org.onosproject.event.EventDeliveryService; | ||
| 34 | +import org.onosproject.event.ListenerRegistry; | ||
| 33 | import org.onosproject.net.intent.Key; | 35 | import org.onosproject.net.intent.Key; |
| 36 | +import org.onosproject.net.intent.PartitionEvent; | ||
| 37 | +import org.onosproject.net.intent.PartitionEventListener; | ||
| 34 | import org.onosproject.net.intent.PartitionService; | 38 | import org.onosproject.net.intent.PartitionService; |
| 35 | import org.slf4j.Logger; | 39 | import org.slf4j.Logger; |
| 36 | import org.slf4j.LoggerFactory; | 40 | import org.slf4j.LoggerFactory; |
| ... | @@ -58,6 +62,9 @@ public class PartitionManager implements PartitionService { | ... | @@ -58,6 +62,9 @@ public class PartitionManager implements PartitionService { |
| 58 | @Reference(cardinality = ReferenceCardinality.MANDATORY_UNARY) | 62 | @Reference(cardinality = ReferenceCardinality.MANDATORY_UNARY) |
| 59 | protected ClusterService clusterService; | 63 | protected ClusterService clusterService; |
| 60 | 64 | ||
| 65 | + @Reference(cardinality = ReferenceCardinality.MANDATORY_UNARY) | ||
| 66 | + protected EventDeliveryService eventDispatcher; | ||
| 67 | + | ||
| 61 | protected final AtomicBoolean rebalanceScheduled = new AtomicBoolean(false); | 68 | protected final AtomicBoolean rebalanceScheduled = new AtomicBoolean(false); |
| 62 | 69 | ||
| 63 | static final int NUM_PARTITIONS = 14; | 70 | static final int NUM_PARTITIONS = 14; |
| ... | @@ -67,6 +74,7 @@ public class PartitionManager implements PartitionService { | ... | @@ -67,6 +74,7 @@ public class PartitionManager implements PartitionService { |
| 67 | 74 | ||
| 68 | private static final String ELECTION_PREFIX = "intent-partition-"; | 75 | private static final String ELECTION_PREFIX = "intent-partition-"; |
| 69 | 76 | ||
| 77 | + private ListenerRegistry<PartitionEvent, PartitionEventListener> listenerRegistry; | ||
| 70 | private LeadershipEventListener leaderListener = new InternalLeadershipListener(); | 78 | private LeadershipEventListener leaderListener = new InternalLeadershipListener(); |
| 71 | private ClusterEventListener clusterListener = new InternalClusterEventListener(); | 79 | private ClusterEventListener clusterListener = new InternalClusterEventListener(); |
| 72 | 80 | ||
| ... | @@ -78,6 +86,9 @@ public class PartitionManager implements PartitionService { | ... | @@ -78,6 +86,9 @@ public class PartitionManager implements PartitionService { |
| 78 | leadershipService.addListener(leaderListener); | 86 | leadershipService.addListener(leaderListener); |
| 79 | clusterService.addListener(clusterListener); | 87 | clusterService.addListener(clusterListener); |
| 80 | 88 | ||
| 89 | + listenerRegistry = new ListenerRegistry<>(); | ||
| 90 | + eventDispatcher.addSink(PartitionEvent.class, listenerRegistry); | ||
| 91 | + | ||
| 81 | for (int i = 0; i < NUM_PARTITIONS; i++) { | 92 | for (int i = 0; i < NUM_PARTITIONS; i++) { |
| 82 | leadershipService.runForLeadership(getPartitionPath(i)); | 93 | leadershipService.runForLeadership(getPartitionPath(i)); |
| 83 | } | 94 | } |
| ... | @@ -90,6 +101,7 @@ public class PartitionManager implements PartitionService { | ... | @@ -90,6 +101,7 @@ public class PartitionManager implements PartitionService { |
| 90 | public void deactivate() { | 101 | public void deactivate() { |
| 91 | executor.shutdownNow(); | 102 | executor.shutdownNow(); |
| 92 | 103 | ||
| 104 | + eventDispatcher.removeSink(PartitionEvent.class); | ||
| 93 | leadershipService.removeListener(leaderListener); | 105 | leadershipService.removeListener(leaderListener); |
| 94 | clusterService.removeListener(clusterListener); | 106 | clusterService.removeListener(clusterListener); |
| 95 | } | 107 | } |
| ... | @@ -133,6 +145,16 @@ public class PartitionManager implements PartitionService { | ... | @@ -133,6 +145,16 @@ public class PartitionManager implements PartitionService { |
| 133 | return leadershipService.getLeader(getPartitionPath(getPartitionForKey(intentKey))); | 145 | return leadershipService.getLeader(getPartitionPath(getPartitionForKey(intentKey))); |
| 134 | } | 146 | } |
| 135 | 147 | ||
| 148 | + @Override | ||
| 149 | + public void addListener(PartitionEventListener listener) { | ||
| 150 | + listenerRegistry.addListener(listener); | ||
| 151 | + } | ||
| 152 | + | ||
| 153 | + @Override | ||
| 154 | + public void removeListener(PartitionEventListener listener) { | ||
| 155 | + listenerRegistry.removeListener(listener); | ||
| 156 | + } | ||
| 157 | + | ||
| 136 | protected void doRebalance() { | 158 | protected void doRebalance() { |
| 137 | rebalanceScheduled.set(false); | 159 | rebalanceScheduled.set(false); |
| 138 | try { | 160 | try { |
| ... | @@ -203,6 +225,9 @@ public class PartitionManager implements PartitionService { | ... | @@ -203,6 +225,9 @@ public class PartitionManager implements PartitionService { |
| 203 | 225 | ||
| 204 | // See if we need to let some partitions go | 226 | // See if we need to let some partitions go |
| 205 | scheduleRebalance(0); | 227 | scheduleRebalance(0); |
| 228 | + | ||
| 229 | + eventDispatcher.post(new PartitionEvent(PartitionEvent.Type.LEADER_CHANGED, | ||
| 230 | + leadership.topic())); | ||
| 206 | } | 231 | } |
| 207 | } | 232 | } |
| 208 | } | 233 | } | ... | ... |
| ... | @@ -28,6 +28,7 @@ import org.onosproject.cluster.LeadershipEventListener; | ... | @@ -28,6 +28,7 @@ import org.onosproject.cluster.LeadershipEventListener; |
| 28 | import org.onosproject.cluster.LeadershipService; | 28 | import org.onosproject.cluster.LeadershipService; |
| 29 | import org.onosproject.cluster.LeadershipServiceAdapter; | 29 | import org.onosproject.cluster.LeadershipServiceAdapter; |
| 30 | import org.onosproject.cluster.NodeId; | 30 | import org.onosproject.cluster.NodeId; |
| 31 | +import org.onosproject.common.event.impl.TestEventDispatcher; | ||
| 31 | import org.onosproject.net.intent.Key; | 32 | import org.onosproject.net.intent.Key; |
| 32 | 33 | ||
| 33 | import java.util.HashMap; | 34 | import java.util.HashMap; |
| ... | @@ -86,6 +87,7 @@ public class PartitionManagerTest { | ... | @@ -86,6 +87,7 @@ public class PartitionManagerTest { |
| 86 | 87 | ||
| 87 | partitionManager.clusterService = new TestClusterService(); | 88 | partitionManager.clusterService = new TestClusterService(); |
| 88 | partitionManager.leadershipService = leadershipService; | 89 | partitionManager.leadershipService = leadershipService; |
| 90 | + partitionManager.eventDispatcher = new TestEventDispatcher(); | ||
| 89 | } | 91 | } |
| 90 | 92 | ||
| 91 | /** | 93 | /** | ... | ... |
| ... | @@ -19,6 +19,8 @@ import com.google.common.collect.ImmutableSet; | ... | @@ -19,6 +19,8 @@ import com.google.common.collect.ImmutableSet; |
| 19 | import org.apache.felix.scr.annotations.Activate; | 19 | import org.apache.felix.scr.annotations.Activate; |
| 20 | import org.apache.felix.scr.annotations.Component; | 20 | import org.apache.felix.scr.annotations.Component; |
| 21 | import org.apache.felix.scr.annotations.Deactivate; | 21 | import org.apache.felix.scr.annotations.Deactivate; |
| 22 | +import org.apache.felix.scr.annotations.Reference; | ||
| 23 | +import org.apache.felix.scr.annotations.ReferenceCardinality; | ||
| 22 | import org.apache.felix.scr.annotations.Service; | 24 | import org.apache.felix.scr.annotations.Service; |
| 23 | import org.joda.time.DateTime; | 25 | import org.joda.time.DateTime; |
| 24 | import org.onlab.packet.IpAddress; | 26 | import org.onlab.packet.IpAddress; |
| ... | @@ -28,7 +30,11 @@ import org.onosproject.cluster.ClusterStoreDelegate; | ... | @@ -28,7 +30,11 @@ import org.onosproject.cluster.ClusterStoreDelegate; |
| 28 | import org.onosproject.cluster.ControllerNode; | 30 | import org.onosproject.cluster.ControllerNode; |
| 29 | import org.onosproject.cluster.DefaultControllerNode; | 31 | import org.onosproject.cluster.DefaultControllerNode; |
| 30 | import org.onosproject.cluster.NodeId; | 32 | import org.onosproject.cluster.NodeId; |
| 33 | +import org.onosproject.event.EventDeliveryService; | ||
| 34 | +import org.onosproject.event.ListenerRegistry; | ||
| 31 | import org.onosproject.net.intent.Key; | 35 | import org.onosproject.net.intent.Key; |
| 36 | +import org.onosproject.net.intent.PartitionEvent; | ||
| 37 | +import org.onosproject.net.intent.PartitionEventListener; | ||
| 32 | import org.onosproject.net.intent.PartitionService; | 38 | import org.onosproject.net.intent.PartitionService; |
| 33 | import org.onosproject.store.AbstractStore; | 39 | import org.onosproject.store.AbstractStore; |
| 34 | import org.slf4j.Logger; | 40 | import org.slf4j.Logger; |
| ... | @@ -55,14 +61,24 @@ public class SimpleClusterStore | ... | @@ -55,14 +61,24 @@ public class SimpleClusterStore |
| 55 | 61 | ||
| 56 | private final DateTime creationTime = DateTime.now(); | 62 | private final DateTime creationTime = DateTime.now(); |
| 57 | 63 | ||
| 64 | + @Reference(cardinality = ReferenceCardinality.MANDATORY_UNARY) | ||
| 65 | + protected EventDeliveryService eventDispatcher; | ||
| 66 | + | ||
| 67 | + private ListenerRegistry<PartitionEvent, PartitionEventListener> listenerRegistry; | ||
| 68 | + | ||
| 58 | @Activate | 69 | @Activate |
| 59 | public void activate() { | 70 | public void activate() { |
| 60 | instance = new DefaultControllerNode(new NodeId("local"), LOCALHOST); | 71 | instance = new DefaultControllerNode(new NodeId("local"), LOCALHOST); |
| 72 | + | ||
| 73 | + listenerRegistry = new ListenerRegistry<>(); | ||
| 74 | + eventDispatcher.addSink(PartitionEvent.class, listenerRegistry); | ||
| 75 | + | ||
| 61 | log.info("Started"); | 76 | log.info("Started"); |
| 62 | } | 77 | } |
| 63 | 78 | ||
| 64 | @Deactivate | 79 | @Deactivate |
| 65 | public void deactivate() { | 80 | public void deactivate() { |
| 81 | + eventDispatcher.removeSink(PartitionEvent.class); | ||
| 66 | log.info("Stopped"); | 82 | log.info("Stopped"); |
| 67 | } | 83 | } |
| 68 | 84 | ||
| ... | @@ -110,4 +126,14 @@ public class SimpleClusterStore | ... | @@ -110,4 +126,14 @@ public class SimpleClusterStore |
| 110 | public NodeId getLeader(Key intentKey) { | 126 | public NodeId getLeader(Key intentKey) { |
| 111 | return instance.id(); | 127 | return instance.id(); |
| 112 | } | 128 | } |
| 129 | + | ||
| 130 | + @Override | ||
| 131 | + public void addListener(PartitionEventListener listener) { | ||
| 132 | + listenerRegistry.addListener(listener); | ||
| 133 | + } | ||
| 134 | + | ||
| 135 | + @Override | ||
| 136 | + public void removeListener(PartitionEventListener listener) { | ||
| 137 | + listenerRegistry.removeListener(listener); | ||
| 138 | + } | ||
| 113 | } | 139 | } | ... | ... |
-
Please register or login to post a comment