blob: 6c2ac33d91d2b2acdf199c71056e55741bfe4329 [file] [log] [blame]
/*
* Licensed to the Apache Software Foundation (ASF) under one
* or more contributor license agreements. See the NOTICE file
* distributed with this work for additional information
* regarding copyright ownership. The ASF licenses this file
* to you 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.apache.brooklyn.rest.resources;
import static com.google.common.base.Preconditions.checkNotNull;
import static com.google.common.collect.Iterables.find;
import org.apache.brooklyn.core.sensor.StaticSensor;
import org.apache.brooklyn.entity.software.base.EmptySoftwareProcess;
import org.apache.brooklyn.util.text.Strings;
import static org.testng.Assert.assertEquals;
import static org.testng.Assert.assertFalse;
import static org.testng.Assert.assertNotNull;
import static org.testng.Assert.assertNull;
import static org.testng.Assert.assertTrue;
import java.io.IOException;
import java.io.InputStream;
import java.net.URI;
import java.util.Collection;
import java.util.LinkedList;
import java.util.List;
import java.util.Map;
import java.util.Set;
import java.util.concurrent.TimeoutException;
import java.util.stream.Collectors;
import javax.ws.rs.WebApplicationException;
import javax.ws.rs.client.Entity;
import javax.ws.rs.core.GenericType;
import javax.ws.rs.core.MediaType;
import javax.ws.rs.core.MultivaluedHashMap;
import javax.ws.rs.core.MultivaluedMap;
import javax.ws.rs.core.Response;
import org.apache.brooklyn.api.entity.Application;
import org.apache.brooklyn.core.entity.Attributes;
import org.apache.brooklyn.core.entity.Entities;
import org.apache.brooklyn.core.entity.EntityFunctions;
import org.apache.brooklyn.core.entity.EntityPredicates;
import org.apache.brooklyn.core.entity.lifecycle.Lifecycle;
import org.apache.brooklyn.core.entity.lifecycle.Lifecycle.Transition;
import org.apache.brooklyn.core.location.AbstractLocation;
import org.apache.brooklyn.core.location.LocationConfigKeys;
import org.apache.brooklyn.core.location.geo.HostGeoInfo;
import org.apache.brooklyn.core.location.internal.LocationInternal;
import org.apache.brooklyn.core.mgmt.internal.IdAlreadyExistsException;
import org.apache.brooklyn.entity.stock.BasicApplication;
import org.apache.brooklyn.entity.stock.BasicEntity;
import org.apache.brooklyn.rest.domain.ApiError;
import org.apache.brooklyn.rest.domain.ApplicationSpec;
import org.apache.brooklyn.rest.domain.ApplicationSummary;
import org.apache.brooklyn.rest.domain.CatalogEntitySummary;
import org.apache.brooklyn.rest.domain.CatalogItemSummary;
import org.apache.brooklyn.rest.domain.EffectorSummary;
import org.apache.brooklyn.rest.domain.EntityConfigSummary;
import org.apache.brooklyn.rest.domain.EntitySpec;
import org.apache.brooklyn.rest.domain.EntitySummary;
import org.apache.brooklyn.rest.domain.PolicySummary;
import org.apache.brooklyn.rest.domain.SensorSummary;
import org.apache.brooklyn.rest.domain.Status;
import org.apache.brooklyn.rest.domain.TaskSummary;
import org.apache.brooklyn.rest.testing.BrooklynRestResourceTest;
import org.apache.brooklyn.rest.testing.mocks.CapitalizePolicy;
import org.apache.brooklyn.rest.testing.mocks.NameMatcherGroup;
import org.apache.brooklyn.rest.testing.mocks.RestMockApp;
import org.apache.brooklyn.rest.testing.mocks.RestMockSimpleEntity;
import org.apache.brooklyn.test.Asserts;
import org.apache.brooklyn.util.collections.CollectionFunctionals;
import org.apache.brooklyn.util.collections.MutableList;
import org.apache.brooklyn.util.collections.MutableMap;
import org.apache.brooklyn.util.exceptions.Exceptions;
import org.apache.brooklyn.util.http.HttpAsserts;
import org.apache.brooklyn.util.stream.Streams;
import org.apache.brooklyn.util.text.StringPredicates;
import org.apache.cxf.jaxrs.client.WebClient;
import org.apache.cxf.jaxrs.ext.multipart.Attachment;
import org.apache.http.HttpHeaders;
import org.apache.http.entity.ContentType;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.testng.Assert;
import org.testng.annotations.Test;
import com.google.common.base.Joiner;
import com.google.common.base.Optional;
import com.google.common.base.Preconditions;
import com.google.common.base.Predicate;
import com.google.common.base.Predicates;
import com.google.common.collect.ImmutableMap;
import com.google.common.collect.ImmutableSet;
import com.google.common.collect.Iterables;
import com.google.common.collect.Maps;
import com.google.common.collect.Range;
@Test(singleThreaded = true,
// by using a different suite name we disallow interleaving other tests between the methods of this test class, which wrecks the test fixtures
suiteName = "ApplicationResourceTest")
public class ApplicationResourceTest extends BrooklynRestResourceTest {
/*
* In simpleSpec, not using EverythingGroup because caused problems! The group is a child of the
* app, and the app is a member of the group. It failed in jenkins with:
* BasicApplicationImpl{id=GSPjBCe4} GSPjBCe4
* service.isUp: true
* service.problems: {service-lifecycle-indicators-from-children-and-members=Required entity not healthy: EverythingGroupImpl{id=KQ4mSEOJ}}
* service.state: on-fire
* service.state.expected: running @ 1412003485617 / Mon Sep 29 15:11:25 UTC 2014
* EverythingGroupImpl{id=KQ4mSEOJ} KQ4mSEOJ
* service.isUp: true
* service.problems: {service-lifecycle-indicators-from-children-and-members=Required entities not healthy: BasicApplicationImpl{id=GSPjBCe4}, EverythingGroupImpl{id=KQ4mSEOJ}}
* service.state: on-fire
* I'm guessing there's a race: the app was not yet healthy because EverythingGroup hadn't set itself to running;
* but then the EverythingGroup would never transition to healthy because one of its members was not healthy.
*/
private static final Logger log = LoggerFactory.getLogger(ApplicationResourceTest.class);
@Override
protected boolean useOsgi() {
return true;
}
private final ApplicationSpec simpleSpec = ApplicationSpec.builder().name("simple-app")
.entities(ImmutableSet.of(
new EntitySpec("simple-ent", RestMockSimpleEntity.class.getName()),
new EntitySpec("simple-group", NameMatcherGroup.class.getName(), ImmutableMap.of("namematchergroup.regex", "simple-ent"))
))
.locations(ImmutableSet.of("localhost"))
.build();
// Convenience for finding an EntitySummary within a collection, based on its name
private static Predicate<EntitySummary> withName(final String name) {
return new Predicate<EntitySummary>() {
@Override
public boolean apply(EntitySummary input) {
return name.equals(input.getName());
}
};
}
@Test
public void testGetUndefinedApplication() {
try {
client().path("/applications/dummy-not-found").get(ApplicationSummary.class);
} catch (WebApplicationException e) {
assertEquals(e.getResponse().getStatus(), 404);
}
}
private static void assertRegexMatches(String actual, String patternExpected) {
if (actual==null) Assert.fail("Actual value is null; expected "+patternExpected);
if (!actual.matches(patternExpected)) {
Assert.fail("Text '"+actual+"' does not match expected pattern "+patternExpected);
}
}
@Test
public void testDeployApplication() throws Exception {
Response response = clientDeploy(simpleSpec);
HttpAsserts.assertHealthyStatusCode(response.getStatus());
assertEquals(getManagementContext().getApplications().size(), 1);
assertRegexMatches(response.getLocation().getPath(), "/applications/.*");
// Object taskO = response.getEntity(Object.class);
TaskSummary task = response.readEntity(TaskSummary.class);
log.info("deployed, got " + task);
assertEquals(task.getEntityId(), getManagementContext().getApplications().iterator().next().getApplicationId());
waitForApplicationToBeRunning(response.getLocation());
}
@Test(dependsOnMethods = { "testDeleteApplication" })
// this must happen after we've deleted the main application, as testLocatedLocations assumes a single location
public void testDeployApplicationImpl() throws Exception {
ApplicationSpec spec = ApplicationSpec.builder()
.type(RestMockApp.class.getCanonicalName())
.name("simple-app-impl")
.locations(ImmutableSet.of("localhost"))
.build();
Response response = clientDeploy(spec);
assertTrue(response.getStatus() / 100 == 2, "response is " + response);
// Expect app to be running
URI appUri = response.getLocation();
waitForApplicationToBeRunning(response.getLocation());
assertEquals(client().path(appUri).get(ApplicationSummary.class).getSpec().getName(), "simple-app-impl");
}
@Test(dependsOnMethods = { "testDeployApplication", "testLocatedLocation" })
// one intermittent failure observed 2024-02-22; possibly a local mvn jar update conflict but flagging in case something else
public void testDeployApplicationFromInterface() throws Exception {
ApplicationSpec spec = ApplicationSpec.builder()
.type(BasicApplication.class.getCanonicalName())
.name("simple-app-interface")
.locations(ImmutableSet.of("localhost"))
.build();
Response response = clientDeploy(spec);
HttpAsserts.assertHealthyStatusCode(response.getStatus());
// Expect app to be running
URI appUri = response.getLocation();
waitForApplicationToBeRunning(response.getLocation());
assertEquals(client().path(appUri).get(ApplicationSummary.class).getSpec().getName(), "simple-app-interface");
}
@Test(dependsOnMethods = { "testDeployApplication", "testLocatedLocation" })
public void testDeployApplicationYaml() throws Exception {
String yaml = "{ name: simple-app-yaml, location: localhost, services: [ { serviceType: "+BasicApplication.class.getCanonicalName()+" } ] }";
Response response = client().path("/applications")
.post(Entity.entity(yaml, "application/x-yaml"));
HttpAsserts.assertHealthyStatusCode(response.getStatus());
// Expect app to be running
URI appUri = response.getLocation();
waitForApplicationToBeRunning(response.getLocation());
assertEquals(client().path(appUri).get(ApplicationSummary.class).getSpec().getName(), "simple-app-yaml");
}
@Test(dependsOnMethods = { "testDeployApplication", "testLocatedLocation" })
public void testDeployWithInvalidSensorType() {
final String SENSOR_TEXT_VALUE = "TEXT";
String yaml = Joiner.on("\n").join(
"name: app-with-invalid-sensor-type",
"location: localhost",
"services:",
"- type: " + EmptySoftwareProcess.class.getName(),
"brooklyn.initializers:",
"- type: " + StaticSensor.class.getName(),
" brooklyn.config:",
" name: service.isUp",
" static.value: " + SENSOR_TEXT_VALUE);
Response response = client().path("/applications")
.post(Entity.entity(yaml, "application/x-yaml"));
HttpAsserts.assertHealthyStatusCode(response.getStatus());
// Fetch applications, find 'simple-app-yaml-with-invalid-sensor-type' application and inspect details.
Collection apps = client().path("/applications/fetch").get(Collection.class);
log.info("Applications fetched are: " + apps);
// Expect application data to be available.
Map app = ((Collection<Map>)apps).stream().filter(m -> "app-with-invalid-sensor-type".equals(m.get("name"))).findFirst().orElse(null);
Assert.assertNotNull(app, "did not find 'app-with-invalid-sensor-type'");
// Expect serviceUp to be NULL because of ClassCoercionException since text has been written to it with the StaticSensor.
Asserts.assertNull(app.get("serviceUp"), "'serviceUp' is expected to be null");
}
@Test(dependsOnMethods = { "testDeployApplication", "testLocatedLocation" })
public void testDeployApplicationYamlWithFormat() throws Exception {
String yaml = "{ name: deploy-format-app-yaml, location: localhost, services: [ { serviceType: "+BasicApplication.class.getCanonicalName()+" } ] }";
List<Attachment> body = new LinkedList<Attachment>();
body.add(new Attachment("plan", "text/plain", yaml));
body.add(new Attachment("format", "text/plain", "brooklyn-camp"));
Response response = client().path("/applications")
.header(HttpHeaders.CONTENT_TYPE, ContentType.MULTIPART_FORM_DATA)
.post(body);
HttpAsserts.assertHealthyStatusCode(response.getStatus());
// Expect app to be running
URI appUri = response.getLocation();
waitForApplicationToBeRunning(response.getLocation());
assertEquals(client().path(appUri).get(ApplicationSummary.class).getSpec().getName(), "deploy-format-app-yaml");
}
@Test
public void testReferenceCatalogEntity() throws Exception {
getManagementContext().getCatalog().addItems(Strings.lines(
"brooklyn.catalog:",
" id: test-reference",
" version: 0.0.1",
" item:",
" name: "+BasicEntity.class.getName(),
" services: [ { type: "+BasicEntity.class.getName()+" } ]"));
String yaml = "{ name: simple-app-yaml, location: localhost, services: [ { type: " + BasicEntity.class.getName() + " } ] }";
Response response = client().path("/applications")
.post(Entity.entity(yaml, "application/x-yaml"));
HttpAsserts.assertHealthyStatusCode(response.getStatus());
// Expect app to be running
URI appUri = response.getLocation();
waitForApplicationToBeRunning(response.getLocation()); // observed to fail here, app ERROR, 2024-02
assertEquals(client().path(appUri).get(ApplicationSummary.class).getSpec().getName(), "simple-app-yaml");
Response response2 = client().path(appUri.getPath())
.delete();
assertEquals(response2.getStatus(), Response.Status.ACCEPTED.getStatusCode());
}
@Test
public void testDeployWithInvalidEntityType() {
try {
clientDeploy(ApplicationSpec.builder()
.name("invalid-app")
.entities(ImmutableSet.of(new EntitySpec("invalid-ent", "not.existing.entity")))
.locations(ImmutableSet.of("localhost"))
.build());
} catch (WebApplicationException e) {
ApiError error = e.getResponse().readEntity(ApiError.class);
assertEquals(error.getMessage(), "Undefined type 'not.existing.entity'");
}
}
@Test
public void testDeployWithInvalidLocation() {
try {
clientDeploy(ApplicationSpec.builder()
.name("invalid-app")
.entities(ImmutableSet.of(new EntitySpec("simple-ent", RestMockSimpleEntity.class.getName())))
.locations(ImmutableSet.of("3423"))
.build());
} catch (WebApplicationException e) {
ApiError error = e.getResponse().readEntity(ApiError.class);
assertEquals(error.getMessage(), "Undefined location '3423'");
}
}
@Test(dependsOnMethods = "testDeployApplication")
public void testListEntities() {
Set<EntitySummary> entities = client().path("/applications/simple-app/entities")
.get(new GenericType<Set<EntitySummary>>() {});
assertEquals(entities.size(), 2);
EntitySummary entity = Iterables.find(entities, withName("simple-ent"), null);
EntitySummary group = Iterables.find(entities, withName("simple-group"), null);
Assert.assertNotNull(entity);
Assert.assertNotNull(group);
client().path(entity.getLinks().get("self")).get();
Set<EntitySummary> children = client().path(entity.getLinks().get("children"))
.get(new GenericType<Set<EntitySummary>>() {});
assertEquals(children.size(), 0);
}
@Test(dependsOnMethods = "testDeployApplication")
public void testListApplications() {
Set<ApplicationSummary> applications = client().path("/applications")
.get(new GenericType<Set<ApplicationSummary>>() { });
log.info("Applications listed are: " + applications);
for (ApplicationSummary app : applications) {
if (simpleSpec.getName().equals(app.getSpec().getName())) return;
}
Assert.fail("simple-app not found in list of applications: "+applications);
}
@Test(dependsOnMethods = "testDeployApplication")
public void testGetApplicationOnFire() {
Application app = Iterables.find(manager.getApplications(), EntityPredicates.displayNameEqualTo(simpleSpec.getName()));
Lifecycle origState = app.getAttribute(Attributes.SERVICE_STATE_ACTUAL);
ApplicationSummary summary = client().path("/applications/"+app.getId())
.get(ApplicationSummary.class);
assertEquals(summary.getStatus(), Status.RUNNING);
app.sensors().set(Attributes.SERVICE_STATE_ACTUAL, Lifecycle.ON_FIRE);
try {
ApplicationSummary summary2 = client().path("/applications/"+app.getId())
.get(ApplicationSummary.class);
log.info("Application: " + summary2);
assertEquals(summary2.getStatus(), Status.ERROR);
} finally {
app.sensors().set(Attributes.SERVICE_STATE_ACTUAL, origState);
}
}
@SuppressWarnings({ "rawtypes", "unchecked" })
@Test(dependsOnMethods = "testDeployApplication")
public void testFetchApplicationsAndEntity() {
Collection apps = client().path("/applications/fetch").get(Collection.class);
log.info("Applications fetched are: " + apps);
Map app = ((Collection<Map>)apps).stream().filter(m -> "simple-app".equals(m.get("name"))).findFirst().orElse(null);
Assert.assertNotNull(app, "did not find 'simple-app'");
Asserts.assertThat((String) app.get("applicationId"), StringPredicates.isNonBlank(), "expected value for applicationId");
Asserts.assertThat((String) app.get("serviceState"), StringPredicates.isNonBlank(), "expected value for serviceState");
Asserts.assertThat(app.get("serviceUp"), Predicates.not(Predicates.isNull()), "expected non-null for serviceUp");
Collection children = (Collection) app.get("children");
Asserts.assertSize(children, 2);
Map entitySummary = (Map) Iterables.find(children, withValueForKey("name", "simple-ent"), null);
Map groupSummary = (Map) Iterables.find(children, withValueForKey("name", "simple-group"), null);
Assert.assertNotNull(entitySummary);
Assert.assertNotNull(groupSummary);
Asserts.assertThat(
((Collection<Map>)apps).stream().filter(m -> ImmutableSet.of("simple-ent", "simple-group").contains(m.get("name"))).collect(Collectors.toList()),
CollectionFunctionals.empty(),
"Some entities should not be listed at high level");
String itemIds = app.get("id") + "," + entitySummary.get("id") + "," + groupSummary.get("id");
Collection entities = client().path("/applications/fetch")
.query("items", itemIds)
.query("sensors", "[ service.state.expected, \"host.address\" ]")
.get(Collection.class);
log.info("Applications+Entities fetched are: " + entities);
Asserts.assertSize(entities, apps.size() + 2);
Map entityDetails = (Map) Iterables.find(entities, withValueForKey("name", "simple-ent"), null);
Map groupDetails = (Map) Iterables.find(entities, withValueForKey("name", "simple-group"), null);
Assert.assertNotNull(entityDetails);
Assert.assertNotNull(groupDetails);
Assert.assertEquals(entityDetails.get("parentId"), app.get("id"));
Assert.assertNull(entityDetails.get("children"));
Assert.assertEquals(groupDetails.get("parentId"), app.get("id"));
Assert.assertNull(groupDetails.get("children"));
Collection entityGroupIds = (Collection) entityDetails.get("groupIds");
Assert.assertNotNull(entityGroupIds);
Assert.assertEquals(entityGroupIds.size(), 1);
Assert.assertEquals(entityGroupIds.iterator().next(), groupDetails.get("id"));
Collection groupMembers = (Collection) groupDetails.get("members");
Assert.assertNotNull(groupMembers);
for (Application appi : getManagementContext().getApplications()) {
Entities.dumpInfo(appi);
}
log.info("MEMBERS: " + groupMembers);
Assert.assertEquals(groupMembers.size(), 1);
Map entityMemberDetails = (Map) Iterables.find(groupMembers, withValueForKey("name", "simple-ent"), null);
Assert.assertNotNull(entityMemberDetails);
Assert.assertEquals(entityMemberDetails.get("id"), entityDetails.get("id"));
Map<String,Object> simpleEntSensors = Preconditions.checkNotNull(Map.class.cast(entityDetails.get("sensors")), "sensors");
org.apache.brooklyn.api.entity.Entity simpleEnt = Preconditions.checkNotNull(getManagementContext().<org.apache.brooklyn.api.entity.Entity>lookup(
EntityPredicates.displayNameEqualTo("simple-ent")));
Assert.assertEquals(simpleEntSensors.get("host.address"), simpleEnt.sensors().get(Attributes.ADDRESS));
Transition expected = simpleEnt.sensors().get(Attributes.SERVICE_STATE_EXPECTED);
Assert.assertEquals(simpleEntSensors.get("service.state.expected"),
MutableMap.of("state", expected.getState().name(),
"timestampUtc", expected.getTimestamp().getTime()));
}
// [{applicationId=zqxybvny1p, id=zqxybvny1p, parentId=null, name=simple-app, type=org.apache.brooklyn.entity.stock.BasicApplication,
// serviceUp=true, serviceState=RUNNING, iconUrl=null,
// children=[{id=pttwaekjhu, name=simple-ent, type=org.apache.brooklyn.rest.testing.mocks.RestMockSimpleEntity, links={self=/applications/zqxybvny1p/entities/pttwaekjhu}}, {id=dtqviudvsg, name=simple-group, type=org.apache.brooklyn.rest.testing.mocks.NameMatcherGroup, links={self=/applications/zqxybvny1p/entities/dtqviudvsg}}], groupIds=[], members=[], links={self=/applications/zqxybvny1p/entities/zqxybvny1p}, tags=[]}]
@SuppressWarnings({ "rawtypes", "unchecked" })
@Test(dependsOnMethods = "testDeployApplication")
public void testApplicationDetailsAndEntity() {
Collection apps = client().path("/applications/details").get(Collection.class);
log.info("Application details are: " + apps);
Map app = ((Collection<Map>)apps).stream().filter(m -> "simple-app".equals(m.get("name"))).findFirst().orElse(null);
Assert.assertNotNull(app, "did not find 'simple-app'");
Asserts.assertThat((String) app.get("applicationId"), StringPredicates.isNonBlank(), "expected value for applicationId");
Asserts.assertThat((String) app.get("serviceState"), StringPredicates.isNonBlank(), "expected value for serviceState");
Asserts.assertThat(app.get("serviceUp"), Predicates.not(Predicates.isNull()), "expected non-null for serviceUp");
Collection children = (Collection) app.get("children");
Asserts.assertSize(children, 2);
Map entitySummary = (Map) Iterables.find(children, withValueForKey("name", "simple-ent"), null);
Map groupSummary = (Map) Iterables.find(children, withValueForKey("name", "simple-group"), null);
Assert.assertNotNull(entitySummary);
Assert.assertNotNull(groupSummary);
Asserts.assertThat(
((Collection<Map>)apps).stream().filter(m -> ImmutableSet.of("simple-ent", "simple-group").contains(m.get("name"))).collect(Collectors.toList()),
CollectionFunctionals.empty(),
"Some entities should not be listed at high level");
String itemIds = entitySummary.get("id") + "," + groupSummary.get("id");
Collection entities1 = client().path("/applications/details")
.query("items", itemIds)
.query("includeAllApps", false)
.query("sensors", "[ service.state.expected, \"host.address\" ]")
.query("config", "*")
.get(Collection.class);
log.info("Applications+Entities details are: " + entities1);
Predicate<Collection> check = (entities) -> {
Asserts.assertSize(entities, 3);
Map entityDetails = (Map) Iterables.find(entities, withValueForKey("name", "simple-ent"), null);
Map groupDetails = (Map) Iterables.find(entities, withValueForKey("name", "simple-group"), null);
Assert.assertNotNull(entityDetails);
Assert.assertNotNull(groupDetails);
Assert.assertEquals(entityDetails.get("parentId"), app.get("id"));
Assert.assertNull(entityDetails.get("children"));
Assert.assertEquals(groupDetails.get("parentId"), app.get("id"));
Assert.assertNull(groupDetails.get("children"));
Collection entityGroupIds = (Collection) entityDetails.get("groupIds");
Assert.assertNotNull(entityGroupIds);
Assert.assertEquals(entityGroupIds.size(), 1);
Assert.assertEquals(entityGroupIds.iterator().next(), groupDetails.get("id"));
Collection groupMembers = (Collection) groupDetails.get("members");
Assert.assertNotNull(groupMembers);
log.info("MEMBERS: " + groupMembers);
Assert.assertEquals(groupMembers.size(), 1);
Map entityMemberDetails = (Map) Iterables.find(groupMembers, withValueForKey("name", "simple-ent"), null);
Assert.assertNotNull(entityMemberDetails);
Assert.assertEquals(entityMemberDetails.get("id"), entityDetails.get("id"));
Map<String,Object> simpleEntSensors = Preconditions.checkNotNull(Map.class.cast(entityDetails.get("sensors")), "sensors");
org.apache.brooklyn.api.entity.Entity simpleEnt = Preconditions.checkNotNull(getManagementContext().<org.apache.brooklyn.api.entity.Entity>lookup(
EntityPredicates.displayNameEqualTo("simple-ent")));
Assert.assertEquals(simpleEntSensors.get("host.address"), simpleEnt.sensors().get(Attributes.ADDRESS));
Transition expected = simpleEnt.sensors().get(Attributes.SERVICE_STATE_EXPECTED);
Assert.assertEquals(simpleEntSensors.get("service.state.expected"),
MutableMap.of("state", expected.getState().name(),
"timestampUtc", expected.getTimestamp().getTime()));
// and config also present
Assert.assertEquals( ((Map)groupDetails.get("config")).get("namematchergroup.regex"), "simple-ent" );
Assert.assertEquals( ((Map)entityDetails.get("config")).get("namematchergroup.regex"), null );
return true;
};
check.apply(entities1);
Collection entities2 = client().path("/applications/details")
.query("items", app.get("id"))
.query("includeAllApps", false)
.query("sensors", "[ service.state.expected, \"host.address\" ]")
.query("config", "n*")
.query("depth", 2)
.get(Collection.class);
log.info("Applications+Entities details 2 are: " + entities2);
// in order to reuse the check, copy the children from the single app to the list
MutableList entities2b = MutableList.copyOf(entities2);
entities2b.addAll((Iterable) ((Map)Iterables.getOnlyElement(entities2)).get("children"));
check.apply(entities2b);
}
@Test(dependsOnMethods = "testDeployApplication")
public void testListSensors() {
Set<SensorSummary> sensors = client().path("/applications/simple-app/entities/simple-ent/sensors")
.get(new GenericType<Set<SensorSummary>>() { });
assertTrue(sensors.size() > 0);
SensorSummary sample = Iterables.find(sensors, new Predicate<SensorSummary>() {
@Override
public boolean apply(SensorSummary sensorSummary) {
return sensorSummary.getName().equals(RestMockSimpleEntity.SAMPLE_SENSOR.getName());
}
});
assertEquals(sample.getType(), "java.lang.String");
}
@Test(dependsOnMethods = "testDeployApplication")
public void testListConfig() {
Set<EntityConfigSummary> config = client().path("/applications/simple-app/entities/simple-ent/config")
.get(new GenericType<Set<EntityConfigSummary>>() { });
assertTrue(config.size() > 0);
System.out.println(("CONFIG: " + config));
}
@Test(dependsOnMethods = "testListConfig")
public void testListConfig2() {
Set<EntityConfigSummary> config = client().path("/applications/simple-app/entities/simple-ent/config")
.get(new GenericType<Set<EntityConfigSummary>>() {});
assertTrue(config.size() > 0);
System.out.println(("CONFIG: " + config));
}
@Test(dependsOnMethods = "testDeployApplication")
public void testListEffectors() {
Set<EffectorSummary> effectors = client().path("/applications/simple-app/entities/simple-ent/effectors")
.get(new GenericType<Set<EffectorSummary>>() {});
assertTrue(effectors.size() > 0);
EffectorSummary sampleEffector = find(effectors, new Predicate<EffectorSummary>() {
@Override
public boolean apply(EffectorSummary input) {
return input.getName().equals("sampleEffector");
}
});
assertEquals(sampleEffector.getReturnType(), "java.lang.String");
}
@Test(dependsOnMethods = "testListSensors")
public void testTriggerSampleEffector() throws InterruptedException, IOException {
Response response = client()
.path("/applications/simple-app/entities/simple-ent/effectors/"+RestMockSimpleEntity.SAMPLE_EFFECTOR.getName())
.type(MediaType.APPLICATION_JSON_TYPE)
.post(ImmutableMap.of("param1", "foo", "param2", 4));
assertEquals(response.getStatus(), Response.Status.ACCEPTED.getStatusCode());
String result = response.readEntity(String.class);
assertEquals(result, "foo4");
}
@Test(dependsOnMethods = "testListSensors")
public void testTriggerSampleEffectorWithFormData() throws InterruptedException, IOException {
MultivaluedMap<String, String> data = new MultivaluedHashMap<>();
data.add("param1", "foo");
data.add("param2", "4");
Response response = client()
.path("/applications/simple-app/entities/simple-ent/effectors/"+RestMockSimpleEntity.SAMPLE_EFFECTOR.getName())
.type(MediaType.APPLICATION_FORM_URLENCODED_TYPE)
.post(data);
assertEquals(response.getStatus(), Response.Status.ACCEPTED.getStatusCode());
String result = response.readEntity(String.class);
assertEquals(result, "foo4");
}
@Test(dependsOnMethods = "testTriggerSampleEffector")
public void testBatchSensorValues() {
WebClient client = client().path("/applications/simple-app/entities/simple-ent/sensors/current-state");
Map<String, Object> sensors = client.get(new GenericType<Map<String, Object>>() {});
assertTrue(sensors.size() > 0);
assertEquals(sensors.get(RestMockSimpleEntity.SAMPLE_SENSOR.getName()), "foo4");
}
@Test(dependsOnMethods = "testBatchSensorValues")
public void testReadEachSensor() {
Set<SensorSummary> sensors = client().path("/applications/simple-app/entities/simple-ent/sensors")
.get(new GenericType<Set<SensorSummary>>() {});
Map<String, String> readings = Maps.newHashMap();
for (SensorSummary sensor : sensors) {
try {
readings.put(sensor.getName(), client().path(sensor.getLinks().get("self")).accept(MediaType.TEXT_PLAIN).get(String.class));
} catch (WebApplicationException uie) {
if (uie.getResponse().getStatus() == 204) { // no content
readings.put(sensor.getName(), null);
} else {
Exceptions.propagate(uie);
}
}
}
assertEquals(readings.get(RestMockSimpleEntity.SAMPLE_SENSOR.getName()), "foo4");
}
@Test(dependsOnMethods = "testTriggerSampleEffector")
public void testPolicyWhichCapitalizes() throws Exception {
String policiesEndpoint = "/applications/simple-app/entities/simple-ent/policies";
Set<PolicySummary> policies = client().path(policiesEndpoint).get(new GenericType<Set<PolicySummary>>(){});
assertEquals(policies.size(), 0);
Response response = client().path(policiesEndpoint)
.query("type", CapitalizePolicy.class.getCanonicalName())
.type(MediaType.APPLICATION_JSON_TYPE)
.post(toJsonEntity(ImmutableMap.of()));
assertEquals(response.getStatus(), 200);
PolicySummary policy = response.readEntity(PolicySummary.class);
assertNotNull(policy.getId());
String newPolicyId = policy.getId();
log.info("POLICY CREATED: " + newPolicyId);
policies = client().path(policiesEndpoint).get(new GenericType<Set<PolicySummary>>() {});
assertEquals(policies.size(), 1);
Lifecycle status = client().path(policiesEndpoint + "/" + newPolicyId).get(Lifecycle.class);
log.info("POLICY STATUS: " + status);
response = client().path(policiesEndpoint+"/"+newPolicyId+"/start")
.post(null);
assertEquals(response.getStatus(), 204);
status = client().path(policiesEndpoint + "/" + newPolicyId).get(Lifecycle.class);
assertEquals(status, Lifecycle.RUNNING);
response = client().path(policiesEndpoint+"/"+newPolicyId+"/stop")
.post(null);
assertEquals(response.getStatus(), 204);
status = client().path(policiesEndpoint + "/" + newPolicyId).get(Lifecycle.class);
assertEquals(status, Lifecycle.STOPPED);
response = client().path(policiesEndpoint+"/"+newPolicyId+"/destroy")
.post(null);
assertEquals(response.getStatus(), 204);
response = client().path(policiesEndpoint+"/"+newPolicyId).get();
log.info("POLICY STATUS RESPONSE AFTER DESTROY: " + response.getStatus());
assertEquals(response.getStatus(), 404);
policies = client().path(policiesEndpoint).get(new GenericType<Set<PolicySummary>>() {});
assertEquals(0, policies.size());
}
@SuppressWarnings({ "rawtypes" })
@Test(dependsOnMethods = "testDeployApplication")
public void testLocatedLocation() {
log.info("starting testLocatedLocations");
testListApplications();
LocationInternal l = (LocationInternal) getManagementContext().getApplications().iterator().next().getLocations().iterator().next();
if (l.config().getLocalRaw(LocationConfigKeys.LATITUDE).isAbsent()) {
log.info("Supplying fake locations for localhost because could not be autodetected");
((AbstractLocation) l).setHostGeoInfo(new HostGeoInfo("localhost", "localhost", 50, 0));
}
Map result = client().path("/locations/usage/LocatedLocations")
.get(Map.class);
log.info("LOCATIONS: " + result);
Assert.assertEquals(result.size(), 1);
Map details = (Map) result.values().iterator().next();
assertEquals(details.get("leafEntityCount"), 2); // observed to fail here once, 2024-02, size 1
}
@Test(dependsOnMethods = {"testListEffectors", "testFetchApplicationsAndEntity", "testApplicationDetailsAndEntity", "testTriggerSampleEffector", "testListApplications","testReadEachSensor","testPolicyWhichCapitalizes","testLocatedLocation"})
public void testDeleteApplication() throws TimeoutException, InterruptedException {
waitForPageFoundResponse("/applications/simple-app", ApplicationSummary.class);
Collection<Application> apps = getManagementContext().getApplications();
log.info("Deleting simple-app from " + apps);
int size = apps.size();
Response response = client().path("/applications/simple-app")
.delete();
assertEquals(response.getStatus(), Response.Status.ACCEPTED.getStatusCode());
TaskSummary task = response.readEntity(TaskSummary.class);
assertTrue(task.getDescription().toLowerCase().contains("destroy"), task.getDescription());
assertTrue(task.getDescription().toLowerCase().contains("simple-app"), task.getDescription());
waitForPageNotFoundResponse("/applications/simple-app", ApplicationSummary.class);
log.info("App appears gone, apps are: " + getManagementContext().getApplications());
// more logging above, for failure in the check below
Asserts.eventually(
EntityFunctions.applications(getManagementContext()),
Predicates.compose(Predicates.equalTo(size-1), CollectionFunctionals.sizeFunction()) );
}
@Test
public void testDisabledApplicationCatalog() throws TimeoutException, InterruptedException {
String itemSymbolicName = "my.catalog.item.id.for.disabling";
String itemVersion = "1.0";
String serviceType = "org.apache.brooklyn.entity.stock.BasicApplication";
CatalogItemSummary itemSummary = testTemplateItem(itemSymbolicName, itemVersion, serviceType);
String itemVersionedId = String.format("%s:%s", itemSummary.getSymbolicName(), itemSummary.getVersion());
assertEquals(itemSummary.getId(), itemVersionedId);
try {
// Create an app before disabling: this should work
String yaml = "{ name: my-app, location: localhost, services: [ { type: \""+itemVersionedId+"\" } ] }";
Response response = client().path("/applications")
.post(Entity.entity(yaml, "application/x-yaml"));
HttpAsserts.assertHealthyStatusCode(response.getStatus());
waitForPageFoundResponse("/applications/my-app", ApplicationSummary.class);
// Deprecate
deprecateCatalogItem(itemSymbolicName, itemVersion, true);
// Create an app when deprecated: this should work
String yaml2 = "{ name: my-app2, location: localhost, services: [ { type: \""+itemVersionedId+"\" } ] }";
Response response2 = client().path("/applications")
.post(Entity.entity(yaml2, "application/x-yaml"));
HttpAsserts.assertHealthyStatusCode(response2.getStatus());
waitForPageFoundResponse("/applications/my-app2", ApplicationSummary.class);
// Disable
disableCatalogItem(itemSymbolicName, itemVersion, true);
// Now try creating an app; this should fail because app is disabled
String yaml3 = "{ name: my-app3, location: localhost, services: [ { type: \""+itemVersionedId+"\" } ] }";
Response response3 = client().path("/applications")
.post(Entity.entity(yaml3, "application/x-yaml"));
HttpAsserts.assertClientErrorStatusCode(response3.getStatus());
assertTrue(response3.readEntity(String.class).toLowerCase().contains("unable to match"));
waitForPageNotFoundResponse("/applications/my-app3", ApplicationSummary.class);
} finally {
client().path("/applications/my-app")
.delete();
client().path("/applications/my-app2")
.delete();
client().path("/applications/my-app3")
.delete();
client().path("/catalog/entities/"+itemVersionedId+"/"+itemVersion)
.delete();
}
}
protected CatalogEntitySummary testTemplateItem(String itemSymbolicName, String itemVersion, String serviceType) {
// Deploy the catalog item
addTestCatalogItem(itemSymbolicName, "template", itemVersion, serviceType);
log.info("Types after adding template: "+manager.getTypeRegistry().getAll());
List<CatalogEntitySummary> itemSummaries = client().path("/catalog/applications")
.query("fragment", itemSymbolicName).query("allVersions", "true").get(new GenericType<List<CatalogEntitySummary>>() {});
return Iterables.getOnlyElement(itemSummaries);
}
@Test(dependsOnMethods = { "testDeployApplication", "testLocatedLocation" })
public void testDeploymentUsesAppId() throws Exception {
String yaml = "{ name: simple-app-yaml, services: [ { type: "+BasicApplication.class.getCanonicalName()+" } ] }";
String appId = "myidtestdeploymentuidisset";
Response response = deployApp(yaml, appId);
assertResponseStatus(response, 201);
BasicApplication app = (BasicApplication) getManagementContext().getEntityManager().getEntity(appId);
assertNotNull(app);
assertEquals(app.getId(), appId);
}
@Test(dependsOnMethods = { "testDeployApplication", "testLocatedLocation" })
public void testDeploymentFailsOnInvalidAppId() throws Exception {
assertAppIdValid("abcdefghijkl");
assertAppIdValid("1234567890");
assertAppIdValid(com.google.common.base.Strings.repeat("a", 63));
assertAppIdInvalid("tooshort"); // must be at least 10 chars
assertAppIdInvalid("a-bcdefghijkl"); // must be letters/digits only
assertAppIdInvalid("Abcdefghijkl"); // must be lower-case only
assertAppIdInvalid(com.google.common.base.Strings.repeat("a", 64)); // too long
}
private void assertAppIdInvalid(String appId) {
String yaml = "{ name: simple-app-yaml, services: [ { type: "+BasicApplication.class.getCanonicalName()+" } ] }";
Response response = deployApp(yaml, appId);
assertResponseStatus(response, 400, StringPredicates.containsLiteral("Invalid entity id '"+appId+"'"));
assertNull(getManagementContext().getEntityManager().getEntity(appId));
}
private void assertAppIdValid(String appId) {
String yaml = "{ name: simple-app-yaml, services: [ { type: "+BasicApplication.class.getCanonicalName()+" } ] }";
Response response = deployApp(yaml, appId);
assertResponseStatus(response, 201);
assertNotNull(getManagementContext().getEntityManager().getEntity(appId));
}
@Test(dependsOnMethods = { "testDeployApplication", "testLocatedLocation" })
public void testDeploymentFailsOnDuplicateAppId() throws Exception {
// First app
String appId = "myuidtestdeploymentuidfailsonduplicate";
String yaml = "{ name: my-name-1, services: [ { type: "+BasicApplication.class.getCanonicalName()+" } ] }";
Response response = deployApp(yaml, appId);
assertResponseStatus(response, 201);
BasicApplication app = (BasicApplication) getManagementContext().getEntityManager().getEntity(appId);
assertNotNull(app);
// Second app should get a conflict response (409)
String yaml2 = "{ name: my-name-2, services: [ { type: "+BasicApplication.class.getCanonicalName()+" } ] }";
Response response2 = deployApp(yaml2, appId);
assertResponseStatus(response2, 409, StringPredicates.containsAllLiterals(
IdAlreadyExistsException.class.getSimpleName(), "already known under that id '"+appId+"'"));
Optional<Application> app2 = Iterables.tryFind(getManagementContext().getApplications(), EntityPredicates.displayNameEqualTo("my-name-2"));
assertFalse(app2.isPresent(), "app2="+app2);
// Third app with different app id should work
String appId3 = "myuiddifferent";
String yaml3 = "{ name: my-name-3, services: [ { type: "+BasicApplication.class.getCanonicalName()+" } ] }";
Response response3 = deployApp(yaml3, appId3);
assertResponseStatus(response3, 201);
BasicApplication app3 = (BasicApplication) getManagementContext().getEntityManager().getEntity(appId3);
assertNotNull(app3);
// Delete app1; then deploying app2 should succeed
Entities.unmanage(app);
Response response2b = deployApp(yaml2, appId);
assertResponseStatus(response2b, 201);
BasicApplication app2b = (BasicApplication) getManagementContext().getEntityManager().getEntity(appId);
assertNotNull(app2b);
assertEquals(app2b.getDisplayName(), "my-name-2");
}
private Response deployApp(String yaml, String appId) {
return client().path("/applications/"+appId)
.put(Entity.entity(yaml, "application/x-yaml"));
}
private void deprecateCatalogItem(String symbolicName, String version, boolean deprecated) {
String id = String.format("%s:%s", symbolicName, version);
Response response = client().path(String.format("/catalog/entities/%s/deprecated", id))
.header(HttpHeaders.CONTENT_TYPE, ContentType.APPLICATION_JSON)
.post(deprecated);
assertEquals(response.getStatus(), Response.Status.NO_CONTENT.getStatusCode());
}
private void disableCatalogItem(String symbolicName, String version, boolean disabled) {
String id = String.format("%s:%s", symbolicName, version);
Response response = client().path(String.format("/catalog/entities/%s/disabled", id))
.header(HttpHeaders.CONTENT_TYPE, ContentType.APPLICATION_JSON)
.post(disabled);
assertEquals(response.getStatus(), Response.Status.NO_CONTENT.getStatusCode());
}
private void addTestCatalogItem(String catalogItemId, String itemType, String version, String service) {
String yaml = Joiner.on("\n").join(
"brooklyn.catalog:",
" id: " + catalogItemId,
" version: " + version,
" itemType: " + checkNotNull(itemType, "itemType"),
" name: My Catalog App",
" description: My description",
" icon_url: classpath:///redis-logo.png",
" item:",
(itemType.equals("template")
?
" services:\n" +
" - type: " + service
:
" type: " + service));
client().path("/catalog").post(yaml);
}
private void assertResponseStatus(Response response, int expectedStatus, Predicate<? super String> expectedBody) {
assertResponseStatus(response, Range.singleton(expectedStatus), expectedBody);
}
private void assertResponseStatus(Response response, int expectedStatus) {
assertResponseStatus(response, Range.singleton(expectedStatus), Predicates.alwaysTrue());
}
private void assertResponseStatus(Response response, Range<Integer> expectedStatusRange, Predicate<? super String> expectedBody) {
String body = Streams.readFullyString((InputStream)response.getEntity());
String errMsg = "response is "+response+"; status="+response.getStatus()+"; reason="+response.getStatusInfo().getReasonPhrase()+"; body="+body;
assertTrue(expectedStatusRange.contains(response.getStatus()), errMsg);
assertTrue(expectedBody.apply(body), errMsg);
}
}