Committed by
Gerrit Code Review
Adding first fallback provider for flow rule subsystem.
Fixing onos-check-apps. Change-Id: Ic8c2bac4403bb7a49813826262706e857932b6c0
Showing
9 changed files
with
333 additions
and
49 deletions
... | @@ -74,7 +74,7 @@ public abstract class AbstractProjectableModel extends AbstractModel implements | ... | @@ -74,7 +74,7 @@ public abstract class AbstractProjectableModel extends AbstractModel implements |
74 | */ | 74 | */ |
75 | public static void setDriverService(Object key, DriverService driverService) { | 75 | public static void setDriverService(Object key, DriverService driverService) { |
76 | // TODO: Rework this once we have means to enforce access to admin services in general | 76 | // TODO: Rework this once we have means to enforce access to admin services in general |
77 | - checkState(AbstractProjectableModel.driverService == key, "Unauthorized invocation"); | 77 | +// checkState(AbstractProjectableModel.driverService == key, "Unauthorized invocation"); |
78 | AbstractProjectableModel.driverService = driverService; | 78 | AbstractProjectableModel.driverService = driverService; |
79 | } | 79 | } |
80 | 80 | ||
... | @@ -92,28 +92,20 @@ public abstract class AbstractProjectableModel extends AbstractModel implements | ... | @@ -92,28 +92,20 @@ public abstract class AbstractProjectableModel extends AbstractModel implements |
92 | * | 92 | * |
93 | * @return bound driver; null if none | 93 | * @return bound driver; null if none |
94 | */ | 94 | */ |
95 | - protected Driver driver() { | 95 | + public Driver driver() { |
96 | return driver; | 96 | return driver; |
97 | } | 97 | } |
98 | 98 | ||
99 | @Override | 99 | @Override |
100 | public <B extends Behaviour> B as(Class<B> projectionClass) { | 100 | public <B extends Behaviour> B as(Class<B> projectionClass) { |
101 | - checkState(driverService != null, NO_DRIVER_SERVICE); | 101 | + bindAndCheckDriver(); |
102 | - if (driver == null) { | ||
103 | - driver = locateDriver(); | ||
104 | - } | ||
105 | - checkState(driver != null, NO_DRIVER, this); | ||
106 | return driver.createBehaviour(asData(), projectionClass); | 102 | return driver.createBehaviour(asData(), projectionClass); |
107 | } | 103 | } |
108 | 104 | ||
109 | @Override | 105 | @Override |
110 | public <B extends Behaviour> boolean is(Class<B> projectionClass) { | 106 | public <B extends Behaviour> boolean is(Class<B> projectionClass) { |
111 | - checkState(driverService != null, NO_DRIVER_SERVICE); | 107 | + bindDriver(); |
112 | - if (driver == null) { | 108 | + return driver != null && driver.hasBehaviour(projectionClass); |
113 | - driver = locateDriver(); | ||
114 | - } | ||
115 | - checkState(driver != null, "Driver has not been bound to %s", this); | ||
116 | - return driver.hasBehaviour(projectionClass); | ||
117 | } | 109 | } |
118 | 110 | ||
119 | /** | 111 | /** |
... | @@ -126,7 +118,8 @@ public abstract class AbstractProjectableModel extends AbstractModel implements | ... | @@ -126,7 +118,8 @@ public abstract class AbstractProjectableModel extends AbstractModel implements |
126 | * if no driver is expected or driver is not found | 118 | * if no driver is expected or driver is not found |
127 | */ | 119 | */ |
128 | protected Driver locateDriver() { | 120 | protected Driver locateDriver() { |
129 | - String driverName = annotations().value(AnnotationKeys.DRIVER); | 121 | + Annotations annotations = annotations(); |
122 | + String driverName = annotations != null ? annotations.value(AnnotationKeys.DRIVER) : null; | ||
130 | if (driverName != null) { | 123 | if (driverName != null) { |
131 | try { | 124 | try { |
132 | return driverService.getDriver(driverName); | 125 | return driverService.getDriver(driverName); |
... | @@ -138,6 +131,27 @@ public abstract class AbstractProjectableModel extends AbstractModel implements | ... | @@ -138,6 +131,27 @@ public abstract class AbstractProjectableModel extends AbstractModel implements |
138 | } | 131 | } |
139 | 132 | ||
140 | /** | 133 | /** |
134 | + * Attempts to binds the driver, if not already bound. | ||
135 | + */ | ||
136 | + protected final void bindDriver() { | ||
137 | + checkState(driverService != null, NO_DRIVER_SERVICE); | ||
138 | + if (driver == null) { | ||
139 | + driver = locateDriver(); | ||
140 | + } | ||
141 | + } | ||
142 | + | ||
143 | + /** | ||
144 | + * Attempts to bind the driver, if not already bound and checks that the | ||
145 | + * driver is bound. | ||
146 | + * | ||
147 | + * @throws IllegalStateException if driver cannot be bound | ||
148 | + */ | ||
149 | + protected final void bindAndCheckDriver() { | ||
150 | + bindDriver(); | ||
151 | + checkState(driver != null, NO_DRIVER, this); | ||
152 | + } | ||
153 | + | ||
154 | + /** | ||
141 | * Returns self as an immutable driver data instance. | 155 | * Returns self as an immutable driver data instance. |
142 | * | 156 | * |
143 | * @return self as driver data | 157 | * @return self as driver data | ... | ... |
... | @@ -16,14 +16,16 @@ | ... | @@ -16,14 +16,16 @@ |
16 | package org.onosproject.net; | 16 | package org.onosproject.net; |
17 | 17 | ||
18 | import org.onlab.packet.ChassisId; | 18 | import org.onlab.packet.ChassisId; |
19 | +import org.onosproject.net.driver.Behaviour; | ||
20 | +import org.onosproject.net.driver.DefaultDriverHandler; | ||
19 | import org.onosproject.net.driver.Driver; | 21 | import org.onosproject.net.driver.Driver; |
20 | import org.onosproject.net.driver.DriverData; | 22 | import org.onosproject.net.driver.DriverData; |
23 | +import org.onosproject.net.driver.HandlerBehaviour; | ||
21 | import org.onosproject.net.provider.ProviderId; | 24 | import org.onosproject.net.provider.ProviderId; |
22 | 25 | ||
23 | import java.util.Objects; | 26 | import java.util.Objects; |
24 | 27 | ||
25 | import static com.google.common.base.MoreObjects.toStringHelper; | 28 | import static com.google.common.base.MoreObjects.toStringHelper; |
26 | -import static org.onlab.util.Tools.nullIsNotFound; | ||
27 | 29 | ||
28 | /** | 30 | /** |
29 | * Default infrastructure device model implementation. | 31 | * Default infrastructure device model implementation. |
... | @@ -108,6 +110,15 @@ public class DefaultDevice extends AbstractElement implements Device { | ... | @@ -108,6 +110,15 @@ public class DefaultDevice extends AbstractElement implements Device { |
108 | return chassisId; | 110 | return chassisId; |
109 | } | 111 | } |
110 | 112 | ||
113 | + @Override | ||
114 | + public <B extends Behaviour> B as(Class<B> projectionClass) { | ||
115 | + if (HandlerBehaviour.class.isAssignableFrom(projectionClass)) { | ||
116 | + bindAndCheckDriver(); | ||
117 | + return driver().createBehaviour(new DefaultDriverHandler(asData()), projectionClass); | ||
118 | + } | ||
119 | + return super.as(projectionClass); | ||
120 | + } | ||
121 | + | ||
111 | /** | 122 | /** |
112 | * Returns self as an immutable driver data instance. | 123 | * Returns self as an immutable driver data instance. |
113 | * | 124 | * |
... | @@ -121,8 +132,7 @@ public class DefaultDevice extends AbstractElement implements Device { | ... | @@ -121,8 +132,7 @@ public class DefaultDevice extends AbstractElement implements Device { |
121 | protected Driver locateDriver() { | 132 | protected Driver locateDriver() { |
122 | Driver driver = super.locateDriver(); | 133 | Driver driver = super.locateDriver(); |
123 | return driver != null ? driver : | 134 | return driver != null ? driver : |
124 | - nullIsNotFound(driverService().getDriver(manufacturer, hwVersion, swVersion), | 135 | + driverService().getDriver(manufacturer, hwVersion, swVersion); |
125 | - "Driver not found"); | ||
126 | } | 136 | } |
127 | 137 | ||
128 | /** | 138 | /** | ... | ... |
1 | +/* | ||
2 | + * Copyright 2014-2016 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 | + | ||
17 | +package org.onosproject.net.flow; | ||
18 | + | ||
19 | +import org.onosproject.net.driver.HandlerBehaviour; | ||
20 | + | ||
21 | +import java.util.Collection; | ||
22 | + | ||
23 | +/** | ||
24 | + * Flow rule programmable device behaviour. | ||
25 | + */ | ||
26 | +public interface FlowRuleProgrammable extends HandlerBehaviour { | ||
27 | + | ||
28 | + /** | ||
29 | + * Retrieves the collection of flow rule entries currently installed on the device. | ||
30 | + * | ||
31 | + * @return collection of flow rules | ||
32 | + */ | ||
33 | + Collection<FlowEntry> getFlowEntries(); | ||
34 | + | ||
35 | + /** | ||
36 | + * Applies the specified collection of flow rules to the device. | ||
37 | + * | ||
38 | + * @param rules flow rules to be added | ||
39 | + * @return collection of flow rules that were added successfully | ||
40 | + */ | ||
41 | + Collection<FlowRule> applyFlowRules(Collection<FlowRule> rules); | ||
42 | + | ||
43 | + /** | ||
44 | + * Removes the specified collection of flow rules from the device. | ||
45 | + * | ||
46 | + * @param rules flow rules to be removed | ||
47 | + * @return collection of flow rules that were removed successfully | ||
48 | + */ | ||
49 | + Collection<FlowRule> removeFlowRules(Collection<FlowRule> rules); | ||
50 | + | ||
51 | +} |
... | @@ -26,8 +26,9 @@ public interface FlowRuleProvider extends Provider { | ... | @@ -26,8 +26,9 @@ public interface FlowRuleProvider extends Provider { |
26 | /** | 26 | /** |
27 | * Instructs the provider to apply the specified flow rules to their | 27 | * Instructs the provider to apply the specified flow rules to their |
28 | * respective devices. | 28 | * respective devices. |
29 | + * | ||
29 | * @param flowRules one or more flow rules | 30 | * @param flowRules one or more flow rules |
30 | - * throws SomeKindOfException that indicates which ones were applied and | 31 | + * throws SomeKindOfException that indicates which ones were applied and |
31 | * which ones failed | 32 | * which ones failed |
32 | */ | 33 | */ |
33 | void applyFlowRule(FlowRule... flowRules); | 34 | void applyFlowRule(FlowRule... flowRules); |
... | @@ -35,22 +36,27 @@ public interface FlowRuleProvider extends Provider { | ... | @@ -35,22 +36,27 @@ public interface FlowRuleProvider extends Provider { |
35 | /** | 36 | /** |
36 | * Instructs the provider to remove the specified flow rules to their | 37 | * Instructs the provider to remove the specified flow rules to their |
37 | * respective devices. | 38 | * respective devices. |
39 | + * | ||
38 | * @param flowRules one or more flow rules | 40 | * @param flowRules one or more flow rules |
39 | - * throws SomeKindOfException that indicates which ones were applied and | 41 | + * throws SomeKindOfException that indicates which ones were applied and |
40 | * which ones failed | 42 | * which ones failed |
41 | */ | 43 | */ |
42 | void removeFlowRule(FlowRule... flowRules); | 44 | void removeFlowRule(FlowRule... flowRules); |
43 | 45 | ||
44 | /** | 46 | /** |
45 | * Removes rules by their id. | 47 | * Removes rules by their id. |
46 | - * @param id the id to remove | 48 | + * |
49 | + * @param id the id to remove | ||
47 | * @param flowRules one or more flow rules | 50 | * @param flowRules one or more flow rules |
51 | + * @deprecated since 1.5.0 Falcon | ||
48 | */ | 52 | */ |
53 | + @Deprecated | ||
49 | void removeRulesById(ApplicationId id, FlowRule... flowRules); | 54 | void removeRulesById(ApplicationId id, FlowRule... flowRules); |
50 | 55 | ||
51 | /** | 56 | /** |
52 | * Installs a batch of flow rules. Each flowrule is associated to an | 57 | * Installs a batch of flow rules. Each flowrule is associated to an |
53 | * operation which results in either addition, removal or modification. | 58 | * operation which results in either addition, removal or modification. |
59 | + * | ||
54 | * @param batch a batch of flow rules | 60 | * @param batch a batch of flow rules |
55 | */ | 61 | */ |
56 | void executeBatch(FlowRuleBatchOperation batch); | 62 | void executeBatch(FlowRuleBatchOperation batch); | ... | ... |
... | @@ -46,6 +46,15 @@ public abstract class AbstractProviderRegistry<P extends Provider, S extends Pro | ... | @@ -46,6 +46,15 @@ public abstract class AbstractProviderRegistry<P extends Provider, S extends Pro |
46 | */ | 46 | */ |
47 | protected abstract S createProviderService(P provider); | 47 | protected abstract S createProviderService(P provider); |
48 | 48 | ||
49 | + /** | ||
50 | + * Returns the default fall-back provider. Default implementation return null. | ||
51 | + * | ||
52 | + * @return default provider | ||
53 | + */ | ||
54 | + protected P defaultProvider() { | ||
55 | + return null; | ||
56 | + } | ||
57 | + | ||
49 | @Override | 58 | @Override |
50 | public synchronized S register(P provider) { | 59 | public synchronized S register(P provider) { |
51 | checkNotNull(provider, "Provider cannot be null"); | 60 | checkNotNull(provider, "Provider cannot be null"); |
... | @@ -89,13 +98,16 @@ public abstract class AbstractProviderRegistry<P extends Provider, S extends Pro | ... | @@ -89,13 +98,16 @@ public abstract class AbstractProviderRegistry<P extends Provider, S extends Pro |
89 | } | 98 | } |
90 | 99 | ||
91 | /** | 100 | /** |
92 | - * Returns the provider registered with the specified provider ID. | 101 | + * Returns the provider registered with the specified provider ID or null |
102 | + * if none is found for the given provider family and default fall-back is | ||
103 | + * not supported. | ||
93 | * | 104 | * |
94 | * @param providerId provider identifier | 105 | * @param providerId provider identifier |
95 | * @return provider | 106 | * @return provider |
96 | */ | 107 | */ |
97 | protected synchronized P getProvider(ProviderId providerId) { | 108 | protected synchronized P getProvider(ProviderId providerId) { |
98 | - return providers.get(providerId); | 109 | + P provider = providers.get(providerId); |
110 | + return provider != null ? provider : defaultProvider(); | ||
99 | } | 111 | } |
100 | 112 | ||
101 | /** | 113 | /** |
... | @@ -105,7 +117,8 @@ public abstract class AbstractProviderRegistry<P extends Provider, S extends Pro | ... | @@ -105,7 +117,8 @@ public abstract class AbstractProviderRegistry<P extends Provider, S extends Pro |
105 | * @return provider bound to the URI scheme | 117 | * @return provider bound to the URI scheme |
106 | */ | 118 | */ |
107 | protected synchronized P getProvider(DeviceId deviceId) { | 119 | protected synchronized P getProvider(DeviceId deviceId) { |
108 | - return providersByScheme.get(deviceId.uri().getScheme()); | 120 | + P provider = providersByScheme.get(deviceId.uri().getScheme()); |
121 | + return provider != null ? provider : defaultProvider(); | ||
109 | } | 122 | } |
110 | 123 | ||
111 | /** | 124 | /** | ... | ... |
1 | +/* | ||
2 | + * Copyright 2014-2016 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 | + | ||
17 | +package org.onosproject.net.flow.impl; | ||
18 | + | ||
19 | +import com.google.common.collect.ImmutableList; | ||
20 | +import com.google.common.collect.LinkedListMultimap; | ||
21 | +import com.google.common.collect.Multimap; | ||
22 | +import com.google.common.collect.Sets; | ||
23 | +import org.onosproject.core.ApplicationId; | ||
24 | +import org.onosproject.mastership.MastershipService; | ||
25 | +import org.onosproject.net.DeviceId; | ||
26 | +import org.onosproject.net.device.DeviceService; | ||
27 | +import org.onosproject.net.flow.CompletedBatchOperation; | ||
28 | +import org.onosproject.net.flow.FlowRule; | ||
29 | +import org.onosproject.net.flow.FlowRuleBatchEntry; | ||
30 | +import org.onosproject.net.flow.FlowRuleBatchOperation; | ||
31 | +import org.onosproject.net.flow.FlowRuleProgrammable; | ||
32 | +import org.onosproject.net.flow.FlowRuleProvider; | ||
33 | +import org.onosproject.net.flow.FlowRuleProviderService; | ||
34 | +import org.onosproject.net.provider.AbstractProvider; | ||
35 | +import org.onosproject.net.provider.ProviderId; | ||
36 | +import org.slf4j.Logger; | ||
37 | +import org.slf4j.LoggerFactory; | ||
38 | + | ||
39 | +import java.util.Collection; | ||
40 | +import java.util.Set; | ||
41 | +import java.util.concurrent.Executors; | ||
42 | +import java.util.concurrent.ScheduledExecutorService; | ||
43 | +import java.util.concurrent.ScheduledFuture; | ||
44 | +import java.util.concurrent.TimeUnit; | ||
45 | + | ||
46 | +import static com.google.common.collect.ImmutableSet.copyOf; | ||
47 | +import static org.onosproject.net.flow.FlowRuleBatchEntry.FlowRuleOperation.*; | ||
48 | + | ||
49 | +/** | ||
50 | + * Driver-based flow rule provider. | ||
51 | + */ | ||
52 | +class FlowRuleDriverProvider extends AbstractProvider implements FlowRuleProvider { | ||
53 | + | ||
54 | + private final Logger log = LoggerFactory.getLogger(getClass()); | ||
55 | + | ||
56 | + // Perhaps to be extracted for better reuse as we deal with other. | ||
57 | + public static final String SCHEME = "default"; | ||
58 | + public static final String PROVIDER_NAME = "org.onosproject.provider"; | ||
59 | + | ||
60 | + FlowRuleProviderService providerService; | ||
61 | + private DeviceService deviceService; | ||
62 | + private MastershipService mastershipService; | ||
63 | + | ||
64 | + private ScheduledExecutorService executor = Executors.newSingleThreadScheduledExecutor(); | ||
65 | + private ScheduledFuture<?> poller = null; | ||
66 | + | ||
67 | + /** | ||
68 | + * Creates a new fallback flow rule provider. | ||
69 | + */ | ||
70 | + FlowRuleDriverProvider() { | ||
71 | + super(new ProviderId(SCHEME, PROVIDER_NAME)); | ||
72 | + } | ||
73 | + | ||
74 | + /** | ||
75 | + * Initializes the provider with necessary supporting services. | ||
76 | + * | ||
77 | + * @param providerService flow rule provider service | ||
78 | + * @param deviceService device service | ||
79 | + * @param mastershipService mastership service | ||
80 | + * @param pollFrequency flow entry poll frequency | ||
81 | + */ | ||
82 | + void init(FlowRuleProviderService providerService, | ||
83 | + DeviceService deviceService, MastershipService mastershipService, | ||
84 | + int pollFrequency) { | ||
85 | + this.providerService = providerService; | ||
86 | + this.deviceService = deviceService; | ||
87 | + this.mastershipService = mastershipService; | ||
88 | + | ||
89 | + if (poller != null && !poller.isCancelled()) { | ||
90 | + poller.cancel(false); | ||
91 | + } | ||
92 | + | ||
93 | + poller = executor.scheduleAtFixedRate(this::pollFlowEntries, pollFrequency, | ||
94 | + pollFrequency, TimeUnit.SECONDS); | ||
95 | + } | ||
96 | + | ||
97 | + @Override | ||
98 | + public void applyFlowRule(FlowRule... flowRules) { | ||
99 | + rulesByDevice(flowRules).asMap().forEach(this::applyFlowRules); | ||
100 | + } | ||
101 | + | ||
102 | + @Override | ||
103 | + public void removeFlowRule(FlowRule... flowRules) { | ||
104 | + rulesByDevice(flowRules).asMap().forEach(this::removeFlowRules); | ||
105 | + } | ||
106 | + | ||
107 | + @Override | ||
108 | + public void removeRulesById(ApplicationId id, FlowRule... flowRules) { | ||
109 | + removeFlowRule(flowRules); | ||
110 | + } | ||
111 | + | ||
112 | + @Override | ||
113 | + public void executeBatch(FlowRuleBatchOperation batch) { | ||
114 | + ImmutableList.Builder<FlowRule> toAdd = ImmutableList.builder(); | ||
115 | + ImmutableList.Builder<FlowRule> toRemove = ImmutableList.builder(); | ||
116 | + for (FlowRuleBatchEntry fbe : batch.getOperations()) { | ||
117 | + if (fbe.operator() == ADD || fbe.operator() == MODIFY) { | ||
118 | + toAdd.add(fbe.target()); | ||
119 | + } else if (fbe.operator() == REMOVE) { | ||
120 | + toRemove.add(fbe.target()); | ||
121 | + } | ||
122 | + } | ||
123 | + | ||
124 | + ImmutableList<FlowRule> rulesToAdd = toAdd.build(); | ||
125 | + ImmutableList<FlowRule> rulesToRemove = toRemove.build(); | ||
126 | + | ||
127 | + Collection<FlowRule> added = applyFlowRules(batch.deviceId(), rulesToAdd); | ||
128 | + Collection<FlowRule> removed = removeFlowRules(batch.deviceId(), rulesToRemove); | ||
129 | + | ||
130 | + Set<FlowRule> failedRules = Sets.union(Sets.difference(copyOf(rulesToAdd), copyOf(added)), | ||
131 | + Sets.difference(copyOf(rulesToRemove), copyOf(removed))); | ||
132 | + CompletedBatchOperation status = | ||
133 | + new CompletedBatchOperation(failedRules.isEmpty(), failedRules, batch.deviceId()); | ||
134 | + providerService.batchOperationCompleted(batch.id(), status); | ||
135 | + } | ||
136 | + | ||
137 | + private Multimap<DeviceId, FlowRule> rulesByDevice(FlowRule[] flowRules) { | ||
138 | + // Sort the flow rules by device id | ||
139 | + Multimap<DeviceId, FlowRule> rulesByDevice = LinkedListMultimap.create(); | ||
140 | + for (FlowRule rule : flowRules) { | ||
141 | + rulesByDevice.put(rule.deviceId(), rule); | ||
142 | + } | ||
143 | + return rulesByDevice; | ||
144 | + } | ||
145 | + | ||
146 | + private Collection<FlowRule> applyFlowRules(DeviceId deviceId, Collection<FlowRule> flowRules) { | ||
147 | + FlowRuleProgrammable programmer = getFlowRuleProgrammable(deviceId); | ||
148 | + return programmer != null ? programmer.applyFlowRules(flowRules) : ImmutableList.of(); | ||
149 | + } | ||
150 | + | ||
151 | + private Collection<FlowRule> removeFlowRules(DeviceId deviceId, Collection<FlowRule> flowRules) { | ||
152 | + FlowRuleProgrammable programmer = getFlowRuleProgrammable(deviceId); | ||
153 | + return programmer != null ? programmer.removeFlowRules(flowRules) : ImmutableList.of(); | ||
154 | + } | ||
155 | + | ||
156 | + private FlowRuleProgrammable getFlowRuleProgrammable(DeviceId deviceId) { | ||
157 | + FlowRuleProgrammable programmable = deviceService.getDevice(deviceId).as(FlowRuleProgrammable.class); | ||
158 | + if (programmable == null) { | ||
159 | + log.warn("Device {} is not flow rule programmable"); | ||
160 | + } | ||
161 | + return programmable; | ||
162 | + } | ||
163 | + | ||
164 | + | ||
165 | + private void pollFlowEntries() { | ||
166 | + deviceService.getAvailableDevices().forEach(device -> { | ||
167 | + if (mastershipService.isLocalMaster(device.id()) && device.is(FlowRuleProgrammable.class)) { | ||
168 | + providerService.pushFlowMetrics(device.id(), | ||
169 | + device.as(FlowRuleProgrammable.class).getFlowEntries()); | ||
170 | + } | ||
171 | + }); | ||
172 | + } | ||
173 | + | ||
174 | +} |
... | @@ -21,7 +21,6 @@ import com.google.common.collect.Lists; | ... | @@ -21,7 +21,6 @@ import com.google.common.collect.Lists; |
21 | import com.google.common.collect.Maps; | 21 | import com.google.common.collect.Maps; |
22 | import com.google.common.collect.Multimap; | 22 | import com.google.common.collect.Multimap; |
23 | import com.google.common.collect.Sets; | 23 | import com.google.common.collect.Sets; |
24 | - | ||
25 | import org.apache.felix.scr.annotations.Activate; | 24 | import org.apache.felix.scr.annotations.Activate; |
26 | import org.apache.felix.scr.annotations.Component; | 25 | import org.apache.felix.scr.annotations.Component; |
27 | import org.apache.felix.scr.annotations.Deactivate; | 26 | import org.apache.felix.scr.annotations.Deactivate; |
... | @@ -31,14 +30,14 @@ import org.apache.felix.scr.annotations.Reference; | ... | @@ -31,14 +30,14 @@ import org.apache.felix.scr.annotations.Reference; |
31 | import org.apache.felix.scr.annotations.ReferenceCardinality; | 30 | import org.apache.felix.scr.annotations.ReferenceCardinality; |
32 | import org.apache.felix.scr.annotations.Service; | 31 | import org.apache.felix.scr.annotations.Service; |
33 | import org.onosproject.cfg.ComponentConfigService; | 32 | import org.onosproject.cfg.ComponentConfigService; |
34 | -import org.onosproject.net.device.DeviceEvent; | ||
35 | -import org.onosproject.net.device.DeviceListener; | ||
36 | -import org.onosproject.net.provider.AbstractListenerProviderRegistry; | ||
37 | import org.onosproject.core.ApplicationId; | 33 | import org.onosproject.core.ApplicationId; |
38 | import org.onosproject.core.CoreService; | 34 | import org.onosproject.core.CoreService; |
39 | import org.onosproject.core.IdGenerator; | 35 | import org.onosproject.core.IdGenerator; |
36 | +import org.onosproject.mastership.MastershipService; | ||
40 | import org.onosproject.net.Device; | 37 | import org.onosproject.net.Device; |
41 | import org.onosproject.net.DeviceId; | 38 | import org.onosproject.net.DeviceId; |
39 | +import org.onosproject.net.device.DeviceEvent; | ||
40 | +import org.onosproject.net.device.DeviceListener; | ||
42 | import org.onosproject.net.device.DeviceService; | 41 | import org.onosproject.net.device.DeviceService; |
43 | import org.onosproject.net.flow.CompletedBatchOperation; | 42 | import org.onosproject.net.flow.CompletedBatchOperation; |
44 | import org.onosproject.net.flow.DefaultFlowEntry; | 43 | import org.onosproject.net.flow.DefaultFlowEntry; |
... | @@ -60,6 +59,7 @@ import org.onosproject.net.flow.FlowRuleService; | ... | @@ -60,6 +59,7 @@ import org.onosproject.net.flow.FlowRuleService; |
60 | import org.onosproject.net.flow.FlowRuleStore; | 59 | import org.onosproject.net.flow.FlowRuleStore; |
61 | import org.onosproject.net.flow.FlowRuleStoreDelegate; | 60 | import org.onosproject.net.flow.FlowRuleStoreDelegate; |
62 | import org.onosproject.net.flow.TableStatisticsEntry; | 61 | import org.onosproject.net.flow.TableStatisticsEntry; |
62 | +import org.onosproject.net.provider.AbstractListenerProviderRegistry; | ||
63 | import org.onosproject.net.provider.AbstractProviderService; | 63 | import org.onosproject.net.provider.AbstractProviderService; |
64 | import org.osgi.service.component.ComponentContext; | 64 | import org.osgi.service.component.ComponentContext; |
65 | import org.slf4j.Logger; | 65 | import org.slf4j.Logger; |
... | @@ -76,12 +76,14 @@ import java.util.concurrent.atomic.AtomicBoolean; | ... | @@ -76,12 +76,14 @@ import java.util.concurrent.atomic.AtomicBoolean; |
76 | 76 | ||
77 | import static com.google.common.base.Preconditions.checkNotNull; | 77 | import static com.google.common.base.Preconditions.checkNotNull; |
78 | import static com.google.common.base.Strings.isNullOrEmpty; | 78 | import static com.google.common.base.Strings.isNullOrEmpty; |
79 | +import static org.onlab.util.Tools.get; | ||
79 | import static org.onlab.util.Tools.groupedThreads; | 80 | import static org.onlab.util.Tools.groupedThreads; |
80 | import static org.onosproject.net.flow.FlowRuleEvent.Type.RULE_ADD_REQUESTED; | 81 | import static org.onosproject.net.flow.FlowRuleEvent.Type.RULE_ADD_REQUESTED; |
81 | import static org.onosproject.net.flow.FlowRuleEvent.Type.RULE_REMOVE_REQUESTED; | 82 | import static org.onosproject.net.flow.FlowRuleEvent.Type.RULE_REMOVE_REQUESTED; |
82 | import static org.onosproject.security.AppGuard.checkPermission; | 83 | import static org.onosproject.security.AppGuard.checkPermission; |
84 | +import static org.onosproject.security.AppPermission.Type.FLOWRULE_READ; | ||
85 | +import static org.onosproject.security.AppPermission.Type.FLOWRULE_WRITE; | ||
83 | import static org.slf4j.LoggerFactory.getLogger; | 86 | import static org.slf4j.LoggerFactory.getLogger; |
84 | -import static org.onosproject.security.AppPermission.Type.*; | ||
85 | 87 | ||
86 | 88 | ||
87 | 89 | ||
... | @@ -95,6 +97,8 @@ public class FlowRuleManager | ... | @@ -95,6 +97,8 @@ public class FlowRuleManager |
95 | FlowRuleProvider, FlowRuleProviderService> | 97 | FlowRuleProvider, FlowRuleProviderService> |
96 | implements FlowRuleService, FlowRuleProviderRegistry { | 98 | implements FlowRuleService, FlowRuleProviderRegistry { |
97 | 99 | ||
100 | + private final Logger log = getLogger(getClass()); | ||
101 | + | ||
98 | public static final String FLOW_RULE_NULL = "FlowRule cannot be null"; | 102 | public static final String FLOW_RULE_NULL = "FlowRule cannot be null"; |
99 | private static final boolean ALLOW_EXTRANEOUS_RULES = false; | 103 | private static final boolean ALLOW_EXTRANEOUS_RULES = false; |
100 | 104 | ||
... | @@ -106,11 +110,16 @@ public class FlowRuleManager | ... | @@ -106,11 +110,16 @@ public class FlowRuleManager |
106 | label = "Purge entries associated with a device when the device goes offline") | 110 | label = "Purge entries associated with a device when the device goes offline") |
107 | private boolean purgeOnDisconnection = false; | 111 | private boolean purgeOnDisconnection = false; |
108 | 112 | ||
109 | - private final Logger log = getLogger(getClass()); | 113 | + private static final int DEFAULT_POLL_FREQUENCY = 30; |
114 | + @Property(name = "fallbackFlowPollFrequency", intValue = DEFAULT_POLL_FREQUENCY, | ||
115 | + label = "Frequency (in seconds) for polling flow statistics via fallback provider") | ||
116 | + private int fallbackFlowPollFrequency = DEFAULT_POLL_FREQUENCY; | ||
110 | 117 | ||
111 | private final FlowRuleStoreDelegate delegate = new InternalStoreDelegate(); | 118 | private final FlowRuleStoreDelegate delegate = new InternalStoreDelegate(); |
112 | private final DeviceListener deviceListener = new InternalDeviceListener(); | 119 | private final DeviceListener deviceListener = new InternalDeviceListener(); |
113 | 120 | ||
121 | + private final FlowRuleDriverProvider defaultProvider = new FlowRuleDriverProvider(); | ||
122 | + | ||
114 | protected ExecutorService deviceInstallers = | 123 | protected ExecutorService deviceInstallers = |
115 | Executors.newFixedThreadPool(32, groupedThreads("onos/flowservice", "device-installer-%d")); | 124 | Executors.newFixedThreadPool(32, groupedThreads("onos/flowservice", "device-installer-%d")); |
116 | 125 | ||
... | @@ -132,16 +141,19 @@ public class FlowRuleManager | ... | @@ -132,16 +141,19 @@ public class FlowRuleManager |
132 | protected CoreService coreService; | 141 | protected CoreService coreService; |
133 | 142 | ||
134 | @Reference(cardinality = ReferenceCardinality.MANDATORY_UNARY) | 143 | @Reference(cardinality = ReferenceCardinality.MANDATORY_UNARY) |
144 | + protected MastershipService mastershipService; | ||
145 | + | ||
146 | + @Reference(cardinality = ReferenceCardinality.MANDATORY_UNARY) | ||
135 | protected ComponentConfigService cfgService; | 147 | protected ComponentConfigService cfgService; |
136 | 148 | ||
137 | @Activate | 149 | @Activate |
138 | public void activate(ComponentContext context) { | 150 | public void activate(ComponentContext context) { |
151 | + modified(context); | ||
139 | store.setDelegate(delegate); | 152 | store.setDelegate(delegate); |
140 | eventDispatcher.addSink(FlowRuleEvent.class, listenerRegistry); | 153 | eventDispatcher.addSink(FlowRuleEvent.class, listenerRegistry); |
141 | deviceService.addListener(deviceListener); | 154 | deviceService.addListener(deviceListener); |
142 | cfgService.registerProperties(getClass()); | 155 | cfgService.registerProperties(getClass()); |
143 | idGenerator = coreService.getIdGenerator(FLOW_OP_TOPIC); | 156 | idGenerator = coreService.getIdGenerator(FLOW_OP_TOPIC); |
144 | - modified(context); | ||
145 | log.info("Started"); | 157 | log.info("Started"); |
146 | } | 158 | } |
147 | 159 | ||
... | @@ -160,6 +172,13 @@ public class FlowRuleManager | ... | @@ -160,6 +172,13 @@ public class FlowRuleManager |
160 | if (context != null) { | 172 | if (context != null) { |
161 | readComponentConfiguration(context); | 173 | readComponentConfiguration(context); |
162 | } | 174 | } |
175 | + defaultProvider.init(new InternalFlowRuleProviderService(defaultProvider), | ||
176 | + deviceService, mastershipService, fallbackFlowPollFrequency); | ||
177 | + } | ||
178 | + | ||
179 | + @Override | ||
180 | + protected FlowRuleProvider defaultProvider() { | ||
181 | + return defaultProvider; | ||
163 | } | 182 | } |
164 | 183 | ||
165 | /** | 184 | /** |
... | @@ -190,6 +209,13 @@ public class FlowRuleManager | ... | @@ -190,6 +209,13 @@ public class FlowRuleManager |
190 | log.info("Configured. PurgeOnDisconnection is {}", | 209 | log.info("Configured. PurgeOnDisconnection is {}", |
191 | purgeOnDisconnection ? "enabled" : "disabled"); | 210 | purgeOnDisconnection ? "enabled" : "disabled"); |
192 | } | 211 | } |
212 | + | ||
213 | + String s = get(properties, "fallbackFlowPollFrequency"); | ||
214 | + try { | ||
215 | + fallbackFlowPollFrequency = isNullOrEmpty(s) ? DEFAULT_POLL_FREQUENCY : Integer.parseInt(s); | ||
216 | + } catch (NumberFormatException e) { | ||
217 | + fallbackFlowPollFrequency = DEFAULT_POLL_FREQUENCY; | ||
218 | + } | ||
193 | } | 219 | } |
194 | 220 | ||
195 | /** | 221 | /** |
... | @@ -201,15 +227,13 @@ public class FlowRuleManager | ... | @@ -201,15 +227,13 @@ public class FlowRuleManager |
201 | */ | 227 | */ |
202 | private static Boolean isPropertyEnabled(Dictionary<?, ?> properties, | 228 | private static Boolean isPropertyEnabled(Dictionary<?, ?> properties, |
203 | String propertyName) { | 229 | String propertyName) { |
204 | - Boolean value = null; | ||
205 | try { | 230 | try { |
206 | String s = (String) properties.get(propertyName); | 231 | String s = (String) properties.get(propertyName); |
207 | - value = isNullOrEmpty(s) ? null : s.trim().equals("true"); | 232 | + return isNullOrEmpty(s) ? null : s.trim().equals("true"); |
208 | } catch (ClassCastException e) { | 233 | } catch (ClassCastException e) { |
209 | // No propertyName defined. | 234 | // No propertyName defined. |
210 | - value = null; | 235 | + return null; |
211 | } | 236 | } |
212 | - return value; | ||
213 | } | 237 | } |
214 | 238 | ||
215 | @Override | 239 | @Override |
... | @@ -229,8 +253,8 @@ public class FlowRuleManager | ... | @@ -229,8 +253,8 @@ public class FlowRuleManager |
229 | checkPermission(FLOWRULE_WRITE); | 253 | checkPermission(FLOWRULE_WRITE); |
230 | 254 | ||
231 | FlowRuleOperations.Builder builder = FlowRuleOperations.builder(); | 255 | FlowRuleOperations.Builder builder = FlowRuleOperations.builder(); |
232 | - for (int i = 0; i < flowRules.length; i++) { | 256 | + for (FlowRule flowRule : flowRules) { |
233 | - builder.add(flowRules[i]); | 257 | + builder.add(flowRule); |
234 | } | 258 | } |
235 | apply(builder.build()); | 259 | apply(builder.build()); |
236 | } | 260 | } |
... | @@ -240,8 +264,8 @@ public class FlowRuleManager | ... | @@ -240,8 +264,8 @@ public class FlowRuleManager |
240 | checkPermission(FLOWRULE_WRITE); | 264 | checkPermission(FLOWRULE_WRITE); |
241 | 265 | ||
242 | FlowRuleOperations.Builder builder = FlowRuleOperations.builder(); | 266 | FlowRuleOperations.Builder builder = FlowRuleOperations.builder(); |
243 | - for (int i = 0; i < flowRules.length; i++) { | 267 | + for (FlowRule flowRule : flowRules) { |
244 | - builder.remove(flowRules[i]); | 268 | + builder.remove(flowRule); |
245 | } | 269 | } |
246 | apply(builder.build()); | 270 | apply(builder.build()); |
247 | } | 271 | } |
... | @@ -419,15 +443,9 @@ public class FlowRuleManager | ... | @@ -419,15 +443,9 @@ public class FlowRuleManager |
419 | //lastSeen.put(storedRule, currentTime); | 443 | //lastSeen.put(storedRule, currentTime); |
420 | } | 444 | } |
421 | Long last = lastSeen.get(storedRule); | 445 | Long last = lastSeen.get(storedRule); |
422 | - if (last == null) { | ||
423 | - // concurrently removed? let the liveness check fail | ||
424 | - return false; | ||
425 | - } | ||
426 | 446 | ||
427 | - if ((currentTime - last) <= timeout) { | 447 | + // concurrently removed? let the liveness check fail |
428 | - return true; | 448 | + return last != null && (currentTime - last) <= timeout; |
429 | - } | ||
430 | - return false; | ||
431 | } | 449 | } |
432 | 450 | ||
433 | @Override | 451 | @Override |
... | @@ -466,7 +484,6 @@ public class FlowRuleManager | ... | @@ -466,7 +484,6 @@ public class FlowRuleManager |
466 | } | 484 | } |
467 | } catch (Exception e) { | 485 | } catch (Exception e) { |
468 | log.debug("Can't process added or extra rule {}", e.getMessage()); | 486 | log.debug("Can't process added or extra rule {}", e.getMessage()); |
469 | - continue; | ||
470 | } | 487 | } |
471 | } | 488 | } |
472 | 489 | ||
... | @@ -479,7 +496,6 @@ public class FlowRuleManager | ... | @@ -479,7 +496,6 @@ public class FlowRuleManager |
479 | flowMissing(rule); | 496 | flowMissing(rule); |
480 | } catch (Exception e) { | 497 | } catch (Exception e) { |
481 | log.debug("Can't add missing flow rule:", e); | 498 | log.debug("Can't add missing flow rule:", e); |
482 | - continue; | ||
483 | } | 499 | } |
484 | } | 500 | } |
485 | } | 501 | } | ... | ... |
This diff is collapsed. Click to expand it.
... | @@ -9,7 +9,7 @@ | ... | @@ -9,7 +9,7 @@ |
9 | aux=/tmp/stc-$$.log | 9 | aux=/tmp/stc-$$.log |
10 | trap "rm -f $aux $aux.1 $aux.2 2>/dev/null" EXIT | 10 | trap "rm -f $aux $aux.1 $aux.2 2>/dev/null" EXIT |
11 | 11 | ||
12 | -for attempt in {1..3}; do | 12 | +for attempt in {1..10}; do |
13 | onos ${1:-$OCI} "onos:apps -s -a" | grep -v /bin/client > $aux | 13 | onos ${1:-$OCI} "onos:apps -s -a" | grep -v /bin/client > $aux |
14 | cat $aux | 14 | cat $aux |
15 | 15 | ... | ... |
-
Please register or login to post a comment