blob: b68928bfe1384aeb0d5939cfe419a9c66b495199 [file] [log] [blame]
Simon Hunta29c87b2015-05-21 09:56:19 -07001/*
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 */
17
18package org.onosproject.cord.gui;
19
Simon Hunt7d02c082015-05-29 12:17:09 -070020import com.fasterxml.jackson.databind.JsonNode;
Simon Hunt09a32db2015-05-21 15:00:42 -070021import com.fasterxml.jackson.databind.node.ArrayNode;
22import com.fasterxml.jackson.databind.node.ObjectNode;
Simon Hunt41b943e2015-05-21 13:52:01 -070023import com.google.common.collect.ImmutableList;
24import org.onosproject.cord.gui.model.Bundle;
25import org.onosproject.cord.gui.model.BundleDescriptor;
26import org.onosproject.cord.gui.model.BundleFactory;
Simon Hunt09a32db2015-05-21 15:00:42 -070027import org.onosproject.cord.gui.model.JsonFactory;
Simon Hunt41b943e2015-05-21 13:52:01 -070028import org.onosproject.cord.gui.model.SubscriberUser;
Simon Hunt09a32db2015-05-21 15:00:42 -070029import org.onosproject.cord.gui.model.UserFactory;
Simon Hunt6c2555b2015-05-21 18:17:56 -070030import org.onosproject.cord.gui.model.XosFunction;
31import org.onosproject.cord.gui.model.XosFunctionDescriptor;
Simon Huntb1246412015-06-01 13:37:26 -070032import org.slf4j.Logger;
33import org.slf4j.LoggerFactory;
Simon Hunt41b943e2015-05-21 13:52:01 -070034
Simon Huntc686c6a2015-06-05 14:33:30 -070035import java.util.HashMap;
36import java.util.Iterator;
Simon Hunt41b943e2015-05-21 13:52:01 -070037import java.util.List;
Simon Hunt87b157c2015-05-22 12:09:59 -070038import java.util.Map;
39import java.util.TreeMap;
40
41import static com.google.common.base.Preconditions.checkNotNull;
Simon Hunt7d02c082015-05-29 12:17:09 -070042import static org.onosproject.cord.gui.model.XosFunctionDescriptor.URL_FILTER;
Simon Hunt41b943e2015-05-21 13:52:01 -070043
Simon Hunta29c87b2015-05-21 09:56:19 -070044/**
45 * In memory cache of the model of the subscriber's account.
46 */
Simon Hunt09a32db2015-05-21 15:00:42 -070047public class CordModelCache extends JsonFactory {
48
Simon Huntc686c6a2015-06-05 14:33:30 -070049 private static final String KEY_SSID_MAP = "ssidmap";
Simon Hunt2739e6f82015-06-05 16:27:45 -070050 private static final String KEY_SSID = "service_specific_id";
Simon Huntc686c6a2015-06-05 14:33:30 -070051 private static final String KEY_SUB_ID = "subscriber_id";
52
53 private static final int DEMO_SSID = 1234;
54
55 private static final String EMAIL_0 = "john@smith.org";
56 private static final String EMAIL_1 = "john@doe.org";
57
58 private static final String EMAIL = "email";
59 private static final String SSID = "ssid";
60 private static final String SUB_ID = "subId";
61
Simon Hunt09a32db2015-05-21 15:00:42 -070062 private static final String BUNDLE = "bundle";
63 private static final String USERS = "users";
Simon Hunt7d02c082015-05-29 12:17:09 -070064 private static final String LEVEL = "level";
Simon Huntc296d2c2015-06-08 12:54:57 -070065 private static final String LOGOUT = "logout";
Simon Hunt41b943e2015-05-21 13:52:01 -070066
Simon Huntc686c6a2015-06-05 14:33:30 -070067 private static final Map<Integer, Integer> LOOKUP = new HashMap<>();
68
Simon Huntc296d2c2015-06-08 12:54:57 -070069 private String email = null;
Simon Huntee6a7372015-05-28 14:04:24 -070070 private int subscriberId;
Simon Huntc686c6a2015-06-05 14:33:30 -070071 private int ssid;
Simon Hunt41b943e2015-05-21 13:52:01 -070072 private Bundle currentBundle;
Simon Hunt87b157c2015-05-22 12:09:59 -070073
Simon Huntb1246412015-06-01 13:37:26 -070074 private final Logger log = LoggerFactory.getLogger(getClass());
75
Simon Hunt87b157c2015-05-22 12:09:59 -070076 // NOTE: use a tree map to maintain sorted order by user ID
77 private final Map<Integer, SubscriberUser> userMap =
78 new TreeMap<Integer, SubscriberUser>();
Simon Hunt41b943e2015-05-21 13:52:01 -070079
80 /**
Simon Huntc686c6a2015-06-05 14:33:30 -070081 * Constructs a model cache, retrieving a mapping of SSID to XOS Subscriber
82 * IDs from the XOS server.
Simon Hunt41b943e2015-05-21 13:52:01 -070083 */
Simon Hunt09a32db2015-05-21 15:00:42 -070084 CordModelCache() {
Simon Huntb1246412015-06-01 13:37:26 -070085 log.info("Initialize model cache");
Simon Huntc686c6a2015-06-05 14:33:30 -070086 ObjectNode map = XosManager.INSTANCE.initXosSubscriberLookups();
87 initLookupMap(map);
88 log.info("{} entries in SSID->SubID lookup map", LOOKUP.size());
89 }
90
91 private void initLookupMap(ObjectNode map) {
92 ArrayNode array = (ArrayNode) map.get(KEY_SSID_MAP);
93 Iterator<JsonNode> iter = array.elements();
Simon Hunt2739e6f82015-06-05 16:27:45 -070094 StringBuilder msg = new StringBuilder();
Simon Huntc686c6a2015-06-05 14:33:30 -070095 while (iter.hasNext()) {
96 ObjectNode node = (ObjectNode) iter.next();
Simon Hunt42366082015-06-08 09:57:05 -070097 String ssidStr = node.get(KEY_SSID).asText();
Simon Huntc686c6a2015-06-05 14:33:30 -070098 int ssid = Integer.valueOf(ssidStr);
99 int subId = node.get(KEY_SUB_ID).asInt();
100 LOOKUP.put(ssid, subId);
Simon Hunt2739e6f82015-06-05 16:27:45 -0700101 msg.append(String.format("\n..binding SSID %s to sub-id %s", ssid, subId));
Simon Huntc686c6a2015-06-05 14:33:30 -0700102 }
Simon Hunt2739e6f82015-06-05 16:27:45 -0700103 log.info(msg.toString());
Simon Huntc686c6a2015-06-05 14:33:30 -0700104 }
105
106 private int lookupSubId(int ssid) {
107 Integer subId = LOOKUP.get(ssid);
108 if (subId == null) {
109 log.error("Unmapped SSID: {}", ssid);
110 return 0;
111 }
112 return subId;
113 }
114
115 /**
116 * Initializes the model for the subscriber account associated with
117 * the given email address.
118 *
119 * @param email the email address
120 */
121 void init(String email) {
122 // defaults to the demo account
123 int ssid = DEMO_SSID;
124
Simon Huntc296d2c2015-06-08 12:54:57 -0700125 this.email = email;
126
Simon Huntc686c6a2015-06-05 14:33:30 -0700127 // obviously not scalable, but good enough for demo code...
128 if (EMAIL_0.equals(email)) {
129 ssid = 0;
130 } else if (EMAIL_1.equals(email)) {
131 ssid = 1;
132 }
133
134 this.ssid = ssid;
135 subscriberId = lookupSubId(ssid);
136 XosManager.INSTANCE.setXosUtilsForSubscriber(subscriberId);
137
138 // if we are using the demo account, tell XOS to reset it...
139 if (ssid == DEMO_SSID) {
140 XosManager.INSTANCE.initDemoSubscriber();
141 }
142
143 // NOTE: I think the following should work for non-DEMO account...
Simon Hunt41b943e2015-05-21 13:52:01 -0700144 currentBundle = new Bundle(BundleFactory.BASIC_BUNDLE);
Simon Hunt7d02c082015-05-29 12:17:09 -0700145 initUsers();
Simon Hunt41b943e2015-05-21 13:52:01 -0700146 }
147
Simon Huntee6a7372015-05-28 14:04:24 -0700148 private void initUsers() {
Simon Huntc296d2c2015-06-08 12:54:57 -0700149 // start with a clean slate
150 userMap.clear();
151
Simon Hunt7d02c082015-05-29 12:17:09 -0700152 ArrayNode users = XosManager.INSTANCE.getUserList();
Simon Huntc686c6a2015-06-05 14:33:30 -0700153 if (users == null) {
154 log.warn("no user list for SSID {} (subid {})", ssid, subscriberId);
155 return;
156 }
157
Simon Huntc296d2c2015-06-08 12:54:57 -0700158 StringBuilder sb = new StringBuilder();
Simon Hunt7d02c082015-05-29 12:17:09 -0700159 for (JsonNode u: users) {
160 ObjectNode user = (ObjectNode) u;
161
162 int id = user.get("id").asInt();
163 String name = user.get("name").asText();
164 String mac = user.get("mac").asText();
165 String level = user.get("level").asText();
166
167 // NOTE: We are just storing the current "url-filter" level.
168 // Since we are starting with the BASIC bundle, (that does
169 // not include URL_FILTER), we don't yet have the URL_FILTER
170 // memento in which to store the level.
171 SubscriberUser su = createUser(id, name, mac, level);
172 userMap.put(id, su);
Simon Huntc296d2c2015-06-08 12:54:57 -0700173 sb.append(String.format("\n..cache user %s [%d], %s, %s",
174 name, id, mac, level));
Simon Hunt7d02c082015-05-29 12:17:09 -0700175 }
Simon Huntc296d2c2015-06-08 12:54:57 -0700176 log.info(sb.toString());
Simon Hunt41b943e2015-05-21 13:52:01 -0700177 }
178
Simon Hunt7d02c082015-05-29 12:17:09 -0700179 private SubscriberUser createUser(int uid, String name, String mac,
180 String level) {
181 SubscriberUser user = new SubscriberUser(uid, name, mac, level);
Simon Hunt6c2555b2015-05-21 18:17:56 -0700182 for (XosFunction f: currentBundle.functions()) {
183 user.setMemento(f.descriptor(), f.createMemento());
184 }
185 return user;
186 }
187
Simon Hunt41b943e2015-05-21 13:52:01 -0700188 /**
189 * Returns the currently selected bundle.
190 *
191 * @return current bundle
192 */
193 public Bundle getCurrentBundle() {
194 return currentBundle;
195 }
196
197 /**
198 * Sets a new bundle.
199 *
200 * @param bundleId bundle identifier
201 * @throws IllegalArgumentException if bundle ID is unknown
202 */
203 public void setCurrentBundle(String bundleId) {
Simon Huntb1246412015-06-01 13:37:26 -0700204 log.info("set new bundle : {}", bundleId);
Simon Hunt6c2555b2015-05-21 18:17:56 -0700205 BundleDescriptor bd = BundleFactory.bundleFromId(bundleId);
206 currentBundle = new Bundle(bd);
207 // update the user mementos
Simon Hunt87b157c2015-05-22 12:09:59 -0700208 for (SubscriberUser user: userMap.values()) {
Simon Hunt6c2555b2015-05-21 18:17:56 -0700209 user.clearMementos();
210 for (XosFunction f: currentBundle.functions()) {
211 user.setMemento(f.descriptor(), f.createMemento());
Simon Hunt7d02c082015-05-29 12:17:09 -0700212 if (f.descriptor().equals(URL_FILTER)) {
213 applyUrlFilterLevel(user, user.urlFilterLevel());
214 }
Simon Hunt6c2555b2015-05-21 18:17:56 -0700215 }
216 }
217
Simon Hunt7d02c082015-05-29 12:17:09 -0700218 XosManager.INSTANCE.setNewBundle(currentBundle);
Simon Hunt41b943e2015-05-21 13:52:01 -0700219 }
220
Simon Hunt6c2555b2015-05-21 18:17:56 -0700221
Simon Hunt41b943e2015-05-21 13:52:01 -0700222 /**
223 * Returns the list of current users for this subscriber account.
224 *
225 * @return the list of users
226 */
227 public List<SubscriberUser> getUsers() {
Simon Hunt87b157c2015-05-22 12:09:59 -0700228 return ImmutableList.copyOf(userMap.values());
Simon Hunt41b943e2015-05-21 13:52:01 -0700229 }
Simon Hunt09a32db2015-05-21 15:00:42 -0700230
Simon Hunt6c2555b2015-05-21 18:17:56 -0700231 /**
Simon Hunt7d02c082015-05-29 12:17:09 -0700232 * Applies a function parameter change for a user, pushing that
233 * change through to XOS.
Simon Hunt6c2555b2015-05-21 18:17:56 -0700234 *
235 * @param userId user identifier
236 * @param funcId function identifier
237 * @param param function parameter to change
238 * @param value new value for function parameter
239 */
240 public void applyPerUserParam(String userId, String funcId,
241 String param, String value) {
Simon Hunt87b157c2015-05-22 12:09:59 -0700242
Simon Hunt6c2555b2015-05-21 18:17:56 -0700243 int uid = Integer.parseInt(userId);
Simon Hunt87b157c2015-05-22 12:09:59 -0700244 SubscriberUser user = userMap.get(uid);
245 checkNotNull(user, "unknown user id: " + uid);
246
Simon Hunt6c2555b2015-05-21 18:17:56 -0700247 XosFunctionDescriptor xfd =
248 XosFunctionDescriptor.valueOf(funcId.toUpperCase());
Simon Hunt87b157c2015-05-22 12:09:59 -0700249
250 XosFunction func = currentBundle.findFunction(xfd);
251 checkNotNull(func, "function not part of bundle: " + funcId);
Simon Hunt7d02c082015-05-29 12:17:09 -0700252 applyParam(func, user, param, value, true);
Simon Hunt6c2555b2015-05-21 18:17:56 -0700253 }
254
255 // =============
256
Simon Hunt7d02c082015-05-29 12:17:09 -0700257 private void applyUrlFilterLevel(SubscriberUser user, String level) {
258 XosFunction urlFilter = currentBundle.findFunction(URL_FILTER);
259 if (urlFilter != null) {
260 applyParam(urlFilter, user, LEVEL, level, false);
261 }
262 }
263
264 private void applyParam(XosFunction func, SubscriberUser user,
265 String param, String value, boolean punchThrough) {
266 func.applyParam(user, param, value);
267 if (punchThrough) {
268 XosManager.INSTANCE.apply(func, user);
269 }
270 }
271
Simon Hunt09a32db2015-05-21 15:00:42 -0700272 private ArrayNode userJsonArray() {
273 ArrayNode userList = arrayNode();
Simon Hunt87b157c2015-05-22 12:09:59 -0700274 for (SubscriberUser user: userMap.values()) {
Simon Hunt09a32db2015-05-21 15:00:42 -0700275 userList.add(UserFactory.toObjectNode(user));
276 }
277 return userList;
278 }
279
280 // ============= generate JSON for GUI rest calls..
281
Simon Huntee6a7372015-05-28 14:04:24 -0700282 private void addSubId(ObjectNode root) {
283 root.put(SUB_ID, subscriberId);
Simon Huntc686c6a2015-06-05 14:33:30 -0700284 root.put(SSID, ssid);
Simon Huntc296d2c2015-06-08 12:54:57 -0700285 root.put(EMAIL, email);
Simon Huntc686c6a2015-06-05 14:33:30 -0700286 }
287
288
289 /**
290 * Returns response JSON for login request.
291 * <p>
292 * Depending on which email is used, will bind the GUI to the
293 * appropriate XOS Subscriber ID.
294 *
295 * @param email the supplied email
296 * @return JSON acknowledgement
297 */
Simon Hunt6f0afae2015-06-08 21:40:41 -0700298 public synchronized String jsonLogin(String email) {
Simon Huntc296d2c2015-06-08 12:54:57 -0700299 log.info("jsonLogin(\"{}\")", email);
Simon Huntc686c6a2015-06-05 14:33:30 -0700300 init(email);
301 ObjectNode root = objectNode();
Simon Huntc686c6a2015-06-05 14:33:30 -0700302 addSubId(root);
303 return root.toString();
Simon Huntee6a7372015-05-28 14:04:24 -0700304 }
305
Simon Hunt09a32db2015-05-21 15:00:42 -0700306 /**
307 * Returns the dashboard page data as JSON.
308 *
309 * @return dashboard page JSON data
310 */
Simon Hunt6f0afae2015-06-08 21:40:41 -0700311 public synchronized String jsonDashboard() {
Simon Huntc296d2c2015-06-08 12:54:57 -0700312 log.info("jsonDashboard()");
313
314 if (email == null) {
315 return jsonLogout();
316 }
317
Simon Hunt09a32db2015-05-21 15:00:42 -0700318 ObjectNode root = objectNode();
319 root.put(BUNDLE, currentBundle.descriptor().displayName());
320 root.set(USERS, userJsonArray());
Simon Huntee6a7372015-05-28 14:04:24 -0700321 addSubId(root);
Simon Hunt09a32db2015-05-21 15:00:42 -0700322 return root.toString();
323 }
324
325 /**
326 * Returns the bundle page data as JSON.
327 *
328 * @return bundle page JSON data
329 */
Simon Hunt6f0afae2015-06-08 21:40:41 -0700330 public synchronized String jsonBundle() {
Simon Huntc296d2c2015-06-08 12:54:57 -0700331 log.info("jsonBundle()");
332
333 if (email == null) {
334 return jsonLogout();
335 }
336
Simon Huntee6a7372015-05-28 14:04:24 -0700337 ObjectNode root = BundleFactory.toObjectNode(currentBundle);
338 addSubId(root);
339 return root.toString();
Simon Hunt09a32db2015-05-21 15:00:42 -0700340 }
341
342 /**
343 * Returns the users page data as JSON.
344 *
345 * @return users page JSON data
346 */
Simon Hunt6f0afae2015-06-08 21:40:41 -0700347 public synchronized String jsonUsers() {
Simon Huntc296d2c2015-06-08 12:54:57 -0700348 log.info("jsonUsers()");
349
350 if (email == null) {
351 return jsonLogout();
352 }
353
Simon Hunt09a32db2015-05-21 15:00:42 -0700354 ObjectNode root = objectNode();
355 root.set(USERS, userJsonArray());
Simon Huntee6a7372015-05-28 14:04:24 -0700356 addSubId(root);
Simon Hunt09a32db2015-05-21 15:00:42 -0700357 return root.toString();
358 }
359
360 /**
Simon Huntc296d2c2015-06-08 12:54:57 -0700361 * Returns logout acknowledgement as JSON.
362 *
363 * @return logout acknowledgement
364 */
Simon Hunt6f0afae2015-06-08 21:40:41 -0700365 public synchronized String jsonLogout() {
Simon Huntc296d2c2015-06-08 12:54:57 -0700366 log.info("jsonLogout()");
367 ObjectNode root = objectNode().put(LOGOUT, true);
368 addSubId(root);
369
370 email = null; // signifies no one logged in
371
372 return root.toString();
373 }
374
375 /**
Simon Hunt09a32db2015-05-21 15:00:42 -0700376 * Singleton instance.
377 */
378 public static final CordModelCache INSTANCE = new CordModelCache();
Simon Hunta29c87b2015-05-21 09:56:19 -0700379}