blob: 9edcc1c08b87caef4343ab9feb881d49fb818c65 [file] [log] [blame]
Thomas Vachuska4f1a60c2014-10-28 13:39:07 -07001/*
Brian O'Connora09fe5b2017-08-03 21:12:30 -07002 * Copyright 2014-present Open Networking Foundation
Thomas Vachuska4f1a60c2014-10-28 13:39:07 -07003 *
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 */
Brian O'Connorabafb502014-12-02 22:26:20 -080016package org.onosproject.store.device.impl;
Madan Jampani61056bc2014-09-27 09:07:26 -070017
18import static org.slf4j.LoggerFactory.getLogger;
19
Yuta HIGUCHI33a847b2017-04-14 15:44:42 -070020import java.util.Optional;
21import java.util.concurrent.TimeUnit;
Madan Jampani15d773c2015-02-25 15:31:55 -080022import java.util.concurrent.atomic.AtomicLong;
Madan Jampani61056bc2014-09-27 09:07:26 -070023
24import org.apache.felix.scr.annotations.Activate;
25import org.apache.felix.scr.annotations.Component;
26import org.apache.felix.scr.annotations.Deactivate;
Madan Jampani565a66a2015-07-25 17:01:13 -070027import org.apache.felix.scr.annotations.Reference;
28import org.apache.felix.scr.annotations.ReferenceCardinality;
Madan Jampani61056bc2014-09-27 09:07:26 -070029import org.apache.felix.scr.annotations.Service;
Madan Jampani565a66a2015-07-25 17:01:13 -070030import org.onosproject.cluster.ClusterService;
31import org.onosproject.cluster.NodeId;
Yuta HIGUCHI33a847b2017-04-14 15:44:42 -070032import org.onosproject.mastership.MastershipEvent;
33import org.onosproject.mastership.MastershipEvent.Type;
34import org.onosproject.mastership.MastershipListener;
35import org.onosproject.mastership.MastershipService;
Brian O'Connorabafb502014-12-02 22:26:20 -080036import org.onosproject.mastership.MastershipTerm;
Madan Jampani565a66a2015-07-25 17:01:13 -070037import org.onosproject.mastership.MastershipTermService;
Brian O'Connorabafb502014-12-02 22:26:20 -080038import org.onosproject.net.DeviceId;
Brian O'Connorabafb502014-12-02 22:26:20 -080039import org.onosproject.net.device.DeviceClockService;
40import org.onosproject.store.Timestamp;
41import org.onosproject.store.impl.MastershipBasedTimestamp;
Madan Jampani61056bc2014-09-27 09:07:26 -070042import org.slf4j.Logger;
43
Yuta HIGUCHI33a847b2017-04-14 15:44:42 -070044import com.google.common.cache.Cache;
45import com.google.common.cache.CacheBuilder;
46
Yuta HIGUCHI67a527f2014-10-02 22:23:54 -070047/**
48 * Clock service to issue Timestamp based on Device Mastership.
49 */
Madan Jampani61056bc2014-09-27 09:07:26 -070050@Component(immediate = true)
51@Service
Madan Jampani565a66a2015-07-25 17:01:13 -070052public class DeviceClockManager implements DeviceClockService {
Yuta HIGUCHI2e963892014-09-27 13:00:39 -070053
Madan Jampani61056bc2014-09-27 09:07:26 -070054 private final Logger log = getLogger(getClass());
55
Madan Jampani565a66a2015-07-25 17:01:13 -070056 @Reference(cardinality = ReferenceCardinality.MANDATORY_UNARY)
57 protected MastershipTermService mastershipTermService;
58
59 @Reference(cardinality = ReferenceCardinality.MANDATORY_UNARY)
Yuta HIGUCHI33a847b2017-04-14 15:44:42 -070060 protected MastershipService mastershipService;
61
62 @Reference(cardinality = ReferenceCardinality.MANDATORY_UNARY)
Madan Jampani565a66a2015-07-25 17:01:13 -070063 protected ClusterService clusterService;
64
65 protected NodeId localNodeId;
66
Madan Jampani15d773c2015-02-25 15:31:55 -080067 private final AtomicLong ticker = new AtomicLong(0);
Yuta HIGUCHI2e963892014-09-27 13:00:39 -070068
Yuta HIGUCHI33a847b2017-04-14 15:44:42 -070069 // Map from DeviceId -> last known term number for this node.
70 // using Cache class but using it as Map which will age out old entries
71 private final Cache<DeviceId, Long> myLastKnownTerm =
72 CacheBuilder.newBuilder()
73 .expireAfterAccess(5, TimeUnit.MINUTES)
74 .build();
75
76 private MastershipListener listener;
77
Madan Jampani61056bc2014-09-27 09:07:26 -070078 @Activate
79 public void activate() {
Madan Jampani565a66a2015-07-25 17:01:13 -070080 localNodeId = clusterService.getLocalNode().id();
Yuta HIGUCHI33a847b2017-04-14 15:44:42 -070081
82 listener = new InnerMastershipListener();
83 mastershipService.addListener(listener);
Madan Jampani61056bc2014-09-27 09:07:26 -070084 log.info("Started");
85 }
86
87 @Deactivate
88 public void deactivate() {
Yuta HIGUCHI33a847b2017-04-14 15:44:42 -070089 mastershipService.removeListener(listener);
Madan Jampani61056bc2014-09-27 09:07:26 -070090 log.info("Stopped");
91 }
Yuta HIGUCHI2e963892014-09-27 13:00:39 -070092
Madan Jampani61056bc2014-09-27 09:07:26 -070093 @Override
94 public Timestamp getTimestamp(DeviceId deviceId) {
Yuta HIGUCHI33a847b2017-04-14 15:44:42 -070095 Long termNumber = refreshLastKnownTerm(deviceId);
96 if (termNumber == null) {
97 log.warn("Requested timestamp for {} which {}"
98 + "doesn't have known recent mastership term",
99 deviceId, localNodeId);
HIGUCHI Yuta3a6f7cd2015-02-25 18:51:25 -0800100 throw new IllegalStateException("Requesting timestamp for " + deviceId + " without mastership");
Madan Jampani61056bc2014-09-27 09:07:26 -0700101 }
Yuta HIGUCHI33a847b2017-04-14 15:44:42 -0700102 return new MastershipBasedTimestamp(termNumber, ticker.incrementAndGet());
Madan Jampani61056bc2014-09-27 09:07:26 -0700103 }
104
105 @Override
Yuta HIGUCHI13c0b872014-10-30 18:09:22 -0700106 public boolean isTimestampAvailable(DeviceId deviceId) {
Yuta HIGUCHI33a847b2017-04-14 15:44:42 -0700107 return myLastKnownTerm.getIfPresent(deviceId) != null ||
108 refreshLastKnownTerm(deviceId) != null;
109 }
110
111 /**
112 * Refreshes this node's last known term number to the latest state.
113 *
114 * @param deviceId of the Device to refresh mastership term
115 * @return latest mastership term number or null if this node
116 * did not have a term number recently
117 */
118 private Long refreshLastKnownTerm(DeviceId deviceId) {
Madan Jampani565a66a2015-07-25 17:01:13 -0700119 MastershipTerm term = mastershipTermService.getMastershipTerm(deviceId);
Yuta HIGUCHI33a847b2017-04-14 15:44:42 -0700120 return myLastKnownTerm.asMap().compute(deviceId, (key, old) -> {
121 if (old == null) {
122 return Optional.ofNullable(term)
123 .filter(t -> localNodeId.equals(t.master()))
124 .map(MastershipTerm::termNumber)
125 .orElse(null);
126 }
127 return Optional.ofNullable(term)
128 .filter(t -> localNodeId.equals(t.master()))
129 .map(MastershipTerm::termNumber)
130 // TODO make following integer wrap-safe
131 .map(tn -> Math.max(old, tn))
132 .orElse(old);
133 });
134 }
135
136 /**
137 * Refreshes {@link DeviceClockManager#myLastKnownTerm} on Master update event.
138 */
139 private final class InnerMastershipListener implements MastershipListener {
140
141 @Override
142 public boolean isRelevant(MastershipEvent event) {
143 return event.type() == Type.MASTER_CHANGED;
144 }
145
146 @Override
147 public void event(MastershipEvent event) {
148 if (localNodeId.equals(event.roleInfo().master())) {
149 refreshLastKnownTerm(event.subject());
150 }
151 }
Yuta HIGUCHI13c0b872014-10-30 18:09:22 -0700152 }
Madan Jampani61056bc2014-09-27 09:07:26 -0700153}