blob: b52312608da34794f10b8cf9d300caaba993c043 [file] [log] [blame]
/*
* Copyright 2015-present Open Networking Foundation
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
package org.onosproject.core;
import com.google.common.collect.ImmutableList;
import com.google.common.collect.ImmutableSet;
import com.google.common.testing.EqualsTester;
import org.junit.Test;
import org.onosproject.security.AppPermission;
import org.onosproject.security.Permission;
import java.util.ArrayList;
import java.util.HashSet;
import java.util.List;
import java.util.Optional;
import java.util.Set;
import static org.onlab.junit.ImmutableClassChecker.assertThatClassIsImmutable;
import static org.onosproject.core.DefaultApplication.Builder;
import static org.junit.Assert.*;
import static org.onosproject.app.DefaultApplicationDescriptionTest.*;
/**
* Basic tests of the default app descriptor.
*/
public class DefaultApplicationTest {
/**
* Checks that the DefaultApplication class is immutable.
*/
@Test
public void testImmutability() {
assertThatClassIsImmutable(DefaultApplication.class);
}
public static final ApplicationId APP_ID = new DefaultApplicationId(2, APP_NAME);
private Builder baseBuilder = DefaultApplication.builder()
.withAppId(APP_ID)
.withVersion(VER)
.withTitle(TITLE)
.withDescription(DESC)
.withOrigin(ORIGIN)
.withCategory(CATEGORY)
.withUrl(URL)
.withReadme(README)
.withIcon(ICON)
.withRole(ROLE)
.withPermissions(PERMS)
.withFeaturesRepo(Optional.of(FURL))
.withFeatures(FEATURES)
.withRequiredApps(APPS);
@Test
public void basics() {
Application app = baseBuilder.build();
assertEquals("incorrect id", APP_ID, app.id());
assertEquals("incorrect version", VER, app.version());
assertEquals("incorrect title", TITLE, app.title());
assertEquals("incorrect description", DESC, app.description());
assertEquals("incorrect origin", ORIGIN, app.origin());
assertEquals("incorrect category", CATEGORY, app.category());
assertEquals("incorrect URL", URL, app.url());
assertEquals("incorrect readme", README, app.readme());
assertArrayEquals("incorrect icon", ICON, app.icon());
assertEquals("incorrect role", ROLE, app.role());
assertEquals("incorrect permissions", PERMS, app.permissions());
assertEquals("incorrect features repo", FURL, app.featuresRepo().get());
assertEquals("incorrect features", FEATURES, app.features());
assertEquals("incorrect apps", APPS, app.requiredApps());
assertTrue("incorrect toString", app.toString().contains(APP_NAME));
}
@Test
public void testEquality() {
Application a1 = baseBuilder.build();
Application a2 = DefaultApplication.builder(a1)
.build();
Application a3 = DefaultApplication.builder(baseBuilder)
.withFeaturesRepo(Optional.empty())
.build();
Application a4 = DefaultApplication.builder(baseBuilder)
.withOrigin(ORIGIN + "asd")
.build();
new EqualsTester()
.addEqualityGroup(a1, a2)
.addEqualityGroup(a3)
.addEqualityGroup(a4)
.testEquals();
}
private static final byte[] ICON_ORIG = new byte[] {1, 2, 3, 4};
@Test
public void immutableIcon() {
byte[] iconSourceData = ICON_ORIG.clone();
Application app = DefaultApplication.builder(baseBuilder)
.withIcon(iconSourceData).build();
// can we modify the icon after getting a reference to the app?
byte[] icon = app.icon();
assertArrayEquals("did not start with orig icon", ICON_ORIG, icon);
// now the hack
for (int i = 0, n = ICON_ORIG.length; i < n; i++) {
icon[i] = 0;
}
// if the reference to the internal array is given out, the hack
// will succeed and this next assertion fails
assertArrayEquals("no longer orig icon", ICON_ORIG, app.icon());
// what if we modify the source data?
for (int i = 0, n = ICON_ORIG.length; i < n; i++) {
iconSourceData[i] = 0;
}
// if the application just saved a reference to the given array
// this next assertion fails
assertArrayEquals("modifying source alters appicon", ICON_ORIG, app.icon());
}
private static final Permission PERM_W =
new Permission(AppPermission.class.getName(), "FLOWRULE_WRITE");
private static final Permission PERM_R =
new Permission(AppPermission.class.getName(), "FLOWRULE_READ");
private static final Permission JUNK_PERM = new Permission("foo", "bar");
private static final Set<Permission> PERMS_ORIG = ImmutableSet.of(PERM_W, PERM_R);
private static final Set<Permission> PERMS_UNSAFE = new HashSet<>(PERMS_ORIG);
@Test
public void immutablePermissions() {
// Set<Permission> p = PERMS_ORIG;
Set<Permission> p = PERMS_UNSAFE;
Application app = baseBuilder.build();
Set<Permission> perms = app.permissions();
try {
perms.add(JUNK_PERM);
} catch (UnsupportedOperationException e) {
// set is immutable
}
assertTrue("no write perm", app.permissions().contains(PERM_W));
assertTrue("no read perm", app.permissions().contains(PERM_R));
assertEquals("extra perms", 2, app.permissions().size());
// DONE: review - is it sufficient to expect caller to pass in ImmutableSet ?
// Issue Resolved with Immutable collections used during construction.
// If we just pass in a HashSet, the contents would be modifiable by
// an external party. (Making the field final just means that the
// reference to the set can never change; the contents may still...)
// Similar reasoning can be applied to these two fields also:
// List<String> features
// List<String> requiredApps
}
private static final String FOO = "foo";
private static final String BAR = "bar";
private static final String FIFI = "fifi";
private static final String EVIL = "Bwahahahaha!";
private static final List<String> FEATURES_ORIG = ImmutableList.of(FOO, BAR);
private static final List<String> FEATURES_UNSAFE = new ArrayList<>(FEATURES_ORIG);
private static final List<String> REQ_APPS_ORIG = ImmutableList.of(FIFI);
private static final List<String> REQ_APPS_UNSAFE = new ArrayList<>(REQ_APPS_ORIG);
@Test
public void immutableFeatures() {
// List<String> f = FEATURES_ORIG;
List<String> f = FEATURES_UNSAFE;
Application app = DefaultApplication.builder(baseBuilder).withFeatures(f).build();
List<String> features = app.features();
try {
features.add(EVIL);
} catch (UnsupportedOperationException e) {
// list is immutable
}
assertTrue("no foo feature", features.contains(FOO));
assertTrue("no bar feature", features.contains(BAR));
assertEquals("extra features!", 2, features.size());
}
@Test
public void immutableRequiredApps() {
// List<String> ra = REQ_APPS_ORIG;
List<String> ra = REQ_APPS_UNSAFE;
Application app = DefaultApplication.builder(baseBuilder).withRequiredApps(ra).build();
List<String> reqApps = app.requiredApps();
try {
reqApps.add(EVIL);
} catch (UnsupportedOperationException e) {
// list is immutable
}
assertTrue("no fifi required app", reqApps.contains(FIFI));
assertEquals("extra required apps!", 1, reqApps.size());
}
@Test
public void nullIcon() {
Application app = DefaultApplication.builder(baseBuilder).withIcon(null).build();
byte[] icon = app.icon();
assertNotNull("null icon", icon);
assertEquals("unexpected size", 0, icon.length);
}
}