blob: 3c2b4133b94b53bb65038d6a071351d65e17dabe [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.net.intent.impl;
Brian O'Connor66630c82014-10-02 21:08:19 -070017
Brian O'Connor64a0369d2015-02-20 22:02:59 -080018import com.google.common.collect.ImmutableList;
Brian O'Connor66630c82014-10-02 21:08:19 -070019import org.apache.felix.scr.annotations.Activate;
20import org.apache.felix.scr.annotations.Component;
21import org.apache.felix.scr.annotations.Deactivate;
22import org.apache.felix.scr.annotations.Reference;
23import org.apache.felix.scr.annotations.ReferenceCardinality;
24import org.apache.felix.scr.annotations.Service;
Brian O'Connorabafb502014-12-02 22:26:20 -080025import org.onosproject.core.CoreService;
26import org.onosproject.core.IdGenerator;
27import org.onosproject.event.AbstractListenerRegistry;
28import org.onosproject.event.EventDeliveryService;
Brian O'Connor0e271dc2015-02-04 18:20:25 -080029import org.onosproject.net.flow.FlowRuleOperations;
Brian O'Connorabafb502014-12-02 22:26:20 -080030import org.onosproject.net.flow.FlowRuleService;
31import org.onosproject.net.intent.Intent;
32import org.onosproject.net.intent.IntentBatchDelegate;
Brian O'Connorabafb502014-12-02 22:26:20 -080033import org.onosproject.net.intent.IntentCompiler;
Brian O'Connorcff03322015-02-03 15:28:59 -080034import org.onosproject.net.intent.IntentData;
Brian O'Connorabafb502014-12-02 22:26:20 -080035import org.onosproject.net.intent.IntentEvent;
Brian O'Connorabafb502014-12-02 22:26:20 -080036import org.onosproject.net.intent.IntentExtensionService;
Brian O'Connorabafb502014-12-02 22:26:20 -080037import org.onosproject.net.intent.IntentInstaller;
38import org.onosproject.net.intent.IntentListener;
Brian O'Connorabafb502014-12-02 22:26:20 -080039import org.onosproject.net.intent.IntentService;
40import org.onosproject.net.intent.IntentState;
41import org.onosproject.net.intent.IntentStore;
Brian O'Connorabafb502014-12-02 22:26:20 -080042import org.onosproject.net.intent.IntentStoreDelegate;
Ray Milkeyf9af43c2015-02-09 16:45:48 -080043import org.onosproject.net.intent.Key;
Sho SHIMIZU36a8a6e2015-02-13 15:38:45 -080044import org.onosproject.net.intent.impl.phase.FinalIntentProcessPhase;
Sho SHIMIZUce49b602015-02-23 19:15:49 -080045import org.onosproject.net.intent.impl.phase.IntentProcessPhase;
Sho SHIMIZU662c3db2015-02-23 16:59:01 -080046import org.onosproject.net.intent.impl.phase.IntentWorker;
Brian O'Connor66630c82014-10-02 21:08:19 -070047import org.slf4j.Logger;
48
Brian O'Connor64a0369d2015-02-20 22:02:59 -080049import java.util.Collection;
Brian O'Connor64a0369d2015-02-20 22:02:59 -080050import java.util.EnumSet;
Brian O'Connor64a0369d2015-02-20 22:02:59 -080051import java.util.List;
52import java.util.Map;
Brian O'Connor64a0369d2015-02-20 22:02:59 -080053import java.util.concurrent.ExecutionException;
54import java.util.concurrent.ExecutorService;
55import java.util.concurrent.Future;
56import java.util.stream.Collectors;
Brian O'Connorfa81eae2014-10-30 13:20:05 -070057
Brian O'Connorfa81eae2014-10-30 13:20:05 -070058import static com.google.common.base.Preconditions.checkNotNull;
Yuta HIGUCHIc2bf3d82014-11-28 18:50:41 -080059import static java.util.concurrent.Executors.newFixedThreadPool;
Brian O'Connordb15b042015-02-04 14:59:28 -080060import static java.util.concurrent.Executors.newSingleThreadExecutor;
Brian O'Connorbdc7f002015-02-18 20:49:41 -080061import static org.onlab.util.Tools.groupedThreads;
Sho SHIMIZUb0a47d42015-02-19 13:26:30 -080062import static org.onosproject.net.intent.IntentState.FAILED;
63import static org.onosproject.net.intent.IntentState.INSTALL_REQ;
Sho SHIMIZUb0a47d42015-02-19 13:26:30 -080064import static org.onosproject.net.intent.IntentState.WITHDRAW_REQ;
Sho SHIMIZUce49b602015-02-23 19:15:49 -080065import static org.onosproject.net.intent.impl.phase.IntentProcessPhase.newInitialPhase;
Brian O'Connorfa81eae2014-10-30 13:20:05 -070066import static org.slf4j.LoggerFactory.getLogger;
Brian O'Connor66630c82014-10-02 21:08:19 -070067
68/**
69 * An implementation of Intent Manager.
70 */
71@Component(immediate = true)
72@Service
73public class IntentManager
74 implements IntentService, IntentExtensionService {
Sho SHIMIZU8b5051d2014-11-05 11:24:13 -080075 private static final Logger log = getLogger(IntentManager.class);
Brian O'Connor66630c82014-10-02 21:08:19 -070076
77 public static final String INTENT_NULL = "Intent cannot be null";
Ray Milkeyf9af43c2015-02-09 16:45:48 -080078 public static final String INTENT_ID_NULL = "Intent key cannot be null";
Brian O'Connor66630c82014-10-02 21:08:19 -070079
Yuta HIGUCHIc2bf3d82014-11-28 18:50:41 -080080 private static final int NUM_THREADS = 12;
81
Brian O'Connor7a71d5d2014-12-02 00:12:27 -080082 private static final EnumSet<IntentState> RECOMPILE
83 = EnumSet.of(INSTALL_REQ, FAILED, WITHDRAW_REQ);
Brian O'Connor7a71d5d2014-12-02 00:12:27 -080084
Brian O'Connor66630c82014-10-02 21:08:19 -070085 private final AbstractListenerRegistry<IntentEvent, IntentListener>
tom95329eb2014-10-06 08:40:06 -070086 listenerRegistry = new AbstractListenerRegistry<>();
Brian O'Connor66630c82014-10-02 21:08:19 -070087
Brian O'Connor520c0522014-11-23 23:50:47 -080088 @Reference(cardinality = ReferenceCardinality.MANDATORY_UNARY)
89 protected CoreService coreService;
Brian O'Connor66630c82014-10-02 21:08:19 -070090
91 @Reference(cardinality = ReferenceCardinality.MANDATORY_UNARY)
92 protected IntentStore store;
93
94 @Reference(cardinality = ReferenceCardinality.MANDATORY_UNARY)
tom85258ee2014-10-07 00:10:02 -070095 protected ObjectiveTrackerService trackerService;
tom95329eb2014-10-06 08:40:06 -070096
97 @Reference(cardinality = ReferenceCardinality.MANDATORY_UNARY)
Brian O'Connor66630c82014-10-02 21:08:19 -070098 protected EventDeliveryService eventDispatcher;
99
Brian O'Connorf2dbde52014-10-10 16:20:24 -0700100 @Reference(cardinality = ReferenceCardinality.MANDATORY_UNARY)
Sho SHIMIZU214ac322015-02-23 19:30:15 -0800101 protected FlowRuleService flowRuleService;
Brian O'Connorf2dbde52014-10-10 16:20:24 -0700102
Brian O'Connor520c0522014-11-23 23:50:47 -0800103
Brian O'Connordb15b042015-02-04 14:59:28 -0800104 private ExecutorService batchExecutor;
105 private ExecutorService workerExecutor;
Brian O'Connor520c0522014-11-23 23:50:47 -0800106
Sho SHIMIZUb0a47d42015-02-19 13:26:30 -0800107 private final CompilerRegistry compilerRegistry = new CompilerRegistry();
108 private final InstallerRegistry installerRegistry = new InstallerRegistry();
109 private final InternalIntentProcessor processor = new InternalIntentProcessor();
Brian O'Connor520c0522014-11-23 23:50:47 -0800110 private final IntentStoreDelegate delegate = new InternalStoreDelegate();
111 private final TopologyChangeDelegate topoDelegate = new InternalTopoChangeDelegate();
112 private final IntentBatchDelegate batchDelegate = new InternalBatchDelegate();
113 private IdGenerator idGenerator;
114
Brian O'Connorb499b352015-02-03 16:46:15 -0800115 private final IntentAccumulator accumulator = new IntentAccumulator(batchDelegate);
Brian O'Connorcff03322015-02-03 15:28:59 -0800116
Brian O'Connor66630c82014-10-02 21:08:19 -0700117 @Activate
118 public void activate() {
119 store.setDelegate(delegate);
tom95329eb2014-10-06 08:40:06 -0700120 trackerService.setDelegate(topoDelegate);
Brian O'Connor66630c82014-10-02 21:08:19 -0700121 eventDispatcher.addSink(IntentEvent.class, listenerRegistry);
Brian O'Connorbdc7f002015-02-18 20:49:41 -0800122 batchExecutor = newSingleThreadExecutor(groupedThreads("onos/intent", "batch"));
123 workerExecutor = newFixedThreadPool(NUM_THREADS, groupedThreads("onos/intent", "worker-%d"));
Brian O'Connor520c0522014-11-23 23:50:47 -0800124 idGenerator = coreService.getIdGenerator("intent-ids");
125 Intent.bindIdGenerator(idGenerator);
Brian O'Connor66630c82014-10-02 21:08:19 -0700126 log.info("Started");
127 }
128
129 @Deactivate
130 public void deactivate() {
131 store.unsetDelegate(delegate);
tom95329eb2014-10-06 08:40:06 -0700132 trackerService.unsetDelegate(topoDelegate);
Brian O'Connor66630c82014-10-02 21:08:19 -0700133 eventDispatcher.removeSink(IntentEvent.class);
Brian O'Connordb15b042015-02-04 14:59:28 -0800134 batchExecutor.shutdown();
Brian O'Connor520c0522014-11-23 23:50:47 -0800135 Intent.unbindIdGenerator(idGenerator);
Brian O'Connor66630c82014-10-02 21:08:19 -0700136 log.info("Stopped");
137 }
138
139 @Override
140 public void submit(Intent intent) {
141 checkNotNull(intent, INTENT_NULL);
Brian O'Connorcff03322015-02-03 15:28:59 -0800142 IntentData data = new IntentData(intent, IntentState.INSTALL_REQ, null);
Brian O'Connorcff03322015-02-03 15:28:59 -0800143 store.addPending(data);
Brian O'Connor66630c82014-10-02 21:08:19 -0700144 }
145
146 @Override
147 public void withdraw(Intent intent) {
148 checkNotNull(intent, INTENT_NULL);
Brian O'Connorcff03322015-02-03 15:28:59 -0800149 IntentData data = new IntentData(intent, IntentState.WITHDRAW_REQ, null);
Brian O'Connorcff03322015-02-03 15:28:59 -0800150 store.addPending(data);
Brian O'Connor66630c82014-10-02 21:08:19 -0700151 }
152
Brian O'Connor66630c82014-10-02 21:08:19 -0700153 @Override
Ray Milkeyf9af43c2015-02-09 16:45:48 -0800154 public Intent getIntent(Key key) {
155 return store.getIntent(key);
156 }
157
158 @Override
Brian O'Connor66630c82014-10-02 21:08:19 -0700159 public Iterable<Intent> getIntents() {
160 return store.getIntents();
161 }
162
163 @Override
164 public long getIntentCount() {
165 return store.getIntentCount();
166 }
167
168 @Override
Ray Milkeyf9af43c2015-02-09 16:45:48 -0800169 public IntentState getIntentState(Key intentKey) {
170 checkNotNull(intentKey, INTENT_ID_NULL);
171 return store.getIntentState(intentKey);
Brian O'Connor66630c82014-10-02 21:08:19 -0700172 }
173
174 @Override
Ray Milkeyf9af43c2015-02-09 16:45:48 -0800175 public List<Intent> getInstallableIntents(Key intentKey) {
176 checkNotNull(intentKey, INTENT_ID_NULL);
177 return store.getInstallableIntents(intentKey);
Thomas Vachuska10d4abc2014-10-21 12:47:26 -0700178 }
179
180 @Override
Brian O'Connorbe28a872015-02-19 21:44:37 -0800181 public boolean isLocal(Key intentKey) {
182 return store.isMaster(intentKey);
183 }
184
185 @Override
Brian O'Connor66630c82014-10-02 21:08:19 -0700186 public void addListener(IntentListener listener) {
187 listenerRegistry.addListener(listener);
188 }
189
190 @Override
191 public void removeListener(IntentListener listener) {
192 listenerRegistry.removeListener(listener);
193 }
194
195 @Override
196 public <T extends Intent> void registerCompiler(Class<T> cls, IntentCompiler<T> compiler) {
Sho SHIMIZUb0a47d42015-02-19 13:26:30 -0800197 compilerRegistry.registerCompiler(cls, compiler);
Brian O'Connor66630c82014-10-02 21:08:19 -0700198 }
199
200 @Override
201 public <T extends Intent> void unregisterCompiler(Class<T> cls) {
Sho SHIMIZUb0a47d42015-02-19 13:26:30 -0800202 compilerRegistry.unregisterCompiler(cls);
Brian O'Connor66630c82014-10-02 21:08:19 -0700203 }
204
205 @Override
206 public Map<Class<? extends Intent>, IntentCompiler<? extends Intent>> getCompilers() {
Sho SHIMIZUb0a47d42015-02-19 13:26:30 -0800207 return compilerRegistry.getCompilers();
Brian O'Connor66630c82014-10-02 21:08:19 -0700208 }
209
210 @Override
Thomas Vachuskab97cf282014-10-20 23:31:12 -0700211 public <T extends Intent> void registerInstaller(Class<T> cls, IntentInstaller<T> installer) {
Sho SHIMIZUb0a47d42015-02-19 13:26:30 -0800212 installerRegistry.registerInstaller(cls, installer);
Brian O'Connor66630c82014-10-02 21:08:19 -0700213 }
214
215 @Override
Thomas Vachuskab97cf282014-10-20 23:31:12 -0700216 public <T extends Intent> void unregisterInstaller(Class<T> cls) {
Sho SHIMIZUb0a47d42015-02-19 13:26:30 -0800217 installerRegistry.unregisterInstaller(cls);
Brian O'Connor66630c82014-10-02 21:08:19 -0700218 }
219
220 @Override
Thomas Vachuskab97cf282014-10-20 23:31:12 -0700221 public Map<Class<? extends Intent>, IntentInstaller<? extends Intent>> getInstallers() {
Sho SHIMIZUb0a47d42015-02-19 13:26:30 -0800222 return installerRegistry.getInstallers();
Brian O'Connor66630c82014-10-02 21:08:19 -0700223 }
224
Brian O'Connor66630c82014-10-02 21:08:19 -0700225 // Store delegate to re-post events emitted from the store.
226 private class InternalStoreDelegate implements IntentStoreDelegate {
227 @Override
228 public void notify(IntentEvent event) {
tom85258ee2014-10-07 00:10:02 -0700229 eventDispatcher.post(event);
Brian O'Connor66630c82014-10-02 21:08:19 -0700230 }
Brian O'Connorea4d7d12015-01-28 16:37:46 -0800231
232 @Override
Brian O'Connorcff03322015-02-03 15:28:59 -0800233 public void process(IntentData data) {
234 accumulator.add(data);
Brian O'Connorea4d7d12015-01-28 16:37:46 -0800235 }
Brian O'Connor66630c82014-10-02 21:08:19 -0700236 }
237
Ray Milkeyf9af43c2015-02-09 16:45:48 -0800238 private void buildAndSubmitBatches(Iterable<Key> intentKeys,
Brian O'Connor72a034c2014-11-26 18:24:23 -0800239 boolean compileAllFailed) {
Brian O'Connor72a034c2014-11-26 18:24:23 -0800240 // Attempt recompilation of the specified intents first.
Ray Milkeyf9af43c2015-02-09 16:45:48 -0800241 for (Key key : intentKeys) {
242 Intent intent = store.getIntent(key);
Brian O'Connor72a034c2014-11-26 18:24:23 -0800243 if (intent == null) {
244 continue;
245 }
Brian O'Connor03406a42015-02-03 17:28:57 -0800246 submit(intent);
Brian O'Connor72a034c2014-11-26 18:24:23 -0800247 }
248
249 if (compileAllFailed) {
250 // If required, compile all currently failed intents.
251 for (Intent intent : getIntents()) {
Ray Milkeyf9af43c2015-02-09 16:45:48 -0800252 IntentState state = getIntentState(intent.key());
Brian O'Connor7a71d5d2014-12-02 00:12:27 -0800253 if (RECOMPILE.contains(state)) {
Brian O'Connor7a71d5d2014-12-02 00:12:27 -0800254 if (state == WITHDRAW_REQ) {
Brian O'Connor03406a42015-02-03 17:28:57 -0800255 withdraw(intent);
Brian O'Connor7a71d5d2014-12-02 00:12:27 -0800256 } else {
Brian O'Connor03406a42015-02-03 17:28:57 -0800257 submit(intent);
Brian O'Connor7a71d5d2014-12-02 00:12:27 -0800258 }
Brian O'Connor72a034c2014-11-26 18:24:23 -0800259 }
260 }
261 }
262
Brian O'Connorb499b352015-02-03 16:46:15 -0800263 //FIXME
264// for (ApplicationId appId : batches.keySet()) {
265// if (batchService.isLocalLeader(appId)) {
266// execute(batches.get(appId).build());
267// }
268// }
Brian O'Connor72a034c2014-11-26 18:24:23 -0800269 }
270
tom95329eb2014-10-06 08:40:06 -0700271 // Topology change delegate
272 private class InternalTopoChangeDelegate implements TopologyChangeDelegate {
273 @Override
Ray Milkeyf9af43c2015-02-09 16:45:48 -0800274 public void triggerCompile(Iterable<Key> intentKeys,
tom85258ee2014-10-07 00:10:02 -0700275 boolean compileAllFailed) {
Ray Milkeyf9af43c2015-02-09 16:45:48 -0800276 buildAndSubmitBatches(intentKeys, compileAllFailed);
tom95329eb2014-10-06 08:40:06 -0700277 }
tom95329eb2014-10-06 08:40:06 -0700278 }
tom85258ee2014-10-07 00:10:02 -0700279
Sho SHIMIZU36a8a6e2015-02-13 15:38:45 -0800280 private Future<FinalIntentProcessPhase> submitIntentData(IntentData data) {
Sho SHIMIZU0cb6fe62015-02-23 16:39:57 -0800281 IntentData current = store.getIntentData(data.key());
Sho SHIMIZUce49b602015-02-23 19:15:49 -0800282 IntentProcessPhase initial = newInitialPhase(processor, data, current);
283 return workerExecutor.submit(new IntentWorker(initial));
Sho SHIMIZU8d9d1362015-02-04 12:28:15 -0800284 }
285
Sho SHIMIZUadf8c482014-12-12 18:23:29 -0800286 private class IntentBatchPreprocess implements Runnable {
287
Sho SHIMIZU5f281a42015-02-04 15:29:11 -0800288 protected final Collection<IntentData> data;
Sho SHIMIZUadf8c482014-12-12 18:23:29 -0800289
Sho SHIMIZU559e4b92015-02-23 16:14:17 -0800290 IntentBatchPreprocess(Collection<IntentData> data) {
Sho SHIMIZU5f281a42015-02-04 15:29:11 -0800291 this.data = checkNotNull(data);
Sho SHIMIZUadf8c482014-12-12 18:23:29 -0800292 }
293
294 @Override
295 public void run() {
296 try {
Brian O'Connor0e271dc2015-02-04 18:20:25 -0800297 /*
298 1. wrap each intentdata in a runnable and submit
299 2. wait for completion of all the work
300 3. accumulate results and submit batch write of IntentData to store
301 (we can also try to update these individually)
302 */
303 submitUpdates(waitForFutures(createIntentUpdates()));
Sho SHIMIZUadf8c482014-12-12 18:23:29 -0800304 } catch (Exception e) {
305 log.error("Error submitting batches:", e);
306 // FIXME incomplete Intents should be cleaned up
307 // (transition to FAILED, etc.)
308
Sho SHIMIZUadf8c482014-12-12 18:23:29 -0800309 // the batch has failed
310 // TODO: maybe we should do more?
311 log.error("Walk the plank, matey...");
Brian O'Connorb499b352015-02-03 16:46:15 -0800312 //FIXME
Sho SHIMIZU5f281a42015-02-04 15:29:11 -0800313// batchService.removeIntentOperations(data);
Sho SHIMIZUadf8c482014-12-12 18:23:29 -0800314 }
315 }
316
Sho SHIMIZU36a8a6e2015-02-13 15:38:45 -0800317 private List<Future<FinalIntentProcessPhase>> createIntentUpdates() {
Sho SHIMIZU5f281a42015-02-04 15:29:11 -0800318 return data.stream()
Brian O'Connor0e271dc2015-02-04 18:20:25 -0800319 .map(IntentManager.this::submitIntentData)
320 .collect(Collectors.toList());
321 }
322
Sho SHIMIZU36a8a6e2015-02-13 15:38:45 -0800323 private List<FinalIntentProcessPhase> waitForFutures(List<Future<FinalIntentProcessPhase>> futures) {
324 ImmutableList.Builder<FinalIntentProcessPhase> updateBuilder = ImmutableList.builder();
325 for (Future<FinalIntentProcessPhase> future : futures) {
Brian O'Connor0e271dc2015-02-04 18:20:25 -0800326 try {
327 updateBuilder.add(future.get());
328 } catch (InterruptedException | ExecutionException e) {
329 //FIXME
330 log.warn("Future failed: {}", e);
331 }
332 }
333 return updateBuilder.build();
334 }
335
Sho SHIMIZU36a8a6e2015-02-13 15:38:45 -0800336 private void submitUpdates(List<FinalIntentProcessPhase> updates) {
Brian O'Connor0e271dc2015-02-04 18:20:25 -0800337 store.batchWrite(updates.stream()
Sho SHIMIZU36a8a6e2015-02-13 15:38:45 -0800338 .map(FinalIntentProcessPhase::data)
Brian O'Connor0e271dc2015-02-04 18:20:25 -0800339 .collect(Collectors.toList()));
Sho SHIMIZUadf8c482014-12-12 18:23:29 -0800340 }
Sho SHIMIZUadf8c482014-12-12 18:23:29 -0800341 }
Yuta HIGUCHIc2bf3d82014-11-28 18:50:41 -0800342
Brian O'Connorfa81eae2014-10-30 13:20:05 -0700343 private class InternalBatchDelegate implements IntentBatchDelegate {
344 @Override
Brian O'Connorb499b352015-02-03 16:46:15 -0800345 public void execute(Collection<IntentData> operations) {
Brian O'Connorab8ef822015-02-17 18:08:54 -0800346 log.debug("Execute {} operation(s).", operations.size());
347 log.trace("Execute operations: {}", operations);
Brian O'Connordb15b042015-02-04 14:59:28 -0800348 batchExecutor.execute(new IntentBatchPreprocess(operations));
349 // TODO ensure that only one batch is in flight at a time
Brian O'Connorfa81eae2014-10-30 13:20:05 -0700350 }
Brian O'Connorfa81eae2014-10-30 13:20:05 -0700351 }
Sho SHIMIZUb0a47d42015-02-19 13:26:30 -0800352
353 private class InternalIntentProcessor implements IntentProcessor {
354 @Override
355 public List<Intent> compile(Intent intent, List<Intent> previousInstallables) {
356 return compilerRegistry.compile(intent, previousInstallables);
357 }
358
359 @Override
360 public FlowRuleOperations coordinate(IntentData current, IntentData pending) {
361 return installerRegistry.coordinate(current, pending, store, trackerService);
362 }
363
364 @Override
365 public FlowRuleOperations uninstallCoordinate(IntentData current, IntentData pending) {
366 return installerRegistry.uninstallCoordinate(current, pending, store, trackerService);
367 }
368
369 @Override
370 public void applyFlowRules(FlowRuleOperations flowRules) {
371 flowRuleService.apply(flowRules);
372 }
373 }
Brian O'Connor66630c82014-10-02 21:08:19 -0700374}