use of org.eclipse.vorto.repository.web.api.v1.dto.ModelFullDetailsDTO in project vorto by eclipse.
the class ModelRepositoryControllerTest method verifyFullModelPayloadForUI.
/**
* This test performs the following:
* <ol>
* <li>
* Creates the {@literal com.test} namespace with a sysadmin user, to hold the models.
* </li>
* <li>
* Adds a non-sysadmin user with {@literal model_creator} role to the {@literal com.test}
* namespace.
* </li>
* <li>
* Creates a "Zone" datatype model from the corresponding file resource.
* </li>
* <li>
* Creates a "Lamp" functionblock model from the corresponding file resource.
* </li>
* <li>
* Creates an "Address" functionblock model from the corresponding file resource.
* </li>
* <li>
* Creates a "StreetLamp" model from the corresponding file resource, using the above
* functionblocks as dependencies.
* </li>
* <li>
* Adds an attachment to the "StreetLamp" model.
* </li>
* <li>
* Adds a link to the "StreetLamp" model.
* </li>
* <li>
* Adds a mapping to the "StreetLamp" model.
* </li>
* <li>
* Loads the "StreetLamp" model with the REST call used by the UI, and verifies / validates:
* <ul>
* <li>
* Basic {@link org.eclipse.vorto.repository.core.ModelInfo} properties.
* </li>
* <li>
* The Base64-encoded model syntax
* (see {@link ModelFullDetailsDTO#getEncodedModelSyntax()})
* </li>
* <li>
* Mapping (see {@link ModelFullDetailsDTO#getMappings()}).
* </li>
* <li>
* Attachment (see {@link ModelFullDetailsDTO#getAttachments()}
* </li>
* <li>
* Link (see {@link ModelFullDetailsDTO#getLinks()}
* </li>
* <li>
* References (see {@link ModelFullDetailsDTO#getReferences()}
* </li>
* <li>
* "Referenced by" (see {@link ModelFullDetailsDTO#getReferencedBy()})
* </li>
* <li>
* Policies (see {@link ModelFullDetailsDTO#getPolicies()} and
* {@link ModelFullDetailsDTO#getBestPolicy()} for the creating user, and conversely, for
* an extraneous user with no access.
* </li>
* <li>
* Workflow actions (see {@link ModelFullDetailsDTO#getActions()} for the creating user,
* and conversely, for an extraneous user with no access.
* </li>
* </ul>
* </li>
* <li>
* Loads the "Lamp" functionblock model with the REST call used by the UI, and verifies the
* "referenced by" node (see {@link ModelFullDetailsDTO#getReferencedBy()}.
* </li>
* <li>
* Finally, cleans up and deletes all 4 models, then the namespace.
* </li>
* </ol>
*
* @throws Exception
*/
@Test
public void verifyFullModelPayloadForUI() throws Exception {
// required for some extra properties in DTOs not annotated with Jackson polymorphism
objectMapper.configure(DeserializationFeature.FAIL_ON_UNKNOWN_PROPERTIES, false);
// model names, id strings and file names
String namespace = "com.test";
String version = "1.0.0";
String idFormat = "%s.%s:%s";
String zoneModelName = "Zone";
String zoneModelID = String.format(idFormat, namespace, zoneModelName, version);
String zoneFileName = zoneModelName.concat(".type");
String lampModelName = "Lamp";
String lampModelID = String.format(idFormat, namespace, lampModelName, version);
String lampFileName = lampModelName.concat(".fbmodel");
String addressModelName = "Address";
String addressModelID = String.format(idFormat, namespace, addressModelName, version);
String addressFileName = addressModelName.concat(".fbmodel");
String streetLampModelName = "StreetLamp";
String streetLampModelID = String.format(idFormat, namespace, streetLampModelName, version);
String streetLampFileName = streetLampModelName.concat(".infomodel");
String streetLampAttachmentFileName = "StreetLampAttachment.json";
String streetLampLinkURL = "https://vorto.eclipse.org/";
String streetLampLinkName = "Vorto";
// creates the namespace as sysadmin
createNamespaceSuccessfully(namespace, userSysadmin);
// creates the collaborator payload to add userModelCreator to the namespace
Collaborator userModelCreatorCollaborator = new Collaborator();
userModelCreatorCollaborator.setAuthenticationProviderId(GITHUB);
userModelCreatorCollaborator.setRoles(Arrays.asList("model_viewer", "model_creator"));
userModelCreatorCollaborator.setTechnicalUser(false);
userModelCreatorCollaborator.setUserId(USER_MODEL_CREATOR_NAME);
// allows creator rights to userCreator on namespace
repositoryServer.perform(put(String.format("/rest/namespaces/%s/users", namespace)).with(userSysadmin).contentType(MediaType.APPLICATION_JSON).content(objectMapper.writeValueAsString(userModelCreatorCollaborator))).andExpect(status().isOk());
// creates the Zone model
createModel(userModelCreator, zoneFileName, zoneModelID);
// creates the Lamp model
createModel(userModelCreator, lampFileName, lampModelID);
// creates the Address model
createModel(userModelCreator, addressFileName, addressModelID);
// creates the StreetLamp model
createModel(userModelCreator, streetLampFileName, streetLampModelID);
// adds an attachment to the StreetLamp model (still requires sysadmin for this)
addAttachment(streetLampModelID, userSysadmin, streetLampAttachmentFileName, MediaType.APPLICATION_JSON).andExpect(status().isOk()).andExpect(content().json(objectMapper.writeValueAsString(AttachResult.success(ModelId.fromPrettyFormat(streetLampModelID), streetLampAttachmentFileName))));
// adds a link to the StreetLamp model
ModelLink link = new ModelLink(streetLampLinkURL, streetLampLinkName);
addLink(streetLampModelID, userModelCreator, link);
// saves a minimal mapping specification for the street lamp model
Infomodel streetLampInfomodel = new Infomodel(ModelId.fromPrettyFormat(streetLampModelID));
streetLampInfomodel.setTargetPlatformKey("myTargetPlatform");
MappingSpecification mapping = new MappingSpecification(streetLampInfomodel);
repositoryServer.perform(put(String.format("/rest/mappings/specifications/%s", streetLampModelID)).contentType(MediaType.APPLICATION_JSON).content(objectMapper.writeValueAsString(mapping)).with(userModelCreator)).andExpect(status().isOk());
// expected ID of the payload mapping model
String mappingId = String.format("%s.%s:%sPayloadMapping:%s", // root namespace
namespace, // virtual namespace for mapping
streetLampModelName.toLowerCase(), // mapping name part 1 matching root model
streetLampModelName, // root model version
version);
// fetches the full model for the UI
repositoryServer.perform(get(String.format("/rest/models/ui/%s", streetLampModelID)).with(userModelCreator)).andExpect(status().isOk()).andDo(mvcResult -> {
ModelFullDetailsDTO output = objectMapper.readValue(mvcResult.getResponse().getContentAsString(), ModelFullDetailsDTO.class);
LOGGER.info(new StringBuilder("\nReceived response body:\n\n").append(objectMapper.writerWithDefaultPrettyPrinter().writeValueAsString(output)).toString());
}).andExpect(jsonPath("$.modelInfo").exists()).andExpect(jsonPath("$.modelInfo.id.name").value(streetLampModelName)).andExpect(jsonPath("$.modelInfo.id.namespace").value(namespace)).andExpect(jsonPath("$.modelInfo.type").value("InformationModel")).andExpect(jsonPath("$.modelInfo.author").value(USER_MODEL_CREATOR_NAME)).andExpect(jsonPath("$.mappings").isNotEmpty()).andExpect(jsonPath("$.mappings[0].id").value(mappingId)).andExpect(jsonPath("$.references", hasSize(2))).andExpect(jsonPath("$.referencedBy", hasSize(1))).andExpect(jsonPath("$.referencedBy[0].id").value(mappingId)).andExpect(jsonPath("$.attachments").exists()).andExpect(jsonPath("$.attachments[0].filename").value(streetLampAttachmentFileName)).andExpect(jsonPath("$.attachments[0].modelId.name").value(streetLampModelName)).andExpect(jsonPath("$.links").exists()).andExpect(jsonPath("$.links[0].url").value(equalTo(link.getUrl()))).andExpect(jsonPath("$.links[0].displayText").value(equalTo(link.getDisplayText()))).andExpect(jsonPath("$.actions").exists()).andExpect(jsonPath("$.actions").isEmpty()).andExpect(jsonPath("$.policies", hasSize(2))).andExpect(jsonPath("$.bestPolicy").exists()).andExpect(jsonPath("$.bestPolicy.principalId").value("model_creator")).andExpect(jsonPath("$.bestPolicy.permission").value(Permission.FULL_ACCESS.toString())).andExpect(jsonPath("$.encodedModelSyntax").value(Base64.getEncoder().encodeToString(createContentAsString(streetLampFileName).getBytes())));
// cleanup: deletes models in reverse order of creation, then namespace
repositoryServer.perform(delete(String.format("/rest/models/%s", mappingId)).with(userModelCreator)).andExpect(status().isOk());
repositoryServer.perform(delete(String.format("/rest/models/%s", streetLampModelID)).with(userModelCreator)).andExpect(status().isOk());
repositoryServer.perform(delete(String.format("/rest/models/%s", lampModelID)).with(userModelCreator)).andExpect(status().isOk());
repositoryServer.perform(delete(String.format("/rest/models/%s", addressModelID)).with(userModelCreator)).andExpect(status().isOk());
repositoryServer.perform(delete(String.format("/rest/models/%s", zoneModelID)).with(userModelCreator)).andExpect(status().isOk());
repositoryServer.perform(delete(String.format("/rest/namespaces/%s", namespace)).with(userSysadmin)).andExpect(status().isNoContent());
// removes test-specific configuration
objectMapper.configure(DeserializationFeature.FAIL_ON_UNKNOWN_PROPERTIES, true);
}
use of org.eclipse.vorto.repository.web.api.v1.dto.ModelFullDetailsDTO in project vorto by eclipse.
the class ModelRepositoryController method getModelForUI.
/**
* Fetches all data required to populate the returned {@link ModelFullDetailsDTO} (see class docs
* for details), in addition the model's "file" contents as file added to the response.<br/>
* Following error cases apply:
* <ul>
* <li>
* If {@link ModelId#fromPrettyFormat(String)} fails throwing {@link IllegalArgumentException},
* returns {@code null} with status {@link HttpStatus#NOT_FOUND}.
* </li>
* <li>
* If {@link ModelRepositoryController#getWorkspaceId(String)} fails throwing
* {@link FatalModelRepositoryException}, returns {@code null} with status
* {@link HttpStatus#NOT_FOUND}.
* </li>
* <li>
* If any operation such as:
* <ul>
* <li>
* {@link IModelRepository#getByIdWithPlatformMappings(ModelId)}
* </li>
* <li>
* {@link IModelRepository#getAttachments(ModelId)}
* </li>
* <li>
* {@link IModelPolicyManager#getPolicyEntries(ModelId)}
* </li>
* </ul>
* ... fails throwing {@link NotAuthorizedException}, returns {@code null} with status
* {@link HttpStatus#FORBIDDEN};
* </li>
* </ul>
*
* @param modelId
* @return
*/
@GetMapping("/ui/{modelId:.+}")
public ResponseEntity<ModelFullDetailsDTO> getModelForUI(@PathVariable String modelId, final HttpServletResponse response) {
try {
// resolve user
Authentication user = SecurityContextHolder.getContext().getAuthentication();
// resolve model ID
ModelId modelID = ModelId.fromPrettyFormat(modelId);
// resolve ModeShape workspace ID
String workspaceId = getWorkspaceId(modelId);
// fetches model info
ModelInfo modelInfo = getModelRepository(modelID).getByIdWithPlatformMappings(modelID);
if (Objects.isNull(modelInfo)) {
LOGGER.warn(String.format("Model resource with id [%s] not found. ", modelId));
return new ResponseEntity<>(null, HttpStatus.NOT_FOUND);
}
// starts spawning threads to retrieve models etc.
final ExecutorService executor = Executors.newCachedThreadPool();
// fetches mappings
Collection<ModelMinimalInfoDTO> mappings = ConcurrentHashMap.newKeySet();
modelInfo.getPlatformMappings().entrySet().stream().forEach(e -> {
executor.submit(new AsyncModelMappingsFetcher(mappings, e).with(SecurityContextHolder.getContext()).with(RequestContextHolder.getRequestAttributes()).with(getModelRepositoryFactory()));
});
// fetches references from model ids built with the root ModelInfo
Collection<ModelMinimalInfoDTO> references = ConcurrentHashMap.newKeySet();
modelInfo.getReferences().stream().forEach(id -> executor.submit(new AsyncModelReferenceFetcher(references, id).with(SecurityContextHolder.getContext()).with(RequestContextHolder.getRequestAttributes()).with(getModelRepositoryFactory())));
// fetches referenced by
Collection<ModelMinimalInfoDTO> referencedBy = ConcurrentHashMap.newKeySet();
modelInfo.getReferencedBy().stream().forEach(id -> executor.submit(new AsyncModelReferenceFetcher(referencedBy, id).with(SecurityContextHolder.getContext()).with(RequestContextHolder.getRequestAttributes()).with(getModelRepositoryFactory())));
// fetches attachments
Collection<Attachment> attachments = ConcurrentHashMap.newKeySet();
executor.submit(new AsyncModelAttachmentsFetcher(attachments, modelID, userRepositoryRoleService.isSysadmin(user.getName())).with(SecurityContextHolder.getContext()).with(RequestContextHolder.getRequestAttributes()).with(getModelRepositoryFactory()));
// fetches links
Collection<ModelLink> links = ConcurrentHashMap.newKeySet();
executor.submit(new AsyncModelLinksFetcher(modelID, links).with(SecurityContextHolder.getContext()).with(RequestContextHolder.getRequestAttributes()).with(getModelRepositoryFactory()));
// fetches available workflow actions
Collection<String> actions = ConcurrentHashMap.newKeySet();
executor.submit(new AsyncWorkflowActionsFetcher(workflowService, actions, modelID, UserContext.user(user, workspaceId)).with(SecurityContextHolder.getContext()).with(RequestContextHolder.getRequestAttributes()));
// fetches model syntax
Future<String> encodedSyntaxFuture = executor.submit(new AsyncModelSyntaxFetcher(modelID, SecurityContextHolder.getContext(), RequestContextHolder.getRequestAttributes(), getModelRepositoryFactory()));
// shuts down executor and waits for completion of tasks until configured timeout
// also retrieves callable content
executor.shutdown();
// single-threaded calls
// fetches policies in this thread
Collection<PolicyEntry> policies = getPolicyManager(workspaceId).getPolicyEntries(modelID).stream().filter(p -> userHasPolicyEntry(p, user, workspaceId)).collect(Collectors.toList());
// getting callables and setting executor timeout
String encodedSyntax = null;
try {
// callable content
encodedSyntax = encodedSyntaxFuture.get();
// timeout
if (!executor.awaitTermination(requestTimeoutInSeconds, TimeUnit.SECONDS)) {
LOGGER.warn(String.format("Requesting UI data for model ID [%s] took over [%d] seconds and programmatically timed out.", modelID, requestTimeoutInSeconds));
return new ResponseEntity<>(null, HttpStatus.GATEWAY_TIMEOUT);
}
} catch (InterruptedException ie) {
LOGGER.error("Awaiting executor termination was interrupted.");
return new ResponseEntity<>(null, HttpStatus.SERVICE_UNAVAILABLE);
} catch (ExecutionException ee) {
LOGGER.error("Failed to retrieve and encode model syntax asynchronously");
return new ResponseEntity<>(null, HttpStatus.SERVICE_UNAVAILABLE);
}
// builds DTO
ModelFullDetailsDTO dto = new ModelFullDetailsDTO().withModelInfo(modelInfo).withMappings(mappings).withReferences(references).withReferencedBy(referencedBy).withAttachments(attachments).withLinks(links).withActions(actions).withEncodedModelSyntax(encodedSyntax).withPolicies(policies);
return new ResponseEntity<>(dto, HttpStatus.OK);
}// could not resolve "pretty format" for given model ID
catch (IllegalArgumentException iae) {
LOGGER.warn(String.format("Could not resolve given model ID [%s]", modelId), iae);
return new ResponseEntity<>(null, HttpStatus.NOT_FOUND);
}// could not find namespace to resolve workspace ID from
catch (FatalModelRepositoryException fmre) {
LOGGER.warn(String.format("Could not resolve workspace ID from namespace inferred by model ID [%s]", modelId), fmre);
return new ResponseEntity<>(null, HttpStatus.NOT_FOUND);
} catch (NotAuthorizedException nae) {
LOGGER.warn(String.format("Could not authorize fetching data from given model ID [%s] for calling user", modelId), nae);
return new ResponseEntity<>(null, HttpStatus.FORBIDDEN);
}
}
Aggregations