blob: f3a5cf0c1fd371015aa321ffc1a3e9defadcf357 [file] [log] [blame]
Thomas Vachuska4f1a60c2014-10-28 13:39:07 -07001/*
2 * Copyright 2014 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 */
Brian O'Connorabafb502014-12-02 22:26:20 -080016package org.onosproject.store.trivial.impl;
Brian O'Connor66630c82014-10-02 21:08:19 -070017
Thomas Vachuskac96058a2014-10-20 23:00:16 -070018import com.google.common.collect.ImmutableSet;
alshabiba9819bf2014-11-30 18:15:52 -080019import com.google.common.collect.Lists;
Brian O'Connor66630c82014-10-02 21:08:19 -070020import org.apache.felix.scr.annotations.Activate;
21import org.apache.felix.scr.annotations.Component;
22import org.apache.felix.scr.annotations.Deactivate;
23import org.apache.felix.scr.annotations.Service;
Sho SHIMIZU64ae11c2014-12-03 15:17:47 -080024import org.onosproject.net.intent.BatchWrite;
Brian O'Connorcff03322015-02-03 15:28:59 -080025import org.onosproject.net.intent.BatchWrite.Operation;
Brian O'Connorabafb502014-12-02 22:26:20 -080026import org.onosproject.net.intent.Intent;
Brian O'Connorcff03322015-02-03 15:28:59 -080027import org.onosproject.net.intent.IntentData;
Brian O'Connorabafb502014-12-02 22:26:20 -080028import org.onosproject.net.intent.IntentEvent;
29import org.onosproject.net.intent.IntentId;
30import org.onosproject.net.intent.IntentState;
31import org.onosproject.net.intent.IntentStore;
32import org.onosproject.net.intent.IntentStoreDelegate;
Brian O'Connorabafb502014-12-02 22:26:20 -080033import org.onosproject.store.AbstractStore;
Brian O'Connor66630c82014-10-02 21:08:19 -070034import org.slf4j.Logger;
35
Sho SHIMIZU2bb988b2015-01-20 13:45:35 -080036import java.util.Collections;
Thomas Vachuskac96058a2014-10-20 23:00:16 -070037import java.util.List;
38import java.util.Map;
39import java.util.concurrent.ConcurrentHashMap;
40
Brian O'Connorcff03322015-02-03 15:28:59 -080041import static com.google.common.base.Preconditions.*;
Brian O'Connorabafb502014-12-02 22:26:20 -080042import static org.onosproject.net.intent.IntentState.WITHDRAWN;
Thomas Vachuskac96058a2014-10-20 23:00:16 -070043import static org.slf4j.LoggerFactory.getLogger;
Brian O'Connor66630c82014-10-02 21:08:19 -070044
45@Component(immediate = true)
46@Service
47public class SimpleIntentStore
tom85258ee2014-10-07 00:10:02 -070048 extends AbstractStore<IntentEvent, IntentStoreDelegate>
49 implements IntentStore {
Brian O'Connor66630c82014-10-02 21:08:19 -070050
51 private final Logger log = getLogger(getClass());
Brian O'Connorcff03322015-02-03 15:28:59 -080052
53 // current state maps FIXME.. make this a IntentData map
Jonathan Hart11096402014-10-20 17:31:49 -070054 private final Map<IntentId, Intent> intents = new ConcurrentHashMap<>();
55 private final Map<IntentId, IntentState> states = new ConcurrentHashMap<>();
Brian O'Connorfa81eae2014-10-30 13:20:05 -070056 private final Map<IntentId, List<Intent>> installable = new ConcurrentHashMap<>();
57
Brian O'Connorcff03322015-02-03 15:28:59 -080058 private final Map<String, IntentData> pending = new ConcurrentHashMap<>(); //String is "key"
Brian O'Connor66630c82014-10-02 21:08:19 -070059
60 @Activate
61 public void activate() {
62 log.info("Started");
63 }
64
65 @Deactivate
66 public void deactivate() {
67 log.info("Stopped");
68 }
69
Jonathan Hartc0363672015-01-20 16:21:08 -080070 private void createIntent(Intent intent) {
Brian O'Connorfa81eae2014-10-30 13:20:05 -070071 if (intents.containsKey(intent.id())) {
alshabiba9819bf2014-11-30 18:15:52 -080072 return;
Brian O'Connorfa81eae2014-10-30 13:20:05 -070073 }
tom85258ee2014-10-07 00:10:02 -070074 intents.put(intent.id(), intent);
Brian O'Connor7a71d5d2014-12-02 00:12:27 -080075 this.setState(intent, IntentState.INSTALL_REQ);
Brian O'Connor66630c82014-10-02 21:08:19 -070076 }
77
Jonathan Hartc0363672015-01-20 16:21:08 -080078 private void removeIntent(IntentId intentId) {
Thomas Vachuskae4b6bb22014-11-25 17:09:43 -080079 checkState(getIntentState(intentId) == WITHDRAWN,
80 "Intent state for {} is not WITHDRAWN.", intentId);
81 intents.remove(intentId);
Brian O'Connor66630c82014-10-02 21:08:19 -070082 installable.remove(intentId);
Brian O'Connor66630c82014-10-02 21:08:19 -070083 states.remove(intentId);
Brian O'Connor66630c82014-10-02 21:08:19 -070084 }
85
86 @Override
87 public long getIntentCount() {
88 return intents.size();
89 }
90
91 @Override
92 public Iterable<Intent> getIntents() {
93 return ImmutableSet.copyOf(intents.values());
94 }
95
96 @Override
97 public Intent getIntent(IntentId intentId) {
98 return intents.get(intentId);
99 }
100
101 @Override
102 public IntentState getIntentState(IntentId id) {
103 return states.get(id);
104 }
105
Jonathan Hartdac30082015-01-22 16:02:48 -0800106 private void setState(Intent intent, IntentState state) {
tom85258ee2014-10-07 00:10:02 -0700107 IntentId id = intent.id();
108 states.put(id, state);
Pavlin Radoslavov2ca9cf22014-10-22 10:39:40 -0700109 IntentEvent.Type type = null;
110
111 switch (state) {
Brian O'Connor7a71d5d2014-12-02 00:12:27 -0800112 case INSTALL_REQ:
113 type = IntentEvent.Type.INSTALL_REQ;
Pavlin Radoslavov2ca9cf22014-10-22 10:39:40 -0700114 break;
115 case INSTALLED:
116 type = IntentEvent.Type.INSTALLED;
117 break;
118 case FAILED:
119 type = IntentEvent.Type.FAILED;
120 break;
Brian O'Connor7a71d5d2014-12-02 00:12:27 -0800121 case WITHDRAW_REQ:
122 type = IntentEvent.Type.WITHDRAW_REQ;
123 break;
Pavlin Radoslavov2ca9cf22014-10-22 10:39:40 -0700124 case WITHDRAWN:
125 type = IntentEvent.Type.WITHDRAWN;
126 break;
127 default:
128 break;
129 }
alshabiba9819bf2014-11-30 18:15:52 -0800130 if (type != null) {
131 notifyDelegate(new IntentEvent(type, intent));
Pavlin Radoslavov2ca9cf22014-10-22 10:39:40 -0700132 }
Brian O'Connor66630c82014-10-02 21:08:19 -0700133 }
134
Jonathan Hartdac30082015-01-22 16:02:48 -0800135 private void setInstallableIntents(IntentId intentId, List<Intent> result) {
Brian O'Connor66630c82014-10-02 21:08:19 -0700136 installable.put(intentId, result);
Brian O'Connor66630c82014-10-02 21:08:19 -0700137 }
138
139 @Override
Thomas Vachuskac96058a2014-10-20 23:00:16 -0700140 public List<Intent> getInstallableIntents(IntentId intentId) {
Brian O'Connor66630c82014-10-02 21:08:19 -0700141 return installable.get(intentId);
142 }
143
Jonathan Hartdac30082015-01-22 16:02:48 -0800144 private void removeInstalledIntents(IntentId intentId) {
Brian O'Connor66630c82014-10-02 21:08:19 -0700145 installable.remove(intentId);
146 }
Jonathan Hartdac30082015-01-22 16:02:48 -0800147
alshabiba9819bf2014-11-30 18:15:52 -0800148 /**
149 * Execute writes in a batch.
150 *
151 * @param batch BatchWrite to execute
152 * @return failed operations
153 */
154 @Override
155 public List<Operation> batchWrite(BatchWrite batch) {
Sho SHIMIZU2bb988b2015-01-20 13:45:35 -0800156 if (batch.isEmpty()) {
157 return Collections.emptyList();
158 }
159
alshabiba9819bf2014-11-30 18:15:52 -0800160 List<Operation> failed = Lists.newArrayList();
161 for (Operation op : batch.operations()) {
162 switch (op.type()) {
163 case CREATE_INTENT:
164 checkArgument(op.args().size() == 1,
165 "CREATE_INTENT takes 1 argument. %s", op);
166 Intent intent = (Intent) op.args().get(0);
167 // TODO: what if it failed?
168 createIntent(intent);
169 break;
Brian O'Connor66630c82014-10-02 21:08:19 -0700170
alshabiba9819bf2014-11-30 18:15:52 -0800171 case REMOVE_INTENT:
172 checkArgument(op.args().size() == 1,
173 "REMOVE_INTENT takes 1 argument. %s", op);
174 IntentId intentId = (IntentId) op.args().get(0);
175 removeIntent(intentId);
176 break;
177
178 case REMOVE_INSTALLED:
179 checkArgument(op.args().size() == 1,
180 "REMOVE_INSTALLED takes 1 argument. %s", op);
181 intentId = (IntentId) op.args().get(0);
182 removeInstalledIntents(intentId);
183 break;
184
185 case SET_INSTALLABLE:
186 checkArgument(op.args().size() == 2,
187 "SET_INSTALLABLE takes 2 arguments. %s", op);
188 intentId = (IntentId) op.args().get(0);
189 @SuppressWarnings("unchecked")
190 List<Intent> installableIntents = (List<Intent>) op.args().get(1);
191 setInstallableIntents(intentId, installableIntents);
192 break;
193
194 case SET_STATE:
195 checkArgument(op.args().size() == 2,
196 "SET_STATE takes 2 arguments. %s", op);
197 intent = (Intent) op.args().get(0);
198 IntentState newState = (IntentState) op.args().get(1);
199 setState(intent, newState);
200 break;
201
202 default:
203 break;
204 }
205 }
206 return failed;
207 }
Brian O'Connorcff03322015-02-03 15:28:59 -0800208
209 @Override
210 public void addPending(IntentData data) {
211 //FIXME need to compare versions
212 pending.put(data.key(), data);
213 checkNotNull(delegate, "Store delegate is not set")
214 .process(data);
215 }
216 // FIXME!!! pending.remove(intent.key()); // TODO check version
217
218
219 @Override
220 public boolean isMaster(Intent intent) {
221 return true;
222 }
Brian O'Connor66630c82014-10-02 21:08:19 -0700223}