blob: aadf66becf437f2391ff7c45f3d9d73c7cd893bd [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.james.webadmin.data.jmap;
import static io.restassured.RestAssured.given;
import static io.restassured.RestAssured.when;
import static io.restassured.RestAssured.with;
import static org.assertj.core.api.Assertions.assertThat;
import static org.hamcrest.Matchers.is;
import static org.hamcrest.Matchers.notNullValue;
import java.util.Optional;
import org.apache.james.core.Username;
import org.apache.james.domainlist.api.DomainList;
import org.apache.james.jmap.api.model.Preview;
import org.apache.james.jmap.api.projections.MessageFastViewPrecomputedProperties;
import org.apache.james.jmap.utils.JsoupHtmlTextExtractor;
import org.apache.james.jmap.memory.projections.MemoryMessageFastViewProjection;
import org.apache.james.json.DTOConverter;
import org.apache.james.mailbox.MailboxSession;
import org.apache.james.mailbox.MessageManager;
import org.apache.james.mailbox.inmemory.InMemoryMailboxManager;
import org.apache.james.mailbox.inmemory.manager.InMemoryIntegrationResources;
import org.apache.james.mailbox.model.ComposedMessageId;
import org.apache.james.mailbox.model.MailboxId;
import org.apache.james.mailbox.model.MailboxPath;
import org.apache.james.metrics.tests.RecordingMetricFactory;
import org.apache.james.task.Hostname;
import org.apache.james.task.MemoryTaskManager;
import org.apache.james.task.TaskManager;
import org.apache.james.user.memory.MemoryUsersRepository;
import org.apache.james.util.html.HtmlTextExtractor;
import org.apache.james.util.mime.MessageContentExtractor;
import org.apache.james.webadmin.Routes;
import org.apache.james.webadmin.WebAdminServer;
import org.apache.james.webadmin.WebAdminUtils;
import org.apache.james.webadmin.routes.TasksRoutes;
import org.apache.james.webadmin.tasks.TaskFromRequestRegistry;
import org.apache.james.webadmin.utils.ErrorResponder;
import org.apache.james.webadmin.utils.JsonTransformer;
import org.eclipse.jetty.http.HttpStatus;
import org.junit.jupiter.api.AfterEach;
import org.junit.jupiter.api.BeforeEach;
import org.junit.jupiter.api.Test;
import io.restassured.RestAssured;
import spark.Service;
class RecomputeAllFastViewProjectionItemsRequestToTaskTest {
private final class JMAPRoutes implements Routes {
private final MessageFastViewProjectionCorrector corrector;
private final TaskManager taskManager;
private JMAPRoutes(MessageFastViewProjectionCorrector corrector, TaskManager taskManager) {
this.corrector = corrector;
this.taskManager = taskManager;
}
@Override
public String getBasePath() {
return BASE_PATH;
}
@Override
public void define(Service service) {
service.post(BASE_PATH,
TaskFromRequestRegistry.builder()
.registrations(new RecomputeAllFastViewProjectionItemsRequestToTask(corrector))
.buildAsRoute(taskManager),
new JsonTransformer());
}
}
static final String BASE_PATH = "/mailboxes";
static final MessageFastViewPrecomputedProperties PROJECTION_ITEM = MessageFastViewPrecomputedProperties.builder()
.preview(Preview.from("body"))
.hasAttachment(false)
.build();
static final DomainList NO_DOMAIN_LIST = null;
static final Username BOB = Username.of("bob");
private WebAdminServer webAdminServer;
private MemoryTaskManager taskManager;
private MemoryMessageFastViewProjection messageFastViewProjection;
private InMemoryMailboxManager mailboxManager;
private MemoryUsersRepository usersRepository;
@BeforeEach
void setUp() {
JsonTransformer jsonTransformer = new JsonTransformer();
taskManager = new MemoryTaskManager(new Hostname("foo"));
messageFastViewProjection = new MemoryMessageFastViewProjection(new RecordingMetricFactory());
mailboxManager = InMemoryIntegrationResources.defaultResources().getMailboxManager();
usersRepository = MemoryUsersRepository.withoutVirtualHosting(NO_DOMAIN_LIST);
MessageContentExtractor messageContentExtractor = new MessageContentExtractor();
HtmlTextExtractor htmlTextExtractor = new JsoupHtmlTextExtractor();
Preview.Factory previewFactory = new Preview.Factory(messageContentExtractor, htmlTextExtractor);
MessageFastViewPrecomputedProperties.Factory projectionItemFactory = new MessageFastViewPrecomputedProperties.Factory(previewFactory);
webAdminServer = WebAdminUtils.createWebAdminServer(
new TasksRoutes(taskManager, jsonTransformer,
DTOConverter.of(RecomputeAllFastViewTaskAdditionalInformationDTO.module())),
new JMAPRoutes(
new MessageFastViewProjectionCorrector(usersRepository, mailboxManager, messageFastViewProjection, projectionItemFactory),
taskManager))
.start();
RestAssured.requestSpecification = WebAdminUtils.buildRequestSpecification(webAdminServer)
.setBasePath(BASE_PATH)
.build();
}
@AfterEach
void afterEach() {
webAdminServer.destroy();
taskManager.stop();
}
@Test
void actionRequestParameterShouldBeCompulsory() {
when()
.post()
.then()
.statusCode(HttpStatus.BAD_REQUEST_400)
.body("statusCode", is(400))
.body("type", is(ErrorResponder.ErrorType.INVALID_ARGUMENT.getType()))
.body("message", is("Invalid arguments supplied in the user request"))
.body("details", is("'action' query parameter is compulsory. Supported values are [recomputeFastViewProjectionItems]"));
}
@Test
void postShouldFailUponEmptyAction() {
given()
.queryParam("action", "")
.post()
.then()
.statusCode(HttpStatus.BAD_REQUEST_400)
.body("statusCode", is(400))
.body("type", is(ErrorResponder.ErrorType.INVALID_ARGUMENT.getType()))
.body("message", is("Invalid arguments supplied in the user request"))
.body("details", is("'action' query parameter cannot be empty or blank. Supported values are [recomputeFastViewProjectionItems]"));
}
@Test
void postShouldFailUponInvalidAction() {
given()
.queryParam("action", "invalid")
.post()
.then()
.statusCode(HttpStatus.BAD_REQUEST_400)
.body("statusCode", is(400))
.body("type", is(ErrorResponder.ErrorType.INVALID_ARGUMENT.getType()))
.body("message", is("Invalid arguments supplied in the user request"))
.body("details", is("Invalid value supplied for query parameter 'action': invalid. Supported values are [recomputeFastViewProjectionItems]"));
}
@Test
void postShouldFailWhenMessagesPerSecondIsNotAnInt() {
given()
.queryParam("action", "recomputeFastViewProjectionItems")
.queryParam("messagesPerSecond", "abc")
.post()
.then()
.statusCode(HttpStatus.BAD_REQUEST_400)
.body("statusCode", is(400))
.body("type", is(ErrorResponder.ErrorType.INVALID_ARGUMENT.getType()))
.body("message", is("Invalid arguments supplied in the user request"))
.body("details", is("Illegal value supplied for query parameter 'messagesPerSecond', expecting a strictly positive optional integer"));
}
@Test
void postShouldFailWhenMessagesPerSecondIsNegative() {
given()
.queryParam("action", "recomputeFastViewProjectionItems")
.queryParam("messagesPerSecond", "-1")
.post()
.then()
.statusCode(HttpStatus.BAD_REQUEST_400)
.body("statusCode", is(400))
.body("type", is(ErrorResponder.ErrorType.INVALID_ARGUMENT.getType()))
.body("message", is("Invalid arguments supplied in the user request"))
.body("details", is("'messagesPerSecond' must be strictly positive"));
}
@Test
void postShouldFailWhenMessagesPerSecondIsZero() {
given()
.queryParam("action", "recomputeFastViewProjectionItems")
.queryParam("messagesPerSecond", "0")
.post()
.then()
.statusCode(HttpStatus.BAD_REQUEST_400)
.body("statusCode", is(400))
.body("type", is(ErrorResponder.ErrorType.INVALID_ARGUMENT.getType()))
.body("message", is("Invalid arguments supplied in the user request"))
.body("details", is("'messagesPerSecond' must be strictly positive"));
}
@Test
void postShouldCreateANewTask() {
given()
.queryParam("action", "recomputeFastViewProjectionItems")
.post()
.then()
.statusCode(HttpStatus.CREATED_201)
.body("taskId", notNullValue());
}
@Test
void postShouldCreateANewTaskWhenConcurrencyParametersSpecified() {
given()
.queryParam("messagesPerSecond", "1")
.queryParam("action", "recomputeFastViewProjectionItems")
.post()
.then()
.statusCode(HttpStatus.CREATED_201)
.body("taskId", notNullValue());
}
@Test
void recomputeAllShouldCompleteWhenNoUser() {
String taskId = with()
.queryParam("action", "recomputeFastViewProjectionItems")
.post()
.jsonPath()
.get("taskId");
given()
.basePath(TasksRoutes.BASE)
.when()
.get(taskId + "/await")
.then()
.body("status", is("completed"))
.body("taskId", is(taskId))
.body("type", is("RecomputeAllFastViewProjectionItemsTask"))
.body("additionalInformation.processedUserCount", is(0))
.body("additionalInformation.processedMessageCount", is(0))
.body("additionalInformation.failedUserCount", is(0))
.body("additionalInformation.failedMessageCount", is(0))
.body("startedDate", is(notNullValue()))
.body("submitDate", is(notNullValue()))
.body("completedDate", is(notNullValue()));
}
@Test
void runningOptionsShouldBePartOfTaskDetails() {
String taskId = with()
.queryParam("action", "recomputeFastViewProjectionItems")
.queryParam("messagesPerSecond", "20")
.post()
.jsonPath()
.get("taskId");
given()
.basePath(TasksRoutes.BASE)
.when()
.get(taskId + "/await")
.then()
.body("taskId", is(taskId))
.body("type", is("RecomputeAllFastViewProjectionItemsTask"))
.body("additionalInformation.runningOptions.messagesPerSecond", is(20));
}
@Test
void recomputeAllShouldCompleteWhenUserWithNoMailbox() throws Exception {
usersRepository.addUser(BOB, "pass");
String taskId = with()
.queryParam("action", "recomputeFastViewProjectionItems")
.post()
.jsonPath()
.get("taskId");
given()
.basePath(TasksRoutes.BASE)
.when()
.get(taskId + "/await")
.then()
.body("status", is("completed"))
.body("taskId", is(taskId))
.body("type", is("RecomputeAllFastViewProjectionItemsTask"))
.body("additionalInformation.processedUserCount", is(1))
.body("additionalInformation.processedMessageCount", is(0))
.body("additionalInformation.failedUserCount", is(0))
.body("additionalInformation.failedMessageCount", is(0))
.body("startedDate", is(notNullValue()))
.body("submitDate", is(notNullValue()))
.body("completedDate", is(notNullValue()));
}
@Test
void recomputeAllShouldCompleteWhenUserWithNoMessage() throws Exception {
usersRepository.addUser(BOB, "pass");
mailboxManager.createMailbox(MailboxPath.inbox(BOB), mailboxManager.createSystemSession(BOB));
String taskId = with()
.queryParam("action", "recomputeFastViewProjectionItems")
.post()
.jsonPath()
.get("taskId");
given()
.basePath(TasksRoutes.BASE)
.when()
.get(taskId + "/await")
.then()
.body("status", is("completed"))
.body("taskId", is(taskId))
.body("type", is("RecomputeAllFastViewProjectionItemsTask"))
.body("additionalInformation.processedUserCount", is(1))
.body("additionalInformation.processedMessageCount", is(0))
.body("additionalInformation.failedUserCount", is(0))
.body("additionalInformation.failedMessageCount", is(0))
.body("startedDate", is(notNullValue()))
.body("submitDate", is(notNullValue()))
.body("completedDate", is(notNullValue()));
}
@Test
void recomputeAllShouldCompleteWhenOneMessage() throws Exception {
usersRepository.addUser(BOB, "pass");
MailboxSession session = mailboxManager.createSystemSession(BOB);
Optional<MailboxId> mailboxId = mailboxManager.createMailbox(MailboxPath.inbox(BOB), session);
mailboxManager.getMailbox(mailboxId.get(), session).appendMessage(
MessageManager.AppendCommand.builder().build("header: value\r\n\r\nbody"),
session);
String taskId = with()
.queryParam("action", "recomputeFastViewProjectionItems")
.post()
.jsonPath()
.get("taskId");
given()
.basePath(TasksRoutes.BASE)
.when()
.get(taskId + "/await")
.then()
.body("status", is("completed"))
.body("taskId", is(taskId))
.body("type", is("RecomputeAllFastViewProjectionItemsTask"))
.body("additionalInformation.processedUserCount", is(1))
.body("additionalInformation.processedMessageCount", is(1))
.body("additionalInformation.failedUserCount", is(0))
.body("additionalInformation.failedMessageCount", is(0))
.body("startedDate", is(notNullValue()))
.body("submitDate", is(notNullValue()))
.body("completedDate", is(notNullValue()));
}
@Test
void recomputeAllShouldUpdateProjection() throws Exception {
usersRepository.addUser(BOB, "pass");
MailboxSession session = mailboxManager.createSystemSession(BOB);
Optional<MailboxId> mailboxId = mailboxManager.createMailbox(MailboxPath.inbox(BOB), session);
ComposedMessageId messageId = mailboxManager.getMailbox(mailboxId.get(), session).appendMessage(
MessageManager.AppendCommand.builder().build("header: value\r\n\r\nbody"),
session).getId();
String taskId = with()
.queryParam("action", "recomputeFastViewProjectionItems")
.post()
.jsonPath()
.get("taskId");
with()
.basePath(TasksRoutes.BASE)
.get(taskId + "/await");
assertThat(messageFastViewProjection.retrieve(messageId.getMessageId()).block())
.isEqualTo(PROJECTION_ITEM);
}
@Test
void recomputeAllShouldBeIdempotent() throws Exception {
usersRepository.addUser(BOB, "pass");
MailboxSession session = mailboxManager.createSystemSession(BOB);
Optional<MailboxId> mailboxId = mailboxManager.createMailbox(MailboxPath.inbox(BOB), session);
ComposedMessageId messageId = mailboxManager.getMailbox(mailboxId.get(), session).appendMessage(
MessageManager.AppendCommand.builder().build("header: value\r\n\r\nbody"),
session).getId();
String taskId1 = with()
.queryParam("action", "recomputeFastViewProjectionItems")
.post()
.jsonPath()
.get("taskId");
with()
.basePath(TasksRoutes.BASE)
.get(taskId1 + "/await");
String taskId2 = with()
.queryParam("action", "recomputeFastViewProjectionItems")
.post()
.jsonPath()
.get("taskId");
with()
.basePath(TasksRoutes.BASE)
.get(taskId2 + "/await");
assertThat(messageFastViewProjection.retrieve(messageId.getMessageId()).block())
.isEqualTo(PROJECTION_ITEM);
}
}