/*******************************************************************************
* Copyright (c) 2012-2017 Codenvy, S.A.
* All rights reserved. This program and the accompanying materials
* are made available under the terms of the Eclipse Public License v1.0
* which accompanies this distribution, and is available at
* http://www.eclipse.org/legal/epl-v10.html
*
* Contributors:
* Codenvy, S.A. - initial API and implementation
*******************************************************************************/
package org.eclipse.che.api.workspace.server.spi.tck;
import com.google.common.collect.ImmutableList;
import com.google.common.collect.ImmutableMap;
import org.eclipse.che.account.spi.AccountImpl;
import org.eclipse.che.api.core.ConflictException;
import org.eclipse.che.api.core.NotFoundException;
import org.eclipse.che.api.core.ServerException;
import org.eclipse.che.api.core.notification.EventService;
import org.eclipse.che.api.machine.server.model.impl.CommandImpl;
import org.eclipse.che.api.workspace.server.event.BeforeWorkspaceRemovedEvent;
import org.eclipse.che.api.workspace.server.event.WorkspaceRemovedEvent;
import org.eclipse.che.api.workspace.server.model.impl.EnvironmentImpl;
import org.eclipse.che.api.workspace.server.model.impl.EnvironmentRecipeImpl;
import org.eclipse.che.api.workspace.server.model.impl.ExtendedMachineImpl;
import org.eclipse.che.api.workspace.server.model.impl.ProjectConfigImpl;
import org.eclipse.che.api.workspace.server.model.impl.ServerConf2Impl;
import org.eclipse.che.api.workspace.server.model.impl.SourceStorageImpl;
import org.eclipse.che.api.workspace.server.model.impl.WorkspaceConfigImpl;
import org.eclipse.che.api.workspace.server.model.impl.WorkspaceImpl;
import org.eclipse.che.api.workspace.server.spi.WorkspaceDao;
import org.eclipse.che.commons.test.tck.TckListener;
import org.eclipse.che.commons.test.tck.repository.TckRepository;
import org.eclipse.che.commons.test.tck.repository.TckRepositoryException;
import org.eclipse.che.core.db.cascade.CascadeEventSubscriber;
import org.eclipse.che.core.db.cascade.event.CascadeEvent;
import org.testng.annotations.AfterMethod;
import org.testng.annotations.BeforeMethod;
import org.testng.annotations.Listeners;
import org.testng.annotations.Test;
import javax.inject.Inject;
import java.util.ArrayList;
import java.util.Arrays;
import java.util.HashMap;
import java.util.HashSet;
import java.util.List;
import java.util.Map;
import java.util.Optional;
import java.util.stream.Stream;
import static java.util.Arrays.asList;
import static java.util.Collections.singletonList;
import static java.util.Collections.singletonMap;
import static java.util.stream.Collectors.toList;
import static org.mockito.Matchers.any;
import static org.mockito.Mockito.doCallRealMethod;
import static org.mockito.Mockito.doThrow;
import static org.mockito.Mockito.mock;
import static org.testng.Assert.assertEquals;
import static org.testng.Assert.assertTrue;
import static org.testng.Assert.fail;
/**
* Tests {@link WorkspaceDao} contract.
*
* @author Yevhenii Voevodin
*/
@Listeners(TckListener.class)
@Test(suiteName = WorkspaceDaoTest.SUITE_NAME)
public class WorkspaceDaoTest {
public static final String SUITE_NAME = "WorkspaceDaoTck";
private static final int COUNT_OF_WORKSPACES = 5;
private static final int COUNT_OF_ACCOUNTS = 3;
@Inject
private TckRepository<WorkspaceImpl> workspaceRepo;
@Inject
private TckRepository<AccountImpl> accountRepo;
@Inject
private WorkspaceDao workspaceDao;
@Inject
private EventService eventService;
private AccountImpl[] accounts;
private WorkspaceImpl[] workspaces;
@AfterMethod
public void removeEntities() throws TckRepositoryException {
workspaceRepo.removeAll();
accountRepo.removeAll();
}
@BeforeMethod
public void createEntities() throws TckRepositoryException {
accounts = new AccountImpl[COUNT_OF_ACCOUNTS];
for (int i = 0; i < COUNT_OF_ACCOUNTS; i++) {
accounts[i] = new AccountImpl("accountId" + i, "accountName" + i, "test");
}
workspaces = new WorkspaceImpl[COUNT_OF_WORKSPACES];
for (int i = 0; i < COUNT_OF_WORKSPACES; i++) {
// 2 workspaces share 1 namespace
workspaces[i] = createWorkspace("workspace-" + i, accounts[i / 2], "name-" + i);
}
accountRepo.createAll(Arrays.asList(accounts));
workspaceRepo.createAll(Stream.of(workspaces).map(WorkspaceImpl::new).collect(toList()));
}
@Test
public void shouldGetWorkspaceById() throws Exception {
final WorkspaceImpl workspace = workspaces[0];
assertEquals(workspaceDao.get(workspace.getId()), new WorkspaceImpl(workspace));
}
@Test(expectedExceptions = NotFoundException.class)
public void shouldThrowNotFoundExceptionWhenGettingNonExistingWorkspaceById() throws Exception {
workspaceDao.get("non-existing-id");
}
@Test(expectedExceptions = NullPointerException.class)
public void shouldThrowNpeWhenGettingWorkspaceByIdWhereIdIsNull() throws Exception {
workspaceDao.get(null);
}
@Test
public void shouldGetWorkspacesByNamespace() throws Exception {
final WorkspaceImpl workspace1 = workspaces[0];
final WorkspaceImpl workspace2 = workspaces[1];
assertEquals(workspace1.getNamespace(), workspace2.getNamespace(), "Namespaces must be the same");
final List<WorkspaceImpl> found = workspaceDao.getByNamespace(workspace1.getNamespace());
assertEquals(new HashSet<>(found), new HashSet<>(asList(workspace1, workspace2)));
}
@Test
public void emptyListShouldBeReturnedWhenThereAreNoWorkspacesInGivenNamespace() throws Exception {
assertTrue(workspaceDao.getByNamespace("non-existing-namespace").isEmpty());
}
@Test(expectedExceptions = NullPointerException.class)
public void shouldThrowNpeWhenGettingWorkspaceByNullNamespace() throws Exception {
workspaceDao.getByNamespace(null);
}
@Test
public void shouldGetWorkspaceByNameAndNamespace() throws Exception {
final WorkspaceImpl workspace = workspaces[0];
assertEquals(workspaceDao.get(workspace.getConfig().getName(), workspace.getNamespace()), new WorkspaceImpl(workspace));
}
@Test(expectedExceptions = NotFoundException.class)
public void shouldThrowNotFoundExceptionWhenWorkspaceWithSuchNameDoesNotExist() throws Exception {
final WorkspaceImpl workspace = workspaces[0];
workspaceDao.get("non-existing-name", workspace.getNamespace());
}
@Test(expectedExceptions = NotFoundException.class)
public void shouldThrowNotFoundExceptionWhenWorkspaceWithSuchNamespaceDoesNotExist() throws Exception {
final WorkspaceImpl workspace = workspaces[0];
workspaceDao.get(workspace.getConfig().getName(), "non-existing-namespace");
}
@Test(expectedExceptions = NotFoundException.class)
public void shouldThrowNotFoundExceptionWhenWorkspaceWithSuchNameDoesNotExistInGiveWorkspace() throws Exception {
final WorkspaceImpl workspace1 = workspaces[0];
final WorkspaceImpl workspace2 = workspaces[2];
workspaceDao.get(workspace1.getConfig().getName(), workspace2.getNamespace());
}
@Test(expectedExceptions = NullPointerException.class)
public void shouldThrowNpeWhenGettingWorkspaceByNameAndNamespaceWhereNameIsNull() throws Exception {
workspaceDao.get(null, workspaces[0].getNamespace());
}
@Test(expectedExceptions = NullPointerException.class)
public void shouldThrowNpeWhenGettingWorkspaceByNameAndNamespaceWhereNamespaceIsNull() throws Exception {
workspaceDao.get(workspaces[0].getConfig().getName(), null);
}
@Test(expectedExceptions = NotFoundException.class,
dependsOnMethods = "shouldThrowNotFoundExceptionWhenGettingNonExistingWorkspaceById")
public void shouldRemoveWorkspace() throws Exception {
final WorkspaceImpl workspace = workspaces[0];
workspaceDao.remove(workspace.getId());
workspaceDao.get(workspace.getId());
}
@Test(dependsOnMethods = "shouldGetWorkspaceById")
public void shouldNotRemoveWorkspaceWhenSubscriberThrowsExceptionOnWorkspaceRemoving() throws Exception {
final WorkspaceImpl workspace = workspaces[0];
CascadeEventSubscriber<BeforeWorkspaceRemovedEvent> subscriber = mockCascadeEventSubscriber();
doThrow(new ServerException("error")).when(subscriber).onCascadeEvent(any());
eventService.subscribe(subscriber, BeforeWorkspaceRemovedEvent.class);
try {
workspaceDao.remove(workspace.getId());
fail("WorkspaceDao#remove had to throw server exception");
} catch (ServerException ignored) {
}
assertEquals(workspaceDao.get(workspace.getId()), workspace);
eventService.unsubscribe(subscriber, BeforeWorkspaceRemovedEvent.class);
}
@Test
public void shouldGetWorkspacesByNonTemporary() throws Exception {
List<WorkspaceImpl> result = workspaceDao.getWorkspaces(false, 0, 2);
assertEquals(result.size(), 2);
assertEquals(new HashSet<>(result), new HashSet<>(asList(workspaces[0], workspaces[1])));
}
@Test
public void shouldGetWorkspacesByTemporary() throws Exception {
final WorkspaceImpl workspace = workspaces[0];
workspace.setTemporary(true);
workspaceDao.update(workspace);
List<WorkspaceImpl> result = workspaceDao.getWorkspaces(true, 0, 0);
assertEquals(result.size(), 1);
assertEquals(result.iterator().next(), workspaceDao.get(workspace.getId()));
}
@Test(expectedExceptions = IllegalArgumentException.class)
public void shouldThrowIllegalStateExceptionOnNegativeLimit() throws Exception {
workspaceDao.getWorkspaces(true, 0, -2);
}
@Test(expectedExceptions = IllegalArgumentException.class)
public void shouldThrowIllegalStateExceptionOnNegativeSkipCount() throws Exception {
workspaceDao.getWorkspaces(true, -2, 0);
}
@Test
public void shouldPublicRemoveWorkspaceEventAfterRemoveWorkspace() throws Exception {
final boolean[] isNotified = new boolean[] {false};
eventService.subscribe(event -> isNotified[0] = true, WorkspaceRemovedEvent.class);
workspaceDao.remove(workspaces[0].getId());
assertTrue(isNotified[0], "Event subscriber notified");
}
@Test
public void shouldNotThrowExceptionWhenRemovingNonExistingWorkspace() throws Exception {
workspaceDao.remove("non-existing-id");
}
@Test(expectedExceptions = NullPointerException.class)
public void shouldThrowNpeWhenRemovingNull() throws Exception {
workspaceDao.remove(null);
}
@Test(dependsOnMethods = "shouldGetWorkspaceById")
public void shouldCreateWorkspace() throws Exception {
final WorkspaceImpl workspace = createWorkspace("new-workspace", accounts[0], "new-name");
workspaceDao.create(workspace);
assertEquals(workspaceDao.get(workspace.getId()), new WorkspaceImpl(workspace, workspace.getAccount()));
}
@Test(expectedExceptions = ConflictException.class)
public void shouldNotCreateWorkspaceWithANameWhichAlreadyExistsInGivenNamespace() throws Exception {
final WorkspaceImpl workspace = workspaces[0];
final WorkspaceImpl newWorkspace = createWorkspace("new-id", workspace.getAccount(), workspace.getConfig().getName());
workspaceDao.create(newWorkspace);
}
@Test
public void shouldCreateWorkspaceWithNameWhichDoesNotExistInGivenNamespace() throws Exception {
final WorkspaceImpl workspace = workspaces[0];
final WorkspaceImpl workspace2 = workspaces[4];
final WorkspaceImpl newWorkspace = createWorkspace("new-id", workspace.getAccount(), workspace2.getConfig().getName());
final WorkspaceImpl expected = new WorkspaceImpl(newWorkspace, newWorkspace.getAccount());
expected.setAccount(newWorkspace.getAccount());
assertEquals(workspaceDao.create(newWorkspace), expected);
}
@Test(expectedExceptions = ConflictException.class)
public void shouldThrowConflictExceptionWhenCreatingWorkspaceWithExistingId() throws Exception {
final WorkspaceImpl workspace = workspaces[0];
final WorkspaceImpl newWorkspace = createWorkspace(workspace.getId(), accounts[0], "new-name");
workspaceDao.create(newWorkspace);
}
@Test(dependsOnMethods = "shouldGetWorkspaceById")
public void shouldUpdateWorkspace() throws Exception {
final WorkspaceImpl workspace = new WorkspaceImpl(workspaces[0], workspaces[0].getAccount());
// Remove an existing project configuration from workspace
workspace.getConfig().getProjects().remove(1);
// Add new project to the workspace configuration
final SourceStorageImpl source3 = new SourceStorageImpl();
source3.setType("type3");
source3.setLocation("location3");
source3.setParameters(new HashMap<>(ImmutableMap.of("param1", "value1",
"param2", "value2",
"param3", "value3")));
final ProjectConfigImpl newProjectCfg = new ProjectConfigImpl();
newProjectCfg.setPath("/path3");
newProjectCfg.setType("type3");
newProjectCfg.setName("project3");
newProjectCfg.setDescription("description3");
newProjectCfg.getMixins().addAll(asList("mixin3", "mixin4"));
newProjectCfg.setSource(source3);
newProjectCfg.getAttributes().put("new-key", asList("1", "2"));
workspace.getConfig().getProjects().add(newProjectCfg);
// Update an existing project configuration
final ProjectConfigImpl projectCfg = workspace.getConfig().getProjects().get(0);
projectCfg.getAttributes().clear();
projectCfg.getSource().setLocation("new-location");
projectCfg.getSource().setType("new-type");
projectCfg.getSource().getParameters().put("new-param", "new-param-value");
projectCfg.getMixins().add("new-mixin");
projectCfg.setPath("/new-path");
projectCfg.setDescription("new project description");
// Remove an existing command
workspace.getConfig().getCommands().remove(1);
// Add a new command
final CommandImpl newCmd = new CommandImpl();
newCmd.setName("name3");
newCmd.setType("type3");
newCmd.setCommandLine("cmd3");
newCmd.getAttributes().putAll(ImmutableMap.of("attr1", "value1",
"attr2", "value2",
"attr3", "value3"));
workspace.getConfig().getCommands().add(newCmd);
// Update an existing command
final CommandImpl command = workspace.getConfig().getCommands().get(0);
command.setName("new-name");
command.setType("new-type");
command.setCommandLine("new-command-line");
command.getAttributes().clear();
// Add a new environment
final EnvironmentRecipeImpl newRecipe = new EnvironmentRecipeImpl();
newRecipe.setLocation("new-location");
newRecipe.setType("new-type");
newRecipe.setContentType("new-content-type");
newRecipe.setContent("new-content");
final ExtendedMachineImpl newMachine = new ExtendedMachineImpl();
final ServerConf2Impl serverConf1 = new ServerConf2Impl("2265", "http", singletonMap("prop1", "val"));
final ServerConf2Impl serverConf2 = new ServerConf2Impl("2266", "ftp", singletonMap("prop1", "val"));
newMachine.setServers(ImmutableMap.of("ref1", serverConf1, "ref2", serverConf2));
newMachine.setAgents(ImmutableList.of("agent5", "agent4"));
newMachine.setAttributes(singletonMap("att1", "val"));
final EnvironmentImpl newEnv = new EnvironmentImpl();
newEnv.setMachines(ImmutableMap.of("new-machine", newMachine));
newEnv.setRecipe(newRecipe);
workspace.getConfig().getEnvironments().put("new-env", newEnv);
// Update an existing environment
final EnvironmentImpl defaultEnv = workspace.getConfig().getEnvironments().get(workspace.getConfig().getDefaultEnv());
// Remove an existing machine config
final List<String> machineNames = new ArrayList<>(defaultEnv.getMachines().keySet());
// Update an existing machine
final ExtendedMachineImpl existingMachine = defaultEnv.getMachines().get(machineNames.get(1));
existingMachine.setAgents(asList("new-agent1", "new-agent2"));
existingMachine.setAttributes(ImmutableMap.of("attr1", "value1",
"attr2", "value2",
"attr3", "value3"));
existingMachine.getServers().clear();
existingMachine.getServers().put("new-ref", new ServerConf2Impl("new-port",
"new-protocol",
ImmutableMap.of("prop1", "value")));
defaultEnv.getMachines().remove(machineNames.get(0));
defaultEnv.getRecipe().setContent("updated-content");
defaultEnv.getRecipe().setContentType("updated-content-type");
defaultEnv.getRecipe().setLocation("updated-location");
defaultEnv.getRecipe().setType("updated-type");
// Remove an existing environment
final Optional<String> nonDefaultEnv = workspace.getConfig()
.getEnvironments()
.keySet()
.stream()
.filter(key -> !key.equals(workspace.getConfig().getDefaultEnv()))
.findAny();
assertTrue(nonDefaultEnv.isPresent());
workspace.getConfig().getEnvironments().remove(nonDefaultEnv.get());
// Update workspace configuration
final WorkspaceConfigImpl wCfg = workspace.getConfig();
wCfg.setDefaultEnv("new-env");
wCfg.setName("new-name");
wCfg.setDescription("This is a new description");
// Update workspace object
workspace.setAccount(new AccountImpl("accId", "new-namespace", "test"));
workspace.getAttributes().clear();
workspaceDao.update(workspace);
assertEquals(workspaceDao.get(workspace.getId()), new WorkspaceImpl(workspace, workspace.getAccount()));
}
@Test(expectedExceptions = NotFoundException.class)
public void shouldNotUpdateWorkspaceWhichDoesNotExist() throws Exception {
final WorkspaceImpl workspace = workspaces[0];
workspace.setId("non-existing-workspace");
workspaceDao.update(workspace);
}
@Test(expectedExceptions = ConflictException.class)
public void shouldNotUpdateWorkspaceWithReservedName() throws Exception {
final WorkspaceImpl workspace1 = workspaces[0];
final WorkspaceImpl workspace2 = workspaces[1];
workspace1.getConfig().setName(workspace2.getConfig().getName());
workspaceDao.update(workspace1);
}
@Test(dependsOnMethods = "shouldGetWorkspaceById")
public void createsWorkspaceWithAProjectConfigContainingLongAttributeValues() throws Exception {
WorkspaceImpl workspace = createWorkspace("new-workspace", accounts[0], "new-name");
ProjectConfigImpl project = workspace.getConfig().getProjects().get(0);
// long string
char[] chars = new char[100_000];
Arrays.fill(chars, 0, chars.length / 2, 'x');
Arrays.fill(chars, chars.length / 2, chars.length, 'y');
String value = new String(chars);
project.getAttributes().put("long_value1", singletonList(value));
project.getAttributes().put("long_value2", singletonList(value));
workspaceDao.create(workspace);
assertEquals(workspaceDao.get(workspace.getId()), new WorkspaceImpl(workspace));
}
@Test(expectedExceptions = NullPointerException.class)
public void shouldThrowNpeWhenUpdatingNull() throws Exception {
workspaceDao.update(null);
}
public static WorkspaceConfigImpl createWorkspaceConfig(String name) {
// Project Sources configuration
final SourceStorageImpl source1 = new SourceStorageImpl();
source1.setType("type1");
source1.setLocation("location1");
source1.setParameters(new HashMap<>(ImmutableMap.of("param1", "value1",
"param2", "value2",
"param3", "value3")));
final SourceStorageImpl source2 = new SourceStorageImpl();
source2.setType("type2");
source2.setLocation("location2");
source2.setParameters(new HashMap<>(ImmutableMap.of("param4", "value1",
"param5", "value2",
"param6", "value3")));
// Project Configuration
final ProjectConfigImpl pCfg1 = new ProjectConfigImpl();
pCfg1.setPath("/path1");
pCfg1.setType("type1");
pCfg1.setName("project1");
pCfg1.setDescription("description1");
pCfg1.getMixins().addAll(asList("mixin1", "mixin2"));
pCfg1.setSource(source1);
pCfg1.getAttributes().putAll(ImmutableMap.of("key1", asList("v1", "v2"), "key2", asList("v1", "v2")));
final ProjectConfigImpl pCfg2 = new ProjectConfigImpl();
pCfg2.setPath("/path2");
pCfg2.setType("type2");
pCfg2.setName("project2");
pCfg2.setDescription("description2");
pCfg2.getMixins().addAll(asList("mixin3", "mixin4"));
pCfg2.setSource(source2);
pCfg2.getAttributes().putAll(ImmutableMap.of("key3", asList("v1", "v2"), "key4", asList("v1", "v2")));
final List<ProjectConfigImpl> projects = new ArrayList<>(asList(pCfg1, pCfg2));
// Commands
final CommandImpl cmd1 = new CommandImpl("name1", "cmd1", "type1");
cmd1.getAttributes().putAll(ImmutableMap.of("key1", "value1",
"key2", "value2",
"key3", "value3"));
final CommandImpl cmd2 = new CommandImpl("name2", "cmd2", "type2");
cmd2.getAttributes().putAll(ImmutableMap.of("key4", "value4",
"key5", "value5",
"key6", "value6"));
final List<CommandImpl> commands = new ArrayList<>(asList(cmd1, cmd2));
// Machine configs
final ExtendedMachineImpl exMachine1 = new ExtendedMachineImpl();
final ServerConf2Impl serverConf1 = new ServerConf2Impl("2265", "http", singletonMap("prop1", "val"));
final ServerConf2Impl serverConf2 = new ServerConf2Impl("2266", "ftp", singletonMap("prop1", "val"));
exMachine1.setServers(ImmutableMap.of("ref1", serverConf1, "ref2", serverConf2));
exMachine1.setAgents(ImmutableList.of("agent5", "agent4"));
exMachine1.setAttributes(singletonMap("att1", "val"));
final ExtendedMachineImpl exMachine2 = new ExtendedMachineImpl();
final ServerConf2Impl serverConf3 = new ServerConf2Impl("2333", "https", singletonMap("prop2", "val"));
final ServerConf2Impl serverConf4 = new ServerConf2Impl("2334", "wss", singletonMap("prop2", "val"));
exMachine2.setServers(ImmutableMap.of("ref1", serverConf3, "ref2", serverConf4));
exMachine2.setAgents(ImmutableList.of("agent2", "agent1"));
exMachine2.setAttributes(singletonMap("att1", "val"));
final ExtendedMachineImpl exMachine3 = new ExtendedMachineImpl();
final ServerConf2Impl serverConf5 = new ServerConf2Impl("2333", "https", singletonMap("prop2", "val"));
exMachine3.setServers(singletonMap("ref1", serverConf5));
exMachine3.setAgents(ImmutableList.of("agent6", "agent2"));
exMachine3.setAttributes(singletonMap("att1", "val"));
// Environments
final EnvironmentRecipeImpl recipe1 = new EnvironmentRecipeImpl();
recipe1.setLocation("https://eclipse.che/Dockerfile");
recipe1.setType("dockerfile");
recipe1.setContentType("text/x-dockerfile");
recipe1.setContent("content");
final EnvironmentImpl env1 = new EnvironmentImpl();
env1.setMachines(new HashMap<>(ImmutableMap.of("machine1", exMachine1,
"machine2", exMachine2,
"machine3", exMachine3)));
env1.setRecipe(recipe1);
final EnvironmentRecipeImpl recipe2 = new EnvironmentRecipeImpl();
recipe2.setLocation("https://eclipse.che/Dockerfile");
recipe2.setType("dockerfile");
recipe2.setContentType("text/x-dockerfile");
recipe2.setContent("content");
final EnvironmentImpl env2 = new EnvironmentImpl();
env2.setMachines(new HashMap<>(ImmutableMap.of("machine1", new ExtendedMachineImpl(exMachine1),
"machine3", new ExtendedMachineImpl(exMachine3))));
env2.setRecipe(recipe2);
final Map<String, EnvironmentImpl> environments = ImmutableMap.of("env1", env1, "env2", env2);
// Workspace configuration
final WorkspaceConfigImpl wCfg = new WorkspaceConfigImpl();
wCfg.setDefaultEnv("env1");
wCfg.setName(name);
wCfg.setDescription("description");
wCfg.setCommands(commands);
wCfg.setProjects(projects);
wCfg.setEnvironments(new HashMap<>(environments));
return wCfg;
}
public static WorkspaceImpl createWorkspace(String id, AccountImpl account, String name) {
final WorkspaceConfigImpl wCfg = createWorkspaceConfig(name);
// Workspace
final WorkspaceImpl workspace = new WorkspaceImpl();
workspace.setId(id);
workspace.setAccount(account);
wCfg.setName(name);
workspace.setConfig(wCfg);
workspace.setAttributes(new HashMap<>(ImmutableMap.of("attr1", "value1",
"attr2", "value2",
"attr3", "value3")));
workspace.setConfig(wCfg);
return workspace;
}
private <T extends CascadeEvent> CascadeEventSubscriber<T> mockCascadeEventSubscriber() {
@SuppressWarnings("unchecked")
CascadeEventSubscriber<T> subscriber = mock(CascadeEventSubscriber.class);
doCallRealMethod().when(subscriber).onEvent(any());
return subscriber;
}
}