diff --git a/backends-common/cassandra/src/main/java/org/apache/james/backends/cassandra/components/CassandraQuotaCurrentValueDao.java b/backends-common/cassandra/src/main/java/org/apache/james/backends/cassandra/components/CassandraQuotaCurrentValueDao.java index 95618be4f25..aec997b27b4 100644 --- a/backends-common/cassandra/src/main/java/org/apache/james/backends/cassandra/components/CassandraQuotaCurrentValueDao.java +++ b/backends-common/cassandra/src/main/java/org/apache/james/backends/cassandra/components/CassandraQuotaCurrentValueDao.java @@ -30,8 +30,6 @@ import static org.apache.james.backends.cassandra.components.CassandraQuotaCurrentValueTable.QUOTA_TYPE; import static org.apache.james.backends.cassandra.components.CassandraQuotaCurrentValueTable.TABLE_NAME; -import java.util.Objects; - import jakarta.inject.Inject; import org.apache.james.backends.cassandra.utils.CassandraAsyncExecutor; @@ -47,66 +45,12 @@ import com.datastax.oss.driver.api.querybuilder.delete.Delete; import com.datastax.oss.driver.api.querybuilder.select.Select; import com.datastax.oss.driver.api.querybuilder.update.Update; -import com.google.common.base.MoreObjects; import reactor.core.publisher.Flux; import reactor.core.publisher.Mono; public class CassandraQuotaCurrentValueDao { - public static class QuotaKey { - - public static QuotaKey of(QuotaComponent component, String identifier, QuotaType quotaType) { - return new QuotaKey(component, identifier, quotaType); - } - - private final QuotaComponent quotaComponent; - private final String identifier; - private final QuotaType quotaType; - - public QuotaComponent getQuotaComponent() { - return quotaComponent; - } - - public String getIdentifier() { - return identifier; - } - - public QuotaType getQuotaType() { - return quotaType; - } - - private QuotaKey(QuotaComponent quotaComponent, String identifier, QuotaType quotaType) { - this.quotaComponent = quotaComponent; - this.identifier = identifier; - this.quotaType = quotaType; - } - - @Override - public final int hashCode() { - return Objects.hash(quotaComponent, identifier, quotaType); - } - - @Override - public final boolean equals(Object o) { - if (o instanceof QuotaKey) { - QuotaKey other = (QuotaKey) o; - return Objects.equals(quotaComponent, other.quotaComponent) - && Objects.equals(identifier, other.identifier) - && Objects.equals(quotaType, other.quotaType); - } - return false; - } - - public String toString() { - return MoreObjects.toStringHelper(this) - .add("quotaComponent", quotaComponent) - .add("identifier", identifier) - .add("quotaType", quotaType) - .toString(); - } - } - private static final Logger LOGGER = LoggerFactory.getLogger(CassandraQuotaCurrentValueDao.class); private final CassandraAsyncExecutor queryExecutor; @@ -126,7 +70,7 @@ public CassandraQuotaCurrentValueDao(CqlSession session) { this.deleteQuotaCurrentValueStatement = session.prepare(deleteQuotaCurrentValueStatement().build()); } - public Mono increase(QuotaKey quotaKey, long amount) { + public Mono increase(QuotaCurrentValue.Key quotaKey, long amount) { return queryExecutor.executeVoid(increaseStatement.bind() .setString(QUOTA_COMPONENT, quotaKey.getQuotaComponent().getValue()) .setString(IDENTIFIER, quotaKey.getIdentifier()) @@ -139,7 +83,7 @@ public Mono increase(QuotaKey quotaKey, long amount) { }); } - public Mono decrease(QuotaKey quotaKey, long amount) { + public Mono decrease(QuotaCurrentValue.Key quotaKey, long amount) { return queryExecutor.executeVoid(decreaseStatement.bind() .setString(QUOTA_COMPONENT, quotaKey.getQuotaComponent().getValue()) .setString(IDENTIFIER, quotaKey.getIdentifier()) @@ -152,7 +96,7 @@ public Mono decrease(QuotaKey quotaKey, long amount) { }); } - public Mono getQuotaCurrentValue(QuotaKey quotaKey) { + public Mono getQuotaCurrentValue(QuotaCurrentValue.Key quotaKey) { return queryExecutor.executeSingleRow(getQuotaCurrentValueStatement.bind() .setString(QUOTA_COMPONENT, quotaKey.getQuotaComponent().getValue()) .setString(IDENTIFIER, quotaKey.getIdentifier()) @@ -160,7 +104,7 @@ public Mono getQuotaCurrentValue(QuotaKey quotaKey) { .map(row -> convertRowToModel(row)); } - public Mono deleteQuotaCurrentValue(QuotaKey quotaKey) { + public Mono deleteQuotaCurrentValue(QuotaCurrentValue.Key quotaKey) { return queryExecutor.executeVoid(deleteQuotaCurrentValueStatement.bind() .setString(QUOTA_COMPONENT, quotaKey.getQuotaComponent().getValue()) .setString(IDENTIFIER, quotaKey.getIdentifier()) diff --git a/backends-common/cassandra/src/main/java/org/apache/james/backends/cassandra/components/CassandraQuotaLimitDao.java b/backends-common/cassandra/src/main/java/org/apache/james/backends/cassandra/components/CassandraQuotaLimitDao.java index c43442ac5ff..2b3090a6403 100644 --- a/backends-common/cassandra/src/main/java/org/apache/james/backends/cassandra/components/CassandraQuotaLimitDao.java +++ b/backends-common/cassandra/src/main/java/org/apache/james/backends/cassandra/components/CassandraQuotaLimitDao.java @@ -31,8 +31,6 @@ import static org.apache.james.backends.cassandra.components.CassandraQuotaLimitTable.QUOTA_TYPE; import static org.apache.james.backends.cassandra.components.CassandraQuotaLimitTable.TABLE_NAME; -import java.util.Objects; - import jakarta.inject.Inject; import org.apache.james.backends.cassandra.utils.CassandraAsyncExecutor; @@ -47,74 +45,11 @@ import com.datastax.oss.driver.api.querybuilder.delete.Delete; import com.datastax.oss.driver.api.querybuilder.insert.Insert; import com.datastax.oss.driver.api.querybuilder.select.Select; -import com.google.common.base.MoreObjects; import reactor.core.publisher.Flux; import reactor.core.publisher.Mono; public class CassandraQuotaLimitDao { - - public static class QuotaLimitKey { - - public static QuotaLimitKey of(QuotaComponent component, QuotaScope scope, String identifier, QuotaType quotaType) { - return new QuotaLimitKey(component, scope, identifier, quotaType); - } - - private final QuotaComponent quotaComponent; - private final QuotaScope quotaScope; - private final String identifier; - private final QuotaType quotaType; - - public QuotaComponent getQuotaComponent() { - return quotaComponent; - } - - public QuotaScope getQuotaScope() { - return quotaScope; - } - - public String getIdentifier() { - return identifier; - } - - public QuotaType getQuotaType() { - return quotaType; - } - - private QuotaLimitKey(QuotaComponent quotaComponent, QuotaScope quotaScope, String identifier, QuotaType quotaType) { - this.quotaComponent = quotaComponent; - this.quotaScope = quotaScope; - this.identifier = identifier; - this.quotaType = quotaType; - } - - @Override - public final int hashCode() { - return Objects.hash(quotaComponent, quotaScope, identifier, quotaType); - } - - @Override - public final boolean equals(Object o) { - if (o instanceof QuotaLimitKey) { - QuotaLimitKey other = (QuotaLimitKey) o; - return Objects.equals(quotaComponent, other.quotaComponent) - && Objects.equals(quotaScope, other.quotaScope) - && Objects.equals(identifier, other.identifier) - && Objects.equals(quotaType, other.quotaType); - } - return false; - } - - public String toString() { - return MoreObjects.toStringHelper(this) - .add("quotaComponent", quotaComponent) - .add("quotaScope", quotaScope) - .add("identifier", identifier) - .add("quotaType", quotaType) - .toString(); - } - } - private final CassandraAsyncExecutor queryExecutor; private final PreparedStatement getQuotaLimitStatement; private final PreparedStatement getQuotaLimitsStatement; @@ -130,7 +65,7 @@ public CassandraQuotaLimitDao(CqlSession session) { this.deleteQuotaLimitStatement = session.prepare((deleteQuotaLimitStatement().build())); } - public Mono getQuotaLimit(QuotaLimitKey quotaKey) { + public Mono getQuotaLimit(QuotaLimit.QuotaLimitKey quotaKey) { return queryExecutor.executeSingleRow(getQuotaLimitStatement.bind() .setString(QUOTA_COMPONENT, quotaKey.getQuotaComponent().getValue()) .setString(QUOTA_SCOPE, quotaKey.getQuotaScope().getValue()) @@ -156,7 +91,7 @@ public Mono setQuotaLimit(QuotaLimit quotaLimit) { .set(QUOTA_LIMIT, quotaLimit.getQuotaLimit().orElse(null), Long.class)); } - public Mono deleteQuotaLimit(QuotaLimitKey quotaKey) { + public Mono deleteQuotaLimit(QuotaLimit.QuotaLimitKey quotaKey) { return queryExecutor.executeVoid(deleteQuotaLimitStatement.bind() .setString(QUOTA_COMPONENT, quotaKey.getQuotaComponent().getValue()) .setString(QUOTA_SCOPE, quotaKey.getQuotaScope().getValue()) @@ -203,7 +138,8 @@ private QuotaLimit convertRowToModel(Row row) { .quotaScope(QuotaScope.of(row.get(QUOTA_SCOPE, String.class))) .identifier(row.get(IDENTIFIER, String.class)) .quotaType(QuotaType.of(row.get(QUOTA_TYPE, String.class))) - .quotaLimit(row.get(QUOTA_LIMIT, Long.class)).build(); + .quotaLimit(row.get(QUOTA_LIMIT, Long.class)) + .build(); } } \ No newline at end of file diff --git a/backends-common/cassandra/src/test/java/org/apache/james/backends/cassandra/quota/CassandraQuotaCurrentValueDaoTest.java b/backends-common/cassandra/src/test/java/org/apache/james/backends/cassandra/quota/CassandraQuotaCurrentValueDaoTest.java index e22b6d4c923..ae7817e42b0 100644 --- a/backends-common/cassandra/src/test/java/org/apache/james/backends/cassandra/quota/CassandraQuotaCurrentValueDaoTest.java +++ b/backends-common/cassandra/src/test/java/org/apache/james/backends/cassandra/quota/CassandraQuotaCurrentValueDaoTest.java @@ -26,7 +26,6 @@ import org.apache.james.backends.cassandra.CassandraClusterExtension; import org.apache.james.backends.cassandra.components.CassandraMutualizedQuotaModule; import org.apache.james.backends.cassandra.components.CassandraQuotaCurrentValueDao; -import org.apache.james.backends.cassandra.components.CassandraQuotaCurrentValueDao.QuotaKey; import org.apache.james.core.quota.QuotaComponent; import org.apache.james.core.quota.QuotaCurrentValue; import org.apache.james.core.quota.QuotaType; @@ -36,7 +35,7 @@ import org.junit.jupiter.api.extension.RegisterExtension; public class CassandraQuotaCurrentValueDaoTest { - private static final QuotaKey QUOTA_KEY = QuotaKey.of(QuotaComponent.MAILBOX, "james@abc.com", QuotaType.SIZE); + private static final QuotaCurrentValue.Key QUOTA_KEY = QuotaCurrentValue.Key.of(QuotaComponent.MAILBOX, "james@abc.com", QuotaType.SIZE); private CassandraQuotaCurrentValueDao cassandraQuotaCurrentValueDao; @@ -92,7 +91,7 @@ void decreaseQuotaCurrentValueShouldDecreaseValueSuccessfully() { @Test void deleteQuotaCurrentValueShouldDeleteSuccessfully() { - QuotaKey quotaKey = QuotaKey.of(QuotaComponent.MAILBOX, "andre@abc.com", QuotaType.SIZE); + QuotaCurrentValue.Key quotaKey = QuotaCurrentValue.Key.of(QuotaComponent.MAILBOX, "andre@abc.com", QuotaType.SIZE); cassandraQuotaCurrentValueDao.increase(quotaKey, 100L).block(); cassandraQuotaCurrentValueDao.deleteQuotaCurrentValue(quotaKey).block(); @@ -125,7 +124,7 @@ void decreaseQuotaCurrentValueShouldNotThrowExceptionWhenQueryExecutorThrowExcep @Test void getQuotasByComponentShouldGetAllQuotaTypesSuccessfully() { - QuotaKey countQuotaKey = QuotaKey.of(QuotaComponent.MAILBOX, "james@abc.com", QuotaType.COUNT); + QuotaCurrentValue.Key countQuotaKey = QuotaCurrentValue.Key.of(QuotaComponent.MAILBOX, "james@abc.com", QuotaType.COUNT); QuotaCurrentValue expectedQuotaSize = QuotaCurrentValue.builder().quotaComponent(QUOTA_KEY.getQuotaComponent()) .identifier(QUOTA_KEY.getIdentifier()).quotaType(QUOTA_KEY.getQuotaType()).currentValue(100L).build(); diff --git a/backends-common/cassandra/src/test/java/org/apache/james/backends/cassandra/quota/CassandraQuotaLimitDaoTest.java b/backends-common/cassandra/src/test/java/org/apache/james/backends/cassandra/quota/CassandraQuotaLimitDaoTest.java index 2c421471756..7fa6f47a462 100644 --- a/backends-common/cassandra/src/test/java/org/apache/james/backends/cassandra/quota/CassandraQuotaLimitDaoTest.java +++ b/backends-common/cassandra/src/test/java/org/apache/james/backends/cassandra/quota/CassandraQuotaLimitDaoTest.java @@ -61,7 +61,7 @@ void setQuotaLimitShouldSaveObjectSuccessfully() { QuotaLimit expected = QuotaLimit.builder().quotaComponent(QuotaComponent.MAILBOX).quotaScope(QuotaScope.DOMAIN).identifier("A").quotaType(QuotaType.COUNT).quotaLimit(100L).build(); cassandraQuotaLimitDao.setQuotaLimit(expected).block(); - assertThat(cassandraQuotaLimitDao.getQuotaLimit(CassandraQuotaLimitDao.QuotaLimitKey.of(QuotaComponent.MAILBOX, QuotaScope.DOMAIN, "A", QuotaType.COUNT)).block()) + assertThat(cassandraQuotaLimitDao.getQuotaLimit(QuotaLimit.QuotaLimitKey.of(QuotaComponent.MAILBOX, QuotaScope.DOMAIN, "A", QuotaType.COUNT)).block()) .isEqualTo(expected); } @@ -70,7 +70,7 @@ void setQuotaLimitWithEmptyQuotaLimitValueShouldNotThrowNullPointerException() { QuotaLimit emptyQuotaLimitValue = QuotaLimit.builder().quotaComponent(QuotaComponent.MAILBOX).quotaScope(QuotaScope.DOMAIN).identifier("A").quotaType(QuotaType.COUNT).build(); cassandraQuotaLimitDao.setQuotaLimit(emptyQuotaLimitValue).block(); - assertThat(cassandraQuotaLimitDao.getQuotaLimit(CassandraQuotaLimitDao.QuotaLimitKey.of(QuotaComponent.MAILBOX, QuotaScope.DOMAIN, "A", QuotaType.COUNT)).block()) + assertThat(cassandraQuotaLimitDao.getQuotaLimit(QuotaLimit.QuotaLimitKey.of(QuotaComponent.MAILBOX, QuotaScope.DOMAIN, "A", QuotaType.COUNT)).block()) .isEqualTo(emptyQuotaLimitValue); } @@ -79,7 +79,7 @@ void setQuotaLimitShouldSaveObjectSuccessfullyWhenLimitIsMinusOne() { QuotaLimit expected = QuotaLimit.builder().quotaComponent(QuotaComponent.MAILBOX).quotaScope(QuotaScope.DOMAIN).identifier("A").quotaType(QuotaType.COUNT).quotaLimit(-1L).build(); cassandraQuotaLimitDao.setQuotaLimit(expected).block(); - assertThat(cassandraQuotaLimitDao.getQuotaLimit(CassandraQuotaLimitDao.QuotaLimitKey.of(QuotaComponent.MAILBOX, QuotaScope.DOMAIN, "A", QuotaType.COUNT)).block()) + assertThat(cassandraQuotaLimitDao.getQuotaLimit(QuotaLimit.QuotaLimitKey.of(QuotaComponent.MAILBOX, QuotaScope.DOMAIN, "A", QuotaType.COUNT)).block()) .isEqualTo(expected); } @@ -87,9 +87,9 @@ void setQuotaLimitShouldSaveObjectSuccessfullyWhenLimitIsMinusOne() { void deleteQuotaLimitShouldDeleteObjectSuccessfully() { QuotaLimit quotaLimit = QuotaLimit.builder().quotaComponent(QuotaComponent.MAILBOX).quotaScope(QuotaScope.DOMAIN).identifier("A").quotaType(QuotaType.COUNT).quotaLimit(100L).build(); cassandraQuotaLimitDao.setQuotaLimit(quotaLimit).block(); - cassandraQuotaLimitDao.deleteQuotaLimit(CassandraQuotaLimitDao.QuotaLimitKey.of(QuotaComponent.MAILBOX, QuotaScope.DOMAIN, "A", QuotaType.COUNT)).block(); + cassandraQuotaLimitDao.deleteQuotaLimit(QuotaLimit.QuotaLimitKey.of(QuotaComponent.MAILBOX, QuotaScope.DOMAIN, "A", QuotaType.COUNT)).block(); - assertThat(cassandraQuotaLimitDao.getQuotaLimit(CassandraQuotaLimitDao.QuotaLimitKey.of(QuotaComponent.MAILBOX, QuotaScope.DOMAIN, "A", QuotaType.COUNT)).block()) + assertThat(cassandraQuotaLimitDao.getQuotaLimit(QuotaLimit.QuotaLimitKey.of(QuotaComponent.MAILBOX, QuotaScope.DOMAIN, "A", QuotaType.COUNT)).block()) .isNull(); } diff --git a/backends-common/pom.xml b/backends-common/pom.xml index a0ae2dc5827..0f46ba17d7a 100644 --- a/backends-common/pom.xml +++ b/backends-common/pom.xml @@ -37,6 +37,7 @@ cassandra jpa opensearch + postgres pulsar rabbitmq redis diff --git a/backends-common/postgres/pom.xml b/backends-common/postgres/pom.xml new file mode 100644 index 00000000000..d3a80cf255a --- /dev/null +++ b/backends-common/postgres/pom.xml @@ -0,0 +1,112 @@ + + + + 4.0.0 + + org.apache.james + james-backends-common + 3.9.0-SNAPSHOT + + + apache-james-backends-postgres + Apache James :: Backends Common :: Postgres + + + 3.19.15 + 1.0.7.RELEASE + + + + + ${james.groupId} + james-core + + + ${james.groupId} + james-server-guice-common + test-jar + test + + + ${james.groupId} + james-server-lifecycle-api + + + ${james.groupId} + james-server-util + + + ${james.groupId} + metrics-api + + + ${james.groupId} + testing-base + test + + + io.r2dbc + r2dbc-pool + 1.0.1.RELEASE + + + jakarta.annotation + jakarta.annotation-api + + + jakarta.inject + jakarta.inject-api + + + org.apache.commons + commons-configuration2 + + + org.jooq + jooq + ${jooq.version} + + + org.jooq + jooq-postgres-extensions + ${jooq.version} + + + org.postgresql + r2dbc-postgresql + ${r2dbc.postgresql.version} + + + org.testcontainers + junit-jupiter + test + + + org.testcontainers + postgresql + test + + + org.testcontainers + testcontainers + test + + + diff --git a/backends-common/postgres/src/main/java/org/apache/james/backends/postgres/PostgresCommons.java b/backends-common/postgres/src/main/java/org/apache/james/backends/postgres/PostgresCommons.java new file mode 100644 index 00000000000..88201ac066c --- /dev/null +++ b/backends-common/postgres/src/main/java/org/apache/james/backends/postgres/PostgresCommons.java @@ -0,0 +1,96 @@ +/**************************************************************** + * 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.backends.postgres; + +import java.time.Instant; +import java.time.LocalDateTime; +import java.time.OffsetDateTime; +import java.time.ZoneId; +import java.time.ZoneOffset; +import java.time.ZonedDateTime; +import java.util.Date; +import java.util.Optional; +import java.util.function.Function; + +import org.jooq.DataType; +import org.jooq.Field; +import org.jooq.Record; +import org.jooq.Table; +import org.jooq.impl.DSL; +import org.jooq.impl.DefaultDataType; +import org.jooq.impl.SQLDataType; +import org.jooq.postgres.extensions.bindings.HstoreBinding; +import org.jooq.postgres.extensions.types.Hstore; + +public class PostgresCommons { + + public interface DataTypes { + + // hstore + DataType HSTORE = DefaultDataType.getDefaultDataType("hstore").asConvertedDataType(new HstoreBinding()); + + // timestamp(6) + DataType TIMESTAMP = SQLDataType.LOCALDATETIME(6); + + DataType TIMESTAMP_WITH_TIMEZONE = SQLDataType.TIMESTAMPWITHTIMEZONE(6); + + // text[] + DataType STRING_ARRAY = SQLDataType.VARCHAR.getArrayDataType(); + } + + + public static Field tableField(Table table, Field field) { + return DSL.field(table.getName() + "." + field.getName(), field.getDataType()); + } + + public static final Function DATE_TO_LOCAL_DATE_TIME = date -> Optional.ofNullable(date) + .map(value -> LocalDateTime.ofInstant(value.toInstant(), ZoneOffset.UTC)) + .orElse(null); + + public static final Function ZONED_DATE_TIME_TO_LOCAL_DATE_TIME = date -> Optional.ofNullable(date) + .map(value -> value.withZoneSameInstant(ZoneOffset.UTC).toLocalDateTime()) + .orElse(null); + + public static final Function INSTANT_TO_LOCAL_DATE_TIME = instant -> Optional.ofNullable(instant) + .map(value -> LocalDateTime.ofInstant(instant, ZoneOffset.UTC)) + .orElse(null); + + public static final Function LOCAL_DATE_TIME_DATE_FUNCTION = localDateTime -> Optional.ofNullable(localDateTime) + .map(value -> value.toInstant(ZoneOffset.UTC)) + .map(Date::from) + .orElse(null); + + public static final Function LOCAL_DATE_TIME_ZONED_DATE_TIME_FUNCTION = localDateTime -> Optional.ofNullable(localDateTime) + .map(value -> value.atZone(ZoneId.of("UTC"))) + .orElse(null); + + public static final Function OFFSET_DATE_TIME_ZONED_DATE_TIME_FUNCTION = offsetDateTime -> Optional.ofNullable(offsetDateTime) + .map(value -> value.atZoneSameInstant(ZoneId.of("UTC"))) + .orElse(null); + + public static final Function LOCAL_DATE_TIME_INSTANT_FUNCTION = localDateTime -> Optional.ofNullable(localDateTime) + .map(value -> value.toInstant(ZoneOffset.UTC)) + .orElse(null); + + public static final Function, Field> UNNEST_FIELD = field -> DSL.function("unnest", field.getType().getComponentType(), field); + + public static final int IN_CLAUSE_MAX_SIZE = 32; + +} diff --git a/backends-common/postgres/src/main/java/org/apache/james/backends/postgres/PostgresConfiguration.java b/backends-common/postgres/src/main/java/org/apache/james/backends/postgres/PostgresConfiguration.java new file mode 100644 index 00000000000..29e5d904762 --- /dev/null +++ b/backends-common/postgres/src/main/java/org/apache/james/backends/postgres/PostgresConfiguration.java @@ -0,0 +1,404 @@ +/**************************************************************** + * 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.backends.postgres; + +import java.time.Duration; +import java.time.temporal.ChronoUnit; +import java.util.Objects; +import java.util.Optional; + +import org.apache.commons.configuration2.Configuration; +import org.apache.james.util.DurationParser; + +import com.google.common.base.Preconditions; + +import io.r2dbc.postgresql.client.SSLMode; + +public class PostgresConfiguration { + public static final String POSTGRES_CONFIGURATION_NAME = "postgres"; + + public static final String DATABASE_NAME = "database.name"; + public static final String DATABASE_NAME_DEFAULT_VALUE = "postgres"; + public static final String DATABASE_SCHEMA = "database.schema"; + public static final String DATABASE_SCHEMA_DEFAULT_VALUE = "public"; + public static final String HOST = "database.host"; + public static final String HOST_DEFAULT_VALUE = "localhost"; + public static final String PORT = "database.port"; + public static final int PORT_DEFAULT_VALUE = 5432; + public static final String USERNAME = "database.username"; + public static final String PASSWORD = "database.password"; + public static final String BY_PASS_RLS_USERNAME = "database.by-pass-rls.username"; + public static final String BY_PASS_RLS_PASSWORD = "database.by-pass-rls.password"; + public static final String RLS_ENABLED = "row.level.security.enabled"; + public static final String POOL_INITIAL_SIZE = "pool.initial.size"; + public static final int POOL_INITIAL_SIZE_DEFAULT_VALUE = 10; + public static final String POOL_MAX_SIZE = "pool.max.size"; + public static final int POOL_MAX_SIZE_DEFAULT_VALUE = 15; + public static final String BY_PASS_RLS_POOL_INITIAL_SIZE = "by-pass-rls.pool.initial.size"; + public static final int BY_PASS_RLS_POOL_INITIAL_SIZE_DEFAULT_VALUE = 5; + public static final String BY_PASS_RLS_POOL_MAX_SIZE = "by-pass-rls.pool.max.size"; + public static final int BY_PASS_RLS_POOL_MAX_SIZE_DEFAULT_VALUE = 10; + public static final String SSL_MODE = "ssl.mode"; + public static final String SSL_MODE_DEFAULT_VALUE = "allow"; + public static final String JOOQ_REACTIVE_TIMEOUT = "jooq.reactive.timeout"; + public static final Duration JOOQ_REACTIVE_TIMEOUT_DEFAULT_VALUE = Duration.ofSeconds(10); + + public static class Credential { + private final String username; + private final String password; + + + public Credential(String username, String password) { + this.username = username; + this.password = password; + } + + public String getUsername() { + return username; + } + + public String getPassword() { + return password; + } + } + + public static class Builder { + private Optional databaseName = Optional.empty(); + private Optional databaseSchema = Optional.empty(); + private Optional host = Optional.empty(); + private Optional port = Optional.empty(); + private Optional username = Optional.empty(); + private Optional password = Optional.empty(); + private Optional byPassRLSUser = Optional.empty(); + private Optional byPassRLSPassword = Optional.empty(); + private Optional rowLevelSecurityEnabled = Optional.empty(); + private Optional poolInitialSize = Optional.empty(); + private Optional poolMaxSize = Optional.empty(); + private Optional byPassRLSPoolInitialSize = Optional.empty(); + private Optional byPassRLSPoolMaxSize = Optional.empty(); + private Optional sslMode = Optional.empty(); + private Optional jooqReactiveTimeout = Optional.empty(); + + public Builder databaseName(String databaseName) { + this.databaseName = Optional.of(databaseName); + return this; + } + + public Builder databaseName(Optional databaseName) { + this.databaseName = databaseName; + return this; + } + + public Builder databaseSchema(String databaseSchema) { + this.databaseSchema = Optional.of(databaseSchema); + return this; + } + + public Builder databaseSchema(Optional databaseSchema) { + this.databaseSchema = databaseSchema; + return this; + } + + public Builder host(String host) { + this.host = Optional.of(host); + return this; + } + + public Builder host(Optional host) { + this.host = host; + return this; + } + + public Builder port(Integer port) { + this.port = Optional.of(port); + return this; + } + + public Builder port(Optional port) { + this.port = port; + return this; + } + + public Builder username(String username) { + this.username = Optional.of(username); + return this; + } + + public Builder username(Optional username) { + this.username = username; + return this; + } + + public Builder password(String password) { + this.password = Optional.of(password); + return this; + } + + public Builder password(Optional password) { + this.password = password; + return this; + } + + public Builder byPassRLSUser(String byPassRLSUser) { + this.byPassRLSUser = Optional.of(byPassRLSUser); + return this; + } + + public Builder byPassRLSUser(Optional byPassRLSUser) { + this.byPassRLSUser = byPassRLSUser; + return this; + } + + public Builder byPassRLSPassword(String byPassRLSPassword) { + this.byPassRLSPassword = Optional.of(byPassRLSPassword); + return this; + } + + public Builder byPassRLSPassword(Optional byPassRLSPassword) { + this.byPassRLSPassword = byPassRLSPassword; + return this; + } + + public Builder rowLevelSecurityEnabled(boolean rlsEnabled) { + this.rowLevelSecurityEnabled = Optional.of(rlsEnabled); + return this; + } + + public Builder rowLevelSecurityEnabled() { + this.rowLevelSecurityEnabled = Optional.of(true); + return this; + } + + public Builder poolInitialSize(Optional poolInitialSize) { + this.poolInitialSize = poolInitialSize; + return this; + } + + public Builder poolInitialSize(Integer poolInitialSize) { + this.poolInitialSize = Optional.of(poolInitialSize); + return this; + } + + public Builder poolMaxSize(Optional poolMaxSize) { + this.poolMaxSize = poolMaxSize; + return this; + } + + public Builder poolMaxSize(Integer poolMaxSize) { + this.poolMaxSize = Optional.of(poolMaxSize); + return this; + } + + public Builder byPassRLSPoolInitialSize(Optional byPassRLSPoolInitialSize) { + this.byPassRLSPoolInitialSize = byPassRLSPoolInitialSize; + return this; + } + + public Builder byPassRLSPoolInitialSize(Integer byPassRLSPoolInitialSize) { + this.byPassRLSPoolInitialSize = Optional.of(byPassRLSPoolInitialSize); + return this; + } + + public Builder byPassRLSPoolMaxSize(Optional byPassRLSPoolMaxSize) { + this.byPassRLSPoolMaxSize = byPassRLSPoolMaxSize; + return this; + } + + public Builder byPassRLSPoolMaxSize(Integer byPassRLSPoolMaxSize) { + this.byPassRLSPoolMaxSize = Optional.of(byPassRLSPoolMaxSize); + return this; + } + + public Builder sslMode(Optional sslMode) { + this.sslMode = sslMode; + return this; + } + + public Builder sslMode(String sslMode) { + this.sslMode = Optional.of(sslMode); + return this; + } + + public Builder jooqReactiveTimeout(Optional jooqReactiveTimeout) { + this.jooqReactiveTimeout = jooqReactiveTimeout; + return this; + } + + public PostgresConfiguration build() { + Preconditions.checkArgument(username.isPresent() && !username.get().isBlank(), "You need to specify username"); + Preconditions.checkArgument(password.isPresent() && !password.get().isBlank(), "You need to specify password"); + + if (rowLevelSecurityEnabled.isPresent() && rowLevelSecurityEnabled.get()) { + Preconditions.checkArgument(byPassRLSUser.isPresent() && !byPassRLSUser.get().isBlank(), "You need to specify byPassRLSUser"); + Preconditions.checkArgument(byPassRLSPassword.isPresent() && !byPassRLSPassword.get().isBlank(), "You need to specify byPassRLSPassword"); + } + + return new PostgresConfiguration(host.orElse(HOST_DEFAULT_VALUE), + port.orElse(PORT_DEFAULT_VALUE), + databaseName.orElse(DATABASE_NAME_DEFAULT_VALUE), + databaseSchema.orElse(DATABASE_SCHEMA_DEFAULT_VALUE), + new Credential(username.get(), password.get()), + new Credential(byPassRLSUser.orElse(username.get()), byPassRLSPassword.orElse(password.get())), + rowLevelSecurityEnabled.filter(rlsEnabled -> rlsEnabled).map(rlsEnabled -> RowLevelSecurity.ENABLED).orElse(RowLevelSecurity.DISABLED), + poolInitialSize.orElse(POOL_INITIAL_SIZE_DEFAULT_VALUE), + poolMaxSize.orElse(POOL_MAX_SIZE_DEFAULT_VALUE), + byPassRLSPoolInitialSize.orElse(BY_PASS_RLS_POOL_INITIAL_SIZE_DEFAULT_VALUE), + byPassRLSPoolMaxSize.orElse(BY_PASS_RLS_POOL_MAX_SIZE_DEFAULT_VALUE), + SSLMode.fromValue(sslMode.orElse(SSL_MODE_DEFAULT_VALUE)), + jooqReactiveTimeout.orElse(JOOQ_REACTIVE_TIMEOUT_DEFAULT_VALUE)); + } + } + + public static Builder builder() { + return new Builder(); + } + + public static PostgresConfiguration from(Configuration propertiesConfiguration) { + return builder() + .databaseName(Optional.ofNullable(propertiesConfiguration.getString(DATABASE_NAME))) + .databaseSchema(Optional.ofNullable(propertiesConfiguration.getString(DATABASE_SCHEMA))) + .host(Optional.ofNullable(propertiesConfiguration.getString(HOST))) + .port(propertiesConfiguration.getInt(PORT, PORT_DEFAULT_VALUE)) + .username(Optional.ofNullable(propertiesConfiguration.getString(USERNAME))) + .password(Optional.ofNullable(propertiesConfiguration.getString(PASSWORD))) + .byPassRLSUser(Optional.ofNullable(propertiesConfiguration.getString(BY_PASS_RLS_USERNAME))) + .byPassRLSPassword(Optional.ofNullable(propertiesConfiguration.getString(BY_PASS_RLS_PASSWORD))) + .rowLevelSecurityEnabled(propertiesConfiguration.getBoolean(RLS_ENABLED, false)) + .poolInitialSize(Optional.ofNullable(propertiesConfiguration.getInteger(POOL_INITIAL_SIZE, null))) + .poolMaxSize(Optional.ofNullable(propertiesConfiguration.getInteger(POOL_MAX_SIZE, null))) + .byPassRLSPoolInitialSize(Optional.ofNullable(propertiesConfiguration.getInteger(BY_PASS_RLS_POOL_INITIAL_SIZE, null))) + .byPassRLSPoolMaxSize(Optional.ofNullable(propertiesConfiguration.getInteger(BY_PASS_RLS_POOL_MAX_SIZE, null))) + .sslMode(Optional.ofNullable(propertiesConfiguration.getString(SSL_MODE))) + .jooqReactiveTimeout(Optional.ofNullable(propertiesConfiguration.getString(JOOQ_REACTIVE_TIMEOUT)) + .map(value -> DurationParser.parse(value, ChronoUnit.SECONDS))) + .build(); + } + + private final String host; + private final int port; + private final String databaseName; + private final String databaseSchema; + private final Credential defaultCredential; + private final Credential byPassRLSCredential; + private final RowLevelSecurity rowLevelSecurity; + private final Integer poolInitialSize; + private final Integer poolMaxSize; + private final Integer byPassRLSPoolInitialSize; + private final Integer byPassRLSPoolMaxSize; + private final SSLMode sslMode; + private final Duration jooqReactiveTimeout; + + private PostgresConfiguration(String host, int port, String databaseName, String databaseSchema, + Credential defaultCredential, Credential byPassRLSCredential, RowLevelSecurity rowLevelSecurity, + Integer poolInitialSize, Integer poolMaxSize, + Integer byPassRLSPoolInitialSize, Integer byPassRLSPoolMaxSize, + SSLMode sslMode, Duration jooqReactiveTimeout) { + this.host = host; + this.port = port; + this.databaseName = databaseName; + this.databaseSchema = databaseSchema; + this.defaultCredential = defaultCredential; + this.byPassRLSCredential = byPassRLSCredential; + this.rowLevelSecurity = rowLevelSecurity; + this.poolInitialSize = poolInitialSize; + this.poolMaxSize = poolMaxSize; + this.byPassRLSPoolInitialSize = byPassRLSPoolInitialSize; + this.byPassRLSPoolMaxSize = byPassRLSPoolMaxSize; + this.sslMode = sslMode; + this.jooqReactiveTimeout = jooqReactiveTimeout; + } + + public String getHost() { + return host; + } + + public int getPort() { + return port; + } + + public String getDatabaseName() { + return databaseName; + } + + public String getDatabaseSchema() { + return databaseSchema; + } + + public Credential getDefaultCredential() { + return defaultCredential; + } + + public Credential getByPassRLSCredential() { + return byPassRLSCredential; + } + + public RowLevelSecurity getRowLevelSecurity() { + return rowLevelSecurity; + } + + public Integer poolInitialSize() { + return poolInitialSize; + } + + public Integer poolMaxSize() { + return poolMaxSize; + } + + public Integer byPassRLSPoolInitialSize() { + return byPassRLSPoolInitialSize; + } + + public Integer byPassRLSPoolMaxSize() { + return byPassRLSPoolMaxSize; + } + + public SSLMode getSslMode() { + return sslMode; + } + + public Duration getJooqReactiveTimeout() { + return jooqReactiveTimeout; + } + + @Override + public final int hashCode() { + return Objects.hash(host, port, databaseName, databaseSchema, defaultCredential, byPassRLSCredential, rowLevelSecurity, poolInitialSize, poolMaxSize, sslMode, jooqReactiveTimeout); + } + + @Override + public final boolean equals(Object o) { + if (o instanceof PostgresConfiguration) { + PostgresConfiguration that = (PostgresConfiguration) o; + + return Objects.equals(this.rowLevelSecurity, that.rowLevelSecurity) + && Objects.equals(this.host, that.host) + && Objects.equals(this.port, that.port) + && Objects.equals(this.defaultCredential, that.defaultCredential) + && Objects.equals(this.byPassRLSCredential, that.byPassRLSCredential) + && Objects.equals(this.databaseName, that.databaseName) + && Objects.equals(this.databaseSchema, that.databaseSchema) + && Objects.equals(this.poolInitialSize, that.poolInitialSize) + && Objects.equals(this.poolMaxSize, that.poolMaxSize) + && Objects.equals(this.sslMode, that.sslMode) + && Objects.equals(this.jooqReactiveTimeout, that.jooqReactiveTimeout); + } + return false; + } +} diff --git a/backends-common/postgres/src/main/java/org/apache/james/backends/postgres/PostgresIndex.java b/backends-common/postgres/src/main/java/org/apache/james/backends/postgres/PostgresIndex.java new file mode 100644 index 00000000000..c1a41f2947e --- /dev/null +++ b/backends-common/postgres/src/main/java/org/apache/james/backends/postgres/PostgresIndex.java @@ -0,0 +1,64 @@ +/**************************************************************** + * 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.backends.postgres; + +import java.util.function.Function; + +import org.jooq.DDLQuery; +import org.jooq.DSLContext; + +import com.google.common.base.Preconditions; + +public class PostgresIndex { + + @FunctionalInterface + public interface RequireCreateIndexStep { + PostgresIndex createIndexStep(CreateIndexFunction createIndexFunction); + } + + @FunctionalInterface + public interface CreateIndexFunction { + DDLQuery createIndex(DSLContext dsl, String indexName); + } + + public static RequireCreateIndexStep name(String indexName) { + Preconditions.checkNotNull(indexName); + String strategyIndexName = indexName.toLowerCase(); + + return createIndexFunction -> new PostgresIndex(strategyIndexName, dsl -> createIndexFunction.createIndex(dsl, strategyIndexName)); + } + + private final String name; + private final Function createIndexStepFunction; + + private PostgresIndex(String name, Function createIndexStepFunction) { + this.name = name; + this.createIndexStepFunction = createIndexStepFunction; + } + + public String getName() { + return name; + } + + public Function getCreateIndexStepFunction() { + return createIndexStepFunction; + } + +} diff --git a/backends-common/postgres/src/main/java/org/apache/james/backends/postgres/PostgresModule.java b/backends-common/postgres/src/main/java/org/apache/james/backends/postgres/PostgresModule.java new file mode 100644 index 00000000000..8f1725fe4b3 --- /dev/null +++ b/backends-common/postgres/src/main/java/org/apache/james/backends/postgres/PostgresModule.java @@ -0,0 +1,130 @@ +/**************************************************************** + * 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.backends.postgres; + + +import java.util.Collection; +import java.util.List; + +import com.google.common.collect.ImmutableList; + +public interface PostgresModule { + + static PostgresModule aggregateModules(PostgresModule... modules) { + return builder() + .modules(modules) + .build(); + } + + static PostgresModule aggregateModules(Collection modules) { + return builder() + .modules(modules) + .build(); + } + + PostgresModule EMPTY_MODULE = builder().build(); + + List tables(); + + List tableIndexes(); + + class Impl implements PostgresModule { + private final List tables; + private final List tableIndexes; + + private Impl(List tables, List tableIndexes) { + this.tables = tables; + this.tableIndexes = tableIndexes; + } + + @Override + public List tables() { + return tables; + } + + @Override + public List tableIndexes() { + return tableIndexes; + } + } + + class Builder { + private final ImmutableList.Builder tables; + private final ImmutableList.Builder tableIndexes; + + public Builder() { + tables = ImmutableList.builder(); + tableIndexes = ImmutableList.builder(); + } + + public Builder addTable(PostgresTable... table) { + tables.add(table); + return this; + } + + public Builder addIndex(PostgresIndex... index) { + tableIndexes.add(index); + return this; + } + + public Builder addTable(List tables) { + this.tables.addAll(tables); + return this; + } + + public Builder addIndex(List indexes) { + this.tableIndexes.addAll(indexes); + return this; + } + + public Builder modules(Collection modules) { + modules.forEach(module -> { + addTable(module.tables()); + addIndex(module.tableIndexes()); + }); + return this; + } + + public Builder modules(PostgresModule... modules) { + return modules(ImmutableList.copyOf(modules)); + } + + public PostgresModule build() { + return new Impl(tables.build(), tableIndexes.build()); + } + } + + static Builder builder() { + return new Builder(); + } + + static PostgresModule table(PostgresTable... tables) { + return builder() + .addTable(ImmutableList.copyOf(tables)) + .build(); + } + + static PostgresModule tableIndex(PostgresIndex... tableIndexes) { + return builder() + .addIndex(ImmutableList.copyOf(tableIndexes)) + .build(); + } + +} diff --git a/backends-common/postgres/src/main/java/org/apache/james/backends/postgres/PostgresTable.java b/backends-common/postgres/src/main/java/org/apache/james/backends/postgres/PostgresTable.java new file mode 100644 index 00000000000..f9bd1308c90 --- /dev/null +++ b/backends-common/postgres/src/main/java/org/apache/james/backends/postgres/PostgresTable.java @@ -0,0 +1,172 @@ +/**************************************************************** + * 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.backends.postgres; + +import java.util.Arrays; +import java.util.List; +import java.util.function.Function; + +import org.jooq.DDLQuery; +import org.jooq.DSLContext; + +import com.google.common.base.Preconditions; +import com.google.common.collect.ImmutableList; + +public class PostgresTable { + @FunctionalInterface + public interface RequireCreateTableStep { + RequireRowLevelSecurity createTableStep(CreateTableFunction createTableFunction); + } + + @FunctionalInterface + public interface CreateTableFunction { + DDLQuery createTable(DSLContext dsl, String tableName); + } + + @FunctionalInterface + public interface RequireRowLevelSecurity { + FinalStage supportsRowLevelSecurity(boolean rowLevelSecurityEnabled); + + default FinalStage disableRowLevelSecurity() { + return supportsRowLevelSecurity(false); + } + + default FinalStage supportsRowLevelSecurity() { + return supportsRowLevelSecurity(true); + } + } + + public abstract static class AdditionalAlterQuery { + private String query; + + public AdditionalAlterQuery(String query) { + this.query = query; + } + + abstract boolean shouldBeApplied(RowLevelSecurity rowLevelSecurity); + + public String getQuery() { + return query; + } + } + + public static class RLSOnlyAdditionalAlterQuery extends AdditionalAlterQuery { + public RLSOnlyAdditionalAlterQuery(String query) { + super(query); + } + + @Override + boolean shouldBeApplied(RowLevelSecurity rowLevelSecurity) { + return rowLevelSecurity.isRowLevelSecurityEnabled(); + } + } + + public static class NonRLSOnlyAdditionalAlterQuery extends AdditionalAlterQuery { + public NonRLSOnlyAdditionalAlterQuery(String query) { + super(query); + } + + @Override + boolean shouldBeApplied(RowLevelSecurity rowLevelSecurity) { + return !rowLevelSecurity.isRowLevelSecurityEnabled(); + } + } + + public static class AllCasesAdditionalAlterQuery extends AdditionalAlterQuery { + public AllCasesAdditionalAlterQuery(String query) { + super(query); + } + + @Override + boolean shouldBeApplied(RowLevelSecurity rowLevelSecurity) { + return true; + } + } + + public static class FinalStage { + private final String tableName; + private final boolean supportsRowLevelSecurity; + private final Function createTableStepFunction; + private final ImmutableList.Builder additionalAlterQueries; + + public FinalStage(String tableName, boolean supportsRowLevelSecurity, Function createTableStepFunction) { + this.tableName = tableName; + this.supportsRowLevelSecurity = supportsRowLevelSecurity; + this.createTableStepFunction = createTableStepFunction; + this.additionalAlterQueries = ImmutableList.builder(); + } + + /** + * Raw SQL ALTER queries in case not supported by jOOQ DSL. + */ + public FinalStage addAdditionalAlterQueries(String... additionalAlterQueries) { + this.additionalAlterQueries.addAll(Arrays.stream(additionalAlterQueries).map(AllCasesAdditionalAlterQuery::new).toList()); + return this; + } + + /** + * Raw SQL ALTER queries in case not supported by jOOQ DSL. + */ + public FinalStage addAdditionalAlterQueries(AdditionalAlterQuery... additionalAlterQueries) { + this.additionalAlterQueries.add(additionalAlterQueries); + return this; + } + + public PostgresTable build() { + return new PostgresTable(tableName, supportsRowLevelSecurity, createTableStepFunction, additionalAlterQueries.build()); + } + } + + public static RequireCreateTableStep name(String tableName) { + Preconditions.checkNotNull(tableName); + String strategyName = tableName.toLowerCase(); + + return createTableFunction -> supportsRowLevelSecurity -> new FinalStage(strategyName, supportsRowLevelSecurity, dsl -> createTableFunction.createTable(dsl, strategyName)); + } + + private final String name; + private final boolean supportsRowLevelSecurity; + private final Function createTableStepFunction; + private final List additionalAlterQueries; + + private PostgresTable(String name, boolean supportsRowLevelSecurity, Function createTableStepFunction, List additionalAlterQueries) { + this.name = name; + this.supportsRowLevelSecurity = supportsRowLevelSecurity; + this.createTableStepFunction = createTableStepFunction; + this.additionalAlterQueries = additionalAlterQueries; + } + + + public String getName() { + return name; + } + + public Function getCreateTableStepFunction() { + return createTableStepFunction; + } + + public boolean supportsRowLevelSecurity() { + return supportsRowLevelSecurity; + } + + public List getAdditionalAlterQueries() { + return additionalAlterQueries; + } +} diff --git a/backends-common/postgres/src/main/java/org/apache/james/backends/postgres/PostgresTableManager.java b/backends-common/postgres/src/main/java/org/apache/james/backends/postgres/PostgresTableManager.java new file mode 100644 index 00000000000..ffb88497682 --- /dev/null +++ b/backends-common/postgres/src/main/java/org/apache/james/backends/postgres/PostgresTableManager.java @@ -0,0 +1,216 @@ +/**************************************************************** + * 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.backends.postgres; + +import java.util.List; + +import jakarta.inject.Inject; + +import org.apache.james.backends.postgres.utils.PostgresExecutor; +import org.apache.james.lifecycle.api.Startable; +import org.jooq.DSLContext; +import org.jooq.exception.DataAccessException; +import org.jooq.impl.DSL; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + +import com.google.common.annotations.VisibleForTesting; + +import io.r2dbc.spi.Connection; +import io.r2dbc.spi.Result; +import reactor.core.publisher.Flux; +import reactor.core.publisher.Mono; + +public class PostgresTableManager implements Startable { + public static final int INITIALIZATION_PRIORITY = 1; + private static final Logger LOGGER = LoggerFactory.getLogger(PostgresTableManager.class); + private final PostgresExecutor postgresExecutor; + private final PostgresModule module; + private final RowLevelSecurity rowLevelSecurity; + + @Inject + public PostgresTableManager(PostgresExecutor postgresExecutor, + PostgresModule module, + PostgresConfiguration postgresConfiguration) { + this.postgresExecutor = postgresExecutor; + this.module = module; + this.rowLevelSecurity = postgresConfiguration.getRowLevelSecurity(); + } + + @VisibleForTesting + public PostgresTableManager(PostgresExecutor postgresExecutor, PostgresModule module, RowLevelSecurity rowLevelSecurity) { + this.postgresExecutor = postgresExecutor; + this.module = module; + this.rowLevelSecurity = rowLevelSecurity; + } + + public void initPostgres() { + initializePostgresExtension() + .then(initializeTables()) + .then(initializeTableIndexes()) + .block(); + } + + public Mono initializePostgresExtension() { + return Mono.usingWhen(postgresExecutor.connectionFactory().getConnection(), + connection -> Mono.just(connection) + .flatMapMany(pgConnection -> pgConnection.createStatement("CREATE EXTENSION IF NOT EXISTS hstore") + .execute()) + .flatMap(Result::getRowsUpdated) + .then(), + connection -> postgresExecutor.connectionFactory().closeConnection(connection)); + } + + public Mono initializeTables() { + return Mono.usingWhen(postgresExecutor.connectionFactory().getConnection(), + connection -> postgresExecutor.dslContext(connection) + .flatMapMany(dsl -> listExistTables() + .flatMapMany(existTables -> Flux.fromIterable(module.tables()) + .filter(table -> !existTables.contains(table.getName())) + .flatMap(table -> createAndAlterTable(table, dsl, connection)))) + .then(), + connection -> postgresExecutor.connectionFactory().closeConnection(connection)); + } + + private Mono createAndAlterTable(PostgresTable table, DSLContext dsl, Connection connection) { + return Mono.from(table.getCreateTableStepFunction().apply(dsl)) + .then(alterTableIfNeeded(table, connection)) + .doOnSuccess(any -> LOGGER.info("Table {} created", table.getName())) + .onErrorResume(exception -> handleTableCreationException(table, exception)); + } + + public Mono> listExistTables() { + return Mono.usingWhen(postgresExecutor.connectionFactory().getConnection(), + connection -> postgresExecutor.dslContext(connection) + .flatMapMany(d -> Flux.from(d.select(DSL.field("tablename")) + .from("pg_tables") + .where(DSL.field("schemaname") + .eq(DSL.currentSchema())))) + .map(r -> r.get(0, String.class)) + .collectList(), + connection -> postgresExecutor.connectionFactory().closeConnection(connection)); + } + + private Mono handleTableCreationException(PostgresTable table, Throwable e) { + if (e instanceof DataAccessException && e.getMessage().contains(String.format("\"%s\" already exists", table.getName()))) { + return Mono.empty(); + } + LOGGER.error("Error while creating table: {}", table.getName(), e); + return Mono.error(e); + } + + private Mono alterTableIfNeeded(PostgresTable table, Connection connection) { + return executeAdditionalAlterQueries(table, connection) + .then(enableRLSIfNeeded(table, connection)); + } + + private Mono executeAdditionalAlterQueries(PostgresTable table, Connection connection) { + return Flux.fromIterable(table.getAdditionalAlterQueries()) + .filter(additionalAlterQuery -> additionalAlterQuery.shouldBeApplied(rowLevelSecurity)) + .map(PostgresTable.AdditionalAlterQuery::getQuery) + .concatMap(alterSQLQuery -> Mono.just(connection) + .flatMapMany(pgConnection -> pgConnection.createStatement(alterSQLQuery) + .execute()) + .flatMap(Result::getRowsUpdated) + .then() + .onErrorResume(e -> { + if (e.getMessage().contains("already exists")) { + return Mono.empty(); + } + LOGGER.error("Error while executing ALTER query for table {}", table.getName(), e); + return Mono.error(e); + })) + .then(); + } + + private Mono enableRLSIfNeeded(PostgresTable table, Connection connection) { + if (rowLevelSecurity.isRowLevelSecurityEnabled() && table.supportsRowLevelSecurity()) { + return alterTableEnableRLS(table, connection); + } + return Mono.empty(); + } + + private Mono alterTableEnableRLS(PostgresTable table, Connection connection) { + return Mono.just(connection) + .flatMapMany(pgConnection -> pgConnection.createStatement(rowLevelSecurityAlterStatement(table.getName())) + .execute()) + .flatMap(Result::getRowsUpdated) + .then(); + } + + private String rowLevelSecurityAlterStatement(String tableName) { + String policyName = "domain_" + tableName + "_policy"; + return "set app.current_domain = ''; alter table " + tableName + " add column if not exists domain varchar(255) not null default current_setting('app.current_domain')::text ;" + + "do $$ \n" + + "begin \n" + + " if not exists( select policyname from pg_policies where policyname = '" + policyName + "') then \n" + + " execute 'alter table " + tableName + " enable row level security; alter table " + tableName + " force row level security; create policy " + policyName + " on " + tableName + " using (domain = current_setting(''app.current_domain'')::text)';\n" + + " end if;\n" + + "end $$;"; + } + + public Mono truncate() { + return Mono.usingWhen(postgresExecutor.connectionFactory().getConnection(), + connection -> postgresExecutor.dslContext(connection) + .flatMap(dsl -> Flux.fromIterable(module.tables()) + .flatMap(table -> Mono.from(dsl.truncateTable(table.getName())) + .doOnSuccess(any -> LOGGER.info("Table {} truncated", table.getName())) + .doOnError(e -> LOGGER.error("Error while truncating table {}", table.getName(), e))) + .then()), + connection -> postgresExecutor.connectionFactory().closeConnection(connection)); + } + + public Mono initializeTableIndexes() { + return Mono.usingWhen(postgresExecutor.connectionFactory().getConnection(), + connection -> postgresExecutor.dslContext(connection) + .flatMapMany(dsl -> listExistIndexes(dsl) + .flatMapMany(existIndexes -> Flux.fromIterable(module.tableIndexes()) + .filter(index -> !existIndexes.contains(index.getName())) + .flatMap(index -> createTableIndex(index, dsl)))) + .then(), + connection -> postgresExecutor.connectionFactory().closeConnection(connection)); + } + + private Mono> listExistIndexes(DSLContext dslContext) { + return Mono.just(dslContext) + .flatMapMany(dsl -> Flux.from(dsl.select(DSL.field("indexname")) + .from("pg_indexes") + .where(DSL.field("schemaname") + .eq(DSL.currentSchema())))) + .map(r -> r.get(0, String.class)) + .collectList(); + } + + private Mono createTableIndex(PostgresIndex index, DSLContext dsl) { + return Mono.from(index.getCreateIndexStepFunction().apply(dsl)) + .doOnSuccess(any -> LOGGER.info("Index {} created", index.getName())) + .onErrorResume(e -> handleIndexCreationException(index, e)) + .then(); + } + + private Mono handleIndexCreationException(PostgresIndex index, Throwable e) { + if (e instanceof DataAccessException && e.getMessage().contains(String.format("\"%s\" already exists", index.getName()))) { + return Mono.empty(); + } + LOGGER.error("Error while creating index {}", index.getName(), e); + return Mono.error(e); + } + +} diff --git a/backends-common/postgres/src/main/java/org/apache/james/backends/postgres/RowLevelSecurity.java b/backends-common/postgres/src/main/java/org/apache/james/backends/postgres/RowLevelSecurity.java new file mode 100644 index 00000000000..2f806b6c74e --- /dev/null +++ b/backends-common/postgres/src/main/java/org/apache/james/backends/postgres/RowLevelSecurity.java @@ -0,0 +1,35 @@ +/**************************************************************** + * 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.backends.postgres; + +public enum RowLevelSecurity { + ENABLED(true), + DISABLED(false); + + private boolean rowLevelSecurityEnabled; + + RowLevelSecurity(boolean rowLevelSecurityEnabled) { + this.rowLevelSecurityEnabled = rowLevelSecurityEnabled; + } + + public boolean isRowLevelSecurityEnabled() { + return rowLevelSecurityEnabled; + } +} diff --git a/backends-common/postgres/src/main/java/org/apache/james/backends/postgres/quota/PostgresQuotaCurrentValueDAO.java b/backends-common/postgres/src/main/java/org/apache/james/backends/postgres/quota/PostgresQuotaCurrentValueDAO.java new file mode 100644 index 00000000000..531f58d8e27 --- /dev/null +++ b/backends-common/postgres/src/main/java/org/apache/james/backends/postgres/quota/PostgresQuotaCurrentValueDAO.java @@ -0,0 +1,158 @@ +/**************************************************************** + * 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.backends.postgres.quota; + +import static org.apache.james.backends.postgres.quota.PostgresQuotaModule.PostgresQuotaCurrentValueTable.COMPONENT; +import static org.apache.james.backends.postgres.quota.PostgresQuotaModule.PostgresQuotaCurrentValueTable.CURRENT_VALUE; +import static org.apache.james.backends.postgres.quota.PostgresQuotaModule.PostgresQuotaCurrentValueTable.IDENTIFIER; +import static org.apache.james.backends.postgres.quota.PostgresQuotaModule.PostgresQuotaCurrentValueTable.PRIMARY_KEY_CONSTRAINT_NAME; +import static org.apache.james.backends.postgres.quota.PostgresQuotaModule.PostgresQuotaCurrentValueTable.TABLE_NAME; +import static org.apache.james.backends.postgres.quota.PostgresQuotaModule.PostgresQuotaCurrentValueTable.TYPE; +import static org.apache.james.backends.postgres.utils.PostgresExecutor.DEFAULT_INJECT; + +import java.util.function.Function; + +import jakarta.inject.Inject; +import jakarta.inject.Named; + +import org.apache.james.backends.postgres.utils.PostgresExecutor; +import org.apache.james.core.quota.QuotaComponent; +import org.apache.james.core.quota.QuotaCurrentValue; +import org.apache.james.core.quota.QuotaType; +import org.jooq.Record; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + +import reactor.core.publisher.Flux; +import reactor.core.publisher.Mono; + +public class PostgresQuotaCurrentValueDAO { + private static final Logger LOGGER = LoggerFactory.getLogger(PostgresQuotaCurrentValueDAO.class); + private static final boolean IS_INCREASE = true; + + private final PostgresExecutor postgresExecutor; + + @Inject + public PostgresQuotaCurrentValueDAO(@Named(DEFAULT_INJECT) PostgresExecutor postgresExecutor) { + this.postgresExecutor = postgresExecutor; + } + + public Mono increase(QuotaCurrentValue.Key quotaKey, long amount) { + return postgresExecutor.executeVoid(dslContext -> Mono.from(dslContext.insertInto(TABLE_NAME) + .set(IDENTIFIER, quotaKey.getIdentifier()) + .set(COMPONENT, quotaKey.getQuotaComponent().getValue()) + .set(TYPE, quotaKey.getQuotaType().getValue()) + .set(CURRENT_VALUE, amount) + .onConflictOnConstraint(PRIMARY_KEY_CONSTRAINT_NAME) + .doUpdate() + .set(CURRENT_VALUE, CURRENT_VALUE.plus(amount)))) + .onErrorResume(ex -> { + LOGGER.warn("Failure when increasing {} {} quota for {}. Quota current value is thus not updated and needs re-computation", + quotaKey.getQuotaComponent().getValue(), quotaKey.getQuotaType().getValue(), quotaKey.getIdentifier(), ex); + return Mono.empty(); + }); + } + + public Mono upsert(QuotaCurrentValue.Key quotaKey, long newCurrentValue) { + return update(quotaKey, newCurrentValue) + .switchIfEmpty(Mono.defer(() -> insert(quotaKey, newCurrentValue, IS_INCREASE))); + } + + public Mono update(QuotaCurrentValue.Key quotaKey, long newCurrentValue) { + return postgresExecutor.executeRow(dslContext -> Mono.from(dslContext.update(TABLE_NAME) + .set(CURRENT_VALUE, newCurrentValue) + .where(IDENTIFIER.eq(quotaKey.getIdentifier()), + COMPONENT.eq(quotaKey.getQuotaComponent().getValue()), + TYPE.eq(quotaKey.getQuotaType().getValue())) + .returning(CURRENT_VALUE))) + .map(record -> record.get(CURRENT_VALUE)); + } + + public Mono insert(QuotaCurrentValue.Key quotaKey, long amount, boolean isIncrease) { + return postgresExecutor.executeRow(dslContext -> Mono.from(dslContext.insertInto(TABLE_NAME) + .set(IDENTIFIER, quotaKey.getIdentifier()) + .set(COMPONENT, quotaKey.getQuotaComponent().getValue()) + .set(TYPE, quotaKey.getQuotaType().getValue()) + .set(CURRENT_VALUE, newCurrentValue(amount, isIncrease)) + .returning(CURRENT_VALUE))) + .map(record -> record.get(CURRENT_VALUE)); + } + + private Long newCurrentValue(long amount, boolean isIncrease) { + if (isIncrease) { + return amount; + } + return -amount; + } + + public Mono decrease(QuotaCurrentValue.Key quotaKey, long amount) { + return postgresExecutor.executeVoid(dslContext -> Mono.from(dslContext.insertInto(TABLE_NAME) + .set(IDENTIFIER, quotaKey.getIdentifier()) + .set(COMPONENT, quotaKey.getQuotaComponent().getValue()) + .set(TYPE, quotaKey.getQuotaType().getValue()) + .set(CURRENT_VALUE, -amount) + .onConflictOnConstraint(PRIMARY_KEY_CONSTRAINT_NAME) + .doUpdate() + .set(CURRENT_VALUE, CURRENT_VALUE.minus(amount)))) + .onErrorResume(ex -> { + LOGGER.warn("Failure when decreasing {} {} quota for {}. Quota current value is thus not updated and needs re-computation", + quotaKey.getQuotaComponent().getValue(), quotaKey.getQuotaType().getValue(), quotaKey.getIdentifier(), ex); + return Mono.empty(); + }); + } + + public Mono getQuotaCurrentValue(QuotaCurrentValue.Key quotaKey) { + return postgresExecutor.executeRow(dslContext -> Mono.from(dslContext.select(CURRENT_VALUE) + .from(TABLE_NAME) + .where(IDENTIFIER.eq(quotaKey.getIdentifier()), + COMPONENT.eq(quotaKey.getQuotaComponent().getValue()), + TYPE.eq(quotaKey.getQuotaType().getValue())))) + .map(toQuotaCurrentValue(quotaKey)); + } + + public Mono deleteQuotaCurrentValue(QuotaCurrentValue.Key quotaKey) { + return postgresExecutor.executeVoid(dslContext -> Mono.from(dslContext.deleteFrom(TABLE_NAME) + .where(IDENTIFIER.eq(quotaKey.getIdentifier()), + COMPONENT.eq(quotaKey.getQuotaComponent().getValue()), + TYPE.eq(quotaKey.getQuotaType().getValue())))); + } + + public Flux getQuotaCurrentValues(QuotaComponent quotaComponent, String identifier) { + return postgresExecutor.executeRows(dslContext -> Flux.from(dslContext.select(TYPE, CURRENT_VALUE) + .from(TABLE_NAME) + .where(IDENTIFIER.eq(identifier), + COMPONENT.eq(quotaComponent.getValue())))) + .map(toQuotaCurrentValue(quotaComponent, identifier)); + } + + private Function toQuotaCurrentValue(QuotaCurrentValue.Key quotaKey) { + return record -> QuotaCurrentValue.builder().quotaComponent(quotaKey.getQuotaComponent()) + .identifier(quotaKey.getIdentifier()) + .quotaType(quotaKey.getQuotaType()) + .currentValue(record.get(CURRENT_VALUE)).build(); + } + + private static Function toQuotaCurrentValue(QuotaComponent quotaComponent, String identifier) { + return record -> QuotaCurrentValue.builder().quotaComponent(quotaComponent) + .identifier(identifier) + .quotaType(QuotaType.of(record.get(TYPE))) + .currentValue(record.get(CURRENT_VALUE)).build(); + } +} diff --git a/backends-common/postgres/src/main/java/org/apache/james/backends/postgres/quota/PostgresQuotaLimitDAO.java b/backends-common/postgres/src/main/java/org/apache/james/backends/postgres/quota/PostgresQuotaLimitDAO.java new file mode 100644 index 00000000000..02523bae40b --- /dev/null +++ b/backends-common/postgres/src/main/java/org/apache/james/backends/postgres/quota/PostgresQuotaLimitDAO.java @@ -0,0 +1,100 @@ +/**************************************************************** + * 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.backends.postgres.quota; + +import static org.apache.james.backends.postgres.quota.PostgresQuotaModule.PostgresQuotaLimitTable.IDENTIFIER; +import static org.apache.james.backends.postgres.quota.PostgresQuotaModule.PostgresQuotaLimitTable.PK_CONSTRAINT_NAME; +import static org.apache.james.backends.postgres.quota.PostgresQuotaModule.PostgresQuotaLimitTable.QUOTA_COMPONENT; +import static org.apache.james.backends.postgres.quota.PostgresQuotaModule.PostgresQuotaLimitTable.QUOTA_LIMIT; +import static org.apache.james.backends.postgres.quota.PostgresQuotaModule.PostgresQuotaLimitTable.QUOTA_SCOPE; +import static org.apache.james.backends.postgres.quota.PostgresQuotaModule.PostgresQuotaLimitTable.QUOTA_TYPE; +import static org.apache.james.backends.postgres.quota.PostgresQuotaModule.PostgresQuotaLimitTable.TABLE_NAME; +import static org.apache.james.backends.postgres.utils.PostgresExecutor.DEFAULT_INJECT; + +import jakarta.inject.Inject; +import jakarta.inject.Named; + +import org.apache.james.backends.postgres.utils.PostgresExecutor; +import org.apache.james.core.quota.QuotaComponent; +import org.apache.james.core.quota.QuotaLimit; +import org.apache.james.core.quota.QuotaScope; +import org.apache.james.core.quota.QuotaType; +import org.jooq.Record; + +import reactor.core.publisher.Flux; +import reactor.core.publisher.Mono; + +public class PostgresQuotaLimitDAO { + private static final Long EMPTY_QUOTA_LIMIT = null; + + private final PostgresExecutor postgresExecutor; + + @Inject + public PostgresQuotaLimitDAO(@Named(DEFAULT_INJECT) PostgresExecutor postgresExecutor) { + this.postgresExecutor = postgresExecutor; + } + + public Mono getQuotaLimit(QuotaLimit.QuotaLimitKey quotaKey) { + return postgresExecutor.executeRow(dsl -> Mono.from(dsl.selectFrom(TABLE_NAME) + .where(QUOTA_COMPONENT.eq(quotaKey.getQuotaComponent().getValue())) + .and(QUOTA_SCOPE.eq(quotaKey.getQuotaScope().getValue())) + .and(IDENTIFIER.eq(quotaKey.getIdentifier())) + .and(QUOTA_TYPE.eq(quotaKey.getQuotaType().getValue())))) + .map(this::asQuotaLimit); + } + + public Flux getQuotaLimits(QuotaComponent quotaComponent, QuotaScope quotaScope, String identifier) { + return postgresExecutor.executeRows(dsl -> Flux.from(dsl.selectFrom(TABLE_NAME) + .where(QUOTA_COMPONENT.eq(quotaComponent.getValue())) + .and(QUOTA_SCOPE.eq(quotaScope.getValue())) + .and(IDENTIFIER.eq(identifier)))) + .map(this::asQuotaLimit); + } + + public Mono setQuotaLimit(QuotaLimit quotaLimit) { + return postgresExecutor.executeVoid(dslContext -> + Mono.from(dslContext.insertInto(TABLE_NAME, QUOTA_SCOPE, IDENTIFIER, QUOTA_COMPONENT, QUOTA_TYPE, QUOTA_LIMIT) + .values(quotaLimit.getQuotaScope().getValue(), + quotaLimit.getIdentifier(), + quotaLimit.getQuotaComponent().getValue(), + quotaLimit.getQuotaType().getValue(), + quotaLimit.getQuotaLimit().orElse(EMPTY_QUOTA_LIMIT)) + .onConflictOnConstraint(PK_CONSTRAINT_NAME) + .doUpdate() + .set(QUOTA_LIMIT, quotaLimit.getQuotaLimit().orElse(EMPTY_QUOTA_LIMIT)))); + } + + public Mono deleteQuotaLimit(QuotaLimit.QuotaLimitKey quotaKey) { + return postgresExecutor.executeVoid(dsl -> Mono.from(dsl.deleteFrom(TABLE_NAME) + .where(QUOTA_COMPONENT.eq(quotaKey.getQuotaComponent().getValue())) + .and(QUOTA_SCOPE.eq(quotaKey.getQuotaScope().getValue())) + .and(IDENTIFIER.eq(quotaKey.getIdentifier())) + .and(QUOTA_TYPE.eq(quotaKey.getQuotaType().getValue())))); + } + + private QuotaLimit asQuotaLimit(Record record) { + return QuotaLimit.builder().quotaComponent(QuotaComponent.of(record.get(QUOTA_COMPONENT))) + .quotaScope(QuotaScope.of(record.get(QUOTA_SCOPE))) + .identifier(record.get(IDENTIFIER)) + .quotaType(QuotaType.of(record.get(QUOTA_TYPE))) + .quotaLimit(record.get(QUOTA_LIMIT)) + .build(); + } +} diff --git a/backends-common/postgres/src/main/java/org/apache/james/backends/postgres/quota/PostgresQuotaModule.java b/backends-common/postgres/src/main/java/org/apache/james/backends/postgres/quota/PostgresQuotaModule.java new file mode 100644 index 00000000000..b0e5c814c56 --- /dev/null +++ b/backends-common/postgres/src/main/java/org/apache/james/backends/postgres/quota/PostgresQuotaModule.java @@ -0,0 +1,84 @@ +/**************************************************************** + * 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.backends.postgres.quota; + +import static org.jooq.impl.DSL.name; +import static org.jooq.impl.SQLDataType.BIGINT; + +import org.apache.james.backends.postgres.PostgresModule; +import org.apache.james.backends.postgres.PostgresTable; +import org.jooq.Field; +import org.jooq.Name; +import org.jooq.Record; +import org.jooq.Table; +import org.jooq.impl.DSL; +import org.jooq.impl.SQLDataType; + +public interface PostgresQuotaModule { + interface PostgresQuotaCurrentValueTable { + Table TABLE_NAME = DSL.table("quota_current_value"); + + Field IDENTIFIER = DSL.field("identifier", SQLDataType.VARCHAR.notNull()); + Field COMPONENT = DSL.field("component", SQLDataType.VARCHAR.notNull()); + Field TYPE = DSL.field("type", SQLDataType.VARCHAR.notNull()); + Field CURRENT_VALUE = DSL.field(name(TABLE_NAME.getName(), "current_value"), BIGINT.notNull()); + + Name PRIMARY_KEY_CONSTRAINT_NAME = DSL.name("quota_current_value_primary_key"); + + PostgresTable TABLE = PostgresTable.name(TABLE_NAME.getName()) + .createTableStep(((dsl, tableName) -> dsl.createTableIfNotExists(tableName) + .column(IDENTIFIER) + .column(COMPONENT) + .column(TYPE) + .column(CURRENT_VALUE) + .constraint(DSL.constraint(PRIMARY_KEY_CONSTRAINT_NAME) + .primaryKey(IDENTIFIER, COMPONENT, TYPE)))) + .disableRowLevelSecurity() + .build(); + } + + interface PostgresQuotaLimitTable { + Table TABLE_NAME = DSL.table("quota_limit"); + + Field QUOTA_SCOPE = DSL.field("quota_scope", SQLDataType.VARCHAR.notNull()); + Field IDENTIFIER = DSL.field("identifier", SQLDataType.VARCHAR.notNull()); + Field QUOTA_COMPONENT = DSL.field("quota_component", SQLDataType.VARCHAR.notNull()); + Field QUOTA_TYPE = DSL.field("quota_type", SQLDataType.VARCHAR.notNull()); + Field QUOTA_LIMIT = DSL.field("quota_limit", SQLDataType.BIGINT); + + Name PK_CONSTRAINT_NAME = DSL.name("quota_limit_pkey"); + + PostgresTable TABLE = PostgresTable.name(TABLE_NAME.getName()) + .createTableStep(((dsl, tableName) -> dsl.createTableIfNotExists(tableName) + .column(QUOTA_SCOPE) + .column(IDENTIFIER) + .column(QUOTA_COMPONENT) + .column(QUOTA_TYPE) + .column(QUOTA_LIMIT) + .constraint(DSL.constraint(PK_CONSTRAINT_NAME).primaryKey(QUOTA_SCOPE, IDENTIFIER, QUOTA_COMPONENT, QUOTA_TYPE)))) + .disableRowLevelSecurity() + .build(); + } + + PostgresModule MODULE = PostgresModule.builder() + .addTable(PostgresQuotaCurrentValueTable.TABLE) + .addTable(PostgresQuotaLimitTable.TABLE) + .build(); +} \ No newline at end of file diff --git a/backends-common/postgres/src/main/java/org/apache/james/backends/postgres/utils/JamesPostgresConnectionFactory.java b/backends-common/postgres/src/main/java/org/apache/james/backends/postgres/utils/JamesPostgresConnectionFactory.java new file mode 100644 index 00000000000..e1b74faf817 --- /dev/null +++ b/backends-common/postgres/src/main/java/org/apache/james/backends/postgres/utils/JamesPostgresConnectionFactory.java @@ -0,0 +1,38 @@ +/**************************************************************** + * 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.backends.postgres.utils; + +import org.apache.james.core.Domain; + +import io.r2dbc.spi.Connection; +import reactor.core.publisher.Mono; + +public interface JamesPostgresConnectionFactory { + String DOMAIN_ATTRIBUTE = "app.current_domain"; + String BY_PASS_RLS_INJECT = "by_pass_rls"; + + Mono getConnection(Domain domain); + + Mono getConnection(); + + Mono closeConnection(Connection connection); + + Mono close(); +} diff --git a/backends-common/postgres/src/main/java/org/apache/james/backends/postgres/utils/PoolBackedPostgresConnectionFactory.java b/backends-common/postgres/src/main/java/org/apache/james/backends/postgres/utils/PoolBackedPostgresConnectionFactory.java new file mode 100644 index 00000000000..465f93a1c38 --- /dev/null +++ b/backends-common/postgres/src/main/java/org/apache/james/backends/postgres/utils/PoolBackedPostgresConnectionFactory.java @@ -0,0 +1,85 @@ +/**************************************************************** + * 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.backends.postgres.utils; + +import org.apache.james.backends.postgres.RowLevelSecurity; +import org.apache.james.core.Domain; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + +import io.r2dbc.pool.ConnectionPool; +import io.r2dbc.pool.ConnectionPoolConfiguration; +import io.r2dbc.spi.Connection; +import io.r2dbc.spi.ConnectionFactory; +import reactor.core.publisher.Mono; + +public class PoolBackedPostgresConnectionFactory implements JamesPostgresConnectionFactory { + private static final Logger LOGGER = LoggerFactory.getLogger(PoolBackedPostgresConnectionFactory.class); + private static final int DEFAULT_INITIAL_SIZE = 10; + private static final int DEFAULT_MAX_SIZE = 20; + + private final RowLevelSecurity rowLevelSecurity; + private final ConnectionPool pool; + + public PoolBackedPostgresConnectionFactory(RowLevelSecurity rowLevelSecurity, int initialSize, int maxSize, ConnectionFactory connectionFactory) { + this.rowLevelSecurity = rowLevelSecurity; + ConnectionPoolConfiguration configuration = ConnectionPoolConfiguration.builder(connectionFactory) + .initialSize(initialSize) + .maxSize(maxSize) + .build(); + LOGGER.info("Creating new postgres ConnectionPool with initialSize {} and maxSize {}", initialSize, maxSize); + pool = new ConnectionPool(configuration); + } + + public PoolBackedPostgresConnectionFactory(RowLevelSecurity rowLevelSecurity, ConnectionFactory connectionFactory) { + this(rowLevelSecurity, DEFAULT_INITIAL_SIZE, DEFAULT_MAX_SIZE, connectionFactory); + } + + @Override + public Mono getConnection(Domain domain) { + if (rowLevelSecurity.isRowLevelSecurityEnabled()) { + return pool.create().flatMap(connection -> setDomainAttributeForConnection(domain.asString(), connection)); + } else { + return pool.create(); + } + } + + @Override + public Mono getConnection() { + return pool.create(); + } + + @Override + public Mono closeConnection(Connection connection) { + return Mono.from(connection.close()); + } + + @Override + public Mono close() { + return pool.close(); + } + + private Mono setDomainAttributeForConnection(String domainAttribute, Connection connection) { + return Mono.from(connection.createStatement("SET " + DOMAIN_ATTRIBUTE + " TO '" + domainAttribute + "'") // It should be set value via Bind, but it doesn't work + .execute()) + .doOnError(e -> LOGGER.error("Error while setting domain attribute for domain {}", domainAttribute, e)) + .then(Mono.just(connection)); + } +} diff --git a/backends-common/postgres/src/main/java/org/apache/james/backends/postgres/utils/PostgresConnectionClosure.java b/backends-common/postgres/src/main/java/org/apache/james/backends/postgres/utils/PostgresConnectionClosure.java new file mode 100644 index 00000000000..0815177f2e7 --- /dev/null +++ b/backends-common/postgres/src/main/java/org/apache/james/backends/postgres/utils/PostgresConnectionClosure.java @@ -0,0 +1,45 @@ +/**************************************************************** + * 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.backends.postgres.utils; + +import jakarta.annotation.PreDestroy; +import jakarta.inject.Inject; +import jakarta.inject.Named; + +import org.apache.james.lifecycle.api.Disposable; + +public class PostgresConnectionClosure implements Disposable { + private final JamesPostgresConnectionFactory factory; + private final JamesPostgresConnectionFactory byPassRLSFactory; + + @Inject + public PostgresConnectionClosure(JamesPostgresConnectionFactory factory, + @Named(JamesPostgresConnectionFactory.BY_PASS_RLS_INJECT) JamesPostgresConnectionFactory byPassRLSFactory) { + this.factory = factory; + this.byPassRLSFactory = byPassRLSFactory; + } + + @PreDestroy + @Override + public void dispose() { + factory.close().block(); + byPassRLSFactory.close().block(); + } +} diff --git a/backends-common/postgres/src/main/java/org/apache/james/backends/postgres/utils/PostgresExecutor.java b/backends-common/postgres/src/main/java/org/apache/james/backends/postgres/utils/PostgresExecutor.java new file mode 100644 index 00000000000..cbe00f80232 --- /dev/null +++ b/backends-common/postgres/src/main/java/org/apache/james/backends/postgres/utils/PostgresExecutor.java @@ -0,0 +1,245 @@ +/**************************************************************** + * 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.backends.postgres.utils; + +import static org.jooq.impl.DSL.exists; +import static org.jooq.impl.DSL.field; + +import java.time.Duration; +import java.util.Optional; +import java.util.concurrent.TimeoutException; +import java.util.function.Function; +import java.util.function.Predicate; + +import jakarta.inject.Inject; + +import org.apache.james.backends.postgres.PostgresConfiguration; +import org.apache.james.core.Domain; +import org.apache.james.metrics.api.MetricFactory; +import org.jooq.DSLContext; +import org.jooq.DeleteResultStep; +import org.jooq.Record; +import org.jooq.Record1; +import org.jooq.SQLDialect; +import org.jooq.SelectConditionStep; +import org.jooq.conf.Settings; +import org.jooq.conf.StatementType; +import org.jooq.impl.DSL; +import org.reactivestreams.Publisher; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + +import com.google.common.annotations.VisibleForTesting; + +import io.r2dbc.spi.Connection; +import io.r2dbc.spi.R2dbcBadGrammarException; +import reactor.core.publisher.Flux; +import reactor.core.publisher.Mono; +import reactor.util.retry.Retry; + +public class PostgresExecutor { + + public static final String DEFAULT_INJECT = "default"; + public static final String BY_PASS_RLS_INJECT = "by_pass_rls"; + public static final int MAX_RETRY_ATTEMPTS = 5; + public static final Duration MIN_BACKOFF = Duration.ofMillis(1); + private static final Logger LOGGER = LoggerFactory.getLogger(PostgresExecutor.class); + private static final String JOOQ_TIMEOUT_ERROR_LOG = "Time out executing Postgres query. May need to check either jOOQ reactive issue or Postgres DB performance."; + public static final boolean EAGER_FETCH = true; + + public static class Factory { + + private final JamesPostgresConnectionFactory jamesPostgresConnectionFactory; + private final PostgresConfiguration postgresConfiguration; + private final MetricFactory metricFactory; + + @Inject + public Factory(JamesPostgresConnectionFactory jamesPostgresConnectionFactory, + PostgresConfiguration postgresConfiguration, + MetricFactory metricFactory) { + this.jamesPostgresConnectionFactory = jamesPostgresConnectionFactory; + this.postgresConfiguration = postgresConfiguration; + this.metricFactory = metricFactory; + } + + public PostgresExecutor create(Optional domain) { + return new PostgresExecutor(domain, jamesPostgresConnectionFactory, postgresConfiguration, metricFactory); + } + + public PostgresExecutor create() { + return create(Optional.empty()); + } + } + + private static final SQLDialect PGSQL_DIALECT = SQLDialect.POSTGRES; + private static final Settings SETTINGS = new Settings() + .withRenderFormatted(true) + .withStatementType(StatementType.PREPARED_STATEMENT); + + private final Optional domain; + private final JamesPostgresConnectionFactory jamesPostgresConnectionFactory; + private final PostgresConfiguration postgresConfiguration; + private final MetricFactory metricFactory; + + private PostgresExecutor(Optional domain, + JamesPostgresConnectionFactory jamesPostgresConnectionFactory, + PostgresConfiguration postgresConfiguration, + MetricFactory metricFactory) { + this.domain = domain; + this.jamesPostgresConnectionFactory = jamesPostgresConnectionFactory; + this.postgresConfiguration = postgresConfiguration; + this.metricFactory = metricFactory; + } + + public Mono dslContext(Connection connection) { + return Mono.fromCallable(() -> DSL.using(connection, PGSQL_DIALECT, SETTINGS)); + } + + public Mono executeVoid(Function> queryFunction) { + return Mono.from(metricFactory.decoratePublisherWithTimerMetric("postgres-execution", + Mono.usingWhen(getConnection(domain), + connection -> dslContext(connection) + .flatMap(queryFunction) + .timeout(postgresConfiguration.getJooqReactiveTimeout()) + .doOnError(TimeoutException.class, e -> LOGGER.error(JOOQ_TIMEOUT_ERROR_LOG, e)) + .retryWhen(Retry.backoff(MAX_RETRY_ATTEMPTS, MIN_BACKOFF) + .filter(preparedStatementConflictException())) + .then(), + jamesPostgresConnectionFactory::closeConnection))); + } + + public Flux executeRows(Function> queryFunction) { + return executeRows(queryFunction, !EAGER_FETCH); + } + + /** + * @param isEagerFetch + * Because an R2DBC postgres connection can only execute one query at a time, when `isEagerFetch` is set to `true`, all elements are fetched and stored in a list (using the `.collectList` API). + *

+ * This approach allows the connection to early complete its current query and return to the pool, helping to avoid Reactor pipeline hanging caused by depleting the available connection pool. + *

+ * It is recommended to set `isEagerFetch` to `true` when the number of elements to fetch is small, such as the number of mailboxes for a single user. + *

+ * In cases where the number of elements is large, consider using pagination-based queries. + *

+ * Reference: + * - james-project + * - r2dbc-postgresql + */ + public Flux executeRows(Function> queryFunction, boolean isEagerFetch) { + return Flux.from(metricFactory.decoratePublisherWithTimerMetric("postgres-execution", + Flux.usingWhen(getConnection(domain), + connection -> { + Flux recordFlux = dslContext(connection) + .flatMapMany(queryFunction) + .timeout(postgresConfiguration.getJooqReactiveTimeout()) + .doOnError(TimeoutException.class, e -> LOGGER.error(JOOQ_TIMEOUT_ERROR_LOG, e)) + .retryWhen(Retry.backoff(MAX_RETRY_ATTEMPTS, MIN_BACKOFF) + .filter(preparedStatementConflictException())); + + if (isEagerFetch) { + return recordFlux.collectList().flatMapIterable(list -> list); + } else { + return recordFlux; + } + }, + jamesPostgresConnectionFactory::closeConnection))); + } + + public Flux executeDeleteAndReturnList(Function> queryFunction) { + return Flux.from(metricFactory.decoratePublisherWithTimerMetric("postgres-execution", + Flux.usingWhen(getConnection(domain), + connection -> dslContext(connection) + .flatMapMany(queryFunction) + .timeout(postgresConfiguration.getJooqReactiveTimeout()) + .doOnError(TimeoutException.class, e -> LOGGER.error(JOOQ_TIMEOUT_ERROR_LOG, e)) + .retryWhen(Retry.backoff(MAX_RETRY_ATTEMPTS, MIN_BACKOFF) + .filter(preparedStatementConflictException())), + jamesPostgresConnectionFactory::closeConnection))); + } + + public Mono executeRow(Function> queryFunction) { + return Mono.from(metricFactory.decoratePublisherWithTimerMetric("postgres-execution", + Mono.usingWhen(getConnection(domain), + connection -> dslContext(connection) + .flatMap(queryFunction.andThen(Mono::from)) + .timeout(postgresConfiguration.getJooqReactiveTimeout()) + .doOnError(TimeoutException.class, e -> LOGGER.error(JOOQ_TIMEOUT_ERROR_LOG, e)) + .retryWhen(Retry.backoff(MAX_RETRY_ATTEMPTS, MIN_BACKOFF) + .filter(preparedStatementConflictException())), + jamesPostgresConnectionFactory::closeConnection))); + } + + public Mono> executeSingleRowOptional(Function> queryFunction) { + return executeRow(queryFunction) + .map(Optional::ofNullable) + .switchIfEmpty(Mono.just(Optional.empty())); + } + + public Mono executeCount(Function>> queryFunction) { + return Mono.from(metricFactory.decoratePublisherWithTimerMetric("postgres-execution", + Mono.usingWhen(getConnection(domain), + connection -> dslContext(connection) + .flatMap(queryFunction) + .timeout(postgresConfiguration.getJooqReactiveTimeout()) + .doOnError(TimeoutException.class, e -> LOGGER.error(JOOQ_TIMEOUT_ERROR_LOG, e)) + .retryWhen(Retry.backoff(MAX_RETRY_ATTEMPTS, MIN_BACKOFF) + .filter(preparedStatementConflictException())) + .map(Record1::value1), + jamesPostgresConnectionFactory::closeConnection))); + } + + public Mono executeExists(Function> queryFunction) { + return executeRow(dslContext -> Mono.from(dslContext.select(field(exists(queryFunction.apply(dslContext)))))) + .map(record -> record.get(0, Boolean.class)); + } + + public Mono executeReturnAffectedRowsCount(Function> queryFunction) { + return Mono.from(metricFactory.decoratePublisherWithTimerMetric("postgres-execution", + Mono.usingWhen(getConnection(domain), + connection -> dslContext(connection) + .flatMap(queryFunction) + .timeout(postgresConfiguration.getJooqReactiveTimeout()) + .doOnError(TimeoutException.class, e -> LOGGER.error(JOOQ_TIMEOUT_ERROR_LOG, e)) + .retryWhen(Retry.backoff(MAX_RETRY_ATTEMPTS, MIN_BACKOFF) + .filter(preparedStatementConflictException())), + jamesPostgresConnectionFactory::closeConnection))); + } + + public JamesPostgresConnectionFactory connectionFactory() { + return jamesPostgresConnectionFactory; + } + + @VisibleForTesting + public void dispose() { + jamesPostgresConnectionFactory.close().block(); + } + + private Predicate preparedStatementConflictException() { + return throwable -> throwable.getCause() instanceof R2dbcBadGrammarException + && throwable.getMessage().contains("prepared statement") + && throwable.getMessage().contains("already exists"); + } + + private Mono getConnection(Optional maybeDomain) { + return maybeDomain.map(jamesPostgresConnectionFactory::getConnection) + .orElseGet(jamesPostgresConnectionFactory::getConnection); + } +} diff --git a/backends-common/postgres/src/main/java/org/apache/james/backends/postgres/utils/PostgresHealthCheck.java b/backends-common/postgres/src/main/java/org/apache/james/backends/postgres/utils/PostgresHealthCheck.java new file mode 100644 index 00000000000..2774c3bc79d --- /dev/null +++ b/backends-common/postgres/src/main/java/org/apache/james/backends/postgres/utils/PostgresHealthCheck.java @@ -0,0 +1,55 @@ +/**************************************************************** + * 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.backends.postgres.utils; + +import java.time.Duration; + +import jakarta.inject.Inject; + +import org.apache.james.core.healthcheck.ComponentName; +import org.apache.james.core.healthcheck.HealthCheck; +import org.apache.james.core.healthcheck.Result; +import org.jooq.impl.DSL; +import org.reactivestreams.Publisher; + +import reactor.core.publisher.Mono; + +public class PostgresHealthCheck implements HealthCheck { + public static final ComponentName COMPONENT_NAME = new ComponentName("Postgres"); + private final PostgresExecutor postgresExecutor; + + @Inject + public PostgresHealthCheck(PostgresExecutor postgresExecutor) { + this.postgresExecutor = postgresExecutor; + } + + @Override + public ComponentName componentName() { + return COMPONENT_NAME; + } + + @Override + public Publisher check() { + return postgresExecutor.executeRow(context -> Mono.from(context.select(DSL.now()))) + .timeout(Duration.ofSeconds(5)) + .map(any -> Result.healthy(COMPONENT_NAME)) + .onErrorResume(e -> Mono.just(Result.unhealthy(COMPONENT_NAME, "Failed to execute request against postgres", e))); + } +} diff --git a/backends-common/postgres/src/main/java/org/apache/james/backends/postgres/utils/PostgresUtils.java b/backends-common/postgres/src/main/java/org/apache/james/backends/postgres/utils/PostgresUtils.java new file mode 100644 index 00000000000..9a6310e00ed --- /dev/null +++ b/backends-common/postgres/src/main/java/org/apache/james/backends/postgres/utils/PostgresUtils.java @@ -0,0 +1,43 @@ +/**************************************************************** + * 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.backends.postgres.utils; + +import java.util.Optional; +import java.util.function.Predicate; + +import org.jooq.exception.DataAccessException; + +import com.google.common.base.Preconditions; + +public class PostgresUtils { + private static final String UNIQUE_CONSTRAINT_VIOLATION_MESSAGE = "duplicate key value violates unique constraint"; + + public static final Predicate UNIQUE_CONSTRAINT_VIOLATION_PREDICATE = + throwable -> throwable instanceof DataAccessException && throwable.getMessage().contains(UNIQUE_CONSTRAINT_VIOLATION_MESSAGE); + + public static final int QUERY_BATCH_SIZE_DEFAULT_VALUE = 5000; + public static final int QUERY_BATCH_SIZE = Optional.ofNullable(System.getProperty("james.postgresql.query.batch.size")) + .map(Integer::valueOf) + .map(batchSize -> { + Preconditions.checkArgument(batchSize > 0, "james.postgresql.query.batch.size must be positive"); + return batchSize; + }) + .orElse(QUERY_BATCH_SIZE_DEFAULT_VALUE); +} diff --git a/backends-common/postgres/src/test/java/org/apache/james/backends/postgres/DockerPostgresSingleton.java b/backends-common/postgres/src/test/java/org/apache/james/backends/postgres/DockerPostgresSingleton.java new file mode 100644 index 00000000000..d51fa296752 --- /dev/null +++ b/backends-common/postgres/src/test/java/org/apache/james/backends/postgres/DockerPostgresSingleton.java @@ -0,0 +1,39 @@ +/**************************************************************** + * 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.backends.postgres; + +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; +import org.testcontainers.containers.PostgreSQLContainer; +import org.testcontainers.containers.output.OutputFrame; + +public class DockerPostgresSingleton { + private static void displayDockerLog(OutputFrame outputFrame) { + LOGGER.info(outputFrame.getUtf8String().trim()); + } + + private static final Logger LOGGER = LoggerFactory.getLogger(DockerPostgresSingleton.class); + public static final PostgreSQLContainer SINGLETON = PostgresFixture.PG_CONTAINER.get() + .withLogConsumer(DockerPostgresSingleton::displayDockerLog); + + static { + SINGLETON.start(); + } +} diff --git a/backends-common/postgres/src/test/java/org/apache/james/backends/postgres/JamesPostgresConnectionFactoryTest.java b/backends-common/postgres/src/test/java/org/apache/james/backends/postgres/JamesPostgresConnectionFactoryTest.java new file mode 100644 index 00000000000..6d27f26ca9d --- /dev/null +++ b/backends-common/postgres/src/test/java/org/apache/james/backends/postgres/JamesPostgresConnectionFactoryTest.java @@ -0,0 +1,78 @@ +/**************************************************************** + * 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.backends.postgres; + +import static org.assertj.core.api.Assertions.assertThat; + +import org.apache.james.backends.postgres.utils.JamesPostgresConnectionFactory; +import org.apache.james.core.Domain; +import org.junit.jupiter.api.Test; + +import com.google.common.collect.ImmutableList; + +import io.r2dbc.spi.Connection; +import reactor.core.publisher.Flux; + +public abstract class JamesPostgresConnectionFactoryTest { + + abstract JamesPostgresConnectionFactory jamesPostgresConnectionFactory(); + + @Test + void getConnectionShouldWork() { + Connection connection = jamesPostgresConnectionFactory().getConnection().block(); + String actual = Flux.from(connection.createStatement("SELECT 1") + .execute()) + .flatMap(result -> result.map((row, rowMetadata) -> row.get(0, String.class))) + .collect(ImmutableList.toImmutableList()) + .block().get(0); + + assertThat(actual).isEqualTo("1"); + } + + @Test + void getConnectionWithDomainShouldWork() { + Connection connection = jamesPostgresConnectionFactory().getConnection(Domain.of("james")).block(); + String actual = Flux.from(connection.createStatement("SELECT 1") + .execute()) + .flatMap(result -> result.map((row, rowMetadata) -> row.get(0, String.class))) + .collect(ImmutableList.toImmutableList()) + .block().get(0); + + assertThat(actual).isEqualTo("1"); + } + + @Test + void getConnectionShouldSetCurrentDomainAttribute() { + Domain domain = Domain.of("james"); + Connection connection = jamesPostgresConnectionFactory().getConnection(domain).block(); + String actual = getDomainAttributeValue(connection); + + assertThat(actual).isEqualTo(domain.asString()); + } + + String getDomainAttributeValue(Connection connection) { + return Flux.from(connection.createStatement("show " + JamesPostgresConnectionFactory.DOMAIN_ATTRIBUTE) + .execute()) + .flatMap(result -> result.map((row, rowMetadata) -> row.get(0, String.class))) + .collect(ImmutableList.toImmutableList()) + .block().get(0); + } + +} diff --git a/backends-common/postgres/src/test/java/org/apache/james/backends/postgres/PoolBackedPostgresConnectionFactoryTest.java b/backends-common/postgres/src/test/java/org/apache/james/backends/postgres/PoolBackedPostgresConnectionFactoryTest.java new file mode 100644 index 00000000000..4e4cb45b7f0 --- /dev/null +++ b/backends-common/postgres/src/test/java/org/apache/james/backends/postgres/PoolBackedPostgresConnectionFactoryTest.java @@ -0,0 +1,34 @@ +/**************************************************************** + * 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.backends.postgres; + +import org.apache.james.backends.postgres.utils.JamesPostgresConnectionFactory; +import org.apache.james.backends.postgres.utils.PoolBackedPostgresConnectionFactory; +import org.junit.jupiter.api.extension.RegisterExtension; + +public class PoolBackedPostgresConnectionFactoryTest extends JamesPostgresConnectionFactoryTest { + @RegisterExtension + static PostgresExtension postgresExtension = PostgresExtension.empty(); + + @Override + JamesPostgresConnectionFactory jamesPostgresConnectionFactory() { + return new PoolBackedPostgresConnectionFactory(RowLevelSecurity.ENABLED, postgresExtension.getConnectionFactory()); + } +} diff --git a/backends-common/postgres/src/test/java/org/apache/james/backends/postgres/PostgresConfigurationTest.java b/backends-common/postgres/src/test/java/org/apache/james/backends/postgres/PostgresConfigurationTest.java new file mode 100644 index 00000000000..08d76a23569 --- /dev/null +++ b/backends-common/postgres/src/test/java/org/apache/james/backends/postgres/PostgresConfigurationTest.java @@ -0,0 +1,125 @@ +/**************************************************************** + * 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.backends.postgres; + +import static org.assertj.core.api.Assertions.assertThat; +import static org.assertj.core.api.Assertions.assertThatThrownBy; + +import org.junit.jupiter.api.Test; + +import io.r2dbc.postgresql.client.SSLMode; + +class PostgresConfigurationTest { + + @Test + void shouldReturnCorrespondingProperties() { + PostgresConfiguration configuration = PostgresConfiguration.builder() + .host("1.1.1.1") + .port(1111) + .databaseName("db") + .databaseSchema("sc") + .username("james") + .password("1") + .byPassRLSUser("bypassrlsjames") + .byPassRLSPassword("2") + .rowLevelSecurityEnabled() + .sslMode("require") + .build(); + + assertThat(configuration.getHost()).isEqualTo("1.1.1.1"); + assertThat(configuration.getPort()).isEqualTo(1111); + assertThat(configuration.getDatabaseName()).isEqualTo("db"); + assertThat(configuration.getDatabaseSchema()).isEqualTo("sc"); + assertThat(configuration.getDefaultCredential().getUsername()).isEqualTo("james"); + assertThat(configuration.getDefaultCredential().getPassword()).isEqualTo("1"); + assertThat(configuration.getByPassRLSCredential().getUsername()).isEqualTo("bypassrlsjames"); + assertThat(configuration.getByPassRLSCredential().getPassword()).isEqualTo("2"); + assertThat(configuration.getRowLevelSecurity()).isEqualTo(RowLevelSecurity.ENABLED); + assertThat(configuration.getSslMode()).isEqualTo(SSLMode.REQUIRE); + } + + @Test + void shouldUseDefaultValues() { + PostgresConfiguration configuration = PostgresConfiguration.builder() + .username("james") + .password("1") + .build(); + + assertThat(configuration.getHost()).isEqualTo(PostgresConfiguration.HOST_DEFAULT_VALUE); + assertThat(configuration.getPort()).isEqualTo(PostgresConfiguration.PORT_DEFAULT_VALUE); + assertThat(configuration.getDatabaseName()).isEqualTo(PostgresConfiguration.DATABASE_NAME_DEFAULT_VALUE); + assertThat(configuration.getDatabaseSchema()).isEqualTo(PostgresConfiguration.DATABASE_SCHEMA_DEFAULT_VALUE); + assertThat(configuration.getByPassRLSCredential().getUsername()).isEqualTo("james"); + assertThat(configuration.getByPassRLSCredential().getPassword()).isEqualTo("1"); + assertThat(configuration.getRowLevelSecurity()).isEqualTo(RowLevelSecurity.DISABLED); + assertThat(configuration.getSslMode()).isEqualTo(SSLMode.ALLOW); + } + + @Test + void shouldThrowWhenMissingUsername() { + assertThatThrownBy(() -> PostgresConfiguration.builder() + .build()) + .isInstanceOf(IllegalArgumentException.class) + .hasMessage("You need to specify username"); + } + + @Test + void shouldThrowWhenMissingPassword() { + assertThatThrownBy(() -> PostgresConfiguration.builder() + .username("james") + .build()) + .isInstanceOf(IllegalArgumentException.class) + .hasMessage("You need to specify password"); + } + + @Test + void shouldThrowWhenMissingByPassRLSUserAndRLSIsEnabled() { + assertThatThrownBy(() -> PostgresConfiguration.builder() + .username("james") + .password("1") + .rowLevelSecurityEnabled() + .build()) + .isInstanceOf(IllegalArgumentException.class) + .hasMessage("You need to specify byPassRLSUser"); + } + + @Test + void shouldThrowWhenMissingByPassRLSPasswordAndRLSIsEnabled() { + assertThatThrownBy(() -> PostgresConfiguration.builder() + .username("james") + .password("1") + .byPassRLSUser("bypassrlsjames") + .rowLevelSecurityEnabled() + .build()) + .isInstanceOf(IllegalArgumentException.class) + .hasMessage("You need to specify byPassRLSPassword"); + } + + @Test + void shouldThrowWhenInvalidSslMode() { + assertThatThrownBy(() -> PostgresConfiguration.builder() + .username("james") + .password("1") + .sslMode("invalid") + .build()) + .isInstanceOf(IllegalArgumentException.class) + .hasMessage("Invalid ssl mode value: invalid"); + } +} diff --git a/backends-common/postgres/src/test/java/org/apache/james/backends/postgres/PostgresExecutorThreadSafetyTest.java b/backends-common/postgres/src/test/java/org/apache/james/backends/postgres/PostgresExecutorThreadSafetyTest.java new file mode 100644 index 00000000000..da1ada6db15 --- /dev/null +++ b/backends-common/postgres/src/test/java/org/apache/james/backends/postgres/PostgresExecutorThreadSafetyTest.java @@ -0,0 +1,202 @@ +/**************************************************************** + * 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.backends.postgres; + +import static org.assertj.core.api.Assertions.assertThat; + +import java.time.Duration; +import java.util.List; +import java.util.Set; +import java.util.Vector; +import java.util.concurrent.atomic.AtomicInteger; +import java.util.function.Function; +import java.util.stream.Stream; + +import org.apache.james.backends.postgres.utils.PostgresExecutor; +import org.apache.james.util.concurrency.ConcurrentTestRunner; +import org.jooq.Record; +import org.jooq.impl.DSL; +import org.jooq.impl.SQLDataType; +import org.junit.jupiter.api.AfterEach; +import org.junit.jupiter.api.BeforeAll; +import org.junit.jupiter.api.BeforeEach; +import org.junit.jupiter.api.Test; +import org.junit.jupiter.api.extension.RegisterExtension; + +import com.google.common.collect.ImmutableSet; + +import reactor.core.publisher.Flux; +import reactor.core.publisher.Mono; + +class PostgresExecutorThreadSafetyTest { + static final int NUMBER_OF_THREAD = 100; + + @RegisterExtension + static PostgresExtension postgresExtension = PostgresExtension.empty(); + + private static PostgresExecutor postgresExecutor; + + @BeforeAll + static void beforeAll() { + postgresExecutor = postgresExtension.getDefaultPostgresExecutor(); + } + + @BeforeEach + void beforeEach() { + postgresExecutor.executeVoid(dslContext -> Mono.from(dslContext.createTableIfNotExists("person") + .column("id", SQLDataType.INTEGER.identity(true)) + .column("name", SQLDataType.VARCHAR(50).nullable(false)) + .constraints(DSL.constraint().primaryKey("id")) + .unique("name"))) + .block(); + } + + @AfterEach + void afterEach() { + postgresExecutor.executeVoid(dslContext -> Mono.from(dslContext.dropTableIfExists("person"))) + .block(); + } + + @Test + void postgresExecutorShouldWorkWellWhenItIsUsedByMultipleThreadsAndAllQueriesAreSelect() throws Exception { + provisionData(NUMBER_OF_THREAD); + + List actual = new Vector<>(); + ConcurrentTestRunner.builder() + .reactorOperation((threadNumber, step) -> getData(threadNumber) + .doOnNext(actual::add) + .then()) + .threadCount(NUMBER_OF_THREAD) + .operationCount(1) + .runSuccessfullyWithin(Duration.ofMinutes(1)); + + Set expected = Stream.iterate(0, i -> i + 1).limit(NUMBER_OF_THREAD).map(i -> i + "|Peter" + i).collect(ImmutableSet.toImmutableSet()); + + assertThat(actual).containsExactlyInAnyOrderElementsOf(expected); + } + + @Test + void postgresExecutorShouldWorkWellWhenItIsUsedByMultipleThreadsAndAllQueriesAreInsert() throws Exception { + ConcurrentTestRunner.builder() + .reactorOperation((threadNumber, step) -> createData(threadNumber)) + .threadCount(NUMBER_OF_THREAD) + .operationCount(1) + .runSuccessfullyWithin(Duration.ofMinutes(1)); + + List actual = getData(0, NUMBER_OF_THREAD); + Set expected = Stream.iterate(0, i -> i + 1).limit(NUMBER_OF_THREAD).map(i -> i + "|Peter" + i).collect(ImmutableSet.toImmutableSet()); + + assertThat(actual).containsExactlyInAnyOrderElementsOf(expected); + } + + @Test + void postgresExecutorShouldWorkWellWhenItIsUsedByMultipleThreadsAndInsertQueriesAreDuplicated() throws Exception { + AtomicInteger numberOfSuccess = new AtomicInteger(0); + AtomicInteger numberOfFail = new AtomicInteger(0); + ConcurrentTestRunner.builder() + .reactorOperation((threadNumber, step) -> createData(threadNumber % 10) + .then(Mono.fromCallable(numberOfSuccess::incrementAndGet)) + .then() + .onErrorResume(throwable -> { + if (throwable.getMessage().contains("duplicate key value violates unique constraint")) { + numberOfFail.incrementAndGet(); + } + return Mono.empty(); + })) + .threadCount(100) + .operationCount(1) + .runSuccessfullyWithin(Duration.ofMinutes(1)); + + List actual = getData(0, 100); + Set expected = Stream.iterate(0, i -> i + 1).limit(10).map(i -> i + "|Peter" + i).collect(ImmutableSet.toImmutableSet()); + + assertThat(actual).containsExactlyInAnyOrderElementsOf(expected); + assertThat(numberOfSuccess.get()).isEqualTo(10); + assertThat(numberOfFail.get()).isEqualTo(90); + } + + @Test + void postgresExecutorShouldWorkWellWhenItIsUsedByMultipleThreadsAndQueriesIncludeBothSelectAndInsert() throws Exception { + provisionData(50); + + List actualSelect = new Vector<>(); + ConcurrentTestRunner.builder() + .reactorOperation((threadNumber, step) -> { + if (threadNumber < 50) { + return getData(threadNumber) + .doOnNext(actualSelect::add) + .then(); + } else { + return createData(threadNumber); + } + }) + .threadCount(NUMBER_OF_THREAD) + .operationCount(1) + .runSuccessfullyWithin(Duration.ofMinutes(1)); + + List actualInsert = getData(50, 100); + + Set expectedSelect = Stream.iterate(0, i -> i + 1).limit(50).map(i -> i + "|Peter" + i).collect(ImmutableSet.toImmutableSet()); + Set expectedInsert = Stream.iterate(50, i -> i + 1).limit(50).map(i -> i + "|Peter" + i).collect(ImmutableSet.toImmutableSet()); + + assertThat(actualSelect).containsExactlyInAnyOrderElementsOf(expectedSelect); + assertThat(actualInsert).containsExactlyInAnyOrderElementsOf(expectedInsert); + } + + public Flux getData(int threadNumber) { + return postgresExecutor.executeRows(dslContext -> Flux.from(dslContext + .select(DSL.field("id"), DSL.field("name")) + .from(DSL.table("person")) + .where(DSL.field("id").eq(threadNumber)))) + .map(recordToString()); + } + + public Mono createData(int threadNumber) { + return postgresExecutor.executeVoid(dslContext -> Mono.from(dslContext + .insertInto(DSL.table("person"), DSL.field("id"), DSL.field("name")) + .values(threadNumber, "Peter" + threadNumber))); + } + + private List getData(int lowerBound, int upperBound) { + return postgresExecutor.executeRows(dslContext -> Flux.from(dslContext + .select(DSL.field("id"), DSL.field("name")) + .from(DSL.table("person")) + .where(DSL.field("id").greaterOrEqual(lowerBound).and(DSL.field("id").lessThan(upperBound))))) + .map(recordToString()) + .collectList() + .block(); + } + + private void provisionData(int upperBound) { + Flux.range(0, upperBound) + .flatMap(i -> insertPerson(i, "Peter" + i)) + .then() + .block(); + } + + private Mono insertPerson(int id, String name) { + return postgresExecutor.executeVoid(dslContext -> Mono.from(dslContext.insertInto(DSL.table("person"), DSL.field("id"), DSL.field("name")) + .values(id, name))); + } + + private Function recordToString() { + return record -> record.get(DSL.field("id", Long.class)) + "|" + record.get(DSL.field("name", String.class)); + } +} diff --git a/backends-common/postgres/src/test/java/org/apache/james/backends/postgres/PostgresExtension.java b/backends-common/postgres/src/test/java/org/apache/james/backends/postgres/PostgresExtension.java new file mode 100644 index 00000000000..85159832a61 --- /dev/null +++ b/backends-common/postgres/src/test/java/org/apache/james/backends/postgres/PostgresExtension.java @@ -0,0 +1,292 @@ +/**************************************************************** + * 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.backends.postgres; + +import static org.apache.james.backends.postgres.PostgresFixture.Database.DEFAULT_DATABASE; +import static org.apache.james.backends.postgres.PostgresFixture.Database.ROW_LEVEL_SECURITY_DATABASE; + +import java.io.IOException; +import java.time.Duration; +import java.util.List; +import java.util.Optional; +import java.util.function.Function; +import java.util.stream.Collectors; + +import org.apache.james.GuiceModuleTestExtension; +import org.apache.james.backends.postgres.utils.JamesPostgresConnectionFactory; +import org.apache.james.backends.postgres.utils.PoolBackedPostgresConnectionFactory; +import org.apache.james.backends.postgres.utils.PostgresExecutor; +import org.apache.james.metrics.tests.RecordingMetricFactory; +import org.junit.jupiter.api.extension.ExtensionContext; +import org.testcontainers.containers.PostgreSQLContainer; + +import com.github.fge.lambdas.Throwing; +import com.google.inject.Module; +import com.google.inject.util.Modules; + +import io.r2dbc.postgresql.PostgresqlConnectionConfiguration; +import io.r2dbc.postgresql.PostgresqlConnectionFactory; +import io.r2dbc.spi.Connection; +import io.r2dbc.spi.ConnectionFactory; +import reactor.core.publisher.Flux; +import reactor.core.publisher.Mono; + +public class PostgresExtension implements GuiceModuleTestExtension { + public enum PoolSize { + SMALL(10, 20), + LARGE(20, 40); + + private final int min; + private final int max; + + PoolSize(int min, int max) { + this.min = min; + this.max = max; + } + + public int getMin() { + return min; + } + + public int getMax() { + return max; + } + } + + public static PostgresExtension withRowLevelSecurity(PostgresModule module) { + return new PostgresExtension(module, RowLevelSecurity.ENABLED); + } + + public static PostgresExtension withoutRowLevelSecurity(PostgresModule module) { + return withoutRowLevelSecurity(module, PoolSize.SMALL); + } + + public static PostgresExtension withoutRowLevelSecurity(PostgresModule module, PoolSize poolSize) { + return new PostgresExtension(module, RowLevelSecurity.DISABLED, Optional.of(poolSize)); + } + + public static PostgresExtension empty() { + return withoutRowLevelSecurity(PostgresModule.EMPTY_MODULE); + } + + public static final PoolSize DEFAULT_POOL_SIZE = PoolSize.SMALL; + public static PostgreSQLContainer PG_CONTAINER = DockerPostgresSingleton.SINGLETON; + private final PostgresModule postgresModule; + private final RowLevelSecurity rowLevelSecurity; + private final PostgresFixture.Database selectedDatabase; + private PoolSize poolSize; + private PostgresConfiguration postgresConfiguration; + private PostgresExecutor defaultPostgresExecutor; + private PostgresExecutor byPassRLSPostgresExecutor; + private PostgresqlConnectionFactory connectionFactory; + private Connection defaultConnection; + private PostgresExecutor.Factory executorFactory; + private PostgresTableManager postgresTableManager; + + public void pause() { + PG_CONTAINER.getDockerClient().pauseContainerCmd(PG_CONTAINER.getContainerId()) + .exec(); + } + + public void unpause() { + PG_CONTAINER.getDockerClient().unpauseContainerCmd(PG_CONTAINER.getContainerId()) + .exec(); + } + + private PostgresExtension(PostgresModule postgresModule, RowLevelSecurity rowLevelSecurity) { + this(postgresModule, rowLevelSecurity, Optional.empty()); + } + + private PostgresExtension(PostgresModule postgresModule, RowLevelSecurity rowLevelSecurity, Optional maybePoolSize) { + this.postgresModule = postgresModule; + this.rowLevelSecurity = rowLevelSecurity; + if (rowLevelSecurity.isRowLevelSecurityEnabled()) { + this.selectedDatabase = PostgresFixture.Database.ROW_LEVEL_SECURITY_DATABASE; + } else { + this.selectedDatabase = DEFAULT_DATABASE; + } + this.poolSize = maybePoolSize.orElse(DEFAULT_POOL_SIZE); + } + + @Override + public void beforeAll(ExtensionContext extensionContext) throws Exception { + if (!PG_CONTAINER.isRunning()) { + PG_CONTAINER.start(); + } + querySettingRowLevelSecurityIfNeed(); + querySettingExtension(); + initPostgresSession(); + } + + private void querySettingRowLevelSecurityIfNeed() { + if (rowLevelSecurity.isRowLevelSecurityEnabled()) { + Throwing.runnable(() -> { + PG_CONTAINER.execInContainer("psql", "-U", DEFAULT_DATABASE.dbUser(), "-c", "create user " + ROW_LEVEL_SECURITY_DATABASE.dbUser() + " WITH PASSWORD '" + ROW_LEVEL_SECURITY_DATABASE.dbPassword() + "';"); + PG_CONTAINER.execInContainer("psql", "-U", DEFAULT_DATABASE.dbUser(), "-c", "create database " + ROW_LEVEL_SECURITY_DATABASE.dbName() + ";"); + PG_CONTAINER.execInContainer("psql", "-U", DEFAULT_DATABASE.dbUser(), "-c", "grant all privileges on database " + ROW_LEVEL_SECURITY_DATABASE.dbName() + " to " + ROW_LEVEL_SECURITY_DATABASE.dbUser() + ";"); + PG_CONTAINER.execInContainer("psql", "-U", ROW_LEVEL_SECURITY_DATABASE.dbUser(), "-d", ROW_LEVEL_SECURITY_DATABASE.dbName(), "-c", "create schema if not exists " + ROW_LEVEL_SECURITY_DATABASE.schema() + ";"); + }).sneakyThrow().run(); + } + } + + private void querySettingExtension() throws IOException, InterruptedException { + PG_CONTAINER.execInContainer("psql", "-U", selectedDatabase.dbUser(), selectedDatabase.dbName(), "-c", String.format("CREATE EXTENSION IF NOT EXISTS hstore SCHEMA %s;", selectedDatabase.schema())); + } + + private void initPostgresSession() { + postgresConfiguration = PostgresConfiguration.builder() + .databaseName(selectedDatabase.dbName()) + .databaseSchema(selectedDatabase.schema()) + .host(getHost()) + .port(getMappedPort()) + .username(selectedDatabase.dbUser()) + .password(selectedDatabase.dbPassword()) + .byPassRLSUser(DEFAULT_DATABASE.dbUser()) + .byPassRLSPassword(DEFAULT_DATABASE.dbPassword()) + .rowLevelSecurityEnabled(rowLevelSecurity.isRowLevelSecurityEnabled()) + .jooqReactiveTimeout(Optional.of(Duration.ofSeconds(20L))) + .build(); + + Function postgresqlConnectionConfigurationFunction = credential -> + PostgresqlConnectionConfiguration.builder() + .host(postgresConfiguration.getHost()) + .port(postgresConfiguration.getPort()) + .database(postgresConfiguration.getDatabaseName()) + .schema(postgresConfiguration.getDatabaseSchema()) + .username(credential.getUsername()) + .password(credential.getPassword()) + .build(); + + RecordingMetricFactory metricFactory = new RecordingMetricFactory(); + + connectionFactory = new PostgresqlConnectionFactory(postgresqlConnectionConfigurationFunction.apply(postgresConfiguration.getDefaultCredential())); + defaultConnection = connectionFactory.create().block(); + executorFactory = new PostgresExecutor.Factory( + getJamesPostgresConnectionFactory(rowLevelSecurity, connectionFactory), + postgresConfiguration, + metricFactory); + + defaultPostgresExecutor = executorFactory.create(); + + PostgresqlConnectionFactory byPassRLSConnectionFactory = new PostgresqlConnectionFactory(postgresqlConnectionConfigurationFunction.apply(postgresConfiguration.getByPassRLSCredential())); + + byPassRLSPostgresExecutor = new PostgresExecutor.Factory( + getJamesPostgresConnectionFactory(RowLevelSecurity.DISABLED, byPassRLSConnectionFactory), + postgresConfiguration, + metricFactory) + .create(); + + this.postgresTableManager = new PostgresTableManager(defaultPostgresExecutor, postgresModule, rowLevelSecurity); + } + + @Override + public void afterAll(ExtensionContext extensionContext) { + disposePostgresSession(); + } + + private void disposePostgresSession() { + defaultPostgresExecutor.dispose(); + byPassRLSPostgresExecutor.dispose(); + Mono.from(defaultConnection.close()).subscribe(); + } + + @Override + public void beforeEach(ExtensionContext extensionContext) { + initTablesAndIndexes(); + } + + @Override + public void afterEach(ExtensionContext extensionContext) { + resetSchema(); + } + + public void restartContainer() { + PG_CONTAINER.stop(); + PG_CONTAINER.start(); + initPostgresSession(); + } + + @Override + public Module getModule() { + return Modules.combine(binder -> binder.bind(PostgresConfiguration.class) + .toInstance(postgresConfiguration)); + } + + public String getHost() { + return PG_CONTAINER.getHost(); + } + + public Integer getMappedPort() { + return PG_CONTAINER.getMappedPort(PostgresFixture.PORT); + } + + public Mono getConnection() { + return Mono.just(defaultConnection); + } + + public PostgresExecutor getDefaultPostgresExecutor() { + return defaultPostgresExecutor; + } + + public PostgresExecutor getByPassRLSPostgresExecutor() { + return byPassRLSPostgresExecutor; + } + + public ConnectionFactory getConnectionFactory() { + return connectionFactory; + } + + public PostgresExecutor.Factory getExecutorFactory() { + return executorFactory; + } + + public PostgresConfiguration getPostgresConfiguration() { + return postgresConfiguration; + } + + private void initTablesAndIndexes() { + postgresTableManager.initializeTables().block(); + postgresTableManager.initializeTableIndexes().block(); + } + + private void resetSchema() { + List tables = postgresTableManager.listExistTables().block(); + dropTables(tables); + } + + private void dropTables(List tables) { + String tablesToDelete = tables.stream() + .map(tableName -> "\"" + tableName + "\"") + .collect(Collectors.joining(", ")); + + Flux.from(defaultConnection.createStatement(String.format("DROP table if exists %s cascade;", tablesToDelete)) + .execute()) + .then() + .block(); + } + + private JamesPostgresConnectionFactory getJamesPostgresConnectionFactory(RowLevelSecurity rowLevelSecurity, PostgresqlConnectionFactory connectionFactory) { + return new PoolBackedPostgresConnectionFactory( + rowLevelSecurity, + poolSize.getMin(), + poolSize.getMax(), + connectionFactory); + } +} diff --git a/backends-common/postgres/src/test/java/org/apache/james/backends/postgres/PostgresExtensionTest.java b/backends-common/postgres/src/test/java/org/apache/james/backends/postgres/PostgresExtensionTest.java new file mode 100644 index 00000000000..619899ed179 --- /dev/null +++ b/backends-common/postgres/src/test/java/org/apache/james/backends/postgres/PostgresExtensionTest.java @@ -0,0 +1,104 @@ +/**************************************************************** + * 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.backends.postgres; + +import static org.assertj.core.api.Assertions.assertThat; + +import java.util.List; + +import org.apache.commons.lang3.tuple.Pair; +import org.jooq.impl.DSL; +import org.jooq.impl.SQLDataType; +import org.junit.jupiter.api.Test; +import org.junit.jupiter.api.extension.RegisterExtension; + +import reactor.core.publisher.Flux; +import reactor.core.publisher.Mono; + +class PostgresExtensionTest { + static PostgresTable TABLE_1 = PostgresTable.name("table1") + .createTableStep((dslContext, tableName) -> dslContext.createTable(tableName) + .column("column1", SQLDataType.UUID.notNull()) + .column("column2", SQLDataType.INTEGER) + .column("column3", SQLDataType.VARCHAR(255).notNull())) + .disableRowLevelSecurity() + .build(); + + static PostgresIndex INDEX_1 = PostgresIndex.name("index1") + .createIndexStep((dslContext, indexName) -> dslContext.createIndex(indexName) + .on(DSL.table("table1"), DSL.field("column1").asc())); + + static PostgresTable TABLE_2 = PostgresTable.name("table2") + .createTableStep((dslContext, tableName) -> dslContext.createTable(tableName) + .column("column1", SQLDataType.INTEGER)) + .disableRowLevelSecurity() + .build(); + + static PostgresIndex INDEX_2 = PostgresIndex.name("index2") + .createIndexStep((dslContext, indexName) -> dslContext.createIndex(indexName) + .on(DSL.table("table2"), DSL.field("column1").desc())); + + static PostgresModule POSTGRES_MODULE = PostgresModule.builder() + .addTable(TABLE_1, TABLE_2) + .addIndex(INDEX_1, INDEX_2) + .build(); + + @RegisterExtension + static PostgresExtension postgresExtension = PostgresExtension.withoutRowLevelSecurity(POSTGRES_MODULE); + + @Test + void postgresExtensionShouldProvisionTablesAndIndexes() { + assertThat(getColumnNameAndDataType("table1")) + .containsExactlyInAnyOrder( + Pair.of("column1", "uuid"), + Pair.of("column2", "integer"), + Pair.of("column3", "character varying")); + + assertThat(getColumnNameAndDataType("table2")) + .containsExactlyInAnyOrder(Pair.of("column1", "integer")); + + assertThat(listIndexToTableMappings()) + .contains( + Pair.of("index1", "table1"), + Pair.of("index2", "table2")); + } + + private List> getColumnNameAndDataType(String tableName) { + return postgresExtension.getConnection() + .flatMapMany(connection -> Flux.from(Mono.from(connection.createStatement("SELECT table_name, column_name, data_type FROM information_schema.columns WHERE table_name = $1;") + .bind("$1", tableName) + .execute()) + .flatMapMany(result -> result.map((row, rowMetadata) -> + Pair.of(row.get("column_name", String.class), row.get("data_type", String.class)))))) + .collectList() + .block(); + } + + private List> listIndexToTableMappings() { + return postgresExtension.getConnection() + .flatMapMany(connection -> Mono.from(connection.createStatement("SELECT indexname, tablename FROM pg_indexes;") + .execute()) + .flatMapMany(result -> + result.map((row, rowMetadata) -> + Pair.of(row.get("indexname", String.class), row.get("tablename", String.class))))) + .collectList() + .block(); + } +} diff --git a/backends-common/postgres/src/test/java/org/apache/james/backends/postgres/PostgresFixture.java b/backends-common/postgres/src/test/java/org/apache/james/backends/postgres/PostgresFixture.java new file mode 100644 index 00000000000..c0c28758e75 --- /dev/null +++ b/backends-common/postgres/src/test/java/org/apache/james/backends/postgres/PostgresFixture.java @@ -0,0 +1,100 @@ +/**************************************************************** + * 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.backends.postgres; + +import static java.util.Collections.singletonMap; +import static org.apache.james.backends.postgres.PostgresFixture.Database.DEFAULT_DATABASE; +import static org.testcontainers.containers.PostgreSQLContainer.POSTGRESQL_PORT; + +import java.util.UUID; +import java.util.function.Supplier; + +import org.testcontainers.containers.PostgreSQLContainer; + +public interface PostgresFixture { + + interface Database { + + Database DEFAULT_DATABASE = new DefaultDatabase(); + Database ROW_LEVEL_SECURITY_DATABASE = new RowLevelSecurityDatabase(); + + String dbUser(); + + String dbPassword(); + + String dbName(); + + String schema(); + + + class DefaultDatabase implements Database { + @Override + public String dbUser() { + return "james"; + } + + @Override + public String dbPassword() { + return "secret1"; + } + + @Override + public String dbName() { + return "james"; + } + + @Override + public String schema() { + return "public"; + } + } + + class RowLevelSecurityDatabase implements Database { + @Override + public String dbUser() { + return "rlsuser"; + } + + @Override + public String dbPassword() { + return "secret1"; + } + + @Override + public String dbName() { + return "rlsdb"; + } + + @Override + public String schema() { + return "rlsschema"; + } + } + } + + String IMAGE = "postgres:16.3"; + Integer PORT = POSTGRESQL_PORT; + Supplier> PG_CONTAINER = () -> new PostgreSQLContainer<>(IMAGE) + .withDatabaseName(DEFAULT_DATABASE.dbName()) + .withUsername(DEFAULT_DATABASE.dbUser()) + .withPassword(DEFAULT_DATABASE.dbPassword()) + .withCreateContainerCmdModifier(cmd -> cmd.withName("james-postgres-test-" + UUID.randomUUID())) + .withTmpFs(singletonMap("/var/lib/postgresql/data", "rw")); +} diff --git a/backends-common/postgres/src/test/java/org/apache/james/backends/postgres/PostgresTableManagerTest.java b/backends-common/postgres/src/test/java/org/apache/james/backends/postgres/PostgresTableManagerTest.java new file mode 100644 index 00000000000..2980885fd8b --- /dev/null +++ b/backends-common/postgres/src/test/java/org/apache/james/backends/postgres/PostgresTableManagerTest.java @@ -0,0 +1,492 @@ +/**************************************************************** + * 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.backends.postgres; + +import static org.assertj.core.api.Assertions.assertThat; +import static org.assertj.core.api.Assertions.assertThatCode; + +import java.util.List; +import java.util.function.Function; +import java.util.function.Supplier; + +import org.apache.commons.lang3.tuple.Pair; +import org.jooq.impl.DSL; +import org.jooq.impl.SQLDataType; +import org.junit.jupiter.api.Test; +import org.junit.jupiter.api.extension.RegisterExtension; + +import reactor.core.publisher.Flux; +import reactor.core.publisher.Mono; + +class PostgresTableManagerTest { + + @RegisterExtension + static PostgresExtension postgresExtension = PostgresExtension.withRowLevelSecurity(PostgresModule.EMPTY_MODULE); + + Function tableManagerFactory = + module -> new PostgresTableManager(postgresExtension.getDefaultPostgresExecutor(), module, RowLevelSecurity.ENABLED); + + @Test + void initializeTableShouldSuccessWhenModuleHasSingleTable() { + String tableName = "tablename1"; + + PostgresTable table = PostgresTable.name(tableName) + .createTableStep((dsl, tbn) -> dsl.createTable(tbn) + .column("colum1", SQLDataType.UUID.notNull()) + .column("colum2", SQLDataType.INTEGER) + .column("colum3", SQLDataType.VARCHAR(255).notNull())) + .disableRowLevelSecurity() + .build(); + + PostgresModule module = PostgresModule.table(table); + + PostgresTableManager testee = tableManagerFactory.apply(module); + + testee.initializeTables() + .block(); + + assertThat(getColumnNameAndDataType(tableName)) + .containsExactlyInAnyOrder( + Pair.of("colum1", "uuid"), + Pair.of("colum2", "integer"), + Pair.of("colum3", "character varying")); + } + + @Test + void initializeTableShouldSuccessWhenModuleHasMultiTables() { + String tableName1 = "tablename1"; + + PostgresTable table1 = PostgresTable.name(tableName1) + .createTableStep((dsl, tbn) -> dsl.createTable(tbn) + .column("columA", SQLDataType.UUID.notNull())).disableRowLevelSecurity() + .build(); + + String tableName2 = "tablename2"; + PostgresTable table2 = PostgresTable.name(tableName2) + .createTableStep((dsl, tbn) -> dsl.createTable(tbn) + .column("columB", SQLDataType.INTEGER)).disableRowLevelSecurity() + .build(); + + PostgresTableManager testee = tableManagerFactory.apply(PostgresModule.table(table1, table2)); + + testee.initializeTables() + .block(); + + assertThat(getColumnNameAndDataType(tableName1)) + .containsExactlyInAnyOrder( + Pair.of("columA", "uuid")); + assertThat(getColumnNameAndDataType(tableName2)) + .containsExactlyInAnyOrder( + Pair.of("columB", "integer")); + } + + @Test + void initializeTableShouldNotThrowWhenTableExists() { + String tableName1 = "tablename1"; + + PostgresTable table1 = PostgresTable.name(tableName1) + .createTableStep((dsl, tbn) -> dsl.createTable(tbn) + .column("columA", SQLDataType.UUID.notNull())).disableRowLevelSecurity() + .build(); + + PostgresTableManager testee = tableManagerFactory.apply(PostgresModule.table(table1)); + + testee.initializeTables() + .block(); + + assertThatCode(() -> testee.initializeTables().block()) + .doesNotThrowAnyException(); + } + + @Test + void initializeTableShouldNotChangeTableStructureOfExistTable() { + String tableName1 = "tablename1"; + PostgresTable table1 = PostgresTable.name(tableName1) + .createTableStep((dsl, tbn) -> dsl.createTable(tbn) + .column("columA", SQLDataType.UUID.notNull())).disableRowLevelSecurity() + .build(); + + tableManagerFactory.apply(PostgresModule.table(table1)) + .initializeTables() + .block(); + + PostgresTable table1Changed = PostgresTable.name(tableName1) + .createTableStep((dsl, tbn) -> dsl.createTable(tbn) + .column("columB", SQLDataType.INTEGER)).disableRowLevelSecurity() + .build(); + + tableManagerFactory.apply(PostgresModule.table(table1Changed)) + .initializeTables() + .block(); + + assertThat(getColumnNameAndDataType(tableName1)) + .containsExactlyInAnyOrder( + Pair.of("columA", "uuid")); + } + + @Test + void initializeIndexShouldSuccessWhenModuleHasSingleIndex() { + String tableName = "tb_test_1"; + + PostgresTable table = PostgresTable.name(tableName) + .createTableStep((dsl, tbn) -> dsl.createTable(tbn) + .column("colum1", SQLDataType.UUID.notNull()) + .column("colum2", SQLDataType.INTEGER) + .column("colum3", SQLDataType.VARCHAR(255).notNull())) + .disableRowLevelSecurity() + .build(); + + String indexName = "idx_test_1"; + PostgresIndex index = PostgresIndex.name(indexName) + .createIndexStep((dsl, idn) -> dsl.createIndex(idn) + .on(DSL.table(tableName), DSL.field("colum1").asc())); + + PostgresModule module = PostgresModule.builder() + .addTable(table) + .addIndex(index) + .build(); + + PostgresTableManager testee = tableManagerFactory.apply(module); + + testee.initializeTables().block(); + + testee.initializeTableIndexes().block(); + + List> listIndexes = listIndexToTableMappings(); + + assertThat(listIndexes) + .contains(Pair.of(indexName, tableName)); + } + + @Test + void initializeIndexShouldSuccessWhenModuleHasMultiIndexes() { + String tableName = "tb_test_1"; + + PostgresTable table = PostgresTable.name(tableName) + .createTableStep((dsl, tbn) -> dsl.createTable(tbn) + .column("colum1", SQLDataType.UUID.notNull()) + .column("colum2", SQLDataType.INTEGER) + .column("colum3", SQLDataType.VARCHAR(255).notNull())) + .disableRowLevelSecurity() + .build(); + + String indexName1 = "idx_test_1"; + PostgresIndex index1 = PostgresIndex.name(indexName1) + .createIndexStep((dsl, idn) -> dsl.createIndex(idn) + .on(DSL.table(tableName), DSL.field("colum1").asc())); + + String indexName2 = "idx_test_2"; + PostgresIndex index2 = PostgresIndex.name(indexName2) + .createIndexStep((dsl, idn) -> dsl.createIndex(idn) + .on(DSL.table(tableName), DSL.field("colum2").desc())); + + PostgresModule module = PostgresModule.builder() + .addTable(table) + .addIndex(index1, index2) + .build(); + + PostgresTableManager testee = tableManagerFactory.apply(module); + + testee.initializeTables().block(); + + testee.initializeTableIndexes().block(); + + List> listIndexes = listIndexToTableMappings(); + + assertThat(listIndexes) + .contains(Pair.of(indexName1, tableName), Pair.of(indexName2, tableName)); + } + + @Test + void initializeIndexShouldNotThrowWhenIndexExists() { + String tableName = "tb_test_1"; + + PostgresTable table = PostgresTable.name(tableName) + .createTableStep((dsl, tbn) -> dsl.createTable(tbn) + .column("colum1", SQLDataType.UUID.notNull()) + .column("colum2", SQLDataType.INTEGER) + .column("colum3", SQLDataType.VARCHAR(255).notNull())) + .disableRowLevelSecurity() + .build(); + + String indexName = "idx_test_1"; + PostgresIndex index = PostgresIndex.name(indexName) + .createIndexStep((dsl, idn) -> dsl.createIndex(idn) + .on(DSL.table(tableName), DSL.field("colum1").asc())); + + PostgresModule module = PostgresModule.builder() + .addTable(table) + .addIndex(index) + .build(); + + PostgresTableManager testee = tableManagerFactory.apply(module); + + testee.initializeTables().block(); + + testee.initializeTableIndexes().block(); + + assertThatCode(() -> testee.initializeTableIndexes().block()) + .doesNotThrowAnyException(); + } + + @Test + void truncateShouldEmptyTableData() { + // Given table tbn1 + String tableName1 = "tbn1"; + PostgresTable table1 = PostgresTable.name(tableName1) + .createTableStep((dsl, tbn) -> dsl.createTable(tbn) + .column("column1", SQLDataType.INTEGER.notNull())).disableRowLevelSecurity() + .build(); + + PostgresTableManager testee = tableManagerFactory.apply(PostgresModule.table(table1)); + testee.initializeTables() + .block(); + + // insert data + postgresExtension.getConnection() + .flatMapMany(connection -> Flux.range(0, 10) + .flatMap(i -> Mono.from(connection.createStatement("INSERT INTO " + tableName1 + " (column1) VALUES ($1);") + .bind("$1", i) + .execute()) + .flatMap(result -> Mono.from(result.getRowsUpdated()))) + .last()) + .collectList() + .block(); + + Supplier getTotalRecordInDB = () -> postgresExtension.getConnection() + .flatMapMany(connection -> Mono.from(connection.createStatement("select count(*) FROM " + tableName1) + .execute()) + .flatMapMany(result -> + result.map((row, rowMetadata) -> row.get("count", Long.class)))) + .last() + .block(); + + assertThat(getTotalRecordInDB.get()).isEqualTo(10L); + + // When truncate table + testee.truncate().block(); + + // Then table is empty + assertThat(getTotalRecordInDB.get()).isEqualTo(0L); + } + + @Test + void createTableShouldCreateRlsColumnWhenEnableRLS() { + String tableName = "tbn1"; + + PostgresTable table = PostgresTable.name(tableName) + .createTableStep((dsl, tbn) -> dsl.createTable(tbn) + .column("clm1", SQLDataType.UUID.notNull()) + .column("clm2", SQLDataType.VARCHAR(255).notNull())) + .supportsRowLevelSecurity() + .build(); + + PostgresModule module = PostgresModule.table(table); + + PostgresTableManager testee = tableManagerFactory.apply(module); + + testee.initializeTables() + .block(); + + assertThat(getColumnNameAndDataType(tableName)) + .containsExactlyInAnyOrder( + Pair.of("clm1", "uuid"), + Pair.of("clm2", "character varying"), + Pair.of("domain", "character varying")); + + List> pgClassCheckResult = postgresExtension.getConnection() + .flatMapMany(connection -> Mono.from(connection.createStatement("select relname, relrowsecurity " + + "from pg_class " + + "where oid = 'tbn1'::regclass;;") + .execute()) + .flatMapMany(result -> + result.map((row, rowMetadata) -> + Pair.of(row.get("relname", String.class), + row.get("relrowsecurity", Boolean.class))))) + .collectList() + .block(); + + assertThat(pgClassCheckResult) + .containsExactlyInAnyOrder( + Pair.of("tbn1", true)); + } + + @Test + void createTableShouldNotCreateRlsColumnWhenDisableRLS() { + String tableName = "tbn1"; + + PostgresTable table = PostgresTable.name(tableName) + .createTableStep((dsl, tbn) -> dsl.createTable(tbn) + .column("clm1", SQLDataType.UUID.notNull()) + .column("clm2", SQLDataType.VARCHAR(255).notNull())) + .supportsRowLevelSecurity() + .build(); + + PostgresModule module = PostgresModule.table(table); + PostgresTableManager testee = new PostgresTableManager(postgresExtension.getDefaultPostgresExecutor(), module, RowLevelSecurity.DISABLED); + + testee.initializeTables() + .block(); + + Pair rlsColumn = Pair.of("domain", "character varying"); + assertThat(getColumnNameAndDataType(tableName)) + .doesNotContain(rlsColumn); + } + + @Test + void recreateRLSColumnWhenExistedShouldNotFail() { + String tableName = "tablename1"; + + PostgresTable rlsTable = PostgresTable.name(tableName) + .createTableStep((dsl, tbn) -> dsl.createTable(tbn) + .column("colum1", SQLDataType.UUID.notNull())) + .supportsRowLevelSecurity() + .build(); + + PostgresModule module = PostgresModule.table(rlsTable); + + PostgresTableManager testee = tableManagerFactory.apply(module); + testee.initializeTables().block(); + + assertThatCode(() -> testee.initializeTables().block()) + .doesNotThrowAnyException(); + } + + @Test + void additionalAlterQueryToCreateConstraintShouldSucceed() { + String constraintName = "exclude_constraint"; + PostgresTable table = PostgresTable.name("tbn1") + .createTableStep((dsl, tbn) -> dsl.createTable(tbn) + .column("clm1", SQLDataType.UUID.notNull()) + .column("clm2", SQLDataType.VARCHAR(255).notNull())) + .disableRowLevelSecurity() + .addAdditionalAlterQueries("ALTER TABLE tbn1 ADD CONSTRAINT " + constraintName + " EXCLUDE (clm2 WITH =)") + .build(); + PostgresModule module = PostgresModule.table(table); + PostgresTableManager testee = new PostgresTableManager(postgresExtension.getDefaultPostgresExecutor(), module, RowLevelSecurity.DISABLED); + + testee.initializeTables().block(); + + boolean constraintExists = postgresExtension.getConnection() + .flatMapMany(connection -> connection.createStatement("SELECT EXISTS(SELECT 1 FROM pg_catalog.pg_constraint WHERE conname = $1) AS constraint_exists;") + .bind("$1", constraintName) + .execute()) + .flatMap(result -> result.map((row, rowMetaData) -> row.get("constraint_exists", Boolean.class))) + .single() + .block(); + + assertThat(constraintExists).isTrue(); + } + + @Test + void additionalAlterQueryToCreateConstraintShouldSucceedWhenSupportCaseIsNonRLSAndRLSIsDisabled() { + String constraintName = "exclude_constraint"; + PostgresTable table = PostgresTable.name("tbn1") + .createTableStep((dsl, tbn) -> dsl.createTable(tbn) + .column("clm1", SQLDataType.UUID.notNull()) + .column("clm2", SQLDataType.VARCHAR(255).notNull())) + .disableRowLevelSecurity() + .addAdditionalAlterQueries(new PostgresTable.NonRLSOnlyAdditionalAlterQuery("ALTER TABLE tbn1 ADD CONSTRAINT " + constraintName + " EXCLUDE (clm2 WITH =)")) + .build(); + PostgresModule module = PostgresModule.table(table); + PostgresTableManager testee = new PostgresTableManager(postgresExtension.getDefaultPostgresExecutor(), module, RowLevelSecurity.DISABLED); + + testee.initializeTables().block(); + + boolean constraintExists = postgresExtension.getConnection() + .flatMapMany(connection -> connection.createStatement("SELECT EXISTS(SELECT 1 FROM pg_catalog.pg_constraint WHERE conname = $1) AS constraint_exists;") + .bind("$1", constraintName) + .execute()) + .flatMap(result -> result.map((row, rowMetaData) -> row.get("constraint_exists", Boolean.class))) + .single() + .block(); + + assertThat(constraintExists).isTrue(); + } + + @Test + void additionalAlterQueryToCreateConstraintShouldNotBeExecutedWhenSupportCaseIsNonRLSAndRLSIsEnabled() { + String constraintName = "exclude_constraint"; + PostgresTable table = PostgresTable.name("tbn1") + .createTableStep((dsl, tbn) -> dsl.createTable(tbn) + .column("clm1", SQLDataType.UUID.notNull()) + .column("clm2", SQLDataType.VARCHAR(255).notNull())) + .disableRowLevelSecurity() + .addAdditionalAlterQueries(new PostgresTable.NonRLSOnlyAdditionalAlterQuery("ALTER TABLE tbn1 ADD CONSTRAINT " + constraintName + " EXCLUDE (clm2 WITH =)")) + .build(); + PostgresModule module = PostgresModule.table(table); + PostgresTableManager testee = new PostgresTableManager(postgresExtension.getDefaultPostgresExecutor(), module, RowLevelSecurity.ENABLED); + + testee.initializeTables().block(); + + boolean constraintExists = postgresExtension.getConnection() + .flatMapMany(connection -> connection.createStatement("SELECT EXISTS(SELECT 1 FROM pg_catalog.pg_constraint WHERE conname = $1) AS constraint_exists;") + .bind("$1", constraintName) + .execute()) + .flatMap(result -> result.map((row, rowMetaData) -> row.get("constraint_exists", Boolean.class))) + .single() + .block(); + + assertThat(constraintExists).isFalse(); + } + + @Test + void additionalAlterQueryToReCreateConstraintShouldNotThrow() { + String constraintName = "exclude_constraint"; + PostgresTable table = PostgresTable.name("tbn1") + .createTableStep((dsl, tbn) -> dsl.createTable(tbn) + .column("clm1", SQLDataType.UUID.notNull()) + .column("clm2", SQLDataType.VARCHAR(255).notNull())) + .disableRowLevelSecurity() + .addAdditionalAlterQueries("ALTER TABLE tbn1 ADD CONSTRAINT " + constraintName + " EXCLUDE (clm2 WITH =)") + .build(); + PostgresModule module = PostgresModule.table(table); + PostgresTableManager testee = new PostgresTableManager(postgresExtension.getDefaultPostgresExecutor(), module, RowLevelSecurity.DISABLED); + + testee.initializeTables().block(); + + assertThatCode(() -> testee.initializeTables().block()) + .doesNotThrowAnyException(); + } + + private List> getColumnNameAndDataType(String tableName) { + return postgresExtension.getConnection() + .flatMapMany(connection -> Flux.from(Mono.from(connection.createStatement("SELECT table_name, column_name, data_type FROM information_schema.columns WHERE table_name = $1;") + .bind("$1", tableName) + .execute()) + .flatMapMany(result -> result.map((row, rowMetadata) -> + Pair.of(row.get("column_name", String.class), row.get("data_type", String.class)))))) + .collectList() + .block(); + } + + // return list> + private List> listIndexToTableMappings() { + return postgresExtension.getConnection() + .flatMapMany(connection -> Mono.from(connection.createStatement("SELECT indexname, tablename FROM pg_indexes;") + .execute()) + .flatMapMany(result -> + result.map((row, rowMetadata) -> + Pair.of(row.get("indexname", String.class), row.get("tablename", String.class))))) + .collectList() + .block(); + } + +} diff --git a/backends-common/postgres/src/test/java/org/apache/james/backends/postgres/quota/PostgresQuotaCurrentValueDAOTest.java b/backends-common/postgres/src/test/java/org/apache/james/backends/postgres/quota/PostgresQuotaCurrentValueDAOTest.java new file mode 100644 index 00000000000..0fc87c8d579 --- /dev/null +++ b/backends-common/postgres/src/test/java/org/apache/james/backends/postgres/quota/PostgresQuotaCurrentValueDAOTest.java @@ -0,0 +1,147 @@ +/**************************************************************** + * 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.backends.postgres.quota; + +import static org.assertj.core.api.Assertions.assertThat; + +import java.util.List; + +import org.apache.james.backends.postgres.PostgresExtension; +import org.apache.james.core.quota.QuotaComponent; +import org.apache.james.core.quota.QuotaCurrentValue; +import org.apache.james.core.quota.QuotaType; +import org.junit.jupiter.api.BeforeEach; +import org.junit.jupiter.api.Test; +import org.junit.jupiter.api.extension.RegisterExtension; + +class PostgresQuotaCurrentValueDAOTest { + @RegisterExtension + static PostgresExtension postgresExtension = PostgresExtension.withoutRowLevelSecurity(PostgresQuotaModule.MODULE); + + private static final QuotaCurrentValue.Key QUOTA_KEY = QuotaCurrentValue.Key.of(QuotaComponent.MAILBOX, "james@abc.com", QuotaType.SIZE); + + private PostgresQuotaCurrentValueDAO postgresQuotaCurrentValueDAO; + + @BeforeEach + void setup() { + postgresQuotaCurrentValueDAO = new PostgresQuotaCurrentValueDAO(postgresExtension.getDefaultPostgresExecutor()); + } + + @Test + void increaseQuotaCurrentValueShouldCreateNewRowSuccessfully() { + postgresQuotaCurrentValueDAO.increase(QUOTA_KEY, 100L).block(); + + assertThat(postgresQuotaCurrentValueDAO.getQuotaCurrentValue(QUOTA_KEY).block().getCurrentValue()) + .isEqualTo(100L); + } + + @Test + void increaseQuotaCurrentValueShouldCreateNewRowSuccessfullyWhenIncreaseAmountIsZero() { + postgresQuotaCurrentValueDAO.increase(QUOTA_KEY, 0L).block(); + + assertThat(postgresQuotaCurrentValueDAO.getQuotaCurrentValue(QUOTA_KEY).block().getCurrentValue()) + .isZero(); + } + + @Test + void increaseQuotaCurrentValueShouldIncreaseValueSuccessfully() { + assertThat(postgresQuotaCurrentValueDAO.getQuotaCurrentValue(QUOTA_KEY).block()).isNull(); + + postgresQuotaCurrentValueDAO.increase(QUOTA_KEY, 100L).block(); + postgresQuotaCurrentValueDAO.increase(QUOTA_KEY, 100L).block(); + + assertThat(postgresQuotaCurrentValueDAO.getQuotaCurrentValue(QUOTA_KEY).block().getCurrentValue()) + .isEqualTo(200L); + } + + @Test + void increaseQuotaCurrentValueShouldDecreaseValueSuccessfullyWhenIncreaseAmountIsNegative() { + postgresQuotaCurrentValueDAO.increase(QUOTA_KEY, 200L).block(); + postgresQuotaCurrentValueDAO.increase(QUOTA_KEY, -100L).block(); + + assertThat(postgresQuotaCurrentValueDAO.getQuotaCurrentValue(QUOTA_KEY).block().getCurrentValue()) + .isEqualTo(100L); + } + + @Test + void decreaseQuotaCurrentValueShouldDecreaseValueSuccessfully() { + postgresQuotaCurrentValueDAO.increase(QUOTA_KEY, 200L).block(); + postgresQuotaCurrentValueDAO.decrease(QUOTA_KEY, 100L).block(); + + assertThat(postgresQuotaCurrentValueDAO.getQuotaCurrentValue(QUOTA_KEY).block().getCurrentValue()) + .isEqualTo(100L); + } + + @Test + void decreaseQuotaCurrentValueDownToNegativeShouldAllowNegativeValue() { + postgresQuotaCurrentValueDAO.increase(QUOTA_KEY, 100L).block(); + postgresQuotaCurrentValueDAO.decrease(QUOTA_KEY, 1000L).block(); + + assertThat(postgresQuotaCurrentValueDAO.getQuotaCurrentValue(QUOTA_KEY).block().getCurrentValue()) + .isEqualTo(-900L); + } + + @Test + void decreaseQuotaCurrentValueWhenNoRecordYetShouldNotFail() { + postgresQuotaCurrentValueDAO.decrease(QUOTA_KEY, 1000L).block(); + + assertThat(postgresQuotaCurrentValueDAO.getQuotaCurrentValue(QUOTA_KEY).block().getCurrentValue()) + .isEqualTo(-1000L); + } + + @Test + void deleteQuotaCurrentValueShouldDeleteSuccessfully() { + QuotaCurrentValue.Key quotaKey = QuotaCurrentValue.Key.of(QuotaComponent.MAILBOX, "andre@abc.com", QuotaType.SIZE); + postgresQuotaCurrentValueDAO.increase(quotaKey, 100L).block(); + postgresQuotaCurrentValueDAO.deleteQuotaCurrentValue(quotaKey).block(); + + assertThat(postgresQuotaCurrentValueDAO.getQuotaCurrentValue(quotaKey).block()) + .isNull(); + } + + @Test + void deleteQuotaCurrentValueShouldResetCounterForever() { + postgresQuotaCurrentValueDAO.increase(QUOTA_KEY, 100L).block(); + postgresQuotaCurrentValueDAO.deleteQuotaCurrentValue(QUOTA_KEY).block(); + postgresQuotaCurrentValueDAO.increase(QUOTA_KEY, 100L).block(); + + assertThat(postgresQuotaCurrentValueDAO.getQuotaCurrentValue(QUOTA_KEY).block().getCurrentValue()) + .isEqualTo(100L); + } + + @Test + void getQuotasByComponentShouldGetAllQuotaTypesSuccessfully() { + QuotaCurrentValue.Key countQuotaKey = QuotaCurrentValue.Key.of(QuotaComponent.MAILBOX, "james@abc.com", QuotaType.COUNT); + + QuotaCurrentValue expectedQuotaSize = QuotaCurrentValue.builder().quotaComponent(QUOTA_KEY.getQuotaComponent()) + .identifier(QUOTA_KEY.getIdentifier()).quotaType(QUOTA_KEY.getQuotaType()).currentValue(100L).build(); + QuotaCurrentValue expectedQuotaCount = QuotaCurrentValue.builder().quotaComponent(countQuotaKey.getQuotaComponent()) + .identifier(countQuotaKey.getIdentifier()).quotaType(countQuotaKey.getQuotaType()).currentValue(56L).build(); + + postgresQuotaCurrentValueDAO.increase(QUOTA_KEY, 100L).block(); + postgresQuotaCurrentValueDAO.increase(countQuotaKey, 56L).block(); + + List actual = postgresQuotaCurrentValueDAO.getQuotaCurrentValues(QUOTA_KEY.getQuotaComponent(), QUOTA_KEY.getIdentifier()) + .collectList() + .block(); + + assertThat(actual).containsExactlyInAnyOrder(expectedQuotaSize, expectedQuotaCount); + } +} diff --git a/backends-common/postgres/src/test/java/org/apache/james/backends/postgres/quota/PostgresQuotaLimitDaoTest.java b/backends-common/postgres/src/test/java/org/apache/james/backends/postgres/quota/PostgresQuotaLimitDaoTest.java new file mode 100644 index 00000000000..b489c194e9e --- /dev/null +++ b/backends-common/postgres/src/test/java/org/apache/james/backends/postgres/quota/PostgresQuotaLimitDaoTest.java @@ -0,0 +1,84 @@ +/**************************************************************** + * 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.backends.postgres.quota; + +import static org.assertj.core.api.Assertions.assertThat; + +import org.apache.james.backends.postgres.PostgresExtension; +import org.apache.james.core.quota.QuotaComponent; +import org.apache.james.core.quota.QuotaLimit; +import org.apache.james.core.quota.QuotaScope; +import org.apache.james.core.quota.QuotaType; +import org.junit.jupiter.api.BeforeEach; +import org.junit.jupiter.api.Test; +import org.junit.jupiter.api.extension.RegisterExtension; + +public class PostgresQuotaLimitDaoTest { + + private PostgresQuotaLimitDAO postgresQuotaLimitDao; + + @RegisterExtension + static PostgresExtension postgresExtension = PostgresExtension.withoutRowLevelSecurity(PostgresQuotaModule.MODULE); + + @BeforeEach + void setup() { + postgresQuotaLimitDao = new PostgresQuotaLimitDAO(postgresExtension.getDefaultPostgresExecutor()); + } + + @Test + void getQuotaLimitsShouldGetSomeQuotaLimitsSuccessfully() { + QuotaLimit expectedOne = QuotaLimit.builder().quotaComponent(QuotaComponent.MAILBOX).quotaScope(QuotaScope.DOMAIN).identifier("A").quotaType(QuotaType.COUNT).quotaLimit(200L).build(); + QuotaLimit expectedTwo = QuotaLimit.builder().quotaComponent(QuotaComponent.MAILBOX).quotaScope(QuotaScope.DOMAIN).identifier("A").quotaType(QuotaType.SIZE).quotaLimit(100L).build(); + postgresQuotaLimitDao.setQuotaLimit(expectedOne).block(); + postgresQuotaLimitDao.setQuotaLimit(expectedTwo).block(); + + assertThat(postgresQuotaLimitDao.getQuotaLimits(QuotaComponent.MAILBOX, QuotaScope.DOMAIN, "A").collectList().block()) + .containsExactlyInAnyOrder(expectedOne, expectedTwo); + } + + @Test + void setQuotaLimitShouldSaveObjectSuccessfully() { + QuotaLimit expected = QuotaLimit.builder().quotaComponent(QuotaComponent.MAILBOX).quotaScope(QuotaScope.DOMAIN).identifier("A").quotaType(QuotaType.COUNT).quotaLimit(100L).build(); + postgresQuotaLimitDao.setQuotaLimit(expected).block(); + + assertThat(postgresQuotaLimitDao.getQuotaLimit(QuotaLimit.QuotaLimitKey.of(QuotaComponent.MAILBOX, QuotaScope.DOMAIN, "A", QuotaType.COUNT)).block()) + .isEqualTo(expected); + } + + @Test + void setQuotaLimitShouldSaveObjectSuccessfullyWhenLimitIsMinusOne() { + QuotaLimit expected = QuotaLimit.builder().quotaComponent(QuotaComponent.MAILBOX).quotaScope(QuotaScope.DOMAIN).identifier("A").quotaType(QuotaType.COUNT).quotaLimit(-1L).build(); + postgresQuotaLimitDao.setQuotaLimit(expected).block(); + + assertThat(postgresQuotaLimitDao.getQuotaLimit(QuotaLimit.QuotaLimitKey.of(QuotaComponent.MAILBOX, QuotaScope.DOMAIN, "A", QuotaType.COUNT)).block()) + .isEqualTo(expected); + } + + @Test + void deleteQuotaLimitShouldDeleteObjectSuccessfully() { + QuotaLimit quotaLimit = QuotaLimit.builder().quotaComponent(QuotaComponent.MAILBOX).quotaScope(QuotaScope.DOMAIN).identifier("A").quotaType(QuotaType.COUNT).quotaLimit(100L).build(); + postgresQuotaLimitDao.setQuotaLimit(quotaLimit).block(); + postgresQuotaLimitDao.deleteQuotaLimit(QuotaLimit.QuotaLimitKey.of(QuotaComponent.MAILBOX, QuotaScope.DOMAIN, "A", QuotaType.COUNT)).block(); + + assertThat(postgresQuotaLimitDao.getQuotaLimit(QuotaLimit.QuotaLimitKey.of(QuotaComponent.MAILBOX, QuotaScope.DOMAIN, "A", QuotaType.COUNT)).block()) + .isNull(); + } + +} \ No newline at end of file diff --git a/backends-common/postgres/src/test/java/org/apache/james/backends/postgres/utils/PostgresHealthCheckTest.java b/backends-common/postgres/src/test/java/org/apache/james/backends/postgres/utils/PostgresHealthCheckTest.java new file mode 100644 index 00000000000..f48f8d5b8c2 --- /dev/null +++ b/backends-common/postgres/src/test/java/org/apache/james/backends/postgres/utils/PostgresHealthCheckTest.java @@ -0,0 +1,61 @@ +/**************************************************************** + * 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.backends.postgres.utils; + +import static org.assertj.core.api.Assertions.assertThat; + +import org.apache.james.backends.postgres.PostgresExtension; +import org.apache.james.backends.postgres.quota.PostgresQuotaModule; +import org.apache.james.core.healthcheck.Result; +import org.apache.james.core.healthcheck.ResultStatus; +import org.junit.jupiter.api.BeforeEach; +import org.junit.jupiter.api.Test; +import org.junit.jupiter.api.extension.RegisterExtension; + +import reactor.core.publisher.Mono; + +public class PostgresHealthCheckTest { + private PostgresHealthCheck testee; + + @RegisterExtension + static PostgresExtension postgresExtension = PostgresExtension.withoutRowLevelSecurity(PostgresQuotaModule.MODULE); + + @BeforeEach + void setup() { + testee = new PostgresHealthCheck(postgresExtension.getDefaultPostgresExecutor()); + } + + @Test + void shouldBeHealthy() { + Result result = Mono.from(testee.check()).block(); + assertThat(result.getStatus()).isEqualTo(ResultStatus.HEALTHY); + } + + @Test + void shouldBeUnhealthyWhenPaused() { + try { + postgresExtension.pause(); + Result result = Mono.from(testee.check()).block(); + assertThat(result.getStatus()).isEqualTo(ResultStatus.UNHEALTHY); + } finally { + postgresExtension.unpause(); + } + } +} \ No newline at end of file diff --git a/core/src/main/java/org/apache/james/core/quota/QuotaCurrentValue.java b/core/src/main/java/org/apache/james/core/quota/QuotaCurrentValue.java index 682f10c7bcb..c1b38bb819f 100644 --- a/core/src/main/java/org/apache/james/core/quota/QuotaCurrentValue.java +++ b/core/src/main/java/org/apache/james/core/quota/QuotaCurrentValue.java @@ -26,6 +26,59 @@ public class QuotaCurrentValue { + public static class Key { + + public static Key of(QuotaComponent component, String identifier, QuotaType quotaType) { + return new Key(component, identifier, quotaType); + } + + private final QuotaComponent quotaComponent; + private final String identifier; + private final QuotaType quotaType; + + public QuotaComponent getQuotaComponent() { + return quotaComponent; + } + + public String getIdentifier() { + return identifier; + } + + public QuotaType getQuotaType() { + return quotaType; + } + + private Key(QuotaComponent quotaComponent, String identifier, QuotaType quotaType) { + this.quotaComponent = quotaComponent; + this.identifier = identifier; + this.quotaType = quotaType; + } + + @Override + public final int hashCode() { + return Objects.hash(quotaComponent, identifier, quotaType); + } + + @Override + public final boolean equals(Object o) { + if (o instanceof Key) { + Key other = (Key) o; + return Objects.equals(quotaComponent, other.quotaComponent) + && Objects.equals(identifier, other.identifier) + && Objects.equals(quotaType, other.quotaType); + } + return false; + } + + public String toString() { + return MoreObjects.toStringHelper(this) + .add("quotaComponent", quotaComponent) + .add("identifier", identifier) + .add("quotaType", quotaType) + .toString(); + } + } + public static class Builder { private QuotaComponent quotaComponent; private String identifier; diff --git a/core/src/main/java/org/apache/james/core/quota/QuotaLimit.java b/core/src/main/java/org/apache/james/core/quota/QuotaLimit.java index 5d49216be7d..0f371a5d051 100644 --- a/core/src/main/java/org/apache/james/core/quota/QuotaLimit.java +++ b/core/src/main/java/org/apache/james/core/quota/QuotaLimit.java @@ -26,6 +26,65 @@ import com.google.common.base.Preconditions; public class QuotaLimit { + public static class QuotaLimitKey { + public static QuotaLimitKey of(QuotaComponent component, QuotaScope scope, String identifier, QuotaType quotaType) { + return new QuotaLimitKey(component, scope, identifier, quotaType); + } + + private final QuotaComponent quotaComponent; + private final QuotaScope quotaScope; + private final String identifier; + private final QuotaType quotaType; + + public QuotaComponent getQuotaComponent() { + return quotaComponent; + } + + public QuotaScope getQuotaScope() { + return quotaScope; + } + + public String getIdentifier() { + return identifier; + } + + public QuotaType getQuotaType() { + return quotaType; + } + + private QuotaLimitKey(QuotaComponent quotaComponent, QuotaScope quotaScope, String identifier, QuotaType quotaType) { + this.quotaComponent = quotaComponent; + this.quotaScope = quotaScope; + this.identifier = identifier; + this.quotaType = quotaType; + } + + @Override + public final int hashCode() { + return Objects.hash(quotaComponent, quotaScope, identifier, quotaType); + } + + @Override + public final boolean equals(Object o) { + if (o instanceof QuotaLimitKey) { + QuotaLimitKey other = (QuotaLimitKey) o; + return Objects.equals(quotaComponent, other.quotaComponent) + && Objects.equals(quotaScope, other.quotaScope) + && Objects.equals(identifier, other.identifier) + && Objects.equals(quotaType, other.quotaType); + } + return false; + } + + public String toString() { + return MoreObjects.toStringHelper(this) + .add("quotaComponent", quotaComponent) + .add("quotaScope", quotaScope) + .add("identifier", identifier) + .add("quotaType", quotaType) + .toString(); + } + } public static class Builder { private QuotaComponent quotaComponent; diff --git a/docs/modules/servers/assets/images/james-imap-base-performance-postgres.png b/docs/modules/servers/assets/images/james-imap-base-performance-postgres.png new file mode 100644 index 00000000000..47bb0eb2c96 Binary files /dev/null and b/docs/modules/servers/assets/images/james-imap-base-performance-postgres.png differ diff --git a/docs/modules/servers/assets/images/postgres_pg_stat_statements.png b/docs/modules/servers/assets/images/postgres_pg_stat_statements.png new file mode 100644 index 00000000000..4cc1e46989d Binary files /dev/null and b/docs/modules/servers/assets/images/postgres_pg_stat_statements.png differ diff --git a/docs/modules/servers/assets/images/specialized-instances-postgres.png b/docs/modules/servers/assets/images/specialized-instances-postgres.png new file mode 100644 index 00000000000..9b1d226257c Binary files /dev/null and b/docs/modules/servers/assets/images/specialized-instances-postgres.png differ diff --git a/docs/modules/servers/assets/images/storage_james_postgres.png b/docs/modules/servers/assets/images/storage_james_postgres.png new file mode 100644 index 00000000000..e846fa4d9c9 Binary files /dev/null and b/docs/modules/servers/assets/images/storage_james_postgres.png differ diff --git a/docs/modules/servers/assets/images/storage_james_postgres.svg b/docs/modules/servers/assets/images/storage_james_postgres.svg new file mode 100644 index 00000000000..bc8203ce1f0 --- /dev/null +++ b/docs/modules/servers/assets/images/storage_james_postgres.svg @@ -0,0 +1 @@ + \ No newline at end of file diff --git a/docs/modules/servers/nav.adoc b/docs/modules/servers/nav.adoc index 7fdb1f8bc13..52a9cda02b8 100644 --- a/docs/modules/servers/nav.adoc +++ b/docs/modules/servers/nav.adoc @@ -77,4 +77,56 @@ *** xref:distributed/benchmark/index.adoc[Performance benchmark] **** xref:distributed/benchmark/db-benchmark.adoc[] **** xref:distributed/benchmark/james-benchmark.adoc[] +** xref:postgres/index.adoc[] +*** xref:postgres/objectives.adoc[] +*** xref:postgres/architecture/index.adoc[] +**** xref:postgres/architecture/implemented-standards.adoc[] +**** xref:postgres/architecture/consistency-model.adoc[] +**** xref:postgres/architecture/specialized-instances.adoc[] +*** xref:postgres/run/index.adoc[] +**** xref:postgres/run/run-java.adoc[Run with Java] +**** xref:postgres/run/run-docker.adoc[Run with Docker] +*** xref:postgres/configure/index.adoc[] +**** Protocols +***** xref:postgres/configure/imap.adoc[imapserver.xml] +***** xref:postgres/configure/jmap.adoc[jmap.properties] +***** xref:postgres/configure/jmx.adoc[jmx.properties] +***** xref:postgres/configure/smtp.adoc[smtpserver.xml & lmtpserver.xml] +***** xref:postgres/configure/smtp-hooks.adoc[Packaged SMTP hooks] +***** xref:postgres/configure/pop3.adoc[pop3server.xml] +***** xref:postgres/configure/webadmin.adoc[webadmin.properties] +***** xref:postgres/configure/ssl.adoc[SSL & TLS] +***** xref:postgres/configure/sieve.adoc[Sieve & ManageSieve] +**** Storage dependencies +***** xref:postgres/configure/blobstore.adoc[blobstore.properties] +***** xref:postgres/configure/opensearch.adoc[opensearch.properties] +***** xref:postgres/configure/rabbitmq.adoc[rabbitmq.properties] +***** xref:postgres/configure/redis.adoc[redis.properties] +***** xref:postgres/configure/tika.adoc[tika.properties] +**** Core components +***** xref:postgres/configure/batchsizes.adoc[batchsizes.properties] +***** xref:postgres/configure/dns.adoc[dnsservice.xml] +***** xref:postgres/configure/domainlist.adoc[domainlist.xml] +***** xref:postgres/configure/droplists.adoc[DropLists] +***** xref:postgres/configure/healthcheck.adoc[healthcheck.properties] +***** xref:postgres/configure/mailetcontainer.adoc[mailetcontainer.xml] +***** xref:postgres/configure/mailets.adoc[Packaged Mailets] +***** xref:postgres/configure/matchers.adoc[Packaged Matchers] +***** xref:postgres/configure/mailrepositorystore.adoc[mailrepositorystore.xml] +***** xref:postgres/configure/recipientrewritetable.adoc[recipientrewritetable.xml] +***** xref:postgres/configure/search.adoc[search.properties] +***** xref:postgres/configure/usersrepository.adoc[usersrepository.xml] +*** xref:postgres/operate/index.adoc[Operate] +**** xref:postgres/operate/guide.adoc[] +**** xref:postgres/operate/performanceChecklist.adoc[] +**** xref:postgres/operate/logging.adoc[] +**** xref:postgres/operate/webadmin.adoc[] +**** xref:postgres/operate/metrics.adoc[] +**** xref:postgres/operate/migrating.adoc[] +**** xref:postgres/operate/cli.adoc[] +**** xref:postgres/operate/security.adoc[] +*** xref:postgres/extending/index.adoc[] +*** xref:postgres/benchmark/index.adoc[] +**** xref:postgres/benchmark/db-benchmark.adoc[] +**** xref:postgres/benchmark/james-benchmark.adoc[] ** xref:test.adoc[] diff --git a/docs/modules/servers/pages/index.adoc b/docs/modules/servers/pages/index.adoc index 3fd055e4367..4c6faf58354 100644 --- a/docs/modules/servers/pages/index.adoc +++ b/docs/modules/servers/pages/index.adoc @@ -16,6 +16,7 @@ The available James Servers are: * <> * <> * <> + * <> * <> If you are just checking out James for the first time, then we highly recommend @@ -79,6 +80,14 @@ and is intended for experts only. +[#postgres] +== James Postgres Mail Server + +The xref:postgres/index.adoc[*Distributed with Postgres Server*] is a one +variant of the distributed server with Postgres as the database. + + + [#test] == James Test Server diff --git a/docs/modules/servers/pages/postgres/architecture/consistency-model.adoc b/docs/modules/servers/pages/postgres/architecture/consistency-model.adoc new file mode 100644 index 00000000000..dfd1687255c --- /dev/null +++ b/docs/modules/servers/pages/postgres/architecture/consistency-model.adoc @@ -0,0 +1,11 @@ += Postgresql James server — Consistency Model +:navtitle: Consistency Model + +:backend-name: postgres +:backend-name-cap: Postgres +:server-name: Postgresql James server +:mailet-repository-path-prefix: postgres +:xref-base: postgres +:data_replication_extend: servers:postgres/architecture/consistency_model_data_replication_extend.adoc + +include::partial$architecture/consistency-model.adoc[] \ No newline at end of file diff --git a/docs/modules/servers/pages/postgres/architecture/consistency_model_data_replication_extend.adoc b/docs/modules/servers/pages/postgres/architecture/consistency_model_data_replication_extend.adoc new file mode 100644 index 00000000000..ab0c01417a7 --- /dev/null +++ b/docs/modules/servers/pages/postgres/architecture/consistency_model_data_replication_extend.adoc @@ -0,0 +1 @@ +// \ No newline at end of file diff --git a/docs/modules/servers/pages/postgres/architecture/implemented-standards.adoc b/docs/modules/servers/pages/postgres/architecture/implemented-standards.adoc new file mode 100644 index 00000000000..e33b3d8a8d4 --- /dev/null +++ b/docs/modules/servers/pages/postgres/architecture/implemented-standards.adoc @@ -0,0 +1,6 @@ += Postgresql James server — Implemented standards +:navtitle: Implemented standards + +:server-name: Postgresql James server + +include::partial$architecture/implemented-standards.adoc[] \ No newline at end of file diff --git a/docs/modules/servers/pages/postgres/architecture/index.adoc b/docs/modules/servers/pages/postgres/architecture/index.adoc new file mode 100644 index 00000000000..8be525750ae --- /dev/null +++ b/docs/modules/servers/pages/postgres/architecture/index.adoc @@ -0,0 +1,13 @@ += Postgresql James server — Architecture +:navtitle: Architecture + +:backend-name: postgres +:server-name: Postgresql James server +:backend-storage-introduce: Postgresql is used for metadata storage. Postgresql is efficient for a very high workload. +:storage-picture-file-name: storage_james_postgres.png +:mailet-repository-path-prefix: postgres +:xref-base: postgres +:mailqueue-combined-extend: servers:postgres/architecture/mailqueue_combined_extend.adoc +:mailqueue-combined-extend-backend: + +include::partial$architecture/index.adoc[] \ No newline at end of file diff --git a/docs/modules/servers/pages/postgres/architecture/mailqueue_combined_extend.adoc b/docs/modules/servers/pages/postgres/architecture/mailqueue_combined_extend.adoc new file mode 100644 index 00000000000..dba010cbb09 --- /dev/null +++ b/docs/modules/servers/pages/postgres/architecture/mailqueue_combined_extend.adoc @@ -0,0 +1 @@ +// 123 \ No newline at end of file diff --git a/docs/modules/servers/pages/postgres/architecture/specialized-instances.adoc b/docs/modules/servers/pages/postgres/architecture/specialized-instances.adoc new file mode 100644 index 00000000000..7f8e3493772 --- /dev/null +++ b/docs/modules/servers/pages/postgres/architecture/specialized-instances.adoc @@ -0,0 +1,7 @@ += Postgresql James server — Specialized instances +:navtitle: Specialized instances + +:server-name: Postgresql James server +:specialized-instances-file-name: specialized-instances-postgres.png + +include::partial$architecture/specialized-instances.adoc[] \ No newline at end of file diff --git a/docs/modules/servers/pages/postgres/benchmark/benchmark_prepare.adoc b/docs/modules/servers/pages/postgres/benchmark/benchmark_prepare.adoc new file mode 100644 index 00000000000..5257ef4840f --- /dev/null +++ b/docs/modules/servers/pages/postgres/benchmark/benchmark_prepare.adoc @@ -0,0 +1,40 @@ +=== Postgresql prepare benchmark + +==== Install extension pg_stat_statements + +The `pg_stat_statements` extension provides a means for tracking execution statistics of all SQL statements executed by a server. +The extension is useful for identifying high-traffic queries and for monitoring the performance of the server. +For more information, see the [PostgreSQL documentation](https://www.postgresql.org/docs/current/pgstatstatements.html). + +To install the extension, connect to the database and run the following query: + +[source,sql] +---- +create extension if not exists pg_stat_statements; +alter system set shared_preload_libraries='pg_stat_statements'; + +-- restart postgres +-- optional +alter system set pg_stat_statements.max = 100000; +alter system set pg_stat_statements.track = 'all'; +---- + +To reset statistics, use: `select pg_stat_statements_reset()`; + +The response fields that we are interested in are: + +- `query`: Text of a representative statement + +- `calls`: Number of times the statement was executed + +- `total_exec_time`, `mean_exec_time`, `min_exec_time`, `max_exec_time` + +To view the statistics, run the following query: + +```sql +select query, mean_exec_time, total_exec_time, calls from pg_stat_statements order by total_exec_time desc; +``` + +The result sample: + +image::postgres_pg_stat_statements.png[Storage responsibilities for the {server-name}] \ No newline at end of file diff --git a/docs/modules/servers/pages/postgres/benchmark/db-benchmark.adoc b/docs/modules/servers/pages/postgres/benchmark/db-benchmark.adoc new file mode 100644 index 00000000000..5e9bf216f77 --- /dev/null +++ b/docs/modules/servers/pages/postgres/benchmark/db-benchmark.adoc @@ -0,0 +1,8 @@ += Postgresql James server -- Database benchmarks +:navtitle: Database benchmarks + +:backend-name: postgres +:server-name: Postgresql James server +:backend-database-extend-sample: PostgreSQL 16 as main database: 1 nodes (OVH instance, 2 CPU / 7 GB RAM, 160 GB SSD) + +include::partial$benchmark/db-benchmark.adoc[] \ No newline at end of file diff --git a/docs/modules/servers/pages/postgres/benchmark/index.adoc b/docs/modules/servers/pages/postgres/benchmark/index.adoc new file mode 100644 index 00000000000..0532346caa2 --- /dev/null +++ b/docs/modules/servers/pages/postgres/benchmark/index.adoc @@ -0,0 +1,7 @@ += Postgresql James server — Performance testing +:navtitle: Performance testing the Postgresql James server + +:xref-base: postgres +:server-name: Postgresql James server + +include::partial$benchmark/index.adoc[] \ No newline at end of file diff --git a/docs/modules/servers/pages/postgres/benchmark/james-benchmark.adoc b/docs/modules/servers/pages/postgres/benchmark/james-benchmark.adoc new file mode 100644 index 00000000000..52bdec9769a --- /dev/null +++ b/docs/modules/servers/pages/postgres/benchmark/james-benchmark.adoc @@ -0,0 +1,10 @@ += Postgresql James server benchmark +:navtitle: James benchmarks + +:server-name: Postgresql James server +:backend-database-extend-sample: PostgreSQL 16 as main database: 1 nodes (OVH instance, 2 CPU / 7 GB RAM, 160 GB SSD) +:provision_file_url: https://github.com/apache/james-project/blob/d8225ed7c5ca8d79cde3b1c8755ee9ffcf462e29/server/apps/postgres-app/provision.sh +:benchmark_prepare_extend: servers:postgres/benchmark/benchmark_prepare.adoc +:james-imap-base-performance-picture: james-imap-base-performance-postgres.png + +include::partial$benchmark/james-benchmark.adoc[] \ No newline at end of file diff --git a/docs/modules/servers/pages/postgres/configure/batchsizes.adoc b/docs/modules/servers/pages/postgres/configure/batchsizes.adoc new file mode 100644 index 00000000000..8c7264ce05a --- /dev/null +++ b/docs/modules/servers/pages/postgres/configure/batchsizes.adoc @@ -0,0 +1,5 @@ += Postgresql James Server — batchsizes.properties +:navtitle: batchsizes.properties + +:sample-configuration-prefix-url: https://github.com/apache/james-project/blob/postgresql/server/apps/postgres-app/sample-configuration +include::partial$configure/batchsizes.adoc[] \ No newline at end of file diff --git a/docs/modules/servers/pages/postgres/configure/blobstore.adoc b/docs/modules/servers/pages/postgres/configure/blobstore.adoc new file mode 100644 index 00000000000..e7c1d341aa1 --- /dev/null +++ b/docs/modules/servers/pages/postgres/configure/blobstore.adoc @@ -0,0 +1,51 @@ += Postgresql James Server — blobstore.properties +:navtitle: blobstore.properties + +:sample-configuration-prefix-url: https://github.com/apache/james-project/blob/postgresql/server/apps/postgres-app/sample-configuration +:pages-path: postgres + +== BlobStore + +This file is optional. If omitted, the *postgres* blob store will be used. + +BlobStore is the dedicated component to store blobs, non-indexable content. +James uses the BlobStore for storing blobs which are usually mail contents, attachments, deleted mails... + +You can choose the underlying implementation of BlobStore to fit with your James setup. + +It could be the implementation on top of Postgres or file storage service S3 compatible like Openstack Swift and AWS S3. + +Consult link:{sample-configuration-prefix-url}/blob.properties[blob.properties] +in GIT to get some examples and hints. + +=== Implementation choice + +*implementation* : + +* postgres: use cassandra based Postgres +* objectstorage: use Swift/AWS S3 based BlobStore +* file: (experimental) use directly the file system. Useful for legacy architecture based on shared ISCI SANs and/or +distributed file system with no object store available. + +*deduplication.enable*: Mandatory. Supported value: true and false. + +If you choose to enable deduplication, the mails with the same content will be stored only once. + +WARNING: Once this feature is enabled, there is no turning back as turning it off will lead to the deletion of all +the mails sharing the same content once one is deleted. + +Deduplication requires a garbage collector mechanism to effectively drop blobs. A first implementation +based on bloom filters can be used and triggered using the WebAdmin REST API. See +xref:{pages-path}/operate/webadmin.adoc#_running_blob_garbage_collection[Running blob garbage collection]. + +In order to avoid concurrency issues upon garbage collection, we slice the blobs in generation, the two more recent +generations are not garbage collected. + +*deduplication.gc.generation.duration*: Allow controlling the duration of one generation. Longer implies better deduplication +but deleted blobs will live longer. Duration, defaults on 30 days, the default unit is in days. + +*deduplication.gc.generation.family*: Every time the duration is changed, this integer counter must be incremented to avoid +conflicts. Defaults to 1. + + +include::partial$configure/blobstore.adoc[] diff --git a/docs/modules/servers/pages/postgres/configure/collecting-contacts.adoc b/docs/modules/servers/pages/postgres/configure/collecting-contacts.adoc new file mode 100644 index 00000000000..b077a2c45ce --- /dev/null +++ b/docs/modules/servers/pages/postgres/configure/collecting-contacts.adoc @@ -0,0 +1,4 @@ += Contact collection + +:sample-configuration-prefix-url: https://github.com/apache/james-project/blob/postgresql/server/apps/postgres-app/sample-configuration +include::partial$configure/collecting-contacts.adoc[] \ No newline at end of file diff --git a/docs/modules/servers/pages/postgres/configure/collecting-events.adoc b/docs/modules/servers/pages/postgres/configure/collecting-events.adoc new file mode 100644 index 00000000000..431f06aa8be --- /dev/null +++ b/docs/modules/servers/pages/postgres/configure/collecting-events.adoc @@ -0,0 +1,4 @@ += Event collection + +:sample-configuration-prefix-url: https://github.com/apache/james-project/blob/postgresql/server/apps/postgres-app/sample-configuration +include::partial$configure/collecting-events.adoc[] \ No newline at end of file diff --git a/docs/modules/servers/pages/postgres/configure/dns.adoc b/docs/modules/servers/pages/postgres/configure/dns.adoc new file mode 100644 index 00000000000..ffff105f3e8 --- /dev/null +++ b/docs/modules/servers/pages/postgres/configure/dns.adoc @@ -0,0 +1,5 @@ += Postgresql James Server — dnsservice.xml +:navtitle: dnsservice.xml + +:sample-configuration-prefix-url: https://github.com/apache/james-project/blob/postgresql/server/apps/postgres-app/sample-configuration +include::partial$configure/dns.adoc[] \ No newline at end of file diff --git a/docs/modules/servers/pages/postgres/configure/domainlist.adoc b/docs/modules/servers/pages/postgres/configure/domainlist.adoc new file mode 100644 index 00000000000..9654c2c6b74 --- /dev/null +++ b/docs/modules/servers/pages/postgres/configure/domainlist.adoc @@ -0,0 +1,5 @@ += Postgresql James Server — domainlist.xml +:navtitle: domainlist.xml + +:sample-configuration-prefix-url: https://github.com/apache/james-project/blob/postgresql/server/apps/postgres-app/sample-configuration +include::partial$configure/domainlist.adoc[] \ No newline at end of file diff --git a/docs/modules/servers/pages/postgres/configure/droplists.adoc b/docs/modules/servers/pages/postgres/configure/droplists.adoc new file mode 100644 index 00000000000..fb1c242047d --- /dev/null +++ b/docs/modules/servers/pages/postgres/configure/droplists.adoc @@ -0,0 +1,6 @@ += Postgresql James Server — DropLists +:navtitle: DropLists + +:sample-configuration-prefix-url: https://github.com/apache/james-project/blob/postgresql/server/apps/postgres-app/sample-configuration +:pages-path: postgres +include::partial$configure/droplists.adoc[] \ No newline at end of file diff --git a/docs/modules/servers/pages/postgres/configure/dsn.adoc b/docs/modules/servers/pages/postgres/configure/dsn.adoc new file mode 100644 index 00000000000..46cdc91803e --- /dev/null +++ b/docs/modules/servers/pages/postgres/configure/dsn.adoc @@ -0,0 +1,7 @@ += Postgresql James Server — Delivery Submission Notifications +:navtitle: ESMTP DSN setup + +:sample-configuration-prefix-url: https://github.com/apache/james-project/blob/postgresql/server/apps/postgres-app/sample-configuration +:pages-path: distributed +:mailet-repository-path-prefix: postgres +include::partial$configure/dsn.adoc[] \ No newline at end of file diff --git a/docs/modules/servers/pages/postgres/configure/extensions.adoc b/docs/modules/servers/pages/postgres/configure/extensions.adoc new file mode 100644 index 00000000000..c99cb4a6289 --- /dev/null +++ b/docs/modules/servers/pages/postgres/configure/extensions.adoc @@ -0,0 +1,6 @@ += Postgresql James Server — extensions.properties +:navtitle: extensions.properties + +:sample-configuration-prefix-url: https://github.com/apache/james-project/blob/postgresql/server/apps/postgres-app/sample-configuration +:pages-path: postgres +include::partial$configure/extensions.adoc[] \ No newline at end of file diff --git a/docs/modules/servers/pages/postgres/configure/healthcheck.adoc b/docs/modules/servers/pages/postgres/configure/healthcheck.adoc new file mode 100644 index 00000000000..dd0a5e4bcb2 --- /dev/null +++ b/docs/modules/servers/pages/postgres/configure/healthcheck.adoc @@ -0,0 +1,5 @@ += Postgresql James Server — healthcheck.properties +:navtitle: healthcheck.properties + +:sample-configuration-prefix-url: https://github.com/apache/james-project/blob/postgresql/server/apps/postgres-app/sample-configuration +include::partial$configure/healthcheck.adoc[] \ No newline at end of file diff --git a/docs/modules/servers/pages/postgres/configure/imap.adoc b/docs/modules/servers/pages/postgres/configure/imap.adoc new file mode 100644 index 00000000000..47b538272fb --- /dev/null +++ b/docs/modules/servers/pages/postgres/configure/imap.adoc @@ -0,0 +1,6 @@ += Postgresql James Server — imapserver.xml +:navtitle: imapserver.xml + +:sample-configuration-prefix-url: https://github.com/apache/james-project/blob/postgresql/server/apps/postgres-app/sample-configuration +:pages-path: postgres +include::partial$configure/imap.adoc[] \ No newline at end of file diff --git a/docs/modules/servers/pages/postgres/configure/index.adoc b/docs/modules/servers/pages/postgres/configure/index.adoc new file mode 100644 index 00000000000..5ef404256d1 --- /dev/null +++ b/docs/modules/servers/pages/postgres/configure/index.adoc @@ -0,0 +1,24 @@ += Postgresql James Server — Configuration +:navtitle: Configuration + +This section presents how to configure the Postgresql James server. + +The Postgresql James Server relies on separated files for configuring various components. Some files follow a *xml* format +and some others follow a *property* format. Some files can be omitted, in which case the functionality can be disabled, +or rely on reasonable defaults. + +The following configuration files are exposed: + +:sample-configuration-prefix-url: https://github.com/apache/james-project/blob/postgresql/server/apps/postgres-app/sample-configuration +:xref-base: postgres/configure +:server-name: Postgresql James server + +include::partial$configure/forProtocolsPartial.adoc[] + +include::partial$configure/forStorageDependenciesPartial.adoc[] + +include::partial$configure/forCoreComponentsPartial.adoc[] + +include::partial$configure/forExtensionsPartial.adoc[] + +include::partial$configure/systemPropertiesPartial.adoc[] \ No newline at end of file diff --git a/docs/modules/servers/pages/postgres/configure/jmap.adoc b/docs/modules/servers/pages/postgres/configure/jmap.adoc new file mode 100644 index 00000000000..912ba217436 --- /dev/null +++ b/docs/modules/servers/pages/postgres/configure/jmap.adoc @@ -0,0 +1,7 @@ += Postgresql James Server — jmap.properties +:navtitle: jmap.properties + +:sample-configuration-prefix-url: https://github.com/apache/james-project/blob/postgresql/server/apps/postgres-app/sample-configuration +:server-name: Postgresql James server +:backend-name: Postgresql +include::partial$configure/jmap.adoc[] \ No newline at end of file diff --git a/docs/modules/servers/pages/postgres/configure/jmx.adoc b/docs/modules/servers/pages/postgres/configure/jmx.adoc new file mode 100644 index 00000000000..0b294bbfa6a --- /dev/null +++ b/docs/modules/servers/pages/postgres/configure/jmx.adoc @@ -0,0 +1,5 @@ += Postgresql James Server — jmx.properties +:navtitle: jmx.properties + +:sample-configuration-prefix-url: https://github.com/apache/james-project/blob/postgresql/server/apps/postgres-app/sample-configuration +include::partial$configure/jmx.adoc[] \ No newline at end of file diff --git a/docs/modules/servers/pages/postgres/configure/jvm.adoc b/docs/modules/servers/pages/postgres/configure/jvm.adoc new file mode 100644 index 00000000000..28611f12800 --- /dev/null +++ b/docs/modules/servers/pages/postgres/configure/jvm.adoc @@ -0,0 +1,5 @@ += Postgresql James Server — jvm.properties +:navtitle: jvm.properties + +:sample-configuration-prefix-url: https://github.com/apache/james-project/blob/postgresql/server/apps/postgres-app/sample-configuration +include::partial$configure/jvm.adoc[] \ No newline at end of file diff --git a/docs/modules/servers/pages/postgres/configure/listeners.adoc b/docs/modules/servers/pages/postgres/configure/listeners.adoc new file mode 100644 index 00000000000..011dd6c3963 --- /dev/null +++ b/docs/modules/servers/pages/postgres/configure/listeners.adoc @@ -0,0 +1,6 @@ += Postgresql James Server — listeners.xml +:navtitle: listeners.xml + +:sample-configuration-prefix-url: https://github.com/apache/james-project/blob/postgresql/server/apps/postgres-app/sample-configuration +:server-name: Postgresql James server +include::partial$configure/listeners.adoc[] \ No newline at end of file diff --git a/docs/modules/servers/pages/postgres/configure/mailetcontainer.adoc b/docs/modules/servers/pages/postgres/configure/mailetcontainer.adoc new file mode 100644 index 00000000000..8b8184fbd95 --- /dev/null +++ b/docs/modules/servers/pages/postgres/configure/mailetcontainer.adoc @@ -0,0 +1,6 @@ += Postgresql James Server — mailetcontainer.xml +:navtitle: mailetcontainer.xml + +:sample-configuration-prefix-url: https://github.com/apache/james-project/blob/postgresql/server/apps/postgres-app/sample-configuration +:pages-path: postgres +include::partial$configure/mailetcontainer.adoc[] \ No newline at end of file diff --git a/docs/modules/servers/pages/postgres/configure/mailets.adoc b/docs/modules/servers/pages/postgres/configure/mailets.adoc new file mode 100644 index 00000000000..07c8f532e56 --- /dev/null +++ b/docs/modules/servers/pages/postgres/configure/mailets.adoc @@ -0,0 +1,6 @@ += Postgresql James Server — Mailets +:navtitle: Mailets + +:sample-configuration-prefix-url: https://github.com/apache/james-project/blob/postgresql/server/apps/postgres-app/sample-configuration +:server-name: Postgresql James server +include::partial$configure/mailets.adoc[] \ No newline at end of file diff --git a/docs/modules/servers/pages/postgres/configure/mailrepositorystore.adoc b/docs/modules/servers/pages/postgres/configure/mailrepositorystore.adoc new file mode 100644 index 00000000000..bba70563b2c --- /dev/null +++ b/docs/modules/servers/pages/postgres/configure/mailrepositorystore.adoc @@ -0,0 +1,9 @@ += Postgresql James Server — mailrepositorystore.xml + +:sample-configuration-prefix-url: https://github.com/apache/james-project/blob/postgresql/server/apps/postgres-app/sample-configuration +:pages-path: postgres +:server-name: Postgresql James server +:mailet-repository-path-prefix: postgres +:mail-repository-protocol: postgres +:mail-repository-class: org.apache.james.mailrepository.postgres.PostgresMailRepository +include::partial$configure/mailrepositorystore.adoc[] \ No newline at end of file diff --git a/docs/modules/servers/pages/postgres/configure/matchers.adoc b/docs/modules/servers/pages/postgres/configure/matchers.adoc new file mode 100644 index 00000000000..d97cc58fd6a --- /dev/null +++ b/docs/modules/servers/pages/postgres/configure/matchers.adoc @@ -0,0 +1,7 @@ += Postgresql James Server — Matchers +:navtitle: Matchers + +:sample-configuration-prefix-url: https://github.com/apache/james-project/blob/postgresql/server/apps/postgres-app/sample-configuration +:pages-path: postgres +:server-name: Postgresql James server +include::partial$configure/matchers.adoc[] \ No newline at end of file diff --git a/docs/modules/servers/pages/postgres/configure/opensearch.adoc b/docs/modules/servers/pages/postgres/configure/opensearch.adoc new file mode 100644 index 00000000000..16314afb10c --- /dev/null +++ b/docs/modules/servers/pages/postgres/configure/opensearch.adoc @@ -0,0 +1,8 @@ += Postgresql James Server — opensearch.properties +:navtitle: opensearch.properties + +:sample-configuration-prefix-url: https://github.com/apache/james-project/blob/postgresql/server/apps/postgres-app/sample-configuration +:pages-path: postgres +:server-name: Postgresql James server +:package-tag: postgres +include::partial$configure/opensearch.adoc[] \ No newline at end of file diff --git a/docs/modules/servers/pages/postgres/configure/pop3.adoc b/docs/modules/servers/pages/postgres/configure/pop3.adoc new file mode 100644 index 00000000000..95da0cfbc9a --- /dev/null +++ b/docs/modules/servers/pages/postgres/configure/pop3.adoc @@ -0,0 +1,7 @@ += Postgresql James Server — pop3server.xml +:navtitle: pop3server.xml + +:sample-configuration-prefix-url: https://github.com/apache/james-project/blob/postgresql/server/apps/postgres-app/sample-configuration +:pages-path: postgres +:server-name: Postgresql James server +include::partial$configure/pop3.adoc[] \ No newline at end of file diff --git a/docs/modules/servers/pages/postgres/configure/queue.adoc b/docs/modules/servers/pages/postgres/configure/queue.adoc new file mode 100644 index 00000000000..09f666e498a --- /dev/null +++ b/docs/modules/servers/pages/postgres/configure/queue.adoc @@ -0,0 +1,5 @@ += Postgresql James Server — queue.properties +:navtitle: queue.properties + +:sample-configuration-prefix-url: https://github.com/apache/james-project/blob/postgresql/server/apps/postgres-app/sample-configuration +include::partial$configure/queue.adoc[] \ No newline at end of file diff --git a/docs/modules/servers/pages/postgres/configure/rabbitmq.adoc b/docs/modules/servers/pages/postgres/configure/rabbitmq.adoc new file mode 100644 index 00000000000..ddee170f82d --- /dev/null +++ b/docs/modules/servers/pages/postgres/configure/rabbitmq.adoc @@ -0,0 +1,5 @@ += Postgresql James Server — rabbitmq.properties +:navtitle: rabbitmq.properties + +:sample-configuration-prefix-url: https://github.com/apache/james-project/blob/postgresql/server/apps/postgres-app/sample-configuration +include::partial$configure/rabbitmq.adoc[] \ No newline at end of file diff --git a/docs/modules/servers/pages/postgres/configure/recipientrewritetable.adoc b/docs/modules/servers/pages/postgres/configure/recipientrewritetable.adoc new file mode 100644 index 00000000000..6cc602f7866 --- /dev/null +++ b/docs/modules/servers/pages/postgres/configure/recipientrewritetable.adoc @@ -0,0 +1,7 @@ += Postgresql James Server — recipientrewritetable.xml +:navtitle: recipientrewritetable.xml + +:sample-configuration-prefix-url: https://github.com/apache/james-project/blob/postgresql/server/apps/postgres-app/sample-configuration +:pages-path: postgres +:server-name: Postgresql James server +include::partial$configure/recipientrewritetable.adoc[] \ No newline at end of file diff --git a/docs/modules/servers/pages/postgres/configure/redis.adoc b/docs/modules/servers/pages/postgres/configure/redis.adoc new file mode 100644 index 00000000000..c3b2558d4b0 --- /dev/null +++ b/docs/modules/servers/pages/postgres/configure/redis.adoc @@ -0,0 +1,5 @@ += Postgresql James Server — redis.properties +:navtitle: redis.properties + +:sample-configuration-prefix-url: https://github.com/apache/james-project/blob/postgresql/server/apps/postgres-app/sample-configuration +include::partial$configure/redis.adoc[] \ No newline at end of file diff --git a/docs/modules/servers/pages/postgres/configure/remote-delivery-error-handling.adoc b/docs/modules/servers/pages/postgres/configure/remote-delivery-error-handling.adoc new file mode 100644 index 00000000000..7500221ac3e --- /dev/null +++ b/docs/modules/servers/pages/postgres/configure/remote-delivery-error-handling.adoc @@ -0,0 +1,8 @@ += Postgresql James Server — About RemoteDelivery error handling +:navtitle: About RemoteDelivery error handling + +:sample-configuration-prefix-url: https://github.com/apache/james-project/blob/postgresql/server/apps/postgres-app/sample-configuration +:pages-path: postgres +:server-name: Postgresql James server +:mailet-repository-path-prefix: postgres +include::partial$configure/remote-delivery-error-handling.adoc[] \ No newline at end of file diff --git a/docs/modules/servers/pages/postgres/configure/search.adoc b/docs/modules/servers/pages/postgres/configure/search.adoc new file mode 100644 index 00000000000..0c329853048 --- /dev/null +++ b/docs/modules/servers/pages/postgres/configure/search.adoc @@ -0,0 +1,5 @@ += Postgresql James Server — Search configuration +:navtitle: Search configuration + +:sample-configuration-prefix-url: https://github.com/apache/james-project/blob/postgresql/server/apps/postgres-app/sample-configuration +include::partial$configure/search.adoc[] \ No newline at end of file diff --git a/docs/modules/servers/pages/postgres/configure/sieve.adoc b/docs/modules/servers/pages/postgres/configure/sieve.adoc new file mode 100644 index 00000000000..8326b2752e4 --- /dev/null +++ b/docs/modules/servers/pages/postgres/configure/sieve.adoc @@ -0,0 +1,7 @@ += Sieve +:navtitle: Sieve + +:sample-configuration-prefix-url: https://github.com/apache/james-project/blob/postgresql/server/apps/postgres-app/sample-configuration +:pages-path: postgres +:server-name: Postgresql James server +include::partial$configure/sieve.adoc[] \ No newline at end of file diff --git a/docs/modules/servers/pages/postgres/configure/smtp-hooks.adoc b/docs/modules/servers/pages/postgres/configure/smtp-hooks.adoc new file mode 100644 index 00000000000..cac323ebc8d --- /dev/null +++ b/docs/modules/servers/pages/postgres/configure/smtp-hooks.adoc @@ -0,0 +1,7 @@ += Postgresql James Server — SMTP Hooks +:navtitle: SMTP Hooks + +:sample-configuration-prefix-url: https://github.com/apache/james-project/blob/postgresql/server/apps/postgres-app/sample-configuration +:pages-path: postgres +:server-name: Postgresql James server +include::partial$configure/smtp-hooks.adoc[] \ No newline at end of file diff --git a/docs/modules/servers/pages/postgres/configure/smtp.adoc b/docs/modules/servers/pages/postgres/configure/smtp.adoc new file mode 100644 index 00000000000..e78cd94302f --- /dev/null +++ b/docs/modules/servers/pages/postgres/configure/smtp.adoc @@ -0,0 +1,7 @@ += Postgresql James Server — smtpserver.xml +:navtitle: smtpserver.xml + +:sample-configuration-prefix-url: https://github.com/apache/james-project/blob/postgresql/server/apps/postgres-app/sample-configuration +:pages-path: postgres +:server-name: Postgresql James server +include::partial$configure/smtp.adoc[] \ No newline at end of file diff --git a/docs/modules/servers/pages/postgres/configure/spam.adoc b/docs/modules/servers/pages/postgres/configure/spam.adoc new file mode 100644 index 00000000000..bce4eb9ae1a --- /dev/null +++ b/docs/modules/servers/pages/postgres/configure/spam.adoc @@ -0,0 +1,8 @@ += Postgresql James Server — Anti-Spam configuration +:navtitle: Anti-Spam configuration + +:sample-configuration-prefix-url: https://github.com/apache/james-project/blob/postgresql/server/apps/postgres-app/sample-configuration +:pages-path: postgres +:server-name: Postgresql James server +:mailet-repository-path-prefix: postgres +include::partial$configure/spam.adoc[] \ No newline at end of file diff --git a/docs/modules/servers/pages/postgres/configure/ssl.adoc b/docs/modules/servers/pages/postgres/configure/ssl.adoc new file mode 100644 index 00000000000..16924ae6b2c --- /dev/null +++ b/docs/modules/servers/pages/postgres/configure/ssl.adoc @@ -0,0 +1,7 @@ += Postgresql James Server — SSL & TLS configuration +:navtitle: SSL & TLS configuration + +:sample-configuration-prefix-url: https://github.com/apache/james-project/blob/postgresql/server/apps/postgres-app/sample-configuration +:pages-path: postgres +:server-name: Postgresql James server +include::partial$configure/ssl.adoc[] \ No newline at end of file diff --git a/docs/modules/servers/pages/postgres/configure/tika.adoc b/docs/modules/servers/pages/postgres/configure/tika.adoc new file mode 100644 index 00000000000..90a68e6eb8f --- /dev/null +++ b/docs/modules/servers/pages/postgres/configure/tika.adoc @@ -0,0 +1,5 @@ += Postgresql James Server — tika.properties +:navtitle: tika.properties + +:sample-configuration-prefix-url: https://github.com/apache/james-project/blob/postgresql/server/apps/postgres-app/sample-configuration +include::partial$configure/tika.adoc[] \ No newline at end of file diff --git a/docs/modules/servers/pages/postgres/configure/usersrepository.adoc b/docs/modules/servers/pages/postgres/configure/usersrepository.adoc new file mode 100644 index 00000000000..8f6d3cba524 --- /dev/null +++ b/docs/modules/servers/pages/postgres/configure/usersrepository.adoc @@ -0,0 +1,5 @@ += Postgresql James Server — usersrepository.xml +:navtitle: usersrepository.xml + +:sample-configuration-prefix-url: https://github.com/apache/james-project/blob/postgresql/server/apps/postgres-app/sample-configuration +include::partial$configure/usersrepository.adoc[] \ No newline at end of file diff --git a/docs/modules/servers/pages/postgres/configure/vault.adoc b/docs/modules/servers/pages/postgres/configure/vault.adoc new file mode 100644 index 00000000000..dcdfc7dd207 --- /dev/null +++ b/docs/modules/servers/pages/postgres/configure/vault.adoc @@ -0,0 +1,8 @@ += Postgresql James Server — deletedMessageVault.properties +:navtitle: deletedMessageVault.properties + +:sample-configuration-prefix-url: https://github.com/apache/james-project/blob/postgresql/server/apps/postgres-app/sample-configuration +:pages-path: postgres +:server-name: Postgresql James server +:backend-name: Postgresql +include::partial$configure/vault.adoc[] \ No newline at end of file diff --git a/docs/modules/servers/pages/postgres/configure/webadmin.adoc b/docs/modules/servers/pages/postgres/configure/webadmin.adoc new file mode 100644 index 00000000000..161652dde4d --- /dev/null +++ b/docs/modules/servers/pages/postgres/configure/webadmin.adoc @@ -0,0 +1,7 @@ += Postgresql James Server — webadmin.properties +:navtitle: webadmin.properties + +:sample-configuration-prefix-url: https://github.com/apache/james-project/blob/postgresql/server/apps/postgres-app/sample-configuration +:pages-path: postgres +:server-name: Postgresql James server +include::partial$configure/webadmin.adoc[] \ No newline at end of file diff --git a/docs/modules/servers/pages/postgres/extending.adoc b/docs/modules/servers/pages/postgres/extending.adoc new file mode 100644 index 00000000000..96f10693b79 --- /dev/null +++ b/docs/modules/servers/pages/postgres/extending.adoc @@ -0,0 +1,4 @@ += Postgres James Mail Server — Extending behaviour +:navtitle: Extending behaviour + +This section can be read xref:customization:index.adoc[this page]. diff --git a/docs/modules/servers/pages/postgres/extending/index.adoc b/docs/modules/servers/pages/postgres/extending/index.adoc new file mode 100644 index 00000000000..c95b2919ad5 --- /dev/null +++ b/docs/modules/servers/pages/postgres/extending/index.adoc @@ -0,0 +1,2 @@ += Distributed James Postgres Server — Extending server behavior +:navtitle: Extending server behavior \ No newline at end of file diff --git a/docs/modules/servers/pages/postgres/index.adoc b/docs/modules/servers/pages/postgres/index.adoc new file mode 100644 index 00000000000..b3a8b8416ff --- /dev/null +++ b/docs/modules/servers/pages/postgres/index.adoc @@ -0,0 +1,15 @@ += Postgres James Mail Server +:navtitle: Distributed Postgres James Application + +The Postgres James server offers an easy way to scale email server. Based on +SQL database solutions, here is https://www.postgresql.org/[Postgres]. + +Postgres is a powerful and versatile database server. Known for its advanced features, scalability, +and robust performance, Postgres is the ideal choice for handling high-throughput and large data sets efficiently. +Its row-level security ensures top-notch data protection, while the flexible architecture allows seamless integration +with various storage and search solutions + +In this section of the documentation, we will introduce you to: + +* xref:postgres/objectives.adoc[Objectives and motivation of the Distributed Postgres Server] +* xref:postgres/run/index.adoc[Run the Postgresql Server] \ No newline at end of file diff --git a/docs/modules/servers/pages/postgres/objectives.adoc b/docs/modules/servers/pages/postgres/objectives.adoc new file mode 100644 index 00000000000..d1dcb91090b --- /dev/null +++ b/docs/modules/servers/pages/postgres/objectives.adoc @@ -0,0 +1,22 @@ += Distributed James Server — Objectives and motivation +:navtitle: Objectives and motivation + +From the outstanding advantages of a distributed mail system, such as scalability and enhancement, +this project aims to implement a backend database version using Postgres. + +Primary Objectives: + +* Provide more options: The current James Distributed server uses Cassandra as the backend database. + This project aims to provide an alternative to Cassandra, using Postgres as the backend database. + This choice aims to offer a highly scalable and reactive James mail server, suitable for small to medium deployments, + while the distributed setup remains more fitting for larger ones. +* Propose an alternative to the jpa-app variant: The jpa-app variant is a simple version of James that uses JPA + to store data and is compatible with various SQL databases. + With the postgres-app, we use the `r2dbc` library to connect to the Postgres database, implementing non-blocking, + reactive APIs for higher performance. +* Leverage advanced Postgres features: Postgres is a powerful database that supports many advanced features. + This project aims to leverage these features to improve the efficiency of the James server. + For example, the implement https://www.postgresql.org/docs/current/ddl-rowsecurity.html[row-level security] + to improve the security of the James server. +* Flexible deployment: The new architecture allows flexible module choices. You can use Postgres directly for + blob storage or use Object Storage (e.g Minio, S3...). \ No newline at end of file diff --git a/docs/modules/servers/pages/postgres/operate/cli.adoc b/docs/modules/servers/pages/postgres/operate/cli.adoc new file mode 100644 index 00000000000..2f008e438b3 --- /dev/null +++ b/docs/modules/servers/pages/postgres/operate/cli.adoc @@ -0,0 +1,6 @@ += Postgresql James Server — Command Line Interface +:navtitle: Command Line Interface + +:xref-base: postgres +:server-name: Postgresql James Server +include::partial$operate/cli.adoc[] \ No newline at end of file diff --git a/docs/modules/servers/pages/postgres/operate/guide.adoc b/docs/modules/servers/pages/postgres/operate/guide.adoc new file mode 100644 index 00000000000..5b829212d66 --- /dev/null +++ b/docs/modules/servers/pages/postgres/operate/guide.adoc @@ -0,0 +1,9 @@ += Postgresql James Server — Operator guide +:navtitle: Operator guide + +:xref-base: postgres +:mailet-repository-path-prefix: postgres +:backend-name: postgres +:sample-configuration-prefix-url: https://github.com/apache/james-project/blob/master/server/apps/postgres-app/sample-configuration +:server-name: Postgresql James Server +include::partial$operate/guide.adoc[] \ No newline at end of file diff --git a/docs/modules/servers/pages/postgres/operate/index.adoc b/docs/modules/servers/pages/postgres/operate/index.adoc new file mode 100644 index 00000000000..484f330aea3 --- /dev/null +++ b/docs/modules/servers/pages/postgres/operate/index.adoc @@ -0,0 +1,7 @@ += Postgresql James Server — Operate the Distributed server +:navtitle: Operate the Distributed server + +:xref-base: postgres +:server-name: Postgresql James Server +:server-tag: postgres +include::partial$operate/index.adoc[] diff --git a/docs/modules/servers/pages/postgres/operate/logging.adoc b/docs/modules/servers/pages/postgres/operate/logging.adoc new file mode 100644 index 00000000000..4b5d3de2453 --- /dev/null +++ b/docs/modules/servers/pages/postgres/operate/logging.adoc @@ -0,0 +1,9 @@ += Postgresql James Server — Logging +:navtitle: Logging + +:xref-base: postgres +:server-name: Postgresql James Server +:server-tag: postgres +:docker-compose-code-block-sample: servers:postgres/operate/logging/docker-compose-block.adoc +:backend-name: postgres +include::partial$operate/logging.adoc[] diff --git a/docs/modules/servers/pages/postgres/operate/logging/docker-compose-block.adoc b/docs/modules/servers/pages/postgres/operate/logging/docker-compose-block.adoc new file mode 100644 index 00000000000..3ff42faf399 --- /dev/null +++ b/docs/modules/servers/pages/postgres/operate/logging/docker-compose-block.adoc @@ -0,0 +1,81 @@ +[source,docker-compose] +---- +version: "3" + +services: + james: + depends_on: + - elasticsearch + - postgres + - rabbitmq + - s3 + image: apache/james:postgres-latest + container_name: james + hostname: james.local + volumes: + - ./extension-jars:/root/extension-jars + - ./conf/logback.xml:/root/conf/logback.xml + - ./logs:/root/logs + ports: + - "80:80" + - "25:25" + - "110:110" + - "143:143" + - "465:465" + - "587:587" + - "993:993" + - "8080:8000" + + elasticsearch: + image: docker.elastic.co/elasticsearch/elasticsearch:7.10.2 + ports: + - "9200:9200" + environment: + - discovery.type=single-node + + postgres: + image: postgres:16.3 + ports: + - "5432:5432" + environment: + - POSTGRES_DB=james + - POSTGRES_USER=james + - POSTGRES_PASSWORD=secret1 + + rabbitmq: + image: rabbitmq:3.13.3-management + ports: + - "5672:5672" + - "15672:15672" + + s3: + image: registry.scality.com/cloudserver/cloudserver:8.7.25 + container_name: s3.docker.test + environment: + - SCALITY_ACCESS_KEY_ID=accessKey1 + - SCALITY_SECRET_ACCESS_KEY=secretKey1 + - S3BACKEND=mem + - LOG_LEVEL=trace + - REMOTE_MANAGEMENT_DISABLE=1 + + fluent-bit: + image: fluent/fluent-bit:1.5.7 + volumes: + - ./fluentbit/fluent-bit.conf:/fluent-bit/etc/fluent-bit.conf + - ./fluentbit/parsers.conf:/fluent-bit/etc/parsers.conf + - ./logs:/fluent-bit/log + ports: + - "24224:24224" + - "24224:24224/udp" + depends_on: + - elasticsearch + + kibana: + image: docker.elastic.co/kibana/kibana:7.10.2 + environment: + ELASTICSEARCH_HOSTS: http://elasticsearch:9200 + ports: + - "5601:5601" + depends_on: + - elasticsearch +---- \ No newline at end of file diff --git a/docs/modules/servers/pages/postgres/operate/metrics.adoc b/docs/modules/servers/pages/postgres/operate/metrics.adoc new file mode 100644 index 00000000000..0bccbb7cc1e --- /dev/null +++ b/docs/modules/servers/pages/postgres/operate/metrics.adoc @@ -0,0 +1,7 @@ += Postgresql James Server — Metrics +:navtitle: Metrics + +:other-metrics: Postgresql Java driver metrics +:xref-base: postgres +:server-name: Postgresql James Server +include::partial$operate/metrics.adoc[] \ No newline at end of file diff --git a/docs/modules/servers/pages/postgres/operate/migrating.adoc b/docs/modules/servers/pages/postgres/operate/migrating.adoc new file mode 100644 index 00000000000..b00a838135c --- /dev/null +++ b/docs/modules/servers/pages/postgres/operate/migrating.adoc @@ -0,0 +1,6 @@ += Postgresql James Server — Migrating existing data +:navtitle: Migrating existing data + +:xref-base: postgres +:server-name: Postgresql James Server +include::partial$operate/migrating.adoc[] \ No newline at end of file diff --git a/docs/modules/servers/pages/postgres/operate/performanceChecklist.adoc b/docs/modules/servers/pages/postgres/operate/performanceChecklist.adoc new file mode 100644 index 00000000000..42f6e9afc9b --- /dev/null +++ b/docs/modules/servers/pages/postgres/operate/performanceChecklist.adoc @@ -0,0 +1,6 @@ += Postgresql James Server — Performance checklist +:navtitle: Performance checklist + +:xref-base: postgres +:backend-name: postgres +include::partial$operate/performanceChecklist.adoc[] diff --git a/docs/modules/servers/pages/postgres/operate/security.adoc b/docs/modules/servers/pages/postgres/operate/security.adoc new file mode 100644 index 00000000000..80c578c9e5c --- /dev/null +++ b/docs/modules/servers/pages/postgres/operate/security.adoc @@ -0,0 +1,6 @@ += Security checklist +:navtitle: Security checklist + +:xref-base: postgres +:backend-name: postgres +include::partial$operate/security.adoc[] diff --git a/docs/modules/servers/pages/postgres/operate/webadmin.adoc b/docs/modules/servers/pages/postgres/operate/webadmin.adoc new file mode 100644 index 00000000000..b8a275de609 --- /dev/null +++ b/docs/modules/servers/pages/postgres/operate/webadmin.adoc @@ -0,0 +1,10 @@ += Postgresql James Server — WebAdmin REST administration API +:navtitle: WebAdmin REST administration API + +:server-name: Postgresql James Server +:xref-base: postgres +:backend-name: postgres +:admin-mail-queues-extend: servers:postgres/operate/webadmin/admin-mail-queues-extend.adoc +:admin-messages-extend: servers:postgres/operate/webadmin/admin-messages-extend.adoc +:admin-mailboxes-extend: servers:postgres/operate/webadmin/admin-mailboxes-extend.adoc +include::partial$operate/webadmin.adoc[] diff --git a/docs/modules/servers/pages/postgres/operate/webadmin/admin-mail-queues-extend.adoc b/docs/modules/servers/pages/postgres/operate/webadmin/admin-mail-queues-extend.adoc new file mode 100644 index 00000000000..ba054a0f3b8 --- /dev/null +++ b/docs/modules/servers/pages/postgres/operate/webadmin/admin-mail-queues-extend.adoc @@ -0,0 +1 @@ +// The document only covers Postgres \ No newline at end of file diff --git a/docs/modules/servers/pages/postgres/operate/webadmin/admin-mailboxes-extend.adoc b/docs/modules/servers/pages/postgres/operate/webadmin/admin-mailboxes-extend.adoc new file mode 100644 index 00000000000..ba054a0f3b8 --- /dev/null +++ b/docs/modules/servers/pages/postgres/operate/webadmin/admin-mailboxes-extend.adoc @@ -0,0 +1 @@ +// The document only covers Postgres \ No newline at end of file diff --git a/docs/modules/servers/pages/postgres/operate/webadmin/admin-messages-extend.adoc b/docs/modules/servers/pages/postgres/operate/webadmin/admin-messages-extend.adoc new file mode 100644 index 00000000000..ba054a0f3b8 --- /dev/null +++ b/docs/modules/servers/pages/postgres/operate/webadmin/admin-messages-extend.adoc @@ -0,0 +1 @@ +// The document only covers Postgres \ No newline at end of file diff --git a/docs/modules/servers/pages/postgres/run/index.adoc b/docs/modules/servers/pages/postgres/run/index.adoc new file mode 100644 index 00000000000..5b2eb998018 --- /dev/null +++ b/docs/modules/servers/pages/postgres/run/index.adoc @@ -0,0 +1,14 @@ += Postgresql James Server — Run +:navtitle: Run + +This sections presents guidance to all current deployment types of Postgresql James Server. + +== Run with Java + +Build your own Apache James Postgresql artifacts and start xref:postgres/run/run-java.adoc[Running it directly on a Java Virtual Machine]. + +== Run with Docker + +We have prepared a docker-compose for Apache James to run with Postgresql & OpenSearch. + +You can start xref:postgres/run/run-docker.adoc[Running James with few simple Docker commands]. \ No newline at end of file diff --git a/docs/modules/servers/pages/postgres/run/run-docker.adoc b/docs/modules/servers/pages/postgres/run/run-docker.adoc new file mode 100644 index 00000000000..1299c241b1f --- /dev/null +++ b/docs/modules/servers/pages/postgres/run/run-docker.adoc @@ -0,0 +1,145 @@ += Postgresql James Server — Run with docker +:navtitle: Run with docker + +== Running via docker-compose + +Requirements: docker & docker-compose installed. + +When you try James this way, you will use the most current state of James. + +=== Running with Postgresql only + +It will be configured to run with Postgresql. +All those components will be started with a single command. + +You can retrieve the docker-compose file : ( docker-compose file and james image name should be changed) + + $ wget https://raw.githubusercontent.com/apache/james-project/master/server/apps/postgres-app/docker-compose.yml + + +Then, you just have to start the services: + + $ docker-compose up -d + +Wait a few seconds in order to have all those services start up. You will see the following log when James is available: +james | Started : true + +A default domain, james.local, has been created. You can see this by running: + + $ docker exec james james-cli -h 127.0.0.1 -p 9999 listdomains + +James will respond to IMAP port 143 and SMTP port 25. +You have to create users before playing with james. You may also want to create other domains. +Follow the xref:postgres/operate/cli.adoc['Useful commands'] section for more information about James CLI. + +=== Running distributed James + +We also have a distributed version of the James postgresql app with: + +* OpenSearch as a search indexer +* S3 as the object storage +* RabbitMQ as the event bus + +To run it, simply type: + + $ docker compose -f docker-compose-distributed.yml up -d + +== Run with docker + +=== Requirements + +Compile the whole project: + + mvn clean install -DskipTests -T 4 + +Then load the James Postgresql server docker image: + + docker load -i server/apps/postgres-app/target/jib-image.tar + +Alternatively we provide convenience distribution for the latest release: + + docker pull apache/james:postgres-3.9.0 + +=== Running with Postgresql only + +Firstly, create your own user network on Docker for the James environment: + + $ docker network create --driver bridge james + +You need a running *Postgresql* in docker which connects to *james* network. To achieve this run: + + $ docker run -d --network james --name=postgres --env 'POSTGRES_DB=james' --env 'POSTGRES_USER=james' --env 'POSTGRES_PASSWORD=secret1' postgres:16.0 + +To run this container : + + $ docker run --network james --hostname HOSTNAME -p "25:25" -p 80:80 -p "110:110" -p "143:143" -p "465:465" -p "587:587" -p "993:993" -p "127.0.0.1:8000:8000" --name james_run + -v $PWD/keystore:/root/conf/keystore -t apache/james:postgres-3.9.0 --generate-keystore + +Where : + +- HOSTNAME: is the hostname you want to give to your James container. This DNS entry will be used to send mail to your James server. + +Webadmin port binding is restricted to loopback as users are not authenticated by default on webadmin server. Thus you should avoid exposing it in production. +Note that the above example assumes `127.0.0.1` is your loopback interface for convenience but you should change it if this is not the case on your machine. + +If you want to pass additional options to the underlying java command, you can configure a _JAVA_TOOL_OPTIONS_ env variable, for example add: + + --env "JAVA_TOOL_OPTIONS=-Xms256m -Xmx2048m" + +To have log file accessible on a volume, add *-v $PWD/logs:/logs* option to the above command line, where *$PWD/logs* is your local directory to put files in. + +=== Running distributed + +Same as above, except that you need to run before James instances of RabbitMQ, S3 object storage and Opensearch. + +You need a running *rabbitmq* in docker which connects to *james* network. To achieve this run: + + $ docker run -d --network james --name=rabbitmq rabbitmq:3.13.3-management + +You need a running *Zenko Cloudserver* objectstorage in docker which connects to *james* network. To achieve this run: + + $ docker run -d --network james --env 'REMOTE_MANAGEMENT_DISABLE=1' --env 'SCALITY_ACCESS_KEY_ID=accessKey1' --env 'SCALITY_SECRET_ACCESS_KEY=secretKey1' --name=s3 registry.scality.com/cloudserver/cloudserver:8.7.25 + +You need a running *OpenSearch* in docker which connects to *james* network. To achieve this run: + +$ docker run -d --network james -p 9200:9200 --name=opensearch --env 'discovery.type=single-node' opensearchproject/opensearch:2.14.0 + +Then run James like in the section above. + +=== Specific keystore + +Alternatively, you can also generate a keystore in your conf folder with the +following command, and drop `--generate-keystore` option: + +[source,bash] +---- +$ keytool -genkey -alias james -keyalg RSA -keystore conf/keystore +---- + +=== Instrumentation +You can use link:https://glowroot.org/[Glowroot] to instrumentalize James. It is packaged as part of the docker distribution to easily enable valuable performances insights. +Disabled by default, its java agent can easily be enabled: + + --env "JAVA_TOOL_OPTIONS=-javaagent:/root/glowroot.jar" -p "4000:4000" + +By default, the Glowroot UI is accessible from every machines in the network as defined in the _destination/admin.json_. +Which you could configure before building the image, if you want to restrict its accessibility to localhost for example. +See the https://github.com/glowroot/glowroot/wiki/Agent-Installation-(with-Embedded-Collector)#user-content-optional-post-installation-steps[Glowroot post installation steps] for more details. + +Or by mapping the 4000 port to the IP of the desired network interface, for example `-p 127.0.0.1:4000:4000`. + + +=== Handling attachment indexing + +You can handle attachment text extraction before indexing in OpenSearch. This makes attachments searchable. To enable this: + +Run tika connect to *james* network: + + $ docker run -d --network james --name tika apache/tika:2.9.2.1 + +Run James: + + $ docker run --network james --hostname HOSTNAME -p "25:25" -p 80:80 -p "110:110" -p "143:143" -p "465:465" -p "587:587" -p "993:993" -p "127.0.0.1:8000:8000" + --name james_run -v $PWD/keystore:/root/conf/keystore -t apache/james:postgres-latest + +You can find more explanation on the need of Tika in this xref:postgres/configure/tika.adoc[page]. diff --git a/docs/modules/servers/pages/postgres/run/run-java.adoc b/docs/modules/servers/pages/postgres/run/run-java.adoc new file mode 100644 index 00000000000..5cd113b8746 --- /dev/null +++ b/docs/modules/servers/pages/postgres/run/run-java.adoc @@ -0,0 +1,108 @@ += Postgresql James Server — Run +:navtitle: Run + +== Building + +=== Requirements + +* Java 21 SDK +* Maven 3 + +=== Building the artifacts + +An usual compilation using maven will produce two artifacts into +server/apps/postgres-app/target directory: + +* james-server-postgres-app.jar +* james-server-postgres-app.lib + +You can for example run in the base of +https://github.com/apache/james-project[this git repository]: + +.... +mvn clean install +.... + +== Running + +=== Running James with Postgresql only + +==== Requirements + +* Postgresql 16.0+ + +==== James launch + +To run james, you have to create a directory containing required +configuration files. + +James requires the configuration to be in a subfolder of working directory that is called conf. +A https://github.com/apache/james-project/tree/master/server/apps/postgres-app/sample-configuration[sample directory] +is provided with some default values you may need to replace. You will need to update its content to match your needs. + +Also you might need to add the files like in the +https://github.com/apache/james-project/tree/master/server/apps/postgres-app/sample-configuration-single[sample directory] +to not have OpenSearch indexing enabled by default for the search. + +You need to have a Postgresql instance running. You can either install the server or launch it via docker: + +[source,bash] +---- +$ docker run -d --network james -p 5432:5432 --name=postgres --env 'POSTGRES_DB=james' --env 'POSTGRES_USER=james' --env 'POSTGRES_PASSWORD=secret1' postgres:16.0 +---- + +Once everything is set up, you just have to run the jar with: + +[source,bash] +---- +$ java -Dworking.directory=. -jar target/james-server-postgres-app.jar --generate-keystore +---- + +Alternatively, you can also generate a keystore in your conf folder with the +following command, and drop `--generate-keystore` option: + +[source,bash] +---- +$ keytool -genkey -alias james -keyalg RSA -keystore conf/keystore +---- + +=== Running distributed James + +==== Requirements + +* Postgresql 16.0+ +* OpenSearch 2.1.0+ +* RabbitMQ-Management 3.8.17+ +* Swift ObjectStorage 2.15.1+ or Zenko Cloudserver or AWS S3 + +==== James Launch + +If you want to use the distributed version of James Postgres app, you will need to add configuration in the conf folder +like in the https://github.com/apache/james-project/tree/master/server/apps/postgres-app/sample-configuration-distributed[sample directory]. + +You need to have a Postgresql, OpenSearch, S3 and RabbitMQ instance +running. You can either install the servers or launch them via docker: + +[source,bash] +---- +$ docker run -d --network james -p 5432:5432 --name=postgres --env 'POSTGRES_DB=james' --env 'POSTGRES_USER=james' --env 'POSTGRES_PASSWORD=secret1' postgres:16.0 +$ docker run -d --network james -p 9200:9200 --name=opensearch --env 'discovery.type=single-node' opensearchproject/opensearch:2.14.0 +$ docker run -d -p 5672:5672 -p 15672:15672 --name=rabbitmq rabbitmq:3.13.3-management +$ docker run -d --env 'REMOTE_MANAGEMENT_DISABLE=1' --env 'SCALITY_ACCESS_KEY_ID=accessKey1' --env 'SCALITY_SECRET_ACCESS_KEY=secretKey1' --name=s3 registry.scality.com/cloudserver/cloudserver:8.7.25 +---- + +Once everything is set up, you just have to run the jar like in the with Postgresql only section. + +==== Using AWS S3 of Zenko Cloudserver + +By default, James is configured with [Zenko Cloudserver](https://hub.docker.com/r/zenko/cloudserver) which is compatible with AWS S3, in `blobstore.propeties` as such: + +[source,bash] +---- +implementation=s3 +objectstorage.namespace=james +objectstorage.s3.endPoint=http://s3.docker.test:8000/ +objectstorage.s3.region=eu-west-1 +objectstorage.s3.accessKeyId=accessKey1 +objectstorage.s3.secretKey=secretKey1 +---- \ No newline at end of file diff --git a/docs/modules/servers/partials/architecture/implemented-standards.adoc b/docs/modules/servers/partials/architecture/implemented-standards.adoc index 5a338bc06e1..1972fd24b00 100644 --- a/docs/modules/servers/partials/architecture/implemented-standards.adoc +++ b/docs/modules/servers/partials/architecture/implemented-standards.adoc @@ -35,6 +35,7 @@ This page details standards implemented by the {server-name}. - link:https://datatracker.ietf.org/doc/html/rfc2197[RFC-2197] SMTP Service Extension for Command Pipelining - link:https://datatracker.ietf.org/doc/html/rfc2554[RFC-2554] ESMTP Service Extension for Authentication - link:https://datatracker.ietf.org/doc/rfc6710/[RFC-6710] SMTP Extension for Message Transfer Priorities +- link:https://datatracker.ietf.org/doc/html/rfc1893[RFC-1893] Enhanced Mail System Status Codes == LMTP diff --git a/docs/modules/servers/partials/configure/collecting-events.adoc b/docs/modules/servers/partials/configure/collecting-events.adoc index f204e6b12f1..7cc79f20f76 100644 --- a/docs/modules/servers/partials/configure/collecting-events.adoc +++ b/docs/modules/servers/partials/configure/collecting-events.adoc @@ -10,6 +10,7 @@ The idea is to write a portion of mailet pipeline extracting Icalendar attachmen can later be sent to other applications over AMQP to be treated in an asynchronous, decoupled fashion. == Configuration + We can achieve this goal by combining simple mailets building blocks. Here is a sample pipeline achieving aforementioned objectives : diff --git a/docs/modules/servers/partials/configure/matchers.adoc b/docs/modules/servers/partials/configure/matchers.adoc index cd7d6ff0749..83b781c2f97 100644 --- a/docs/modules/servers/partials/configure/matchers.adoc +++ b/docs/modules/servers/partials/configure/matchers.adoc @@ -123,6 +123,8 @@ include::partial$CompareNumericHeaderValue.adoc[] include::partial$FileRegexMatcher.adoc[] +include::partial$HasHabeasWarrantMark.adoc[] + include::partial$InSpammerBlacklist.adoc[] include::partial$NESSpamCheck.adoc[] diff --git a/event-bus/pom.xml b/event-bus/pom.xml index 64b10dcded6..16a649f4322 100644 --- a/event-bus/pom.xml +++ b/event-bus/pom.xml @@ -34,5 +34,6 @@ cassandra distributed in-vm + postgres diff --git a/event-bus/postgres/pom.xml b/event-bus/postgres/pom.xml new file mode 100644 index 00000000000..033ab6dafc1 --- /dev/null +++ b/event-bus/postgres/pom.xml @@ -0,0 +1,70 @@ + + + + 4.0.0 + + org.apache.james + event-bus + 3.9.0-SNAPSHOT + + + dead-letter-postgres + Apache James :: Event Bus :: Dead Letter :: Postgres + In Postgres implementation for the eventDeadLetter API + + + + ${james.groupId} + apache-james-backends-postgres + + + ${james.groupId} + apache-james-backends-postgres + test-jar + test + + + ${james.groupId} + event-bus-api + + + ${james.groupId} + event-bus-api + test-jar + test + + + ${james.groupId} + james-server-guice-common + test-jar + test + + + ${james.groupId} + testing-base + test + + + org.testcontainers + postgresql + test + + + diff --git a/event-bus/postgres/src/main/java/org/apache/james/events/PostgresEventDeadLetters.java b/event-bus/postgres/src/main/java/org/apache/james/events/PostgresEventDeadLetters.java new file mode 100644 index 00000000000..01d7271cb71 --- /dev/null +++ b/event-bus/postgres/src/main/java/org/apache/james/events/PostgresEventDeadLetters.java @@ -0,0 +1,118 @@ +/**************************************************************** + * 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.events; + +import static org.apache.james.events.PostgresEventDeadLettersModule.PostgresEventDeadLettersTable.EVENT; +import static org.apache.james.events.PostgresEventDeadLettersModule.PostgresEventDeadLettersTable.GROUP; +import static org.apache.james.events.PostgresEventDeadLettersModule.PostgresEventDeadLettersTable.INSERTION_ID; +import static org.apache.james.events.PostgresEventDeadLettersModule.PostgresEventDeadLettersTable.TABLE_NAME; + +import jakarta.inject.Inject; + +import org.apache.james.backends.postgres.utils.PostgresExecutor; +import org.jooq.Record; + +import com.github.fge.lambdas.Throwing; +import com.google.common.base.Preconditions; + +import reactor.core.publisher.Flux; +import reactor.core.publisher.Mono; + +public class PostgresEventDeadLetters implements EventDeadLetters { + private final PostgresExecutor postgresExecutor; + private final EventSerializer eventSerializer; + + @Inject + public PostgresEventDeadLetters(PostgresExecutor postgresExecutor, EventSerializer eventSerializer) { + this.postgresExecutor = postgresExecutor; + this.eventSerializer = eventSerializer; + } + + @Override + public Mono store(Group registeredGroup, Event failDeliveredEvent) { + Preconditions.checkArgument(registeredGroup != null, REGISTERED_GROUP_CANNOT_BE_NULL); + Preconditions.checkArgument(failDeliveredEvent != null, FAIL_DELIVERED_EVENT_CANNOT_BE_NULL); + + InsertionId insertionId = InsertionId.random(); + return postgresExecutor.executeVoid(dslContext -> Mono.from(dslContext.insertInto(TABLE_NAME) + .set(INSERTION_ID, insertionId.getId()) + .set(GROUP, registeredGroup.asString()) + .set(EVENT, eventSerializer.toJson(failDeliveredEvent)))) + .thenReturn(insertionId); + } + + @Override + public Mono remove(Group registeredGroup, InsertionId failDeliveredInsertionId) { + Preconditions.checkArgument(registeredGroup != null, REGISTERED_GROUP_CANNOT_BE_NULL); + Preconditions.checkArgument(failDeliveredInsertionId != null, FAIL_DELIVERED_ID_INSERTION_CANNOT_BE_NULL); + + return postgresExecutor.executeVoid(dslContext -> Mono.from(dslContext.deleteFrom(TABLE_NAME) + .where(INSERTION_ID.eq(failDeliveredInsertionId.getId())))); + } + + @Override + public Mono remove(Group registeredGroup) { + Preconditions.checkArgument(registeredGroup != null, REGISTERED_GROUP_CANNOT_BE_NULL); + + return postgresExecutor.executeVoid(dslContext -> Mono.from(dslContext.deleteFrom(TABLE_NAME) + .where(GROUP.eq(registeredGroup.asString())))); + } + + @Override + public Mono failedEvent(Group registeredGroup, InsertionId failDeliveredInsertionId) { + Preconditions.checkArgument(registeredGroup != null, REGISTERED_GROUP_CANNOT_BE_NULL); + Preconditions.checkArgument(failDeliveredInsertionId != null, FAIL_DELIVERED_ID_INSERTION_CANNOT_BE_NULL); + + return postgresExecutor.executeRow(dslContext -> Mono.from(dslContext.select(EVENT) + .from(TABLE_NAME) + .where(INSERTION_ID.eq(failDeliveredInsertionId.getId())))) + .map(this::deserializeEvent); + } + + private Event deserializeEvent(Record record) { + return eventSerializer.asEvent(record.get(EVENT)); + } + + @Override + public Flux failedIds(Group registeredGroup) { + Preconditions.checkArgument(registeredGroup != null, REGISTERED_GROUP_CANNOT_BE_NULL); + + return postgresExecutor.executeRows(dslContext -> Flux.from(dslContext + .select(INSERTION_ID) + .from(TABLE_NAME) + .where(GROUP.eq(registeredGroup.asString())))) + .map(record -> InsertionId.of(record.get(INSERTION_ID))); + } + + @Override + public Flux groupsWithFailedEvents() { + return postgresExecutor.executeRows(dslContext -> Flux.from(dslContext + .selectDistinct(GROUP) + .from(TABLE_NAME))) + .map(Throwing.function(record -> Group.deserialize(record.get(GROUP)))); + } + + @Override + public Mono containEvents() { + return postgresExecutor.executeExists(dslContext -> dslContext.selectOne() + .from(TABLE_NAME) + .where()); + } +} diff --git a/event-bus/postgres/src/main/java/org/apache/james/events/PostgresEventDeadLettersModule.java b/event-bus/postgres/src/main/java/org/apache/james/events/PostgresEventDeadLettersModule.java new file mode 100644 index 00000000000..28d5809c26a --- /dev/null +++ b/event-bus/postgres/src/main/java/org/apache/james/events/PostgresEventDeadLettersModule.java @@ -0,0 +1,59 @@ +/**************************************************************** + * 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.events; + +import java.util.UUID; + +import org.apache.james.backends.postgres.PostgresIndex; +import org.apache.james.backends.postgres.PostgresModule; +import org.apache.james.backends.postgres.PostgresTable; +import org.jooq.Field; +import org.jooq.Record; +import org.jooq.Table; +import org.jooq.impl.DSL; +import org.jooq.impl.SQLDataType; + +public interface PostgresEventDeadLettersModule { + interface PostgresEventDeadLettersTable { + Table TABLE_NAME = DSL.table("event_dead_letters"); + + Field INSERTION_ID = DSL.field("insertion_id", SQLDataType.UUID.notNull()); + Field GROUP = DSL.field("\"group\"", SQLDataType.VARCHAR.notNull()); + Field EVENT = DSL.field("event", SQLDataType.VARCHAR.notNull()); + + PostgresTable TABLE = PostgresTable.name(TABLE_NAME.getName()) + .createTableStep(((dsl, tableName) -> dsl.createTableIfNotExists(tableName) + .column(INSERTION_ID) + .column(GROUP) + .column(EVENT) + .primaryKey(INSERTION_ID))) + .disableRowLevelSecurity() + .build(); + + PostgresIndex GROUP_INDEX = PostgresIndex.name("event_dead_letters_group_index") + .createIndexStep((dsl, indexName) -> dsl.createIndexIfNotExists(indexName) + .on(TABLE_NAME, GROUP)); + } + + PostgresModule MODULE = PostgresModule.builder() + .addTable(PostgresEventDeadLettersTable.TABLE) + .addIndex(PostgresEventDeadLettersTable.GROUP_INDEX) + .build(); +} diff --git a/event-bus/postgres/src/test/java/org/apache/james/events/PostgresEventDeadLettersTest.java b/event-bus/postgres/src/test/java/org/apache/james/events/PostgresEventDeadLettersTest.java new file mode 100644 index 00000000000..6dff2be8e11 --- /dev/null +++ b/event-bus/postgres/src/test/java/org/apache/james/events/PostgresEventDeadLettersTest.java @@ -0,0 +1,35 @@ +/**************************************************************** + * 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.events; + +import org.apache.james.backends.postgres.PostgresExtension; +import org.apache.james.backends.postgres.PostgresModule; +import org.junit.jupiter.api.extension.RegisterExtension; + +public class PostgresEventDeadLettersTest implements EventDeadLettersContract.AllContracts { + @RegisterExtension + static PostgresExtension postgresExtension = PostgresExtension.withoutRowLevelSecurity( + PostgresModule.aggregateModules(PostgresEventDeadLettersModule.MODULE)); + + @Override + public EventDeadLetters eventDeadLetters() { + return new PostgresEventDeadLetters(postgresExtension.getDefaultPostgresExecutor(), new EventBusTestFixture.TestEventSerializer()); + } +} diff --git a/event-sourcing/event-store-postgres/pom.xml b/event-sourcing/event-store-postgres/pom.xml new file mode 100644 index 00000000000..daf4273b9c6 --- /dev/null +++ b/event-sourcing/event-store-postgres/pom.xml @@ -0,0 +1,103 @@ + + + + 4.0.0 + + + org.apache.james + event-sourcing + 3.9.0-SNAPSHOT + + + event-sourcing-event-store-postgres + + Apache James :: Event sourcing :: Event Store :: Postgres + Postgres implementation for James Event Store + + + + ${james.groupId} + apache-james-backends-postgres + + + ${james.groupId} + apache-james-backends-postgres + test-jar + test + + + ${james.groupId} + event-sourcing-core + test-jar + test + + + ${james.groupId} + event-sourcing-event-store-api + + + ${james.groupId} + event-sourcing-event-store-api + test-jar + test + + + ${james.groupId} + event-sourcing-pojo + test-jar + test + + + ${james.groupId} + james-json + + + ${james.groupId} + james-server-guice-common + test-jar + test + + + net.javacrumbs.json-unit + json-unit-assertj + test + + + org.assertj + assertj-core + test + + + org.junit.jupiter + junit-jupiter-engine + test + + + org.mockito + mockito-core + test + + + org.testcontainers + postgresql + test + + + diff --git a/event-sourcing/event-store-postgres/src/main/java/org/apache/james/eventsourcing/eventstore/postgres/PostgresEventStore.java b/event-sourcing/event-store-postgres/src/main/java/org/apache/james/eventsourcing/eventstore/postgres/PostgresEventStore.java new file mode 100644 index 00000000000..5d408d0ab68 --- /dev/null +++ b/event-sourcing/event-store-postgres/src/main/java/org/apache/james/eventsourcing/eventstore/postgres/PostgresEventStore.java @@ -0,0 +1,81 @@ +/**************************************************************** + * 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.eventsourcing.eventstore.postgres; + +import static org.apache.james.backends.postgres.utils.PostgresUtils.UNIQUE_CONSTRAINT_VIOLATION_PREDICATE; + +import java.util.List; +import java.util.Optional; + +import jakarta.inject.Inject; + +import org.apache.james.eventsourcing.AggregateId; +import org.apache.james.eventsourcing.Event; +import org.apache.james.eventsourcing.EventId; +import org.apache.james.eventsourcing.eventstore.EventStore; +import org.apache.james.eventsourcing.eventstore.EventStoreFailedException; +import org.apache.james.eventsourcing.eventstore.History; +import org.reactivestreams.Publisher; + +import com.google.common.base.Preconditions; +import com.google.common.collect.ImmutableList; + +import reactor.core.publisher.Mono; +import scala.jdk.javaapi.CollectionConverters; + +public class PostgresEventStore implements EventStore { + private final PostgresEventStoreDAO eventStoreDAO; + + @Inject + public PostgresEventStore(PostgresEventStoreDAO eventStoreDAO) { + this.eventStoreDAO = eventStoreDAO; + } + + @Override + public Publisher appendAll(scala.collection.Iterable scalaEvents) { + if (scalaEvents.isEmpty()) { + return Mono.empty(); + } + Preconditions.checkArgument(Event.belongsToSameAggregate(scalaEvents)); + List events = ImmutableList.copyOf(CollectionConverters.asJava(scalaEvents)); + Optional snapshotId = events.stream().filter(Event::isASnapshot).map(Event::eventId).findFirst(); + return eventStoreDAO.appendAll(events, snapshotId) + .onErrorMap(UNIQUE_CONSTRAINT_VIOLATION_PREDICATE, + e -> new EventStoreFailedException("Concurrent update to the EventStore detected")); + } + + @Override + public Publisher getEventsOfAggregate(AggregateId aggregateId) { + return eventStoreDAO.getSnapshot(aggregateId) + .flatMap(snapshotId -> eventStoreDAO.getEventsOfAggregate(aggregateId, snapshotId)) + .flatMap(history -> { + if (history.getEventsJava().isEmpty()) { + return Mono.from(eventStoreDAO.getEventsOfAggregate(aggregateId)); + } else { + return Mono.just(history); + } + }).defaultIfEmpty(History.empty()); + } + + @Override + public Publisher remove(AggregateId aggregateId) { + return eventStoreDAO.delete(aggregateId); + } +} diff --git a/event-sourcing/event-store-postgres/src/main/java/org/apache/james/eventsourcing/eventstore/postgres/PostgresEventStoreDAO.java b/event-sourcing/event-store-postgres/src/main/java/org/apache/james/eventsourcing/eventstore/postgres/PostgresEventStoreDAO.java new file mode 100644 index 00000000000..cd5f8257a84 --- /dev/null +++ b/event-sourcing/event-store-postgres/src/main/java/org/apache/james/eventsourcing/eventstore/postgres/PostgresEventStoreDAO.java @@ -0,0 +1,124 @@ +/**************************************************************** + * 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.eventsourcing.eventstore.postgres; + +import static org.apache.james.eventsourcing.eventstore.postgres.PostgresEventStoreModule.PostgresEventStoreTable.AGGREGATE_ID; +import static org.apache.james.eventsourcing.eventstore.postgres.PostgresEventStoreModule.PostgresEventStoreTable.EVENT; +import static org.apache.james.eventsourcing.eventstore.postgres.PostgresEventStoreModule.PostgresEventStoreTable.EVENT_ID; +import static org.apache.james.eventsourcing.eventstore.postgres.PostgresEventStoreModule.PostgresEventStoreTable.SNAPSHOT; +import static org.apache.james.eventsourcing.eventstore.postgres.PostgresEventStoreModule.PostgresEventStoreTable.TABLE_NAME; + +import java.util.List; +import java.util.Optional; + +import jakarta.inject.Inject; + +import org.apache.james.backends.postgres.utils.PostgresExecutor; +import org.apache.james.eventsourcing.AggregateId; +import org.apache.james.eventsourcing.Event; +import org.apache.james.eventsourcing.EventId; +import org.apache.james.eventsourcing.eventstore.History; +import org.apache.james.eventsourcing.eventstore.JsonEventSerializer; +import org.apache.james.util.ReactorUtils; +import org.jooq.JSON; +import org.jooq.Record; + +import com.fasterxml.jackson.core.JsonProcessingException; +import com.google.common.collect.ImmutableList; + +import reactor.core.publisher.Flux; +import reactor.core.publisher.Mono; +import scala.jdk.javaapi.CollectionConverters; + +public class PostgresEventStoreDAO { + private PostgresExecutor postgresExecutor; + private JsonEventSerializer jsonEventSerializer; + + @Inject + public PostgresEventStoreDAO(PostgresExecutor postgresExecutor, JsonEventSerializer jsonEventSerializer) { + this.postgresExecutor = postgresExecutor; + this.jsonEventSerializer = jsonEventSerializer; + } + + public Mono appendAll(List events, Optional lastSnapshot) { + return postgresExecutor.executeVoid(dslContext -> Mono.from(dslContext.insertInto(TABLE_NAME, AGGREGATE_ID, EVENT_ID, EVENT) + .valuesOfRecords(events.stream().map(event -> dslContext.newRecord(AGGREGATE_ID, EVENT_ID, EVENT) + .value1(event.getAggregateId().asAggregateKey()) + .value2(event.eventId().serialize()) + .value3(convertToJooqJson(event))) + .collect(ImmutableList.toImmutableList())))) + .then(lastSnapshot.map(eventId -> insertSnapshot(events.iterator().next().getAggregateId(), eventId)).orElse(Mono.empty())); + } + + private Mono insertSnapshot(AggregateId aggregateId, EventId snapshotId) { + return postgresExecutor.executeVoid(dslContext -> Mono.from(dslContext.update(TABLE_NAME) + .set(SNAPSHOT, snapshotId.serialize()) + .where(AGGREGATE_ID.eq(aggregateId.asAggregateKey())))); + } + + private JSON convertToJooqJson(Event event) { + try { + return JSON.json(jsonEventSerializer.serialize(event)); + } catch (JsonProcessingException e) { + throw new RuntimeException(e); + } + } + + public Mono getSnapshot(AggregateId aggregateId) { + return postgresExecutor.executeRow(dslContext -> Mono.from(dslContext.select(SNAPSHOT) + .from(TABLE_NAME) + .where(AGGREGATE_ID.eq(aggregateId.asAggregateKey())) + .limit(1))) + .map(record -> EventId.fromSerialized(Optional.ofNullable(record.get(SNAPSHOT)).orElse(0))); + } + + public Mono getEventsOfAggregate(AggregateId aggregateId, EventId snapshotId) { + return postgresExecutor.executeRows(dslContext -> Flux.from(dslContext.selectFrom(TABLE_NAME) + .where(AGGREGATE_ID.eq(aggregateId.asAggregateKey())) + .and(EVENT_ID.greaterOrEqual(snapshotId.value())) + .orderBy(EVENT_ID))) + .concatMap(this::toEvent) + .collect(ImmutableList.toImmutableList()) + .map(this::asHistory); + } + + public Mono getEventsOfAggregate(AggregateId aggregateId) { + return postgresExecutor.executeRows(dslContext -> Flux.from(dslContext.selectFrom(TABLE_NAME) + .where(AGGREGATE_ID.eq(aggregateId.asAggregateKey())) + .orderBy(EVENT_ID))) + .concatMap(this::toEvent) + .collect(ImmutableList.toImmutableList()) + .map(this::asHistory); + } + + public Mono delete(AggregateId aggregateId) { + return postgresExecutor.executeVoid(dslContext -> Mono.from(dslContext.deleteFrom(TABLE_NAME) + .where(AGGREGATE_ID.eq(aggregateId.asAggregateKey())))); + } + + private History asHistory(List events) { + return History.of(CollectionConverters.asScala(events).toList()); + } + + private Mono toEvent(Record record) { + return Mono.fromCallable(() -> jsonEventSerializer.deserialize(record.get(EVENT).data())) + .subscribeOn(ReactorUtils.BLOCKING_CALL_WRAPPER); + } +} diff --git a/event-sourcing/event-store-postgres/src/main/java/org/apache/james/eventsourcing/eventstore/postgres/PostgresEventStoreModule.java b/event-sourcing/event-store-postgres/src/main/java/org/apache/james/eventsourcing/eventstore/postgres/PostgresEventStoreModule.java new file mode 100644 index 00000000000..f90eb5c1cc1 --- /dev/null +++ b/event-sourcing/event-store-postgres/src/main/java/org/apache/james/eventsourcing/eventstore/postgres/PostgresEventStoreModule.java @@ -0,0 +1,63 @@ +/**************************************************************** + * 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.eventsourcing.eventstore.postgres; + +import static org.apache.james.eventsourcing.eventstore.postgres.PostgresEventStoreModule.PostgresEventStoreTable.INDEX; +import static org.apache.james.eventsourcing.eventstore.postgres.PostgresEventStoreModule.PostgresEventStoreTable.TABLE; + +import org.apache.james.backends.postgres.PostgresIndex; +import org.apache.james.backends.postgres.PostgresModule; +import org.apache.james.backends.postgres.PostgresTable; +import org.jooq.Field; +import org.jooq.JSON; +import org.jooq.Record; +import org.jooq.Table; +import org.jooq.impl.DSL; +import org.jooq.impl.SQLDataType; + +public interface PostgresEventStoreModule { + interface PostgresEventStoreTable { + Table TABLE_NAME = DSL.table("event_store"); + + Field AGGREGATE_ID = DSL.field("aggregate_id", SQLDataType.VARCHAR.notNull()); + Field EVENT_ID = DSL.field("event_id", SQLDataType.INTEGER.notNull()); + Field SNAPSHOT = DSL.field("snapshot", SQLDataType.INTEGER); + Field EVENT = DSL.field("event", SQLDataType.JSON.notNull()); + + PostgresTable TABLE = PostgresTable.name(TABLE_NAME.getName()) + .createTableStep(((dsl, tableName) -> dsl.createTableIfNotExists(tableName) + .column(AGGREGATE_ID) + .column(EVENT_ID) + .column(SNAPSHOT) + .column(EVENT) + .constraint(DSL.primaryKey(AGGREGATE_ID, EVENT_ID)))) + .disableRowLevelSecurity() + .build(); + + PostgresIndex INDEX = PostgresIndex.name("event_store_aggregate_id_index") + .createIndexStep((dslContext, indexName) -> dslContext.createIndexIfNotExists(indexName) + .on(TABLE_NAME, AGGREGATE_ID)); + } + + PostgresModule MODULE = PostgresModule.builder() + .addTable(TABLE) + .addIndex(INDEX) + .build(); +} diff --git a/event-sourcing/event-store-postgres/src/test/java/org/apache/james/eventsourcing/eventstore/postgres/PostgresEventSourcingSystemTest.java b/event-sourcing/event-store-postgres/src/test/java/org/apache/james/eventsourcing/eventstore/postgres/PostgresEventSourcingSystemTest.java new file mode 100644 index 00000000000..1faf9842e2d --- /dev/null +++ b/event-sourcing/event-store-postgres/src/test/java/org/apache/james/eventsourcing/eventstore/postgres/PostgresEventSourcingSystemTest.java @@ -0,0 +1,27 @@ +/**************************************************************** + * 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.eventsourcing.eventstore.postgres; + +import org.apache.james.eventsourcing.EventSourcingSystemTest; +import org.junit.jupiter.api.extension.ExtendWith; + +@ExtendWith(PostgresEventStoreExtensionForTestEvents.class) +public class PostgresEventSourcingSystemTest implements EventSourcingSystemTest { +} diff --git a/event-sourcing/event-store-postgres/src/test/java/org/apache/james/eventsourcing/eventstore/postgres/PostgresEventStoreExtension.java b/event-sourcing/event-store-postgres/src/test/java/org/apache/james/eventsourcing/eventstore/postgres/PostgresEventStoreExtension.java new file mode 100644 index 00000000000..6f5ea91e1c7 --- /dev/null +++ b/event-sourcing/event-store-postgres/src/test/java/org/apache/james/eventsourcing/eventstore/postgres/PostgresEventStoreExtension.java @@ -0,0 +1,72 @@ +/**************************************************************** + * 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.eventsourcing.eventstore.postgres; + +import org.apache.james.backends.postgres.PostgresExtension; +import org.apache.james.eventsourcing.eventstore.EventStore; +import org.apache.james.eventsourcing.eventstore.JsonEventSerializer; +import org.junit.jupiter.api.extension.AfterAllCallback; +import org.junit.jupiter.api.extension.AfterEachCallback; +import org.junit.jupiter.api.extension.BeforeAllCallback; +import org.junit.jupiter.api.extension.BeforeEachCallback; +import org.junit.jupiter.api.extension.ExtensionContext; +import org.junit.jupiter.api.extension.ParameterContext; +import org.junit.jupiter.api.extension.ParameterResolutionException; +import org.junit.jupiter.api.extension.ParameterResolver; + +public class PostgresEventStoreExtension implements AfterAllCallback, BeforeAllCallback, AfterEachCallback, BeforeEachCallback, ParameterResolver { + private PostgresExtension postgresExtension; + private JsonEventSerializer jsonEventSerializer; + + public PostgresEventStoreExtension(JsonEventSerializer jsonEventSerializer) { + this.jsonEventSerializer = jsonEventSerializer; + this.postgresExtension = PostgresExtension.withoutRowLevelSecurity(PostgresEventStoreModule.MODULE); + } + + @Override + public void afterAll(ExtensionContext extensionContext) { + postgresExtension.afterAll(extensionContext); + } + + @Override + public void afterEach(ExtensionContext extensionContext) { + postgresExtension.afterEach(extensionContext); + } + + @Override + public void beforeAll(ExtensionContext extensionContext) throws Exception { + postgresExtension.beforeAll(extensionContext); + } + + @Override + public void beforeEach(ExtensionContext extensionContext) { + postgresExtension.beforeEach(extensionContext); + } + + @Override + public boolean supportsParameter(ParameterContext parameterContext, ExtensionContext extensionContext) throws ParameterResolutionException { + return parameterContext.getParameter().getType() == EventStore.class; + } + + @Override + public PostgresEventStore resolveParameter(ParameterContext parameterContext, ExtensionContext extensionContext) throws ParameterResolutionException { + return new PostgresEventStore(new PostgresEventStoreDAO(postgresExtension.getDefaultPostgresExecutor(), jsonEventSerializer)); + } +} diff --git a/event-sourcing/event-store-postgres/src/test/java/org/apache/james/eventsourcing/eventstore/postgres/PostgresEventStoreExtensionForTestEvents.java b/event-sourcing/event-store-postgres/src/test/java/org/apache/james/eventsourcing/eventstore/postgres/PostgresEventStoreExtensionForTestEvents.java new file mode 100644 index 00000000000..dcebb2932ad --- /dev/null +++ b/event-sourcing/event-store-postgres/src/test/java/org/apache/james/eventsourcing/eventstore/postgres/PostgresEventStoreExtensionForTestEvents.java @@ -0,0 +1,29 @@ +/**************************************************************** + * 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.eventsourcing.eventstore.postgres; + +import org.apache.james.eventsourcing.eventstore.JsonEventSerializer; +import org.apache.james.eventsourcing.eventstore.dto.TestEventDTOModules; + +public class PostgresEventStoreExtensionForTestEvents extends PostgresEventStoreExtension { + public PostgresEventStoreExtensionForTestEvents() { + super(JsonEventSerializer.forModules(TestEventDTOModules.TEST_TYPE(), TestEventDTOModules.SNAPSHOT_TYPE()).withoutNestedType()); + } +} diff --git a/event-sourcing/event-store-postgres/src/test/java/org/apache/james/eventsourcing/eventstore/postgres/PostgresEventStoreTest.java b/event-sourcing/event-store-postgres/src/test/java/org/apache/james/eventsourcing/eventstore/postgres/PostgresEventStoreTest.java new file mode 100644 index 00000000000..a1a00f8a3d4 --- /dev/null +++ b/event-sourcing/event-store-postgres/src/test/java/org/apache/james/eventsourcing/eventstore/postgres/PostgresEventStoreTest.java @@ -0,0 +1,65 @@ +/**************************************************************** + * 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.eventsourcing.eventstore.postgres; + +import static org.assertj.core.api.Assertions.assertThat; + +import org.apache.james.eventsourcing.Event; +import org.apache.james.eventsourcing.EventId; +import org.apache.james.eventsourcing.TestEvent; +import org.apache.james.eventsourcing.eventstore.EventStore; +import org.apache.james.eventsourcing.eventstore.EventStoreContract; +import org.apache.james.eventsourcing.eventstore.History; +import org.apache.james.eventsourcing.eventstore.dto.SnapshotEvent; +import org.junit.jupiter.api.Test; +import org.junit.jupiter.api.extension.ExtendWith; + +import reactor.core.publisher.Mono; + +@ExtendWith(PostgresEventStoreExtensionForTestEvents.class) +public class PostgresEventStoreTest implements EventStoreContract { + @Test + void getEventsOfAggregateShouldResumeFromSnapshot(EventStore testee) { + Event event1 = new TestEvent(EventId.first(), EventStoreContract.AGGREGATE_1(), "first"); + Event event2 = new SnapshotEvent(EventId.first().next(), EventStoreContract.AGGREGATE_1(), "second"); + Event event3 = new TestEvent(EventId.first().next().next(), EventStoreContract.AGGREGATE_1(), "third"); + + Mono.from(testee.append(event1)).block(); + Mono.from(testee.append(event2)).block(); + Mono.from(testee.append(event3)).block(); + + assertThat(Mono.from(testee.getEventsOfAggregate(EventStoreContract.AGGREGATE_1())).block()) + .isEqualTo(History.of(event2, event3)); + } + + @Test + void getEventsOfAggregateShouldResumeFromLatestSnapshot(EventStore testee) { + Event event1 = new SnapshotEvent(EventId.first(), EventStoreContract.AGGREGATE_1(), "first"); + Event event2 = new TestEvent(EventId.first().next(), EventStoreContract.AGGREGATE_1(), "second"); + Event event3 = new SnapshotEvent(EventId.first().next().next(), EventStoreContract.AGGREGATE_1(), "third"); + + Mono.from(testee.append(event1)).block(); + Mono.from(testee.append(event2)).block(); + Mono.from(testee.append(event3)).block(); + + assertThat(Mono.from(testee.getEventsOfAggregate(EventStoreContract.AGGREGATE_1())).block()) + .isEqualTo(History.of(event3)); + } +} \ No newline at end of file diff --git a/event-sourcing/pom.xml b/event-sourcing/pom.xml index c4033287a87..17e3c3747d9 100644 --- a/event-sourcing/pom.xml +++ b/event-sourcing/pom.xml @@ -38,6 +38,7 @@ event-store-cassandra event-store-jpa event-store-memory + event-store-postgres diff --git a/mailbox/api/src/main/java/org/apache/james/mailbox/MessageManager.java b/mailbox/api/src/main/java/org/apache/james/mailbox/MessageManager.java index bab6c535309..c87729aed68 100644 --- a/mailbox/api/src/main/java/org/apache/james/mailbox/MessageManager.java +++ b/mailbox/api/src/main/java/org/apache/james/mailbox/MessageManager.java @@ -38,7 +38,6 @@ import jakarta.mail.internet.SharedInputStream; import org.apache.commons.io.IOUtils; -import org.apache.james.mailbox.MailboxManager.MessageCapabilities; import org.apache.james.mailbox.MessageManager.MailboxMetaData.RecentMode; import org.apache.james.mailbox.exception.MailboxException; import org.apache.james.mailbox.exception.UnsupportedCriteriaException; @@ -441,7 +440,6 @@ default Publisher getMessagesReactive(MessageRange set, FetchGrou */ Mailbox getMailboxEntity() throws MailboxException; - EnumSet getSupportedMessageCapabilities(); /** * Gets the id of the referenced mailbox diff --git a/mailbox/api/src/main/java/org/apache/james/mailbox/UuidBackedAttachmentIdFactory.java b/mailbox/api/src/main/java/org/apache/james/mailbox/UuidBackedAttachmentIdFactory.java new file mode 100644 index 00000000000..3cebb5ab36e --- /dev/null +++ b/mailbox/api/src/main/java/org/apache/james/mailbox/UuidBackedAttachmentIdFactory.java @@ -0,0 +1,34 @@ +/**************************************************************** + * 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.mailbox; + +import org.apache.james.mailbox.model.UuidBackedAttachmentId; + +public class UuidBackedAttachmentIdFactory implements AttachmentIdFactory { + @Override + public UuidBackedAttachmentId random() { + return UuidBackedAttachmentId.random(); + } + + @Override + public UuidBackedAttachmentId from(String id) { + return UuidBackedAttachmentId.from(id); + } +} diff --git a/mailbox/api/src/main/java/org/apache/james/mailbox/model/UuidBackedAttachmentId.java b/mailbox/api/src/main/java/org/apache/james/mailbox/model/UuidBackedAttachmentId.java new file mode 100644 index 00000000000..12186a28821 --- /dev/null +++ b/mailbox/api/src/main/java/org/apache/james/mailbox/model/UuidBackedAttachmentId.java @@ -0,0 +1,76 @@ +/**************************************************************** + * 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.mailbox.model; + +import java.util.UUID; + +import com.google.common.base.MoreObjects; +import com.google.common.base.Objects; + +public class UuidBackedAttachmentId implements AttachmentId { + public static UuidBackedAttachmentId random() { + return new UuidBackedAttachmentId(UUID.randomUUID()); + } + + public static UuidBackedAttachmentId from(String id) { + return new UuidBackedAttachmentId(UUID.fromString(id)); + } + + public static UuidBackedAttachmentId from(UUID id) { + return new UuidBackedAttachmentId(id); + } + + private final UUID id; + + private UuidBackedAttachmentId(UUID id) { + this.id = id; + } + + @Override + public String getId() { + return id.toString(); + } + + @Override + public UUID asUUID() { + return id; + } + + @Override + public boolean equals(Object obj) { + if (obj instanceof UuidBackedAttachmentId) { + UuidBackedAttachmentId other = (UuidBackedAttachmentId) obj; + return Objects.equal(id, other.id); + } + return false; + } + + @Override + public int hashCode() { + return Objects.hashCode(id); + } + + @Override + public String toString() { + return MoreObjects + .toStringHelper(this) + .add("id", id) + .toString(); + } +} diff --git a/mailbox/cassandra/src/main/java/org/apache/james/mailbox/cassandra/quota/Limits.java b/mailbox/api/src/main/java/org/apache/james/mailbox/quota/Limits.java similarity index 97% rename from mailbox/cassandra/src/main/java/org/apache/james/mailbox/cassandra/quota/Limits.java rename to mailbox/api/src/main/java/org/apache/james/mailbox/quota/Limits.java index 3ef7aec0975..f278d03ed75 100644 --- a/mailbox/cassandra/src/main/java/org/apache/james/mailbox/cassandra/quota/Limits.java +++ b/mailbox/api/src/main/java/org/apache/james/mailbox/quota/Limits.java @@ -17,7 +17,7 @@ * under the License. * ****************************************************************/ -package org.apache.james.mailbox.cassandra.quota; +package org.apache.james.mailbox.quota; import java.util.Optional; diff --git a/mailbox/cassandra/src/main/java/org/apache/james/mailbox/cassandra/quota/QuotaCodec.java b/mailbox/api/src/main/java/org/apache/james/mailbox/quota/QuotaCodec.java similarity index 90% rename from mailbox/cassandra/src/main/java/org/apache/james/mailbox/cassandra/quota/QuotaCodec.java rename to mailbox/api/src/main/java/org/apache/james/mailbox/quota/QuotaCodec.java index 87b6cdcef79..d3d9b5cd67a 100644 --- a/mailbox/cassandra/src/main/java/org/apache/james/mailbox/cassandra/quota/QuotaCodec.java +++ b/mailbox/api/src/main/java/org/apache/james/mailbox/quota/QuotaCodec.java @@ -16,7 +16,7 @@ * specific language governing permissions and limitations * * under the License. * ****************************************************************/ -package org.apache.james.mailbox.cassandra.quota; +package org.apache.james.mailbox.quota; import java.util.Optional; import java.util.function.Function; @@ -30,18 +30,18 @@ public class QuotaCodec { private static final long INFINITE = -1; private static final long NO_RIGHT = 0L; - static Long quotaValueToLong(QuotaLimitValue value) { + public static Long quotaValueToLong(QuotaLimitValue value) { if (value.isUnlimited()) { return INFINITE; } return value.asLong(); } - static Optional longToQuotaSize(Long value) { + public static Optional longToQuotaSize(Long value) { return longToQuotaValue(value, QuotaSizeLimit.unlimited(), QuotaSizeLimit::size); } - static Optional longToQuotaCount(Long value) { + public static Optional longToQuotaCount(Long value) { return longToQuotaValue(value, QuotaCountLimit.unlimited(), QuotaCountLimit::count); } diff --git a/mailbox/api/src/test/java/org/apache/james/mailbox/MailboxManagerTest.java b/mailbox/api/src/test/java/org/apache/james/mailbox/MailboxManagerTest.java index 422fe64e630..7c96b5b5aa2 100644 --- a/mailbox/api/src/test/java/org/apache/james/mailbox/MailboxManagerTest.java +++ b/mailbox/api/src/test/java/org/apache/james/mailbox/MailboxManagerTest.java @@ -742,7 +742,9 @@ void getAllAnnotationsShouldRetrieveStoredAnnotations() throws Exception { mailboxManager.updateAnnotations(inbox, session, annotations); - assertThat(mailboxManager.getAllAnnotations(inbox, session)).isEqualTo(annotations); + assertThat(mailboxManager.getAllAnnotations(inbox, session)) + .hasSize(annotations.size()) + .containsAnyElementsOf(annotations); } @Test diff --git a/mailbox/cassandra/src/main/java/org/apache/james/mailbox/cassandra/CassandraMailboxSessionMapperFactory.java b/mailbox/cassandra/src/main/java/org/apache/james/mailbox/cassandra/CassandraMailboxSessionMapperFactory.java index 6c8e39c5b3c..55a27497830 100644 --- a/mailbox/cassandra/src/main/java/org/apache/james/mailbox/cassandra/CassandraMailboxSessionMapperFactory.java +++ b/mailbox/cassandra/src/main/java/org/apache/james/mailbox/cassandra/CassandraMailboxSessionMapperFactory.java @@ -180,12 +180,12 @@ public SubscriptionMapper createSubscriptionMapper(MailboxSession mailboxSession } @Override - public ModSeqProvider getModSeqProvider() { + public ModSeqProvider getModSeqProvider(MailboxSession session) { return modSeqProvider; } @Override - public UidProvider getUidProvider() { + public UidProvider getUidProvider(MailboxSession session) { return uidProvider; } diff --git a/mailbox/cassandra/src/main/java/org/apache/james/mailbox/cassandra/quota/CassandraCurrentQuotaManagerV2.java b/mailbox/cassandra/src/main/java/org/apache/james/mailbox/cassandra/quota/CassandraCurrentQuotaManagerV2.java index fba7cc01ab6..d455706f429 100644 --- a/mailbox/cassandra/src/main/java/org/apache/james/mailbox/cassandra/quota/CassandraCurrentQuotaManagerV2.java +++ b/mailbox/cassandra/src/main/java/org/apache/james/mailbox/cassandra/quota/CassandraCurrentQuotaManagerV2.java @@ -26,7 +26,6 @@ import jakarta.inject.Inject; import org.apache.james.backends.cassandra.components.CassandraQuotaCurrentValueDao; -import org.apache.james.backends.cassandra.components.CassandraQuotaCurrentValueDao.QuotaKey; import org.apache.james.core.quota.QuotaComponent; import org.apache.james.core.quota.QuotaCountUsage; import org.apache.james.core.quota.QuotaCurrentValue; @@ -117,16 +116,16 @@ public Mono setCurrentQuotas(QuotaOperation quotaOperation) { }); } - private QuotaKey asQuotaKeyCount(QuotaRoot quotaRoot) { + private QuotaCurrentValue.Key asQuotaKeyCount(QuotaRoot quotaRoot) { return asQuotaKey(quotaRoot, QuotaType.COUNT); } - private QuotaKey asQuotaKeySize(QuotaRoot quotaRoot) { + private QuotaCurrentValue.Key asQuotaKeySize(QuotaRoot quotaRoot) { return asQuotaKey(quotaRoot, QuotaType.SIZE); } - private QuotaKey asQuotaKey(QuotaRoot quotaRoot, QuotaType quotaType) { - return QuotaKey.of( + private QuotaCurrentValue.Key asQuotaKey(QuotaRoot quotaRoot, QuotaType quotaType) { + return QuotaCurrentValue.Key.of( QuotaComponent.MAILBOX, quotaRoot.asString(), quotaType); diff --git a/mailbox/cassandra/src/main/java/org/apache/james/mailbox/cassandra/quota/CassandraGlobalMaxQuotaDao.java b/mailbox/cassandra/src/main/java/org/apache/james/mailbox/cassandra/quota/CassandraGlobalMaxQuotaDao.java index 02e777d7f18..3be44c6c6d6 100644 --- a/mailbox/cassandra/src/main/java/org/apache/james/mailbox/cassandra/quota/CassandraGlobalMaxQuotaDao.java +++ b/mailbox/cassandra/src/main/java/org/apache/james/mailbox/cassandra/quota/CassandraGlobalMaxQuotaDao.java @@ -39,6 +39,8 @@ import org.apache.james.backends.cassandra.utils.CassandraAsyncExecutor; import org.apache.james.core.quota.QuotaCountLimit; import org.apache.james.core.quota.QuotaSizeLimit; +import org.apache.james.mailbox.quota.Limits; +import org.apache.james.mailbox.quota.QuotaCodec; import com.datastax.oss.driver.api.core.CqlSession; import com.datastax.oss.driver.api.core.cql.PreparedStatement; diff --git a/mailbox/cassandra/src/main/java/org/apache/james/mailbox/cassandra/quota/CassandraPerDomainMaxQuotaDao.java b/mailbox/cassandra/src/main/java/org/apache/james/mailbox/cassandra/quota/CassandraPerDomainMaxQuotaDao.java index c583a6a487d..53267376eda 100644 --- a/mailbox/cassandra/src/main/java/org/apache/james/mailbox/cassandra/quota/CassandraPerDomainMaxQuotaDao.java +++ b/mailbox/cassandra/src/main/java/org/apache/james/mailbox/cassandra/quota/CassandraPerDomainMaxQuotaDao.java @@ -35,6 +35,8 @@ import org.apache.james.core.quota.QuotaCountLimit; import org.apache.james.core.quota.QuotaSizeLimit; import org.apache.james.mailbox.cassandra.table.CassandraDomainMaxQuota; +import org.apache.james.mailbox.quota.Limits; +import org.apache.james.mailbox.quota.QuotaCodec; import com.datastax.oss.driver.api.core.CqlSession; import com.datastax.oss.driver.api.core.cql.PreparedStatement; diff --git a/mailbox/cassandra/src/main/java/org/apache/james/mailbox/cassandra/quota/CassandraPerUserMaxQuotaDao.java b/mailbox/cassandra/src/main/java/org/apache/james/mailbox/cassandra/quota/CassandraPerUserMaxQuotaDao.java index db1c36eeaa0..932da5ea912 100644 --- a/mailbox/cassandra/src/main/java/org/apache/james/mailbox/cassandra/quota/CassandraPerUserMaxQuotaDao.java +++ b/mailbox/cassandra/src/main/java/org/apache/james/mailbox/cassandra/quota/CassandraPerUserMaxQuotaDao.java @@ -35,6 +35,8 @@ import org.apache.james.core.quota.QuotaSizeLimit; import org.apache.james.mailbox.cassandra.table.CassandraMaxQuota; import org.apache.james.mailbox.model.QuotaRoot; +import org.apache.james.mailbox.quota.Limits; +import org.apache.james.mailbox.quota.QuotaCodec; import com.datastax.oss.driver.api.core.CqlSession; import com.datastax.oss.driver.api.core.cql.PreparedStatement; diff --git a/mailbox/cassandra/src/main/java/org/apache/james/mailbox/cassandra/quota/CassandraPerUserMaxQuotaManagerV1.java b/mailbox/cassandra/src/main/java/org/apache/james/mailbox/cassandra/quota/CassandraPerUserMaxQuotaManagerV1.java index d6dead22c40..6016288f5d5 100644 --- a/mailbox/cassandra/src/main/java/org/apache/james/mailbox/cassandra/quota/CassandraPerUserMaxQuotaManagerV1.java +++ b/mailbox/cassandra/src/main/java/org/apache/james/mailbox/cassandra/quota/CassandraPerUserMaxQuotaManagerV1.java @@ -32,6 +32,7 @@ import org.apache.james.core.quota.QuotaSizeLimit; import org.apache.james.mailbox.model.Quota; import org.apache.james.mailbox.model.QuotaRoot; +import org.apache.james.mailbox.quota.Limits; import org.apache.james.mailbox.quota.MaxQuotaManager; import com.google.common.collect.ImmutableMap; diff --git a/mailbox/cassandra/src/main/java/org/apache/james/mailbox/cassandra/quota/CassandraPerUserMaxQuotaManagerV2.java b/mailbox/cassandra/src/main/java/org/apache/james/mailbox/cassandra/quota/CassandraPerUserMaxQuotaManagerV2.java index 0895b24c684..ba5102f7020 100644 --- a/mailbox/cassandra/src/main/java/org/apache/james/mailbox/cassandra/quota/CassandraPerUserMaxQuotaManagerV2.java +++ b/mailbox/cassandra/src/main/java/org/apache/james/mailbox/cassandra/quota/CassandraPerUserMaxQuotaManagerV2.java @@ -19,7 +19,6 @@ package org.apache.james.mailbox.cassandra.quota; -import static org.apache.james.backends.cassandra.components.CassandraQuotaLimitDao.QuotaLimitKey; import static org.apache.james.util.ReactorUtils.publishIfPresent; import java.util.Map; @@ -41,8 +40,10 @@ import org.apache.james.core.quota.QuotaType; import org.apache.james.mailbox.model.Quota; import org.apache.james.mailbox.model.QuotaRoot; +import org.apache.james.mailbox.quota.Limits; import org.apache.james.mailbox.quota.MaxQuotaManager; import org.apache.james.mailbox.quota.QuotaChangeNotifier; +import org.apache.james.mailbox.quota.QuotaCodec; import com.google.common.collect.ImmutableMap; @@ -137,7 +138,7 @@ public void removeDomainMaxMessage(Domain domain) { @Override public Mono removeDomainMaxMessageReactive(Domain domain) { - return cassandraQuotaLimitDao.deleteQuotaLimit(QuotaLimitKey.of(QuotaComponent.MAILBOX, QuotaScope.DOMAIN, domain.asString(), QuotaType.COUNT)) + return cassandraQuotaLimitDao.deleteQuotaLimit(QuotaLimit.QuotaLimitKey.of(QuotaComponent.MAILBOX, QuotaScope.DOMAIN, domain.asString(), QuotaType.COUNT)) .then(Mono.from(quotaChangeNotifier.notifyUpdate(domain))); } @@ -148,7 +149,7 @@ public void removeDomainMaxStorage(Domain domain) { @Override public Mono removeDomainMaxStorageReactive(Domain domain) { - return cassandraQuotaLimitDao.deleteQuotaLimit(QuotaLimitKey.of(QuotaComponent.MAILBOX, QuotaScope.DOMAIN, domain.asString(), QuotaType.SIZE)) + return cassandraQuotaLimitDao.deleteQuotaLimit(QuotaLimit.QuotaLimitKey.of(QuotaComponent.MAILBOX, QuotaScope.DOMAIN, domain.asString(), QuotaType.SIZE)) .then(Mono.from(quotaChangeNotifier.notifyUpdate(domain))); } @@ -179,7 +180,7 @@ public void removeMaxMessage(QuotaRoot quotaRoot) { @Override public Mono removeMaxMessageReactive(QuotaRoot quotaRoot) { - return cassandraQuotaLimitDao.deleteQuotaLimit(QuotaLimitKey.of(QuotaComponent.MAILBOX, QuotaScope.USER, quotaRoot.getValue(), QuotaType.COUNT)) + return cassandraQuotaLimitDao.deleteQuotaLimit(QuotaLimit.QuotaLimitKey.of(QuotaComponent.MAILBOX, QuotaScope.USER, quotaRoot.getValue(), QuotaType.COUNT)) .then(Mono.from(quotaChangeNotifier.notifyUpdate(quotaRoot))); } @@ -190,7 +191,7 @@ public void removeMaxStorage(QuotaRoot quotaRoot) { @Override public Mono removeMaxStorageReactive(QuotaRoot quotaRoot) { - return cassandraQuotaLimitDao.deleteQuotaLimit(QuotaLimitKey.of(QuotaComponent.MAILBOX, QuotaScope.USER, quotaRoot.getValue(), QuotaType.SIZE)) + return cassandraQuotaLimitDao.deleteQuotaLimit(QuotaLimit.QuotaLimitKey.of(QuotaComponent.MAILBOX, QuotaScope.USER, quotaRoot.getValue(), QuotaType.SIZE)) .then(Mono.from(quotaChangeNotifier.notifyUpdate(quotaRoot))); } @@ -217,7 +218,7 @@ public void removeGlobalMaxStorage() { @Override public Mono removeGlobalMaxStorageReactive() { - return cassandraQuotaLimitDao.deleteQuotaLimit(QuotaLimitKey.of(QuotaComponent.MAILBOX, QuotaScope.GLOBAL, GLOBAL_IDENTIFIER, QuotaType.SIZE)) + return cassandraQuotaLimitDao.deleteQuotaLimit(QuotaLimit.QuotaLimitKey.of(QuotaComponent.MAILBOX, QuotaScope.GLOBAL, GLOBAL_IDENTIFIER, QuotaType.SIZE)) .then(Mono.from(quotaChangeNotifier.notifyGlobalUpdate())); } @@ -244,7 +245,7 @@ public void removeGlobalMaxMessage() { @Override public Mono removeGlobalMaxMessageReactive() { - return cassandraQuotaLimitDao.deleteQuotaLimit(QuotaLimitKey.of(QuotaComponent.MAILBOX, QuotaScope.GLOBAL, GLOBAL_IDENTIFIER, QuotaType.COUNT)) + return cassandraQuotaLimitDao.deleteQuotaLimit(QuotaLimit.QuotaLimitKey.of(QuotaComponent.MAILBOX, QuotaScope.GLOBAL, GLOBAL_IDENTIFIER, QuotaType.COUNT)) .then(Mono.from(quotaChangeNotifier.notifyGlobalUpdate())); } @@ -337,7 +338,7 @@ private Mono getLimits(QuotaScope quotaScope, String identifier) { } private Mono getMaxMessageReactive(QuotaScope quotaScope, String identifier) { - return cassandraQuotaLimitDao.getQuotaLimit(QuotaLimitKey.of(QuotaComponent.MAILBOX, quotaScope, identifier, QuotaType.COUNT)) + return cassandraQuotaLimitDao.getQuotaLimit(QuotaLimit.QuotaLimitKey.of(QuotaComponent.MAILBOX, quotaScope, identifier, QuotaType.COUNT)) .map(QuotaLimit::getQuotaLimit) .handle(publishIfPresent()) .map(QuotaCodec::longToQuotaCount) @@ -345,7 +346,7 @@ private Mono getMaxMessageReactive(QuotaScope quotaScope, Strin } public Mono getMaxStorageReactive(QuotaScope quotaScope, String identifier) { - return cassandraQuotaLimitDao.getQuotaLimit(QuotaLimitKey.of(QuotaComponent.MAILBOX, quotaScope, identifier, QuotaType.SIZE)) + return cassandraQuotaLimitDao.getQuotaLimit(QuotaLimit.QuotaLimitKey.of(QuotaComponent.MAILBOX, quotaScope, identifier, QuotaType.SIZE)) .map(QuotaLimit::getQuotaLimit) .handle(publishIfPresent()) .map(QuotaCodec::longToQuotaSize) diff --git a/mailbox/cassandra/src/test/java/org/apache/james/mailbox/cassandra/CassandraThreadIdGuessingAlgorithmTest.java b/mailbox/cassandra/src/test/java/org/apache/james/mailbox/cassandra/CassandraThreadIdGuessingAlgorithmTest.java index 546657676f9..1d7e8dd5a4e 100644 --- a/mailbox/cassandra/src/test/java/org/apache/james/mailbox/cassandra/CassandraThreadIdGuessingAlgorithmTest.java +++ b/mailbox/cassandra/src/test/java/org/apache/james/mailbox/cassandra/CassandraThreadIdGuessingAlgorithmTest.java @@ -52,8 +52,6 @@ import org.junit.jupiter.api.Test; import org.junit.jupiter.api.extension.RegisterExtension; -import reactor.core.publisher.Flux; - public class CassandraThreadIdGuessingAlgorithmTest extends ThreadIdGuessingAlgorithmContract { private CassandraMailboxManager mailboxManager; private CassandraThreadDAO threadDAO; @@ -94,8 +92,10 @@ protected MessageId initOtherBasedMessageId() { } @Override - protected Flux saveThreadData(Username username, Set mimeMessageIds, MessageId messageId, ThreadId threadId, Optional baseSubject) { - return threadDAO.insertSome(username, hashMimeMessagesIds(mimeMessageIds), messageId, threadId, hashSubject(baseSubject)); + protected void saveThreadData(Username username, Set mimeMessageIds, MessageId messageId, ThreadId threadId, Optional baseSubject) { + threadDAO.insertSome(username, hashMimeMessagesIds(mimeMessageIds), messageId, threadId, hashSubject(baseSubject)) + .then() + .block(); } @Test diff --git a/mailbox/cassandra/src/test/java/org/apache/james/mailbox/cassandra/mail/CassandraMapperProvider.java b/mailbox/cassandra/src/test/java/org/apache/james/mailbox/cassandra/mail/CassandraMapperProvider.java index e3e7cd9f2b8..4f5c310b181 100644 --- a/mailbox/cassandra/src/test/java/org/apache/james/mailbox/cassandra/mail/CassandraMapperProvider.java +++ b/mailbox/cassandra/src/test/java/org/apache/james/mailbox/cassandra/mail/CassandraMapperProvider.java @@ -40,8 +40,8 @@ import org.apache.james.mailbox.store.mail.MailboxMapper; import org.apache.james.mailbox.store.mail.MessageIdMapper; import org.apache.james.mailbox.store.mail.MessageMapper; +import org.apache.james.mailbox.store.mail.UidProvider; import org.apache.james.mailbox.store.mail.model.MapperProvider; -import org.apache.james.mailbox.store.mail.model.MessageUidProvider; import org.apache.james.utils.UpdatableTickingClock; import com.google.common.collect.ImmutableList; @@ -51,7 +51,7 @@ public class CassandraMapperProvider implements MapperProvider { private static final Factory MESSAGE_ID_FACTORY = new CassandraMessageId.Factory(); private final CassandraCluster cassandra; - private final MessageUidProvider messageUidProvider; + private final UidProvider messageUidProvider; private final CassandraModSeqProvider cassandraModSeqProvider; private final UpdatableTickingClock updatableTickingClock; private final MailboxSession mailboxSession = MailboxSessionUtil.create(Username.of("benwa")); @@ -60,7 +60,7 @@ public class CassandraMapperProvider implements MapperProvider { public CassandraMapperProvider(CassandraCluster cassandra, CassandraConfiguration cassandraConfiguration) { this.cassandra = cassandra; - messageUidProvider = new MessageUidProvider(); + messageUidProvider = new CassandraUidProvider(this.cassandra.getConf(), cassandraConfiguration); cassandraModSeqProvider = new CassandraModSeqProvider( this.cassandra.getConf(), cassandraConfiguration); @@ -116,8 +116,12 @@ public List getSupportedCapabilities() { } @Override - public MessageUid generateMessageUid() { - return messageUidProvider.next(); + public MessageUid generateMessageUid(Mailbox mailbox) { + try { + return messageUidProvider.nextUid(mailbox); + } catch (MailboxException e) { + throw new RuntimeException(e); + } } @Override diff --git a/mailbox/cassandra/src/test/java/org/apache/james/mailbox/cassandra/mail/CassandraMessageIdMapperTest.java b/mailbox/cassandra/src/test/java/org/apache/james/mailbox/cassandra/mail/CassandraMessageIdMapperTest.java index 00200d3b214..33e42502d04 100644 --- a/mailbox/cassandra/src/test/java/org/apache/james/mailbox/cassandra/mail/CassandraMessageIdMapperTest.java +++ b/mailbox/cassandra/src/test/java/org/apache/james/mailbox/cassandra/mail/CassandraMessageIdMapperTest.java @@ -152,7 +152,7 @@ void retrieveMessagesShouldNotReturnMessagesWhenFailToPersistInMessageDAO(Cassan .whenQueryStartsWith("UPDATE messagev3")); try { - message1.setUid(mapperProvider.generateMessageUid()); + message1.setUid(mapperProvider.generateMessageUid(benwaInboxMailbox)); message1.setModSeq(mapperProvider.generateModSeq(benwaInboxMailbox)); sut.save(message1); } catch (Exception e) { @@ -176,7 +176,7 @@ void retrieveMessagesShouldNotReturnMessagesWhenFailsToPersistBlobParts(Cassandr .whenQueryStartsWith("INSERT INTO blobparts (id,chunknumber,data)")); try { - message1.setUid(mapperProvider.generateMessageUid()); + message1.setUid(mapperProvider.generateMessageUid(benwaInboxMailbox)); message1.setModSeq(mapperProvider.generateModSeq(benwaInboxMailbox)); sut.save(message1); } catch (Exception e) { @@ -200,7 +200,7 @@ void retrieveMessagesShouldNotReturnMessagesWhenFailsToPersistBlobs(CassandraClu .whenQueryStartsWith("INSERT INTO blobs (id,position) VALUES (:id,:position)")); try { - message1.setUid(mapperProvider.generateMessageUid()); + message1.setUid(mapperProvider.generateMessageUid(benwaInboxMailbox)); message1.setModSeq(mapperProvider.generateModSeq(benwaInboxMailbox)); sut.save(message1); } catch (Exception e) { @@ -224,7 +224,7 @@ void retrieveMessagesShouldNotReturnMessagesWhenFailsToPersistInImapUidTable(Cas .whenQueryStartsWith("INSERT INTO imapuidtable")); try { - message1.setUid(mapperProvider.generateMessageUid()); + message1.setUid(mapperProvider.generateMessageUid(benwaInboxMailbox)); message1.setModSeq(mapperProvider.generateModSeq(benwaInboxMailbox)); sut.save(message1); } catch (Exception e) { @@ -248,7 +248,7 @@ void addShouldPersistInTableOfTruthWhenMessageIdTableWritesFails(CassandraCluste .whenQueryStartsWith("INSERT INTO messageidtable")); try { - message1.setUid(mapperProvider.generateMessageUid()); + message1.setUid(mapperProvider.generateMessageUid(benwaInboxMailbox)); message1.setModSeq(mapperProvider.generateModSeq(benwaInboxMailbox)); sut.save(message1); } catch (Exception e) { @@ -275,7 +275,7 @@ void addShouldRetryMessageDenormalization(CassandraCluster cassandra) throws Exc .times(5) .whenQueryStartsWith("INSERT INTO messageidtable")); - message1.setUid(mapperProvider.generateMessageUid()); + message1.setUid(mapperProvider.generateMessageUid(benwaInboxMailbox)); message1.setModSeq(mapperProvider.generateModSeq(benwaInboxMailbox)); sut.save(message1); diff --git a/mailbox/cassandra/src/test/java/org/apache/james/mailbox/cassandra/mail/CassandraMessageMapperRelaxedConsistencyTest.java b/mailbox/cassandra/src/test/java/org/apache/james/mailbox/cassandra/mail/CassandraMessageMapperRelaxedConsistencyTest.java index a71d7318973..202ce3797ac 100644 --- a/mailbox/cassandra/src/test/java/org/apache/james/mailbox/cassandra/mail/CassandraMessageMapperRelaxedConsistencyTest.java +++ b/mailbox/cassandra/src/test/java/org/apache/james/mailbox/cassandra/mail/CassandraMessageMapperRelaxedConsistencyTest.java @@ -98,5 +98,20 @@ public void setFlagsShouldWorkWithConcurrencyWithRemove() throws Exception { public void userFlagsUpdateShouldWorkInConcurrentEnvironment() throws Exception { super.userFlagsUpdateShouldWorkInConcurrentEnvironment(); } + + @Disabled("JAMES-3435 Without strong consistency flags update is not thread safe as long as it follows a read-before-write pattern") + @Override + public void updateFlagsWithRangeAllRangeShouldReturnUpdatedFlagsWithUidOrderAsc() { + } + + @Disabled("JAMES-3435 Without strong consistency flags update is not thread safe as long as it follows a read-before-write pattern") + @Override + public void updateFlagsOnRangeShouldReturnUpdatedFlagsWithUidOrderAsc() { + } + + @Disabled("JAMES-3435 Without strong consistency flags update is not thread safe as long as it follows a read-before-write pattern") + @Override + public void updateFlagsWithRangeFromShouldReturnUpdatedFlagsWithUidOrderAsc() { + } } } diff --git a/mailbox/jpa/src/main/java/org/apache/james/mailbox/jpa/JPAMailboxSessionMapperFactory.java b/mailbox/jpa/src/main/java/org/apache/james/mailbox/jpa/JPAMailboxSessionMapperFactory.java index 7f4f05d6c24..233d0e45a6d 100644 --- a/mailbox/jpa/src/main/java/org/apache/james/mailbox/jpa/JPAMailboxSessionMapperFactory.java +++ b/mailbox/jpa/src/main/java/org/apache/james/mailbox/jpa/JPAMailboxSessionMapperFactory.java @@ -102,12 +102,12 @@ public AnnotationMapper createAnnotationMapper(MailboxSession session) { } @Override - public UidProvider getUidProvider() { + public UidProvider getUidProvider(MailboxSession session) { return uidProvider; } @Override - public ModSeqProvider getModSeqProvider() { + public ModSeqProvider getModSeqProvider(MailboxSession session) { return modSeqProvider; } diff --git a/mailbox/jpa/src/test/java/org/apache/james/mailbox/jpa/mail/JPAMapperProvider.java b/mailbox/jpa/src/test/java/org/apache/james/mailbox/jpa/mail/JPAMapperProvider.java index fdad8e414ac..8bbf83238c0 100644 --- a/mailbox/jpa/src/test/java/org/apache/james/mailbox/jpa/mail/JPAMapperProvider.java +++ b/mailbox/jpa/src/test/java/org/apache/james/mailbox/jpa/mail/JPAMapperProvider.java @@ -105,7 +105,7 @@ public MessageIdMapper createMessageIdMapper() throws MailboxException { } @Override - public MessageUid generateMessageUid() { + public MessageUid generateMessageUid(Mailbox mailbox) { throw new NotImplementedException("not implemented"); } diff --git a/mailbox/memory/src/main/java/org/apache/james/mailbox/inmemory/InMemoryMailboxSessionMapperFactory.java b/mailbox/memory/src/main/java/org/apache/james/mailbox/inmemory/InMemoryMailboxSessionMapperFactory.java index 84250a64512..bef77415878 100644 --- a/mailbox/memory/src/main/java/org/apache/james/mailbox/inmemory/InMemoryMailboxSessionMapperFactory.java +++ b/mailbox/memory/src/main/java/org/apache/james/mailbox/inmemory/InMemoryMailboxSessionMapperFactory.java @@ -103,12 +103,12 @@ public AnnotationMapper createAnnotationMapper(MailboxSession session) { } @Override - public UidProvider getUidProvider() { + public UidProvider getUidProvider(MailboxSession session) { return uidProvider; } @Override - public ModSeqProvider getModSeqProvider() { + public ModSeqProvider getModSeqProvider(MailboxSession session) { return modSeqProvider; } diff --git a/mailbox/memory/src/test/java/org/apache/james/mailbox/inmemory/mail/InMemoryMapperProvider.java b/mailbox/memory/src/test/java/org/apache/james/mailbox/inmemory/mail/InMemoryMapperProvider.java index 286057ee167..e0eeeb3bfd0 100644 --- a/mailbox/memory/src/test/java/org/apache/james/mailbox/inmemory/mail/InMemoryMapperProvider.java +++ b/mailbox/memory/src/test/java/org/apache/james/mailbox/inmemory/mail/InMemoryMapperProvider.java @@ -90,7 +90,7 @@ public InMemoryId generateId() { } @Override - public MessageUid generateMessageUid() { + public MessageUid generateMessageUid(Mailbox mailbox) { return messageUidProvider.next(); } @@ -119,13 +119,13 @@ public List getSupportedCapabilities() { @Override public ModSeq generateModSeq(Mailbox mailbox) throws MailboxException { - return inMemoryMailboxSessionMapperFactory.getModSeqProvider() + return inMemoryMailboxSessionMapperFactory.getModSeqProvider(null) .nextModSeq(mailbox); } @Override public ModSeq highestModSeq(Mailbox mailbox) throws MailboxException { - return inMemoryMailboxSessionMapperFactory.getModSeqProvider() + return inMemoryMailboxSessionMapperFactory.getModSeqProvider(null) .highestModSeq(mailbox); } diff --git a/mailbox/plugin/deleted-messages-vault-postgres/pom.xml b/mailbox/plugin/deleted-messages-vault-postgres/pom.xml new file mode 100644 index 00000000000..856b49aa565 --- /dev/null +++ b/mailbox/plugin/deleted-messages-vault-postgres/pom.xml @@ -0,0 +1,83 @@ + + + + 4.0.0 + + org.apache.james + apache-james-mailbox + 3.9.0-SNAPSHOT + ../../pom.xml + + + apache-james-mailbox-deleted-messages-vault-postgres + Apache James :: Mailbox :: Plugin :: Deleted Messages Vault :: Postgres + Apache James Mailbox Deleted Messages Vault metadata on top of Postgres + + + + ${james.groupId} + apache-james-backends-postgres + + + ${james.groupId} + apache-james-backends-postgres + test-jar + test + + + ${james.groupId} + apache-james-mailbox-deleted-messages-vault + + + ${james.groupId} + apache-james-mailbox-deleted-messages-vault + test-jar + test + + + ${james.groupId} + apache-james-mailbox-memory + test + + + ${james.groupId} + apache-james-mailbox-postgres + + + ${james.groupId} + james-server-guice-common + test-jar + test + + + ${james.groupId} + james-server-testing + test + + + ${james.groupId} + testing-base + test + + + org.testcontainers + postgresql + test + + + diff --git a/mailbox/plugin/deleted-messages-vault-postgres/src/main/java/org/apache/james/vault/metadata/PostgresDeletedMessageMetadataModule.java b/mailbox/plugin/deleted-messages-vault-postgres/src/main/java/org/apache/james/vault/metadata/PostgresDeletedMessageMetadataModule.java new file mode 100644 index 00000000000..de041482a47 --- /dev/null +++ b/mailbox/plugin/deleted-messages-vault-postgres/src/main/java/org/apache/james/vault/metadata/PostgresDeletedMessageMetadataModule.java @@ -0,0 +1,65 @@ +/**************************************************************** + * 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.vault.metadata; + +import static org.apache.james.vault.metadata.PostgresDeletedMessageMetadataModule.DeletedMessageMetadataTable.OWNER_MESSAGE_ID_INDEX; +import static org.apache.james.vault.metadata.PostgresDeletedMessageMetadataModule.DeletedMessageMetadataTable.TABLE; + +import org.apache.james.backends.postgres.PostgresIndex; +import org.apache.james.backends.postgres.PostgresModule; +import org.apache.james.backends.postgres.PostgresTable; +import org.jooq.Field; +import org.jooq.JSONB; +import org.jooq.Record; +import org.jooq.Table; +import org.jooq.impl.DSL; +import org.jooq.impl.SQLDataType; + +public interface PostgresDeletedMessageMetadataModule { + interface DeletedMessageMetadataTable { + Table TABLE_NAME = DSL.table("deleted_messages_metadata"); + + Field BUCKET_NAME = DSL.field("bucket_name", SQLDataType.VARCHAR.notNull()); + Field OWNER = DSL.field("owner", SQLDataType.VARCHAR.notNull()); + Field MESSAGE_ID = DSL.field("messageId", SQLDataType.VARCHAR.notNull()); + Field BLOB_ID = DSL.field("blob_id", SQLDataType.VARCHAR.notNull()); + Field METADATA = DSL.field("metadata", SQLDataType.JSONB.notNull()); + + PostgresTable TABLE = PostgresTable.name(TABLE_NAME.getName()) + .createTableStep(((dsl, tableName) -> dsl.createTableIfNotExists(tableName) + .column(BUCKET_NAME) + .column(OWNER) + .column(MESSAGE_ID) + .column(BLOB_ID) + .column(METADATA) + .primaryKey(BUCKET_NAME, OWNER, MESSAGE_ID))) + .disableRowLevelSecurity() + .build(); + + PostgresIndex OWNER_MESSAGE_ID_INDEX = PostgresIndex.name("owner_messageId_index") + .createIndexStep((dsl, indexName) -> dsl.createUniqueIndexIfNotExists(indexName) + .on(TABLE_NAME, OWNER, MESSAGE_ID)); + } + + PostgresModule MODULE = PostgresModule.builder() + .addTable(TABLE) + .addIndex(OWNER_MESSAGE_ID_INDEX) + .build(); +} diff --git a/mailbox/plugin/deleted-messages-vault-postgres/src/main/java/org/apache/james/vault/metadata/PostgresDeletedMessageMetadataVault.java b/mailbox/plugin/deleted-messages-vault-postgres/src/main/java/org/apache/james/vault/metadata/PostgresDeletedMessageMetadataVault.java new file mode 100644 index 00000000000..76e4d672466 --- /dev/null +++ b/mailbox/plugin/deleted-messages-vault-postgres/src/main/java/org/apache/james/vault/metadata/PostgresDeletedMessageMetadataVault.java @@ -0,0 +1,115 @@ +/**************************************************************** + * 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.vault.metadata; + +import static org.apache.james.util.ReactorUtils.publishIfPresent; +import static org.apache.james.vault.metadata.PostgresDeletedMessageMetadataModule.DeletedMessageMetadataTable.BLOB_ID; +import static org.apache.james.vault.metadata.PostgresDeletedMessageMetadataModule.DeletedMessageMetadataTable.BUCKET_NAME; +import static org.apache.james.vault.metadata.PostgresDeletedMessageMetadataModule.DeletedMessageMetadataTable.MESSAGE_ID; +import static org.apache.james.vault.metadata.PostgresDeletedMessageMetadataModule.DeletedMessageMetadataTable.METADATA; +import static org.apache.james.vault.metadata.PostgresDeletedMessageMetadataModule.DeletedMessageMetadataTable.OWNER; +import static org.apache.james.vault.metadata.PostgresDeletedMessageMetadataModule.DeletedMessageMetadataTable.TABLE_NAME; +import static org.jooq.JSONB.jsonb; + +import java.util.function.Function; + +import jakarta.inject.Inject; + +import org.apache.james.backends.postgres.utils.PostgresExecutor; +import org.apache.james.blob.api.BlobId; +import org.apache.james.blob.api.BucketName; +import org.apache.james.core.Username; +import org.apache.james.mailbox.model.MessageId; +import org.jooq.Record; +import org.reactivestreams.Publisher; + +import reactor.core.publisher.Flux; +import reactor.core.publisher.Mono; + +public class PostgresDeletedMessageMetadataVault implements DeletedMessageMetadataVault { + private final PostgresExecutor postgresExecutor; + private final MetadataSerializer metadataSerializer; + private final BlobId.Factory blobIdFactory; + + @Inject + public PostgresDeletedMessageMetadataVault(PostgresExecutor postgresExecutor, + MetadataSerializer metadataSerializer, + BlobId.Factory blobIdFactory) { + this.postgresExecutor = postgresExecutor; + this.metadataSerializer = metadataSerializer; + this.blobIdFactory = blobIdFactory; + } + + @Override + public Publisher store(DeletedMessageWithStorageInformation deletedMessage) { + return postgresExecutor.executeVoid(context -> Mono.from(context.insertInto(TABLE_NAME) + .set(OWNER, deletedMessage.getDeletedMessage().getOwner().asString()) + .set(MESSAGE_ID, deletedMessage.getDeletedMessage().getMessageId().serialize()) + .set(BUCKET_NAME, deletedMessage.getStorageInformation().getBucketName().asString()) + .set(BLOB_ID, deletedMessage.getStorageInformation().getBlobId().asString()) + .set(METADATA, jsonb(metadataSerializer.serialize(deletedMessage))))); + } + + @Override + public Publisher removeMetadataRelatedToBucket(BucketName bucketName) { + return postgresExecutor.executeVoid(context -> Mono.from(context.deleteFrom(TABLE_NAME) + .where(BUCKET_NAME.eq(bucketName.asString())))); + } + + @Override + public Publisher remove(BucketName bucketName, Username username, MessageId messageId) { + return postgresExecutor.executeVoid(context -> Mono.from(context.deleteFrom(TABLE_NAME) + .where(BUCKET_NAME.eq(bucketName.asString()), + OWNER.eq(username.asString()), + MESSAGE_ID.eq(messageId.serialize())))); + } + + @Override + public Publisher retrieveStorageInformation(Username username, MessageId messageId) { + return postgresExecutor.executeRow(context -> Mono.from(context.select(BUCKET_NAME, BLOB_ID) + .from(TABLE_NAME) + .where(OWNER.eq(username.asString()), + MESSAGE_ID.eq(messageId.serialize())))) + .map(toStorageInformation()); + } + + private Function toStorageInformation() { + return record -> StorageInformation.builder() + .bucketName(BucketName.of(record.get(BUCKET_NAME))) + .blobId(blobIdFactory.parse(record.get(BLOB_ID))); + } + + @Override + public Publisher listMessages(BucketName bucketName, Username username) { + return postgresExecutor.executeRows(context -> Flux.from(context.select(METADATA) + .from(TABLE_NAME) + .where(BUCKET_NAME.eq(bucketName.asString()), + OWNER.eq(username.asString())))) + .map(record -> metadataSerializer.deserialize(record.get(METADATA).data())) + .handle(publishIfPresent()); + } + + @Override + public Publisher listRelatedBuckets() { + return postgresExecutor.executeRows(context -> Flux.from(context.selectDistinct(BUCKET_NAME) + .from(TABLE_NAME))) + .map(record -> BucketName.of(record.get(BUCKET_NAME))); + } +} diff --git a/mailbox/plugin/deleted-messages-vault-postgres/src/main/java/org/apache/james/vault/metadata/PostgresDeletedMessageVaultDeletionCallback.java b/mailbox/plugin/deleted-messages-vault-postgres/src/main/java/org/apache/james/vault/metadata/PostgresDeletedMessageVaultDeletionCallback.java new file mode 100644 index 00000000000..224d2a492ed --- /dev/null +++ b/mailbox/plugin/deleted-messages-vault-postgres/src/main/java/org/apache/james/vault/metadata/PostgresDeletedMessageVaultDeletionCallback.java @@ -0,0 +1,123 @@ +/**************************************************************** + * 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.vault.metadata; + +import java.io.IOException; +import java.io.InputStream; +import java.io.SequenceInputStream; +import java.time.Clock; +import java.time.ZoneOffset; +import java.time.ZonedDateTime; +import java.util.Optional; +import java.util.Set; + +import jakarta.inject.Inject; + +import org.apache.james.blob.api.BlobStore; +import org.apache.james.core.MailAddress; +import org.apache.james.core.MaybeSender; +import org.apache.james.core.Username; +import org.apache.james.mailbox.model.MailboxId; +import org.apache.james.mailbox.model.MessageId; +import org.apache.james.mailbox.postgres.DeleteMessageListener; +import org.apache.james.mailbox.postgres.mail.MessageRepresentation; +import org.apache.james.mime4j.MimeIOException; +import org.apache.james.mime4j.codec.DecodeMonitor; +import org.apache.james.mime4j.dom.Message; +import org.apache.james.mime4j.dom.address.Mailbox; +import org.apache.james.mime4j.message.DefaultMessageBuilder; +import org.apache.james.mime4j.stream.MimeConfig; +import org.apache.james.server.core.Envelope; +import org.apache.james.vault.DeletedMessage; +import org.apache.james.vault.DeletedMessageVault; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + +import com.github.fge.lambdas.Throwing; +import com.google.common.collect.ImmutableSet; + +import reactor.core.publisher.Mono; + +public class PostgresDeletedMessageVaultDeletionCallback implements DeleteMessageListener.DeletionCallback { + private static final Logger LOGGER = LoggerFactory.getLogger(PostgresDeletedMessageVaultDeletionCallback.class); + + private final DeletedMessageVault deletedMessageVault; + private final BlobStore blobStore; + private final Clock clock; + + @Inject + public PostgresDeletedMessageVaultDeletionCallback(DeletedMessageVault deletedMessageVault, BlobStore blobStore, Clock clock) { + this.deletedMessageVault = deletedMessageVault; + this.blobStore = blobStore; + this.clock = clock; + } + + @Override + public Mono forMessage(MessageRepresentation message, MailboxId mailboxId, Username owner) { + return Mono.fromSupplier(Throwing.supplier(() -> message.getHeaderContent().getInputStream())) + .flatMap(headerStream -> { + Optional mimeMessage = parseMessage(headerStream, message.getMessageId()); + DeletedMessage deletedMessage = DeletedMessage.builder() + .messageId(message.getMessageId()) + .originMailboxes(mailboxId) + .user(owner) + .deliveryDate(ZonedDateTime.ofInstant(message.getInternalDate().toInstant(), ZoneOffset.UTC)) + .deletionDate(ZonedDateTime.ofInstant(clock.instant(), ZoneOffset.UTC)) + .sender(retrieveSender(mimeMessage)) + .recipients(retrieveRecipients(mimeMessage)) + .hasAttachment(!message.getAttachments().isEmpty()) + .size(message.getSize()) + .subject(mimeMessage.map(Message::getSubject)) + .build(); + + return Mono.from(blobStore.readReactive(blobStore.getDefaultBucketName(), message.getBodyBlobId(), BlobStore.StoragePolicy.LOW_COST)) + .map(bodyStream -> new SequenceInputStream(headerStream, bodyStream)) + .flatMap(bodyStream -> Mono.from(deletedMessageVault.append(deletedMessage, bodyStream))); + }); + } + + private Optional parseMessage(InputStream inputStream, MessageId messageId) { + DefaultMessageBuilder messageBuilder = new DefaultMessageBuilder(); + messageBuilder.setMimeEntityConfig(MimeConfig.PERMISSIVE); + messageBuilder.setDecodeMonitor(DecodeMonitor.SILENT); + try { + return Optional.ofNullable(messageBuilder.parseMessage(inputStream)); + } catch (MimeIOException e) { + LOGGER.warn("Can not parse the message {}", messageId, e); + return Optional.empty(); + } catch (IOException e) { + throw new RuntimeException(e); + } + } + + private MaybeSender retrieveSender(Optional mimeMessage) { + return mimeMessage + .map(Message::getSender) + .map(Mailbox::getAddress) + .map(MaybeSender::getMailSender) + .orElse(MaybeSender.nullSender()); + } + + private Set retrieveRecipients(Optional maybeMessage) { + return maybeMessage.map(message -> Envelope.fromMime4JMessage(message, Envelope.ValidationPolicy.IGNORE)) + .map(Envelope::getRecipients) + .orElse(ImmutableSet.of()); + } +} diff --git a/mailbox/plugin/deleted-messages-vault-postgres/src/test/java/org/apache/james/vault/metadata/PostgresDeletedMessageMetadataVaultTest.java b/mailbox/plugin/deleted-messages-vault-postgres/src/test/java/org/apache/james/vault/metadata/PostgresDeletedMessageMetadataVaultTest.java new file mode 100644 index 00000000000..9a6a4b738be --- /dev/null +++ b/mailbox/plugin/deleted-messages-vault-postgres/src/test/java/org/apache/james/vault/metadata/PostgresDeletedMessageMetadataVaultTest.java @@ -0,0 +1,47 @@ +/**************************************************************** + * 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.vault.metadata; + +import org.apache.james.backends.postgres.PostgresExtension; +import org.apache.james.backends.postgres.PostgresModule; +import org.apache.james.blob.api.BlobId; +import org.apache.james.blob.api.PlainBlobId; +import org.apache.james.mailbox.inmemory.InMemoryId; +import org.apache.james.mailbox.inmemory.InMemoryMessageId; +import org.apache.james.vault.dto.DeletedMessageWithStorageInformationConverter; +import org.junit.jupiter.api.extension.RegisterExtension; + +class PostgresDeletedMessageMetadataVaultTest implements DeletedMessageMetadataVaultContract { + @RegisterExtension + static PostgresExtension postgresExtension = PostgresExtension.withoutRowLevelSecurity( + PostgresModule.aggregateModules(PostgresDeletedMessageMetadataModule.MODULE)); + + @Override + public DeletedMessageMetadataVault metadataVault() { + BlobId.Factory blobIdFactory = new PlainBlobId.Factory(); + InMemoryMessageId.Factory messageIdFactory = new InMemoryMessageId.Factory(); + DeletedMessageWithStorageInformationConverter dtoConverter = new DeletedMessageWithStorageInformationConverter(blobIdFactory, + messageIdFactory, new InMemoryId.Factory()); + + return new PostgresDeletedMessageMetadataVault(postgresExtension.getDefaultPostgresExecutor(), + new MetadataSerializer(dtoConverter), + blobIdFactory); + } +} diff --git a/mailbox/plugin/deleted-messages-vault-cassandra/src/main/java/org/apache/james/vault/metadata/MetadataSerializer.java b/mailbox/plugin/deleted-messages-vault/src/main/java/org/apache/james/vault/metadata/MetadataSerializer.java similarity index 100% rename from mailbox/plugin/deleted-messages-vault-cassandra/src/main/java/org/apache/james/vault/metadata/MetadataSerializer.java rename to mailbox/plugin/deleted-messages-vault/src/main/java/org/apache/james/vault/metadata/MetadataSerializer.java diff --git a/mailbox/plugin/quota-mailing-cassandra/src/test/java/org/apache/james/mailbox/quota/cassandra/dto/DTOTest.java b/mailbox/plugin/quota-mailing-cassandra/src/test/java/org/apache/james/mailbox/quota/cassandra/dto/DTOTest.java index 2673f28d1a0..be60527068a 100644 --- a/mailbox/plugin/quota-mailing-cassandra/src/test/java/org/apache/james/mailbox/quota/cassandra/dto/DTOTest.java +++ b/mailbox/plugin/quota-mailing-cassandra/src/test/java/org/apache/james/mailbox/quota/cassandra/dto/DTOTest.java @@ -20,7 +20,7 @@ package org.apache.james.mailbox.quota.cassandra.dto; import static net.javacrumbs.jsonunit.assertj.JsonAssertions.assertThatJson; -import static org.apache.james.mailbox.quota.cassandra.dto.QuotaEventDTOModules.QUOTA_THRESHOLD_CHANGE; +import static org.apache.james.mailbox.quota.mailing.events.QuotaEventDTOModules.QUOTA_THRESHOLD_CHANGE; import static org.apache.james.mailbox.quota.model.QuotaThresholdFixture._75; import static org.apache.james.mailbox.quota.model.QuotaThresholdFixture._80; import static org.assertj.core.api.Assertions.assertThat; @@ -36,7 +36,10 @@ import org.apache.james.eventsourcing.EventId; import org.apache.james.mailbox.model.Quota; import org.apache.james.mailbox.quota.mailing.aggregates.UserQuotaThresholds; +import org.apache.james.mailbox.quota.mailing.events.HistoryEvolutionDTO; +import org.apache.james.mailbox.quota.mailing.events.QuotaDTO; import org.apache.james.mailbox.quota.mailing.events.QuotaThresholdChangedEvent; +import org.apache.james.mailbox.quota.mailing.events.QuotaThresholdChangedEventDTO; import org.apache.james.mailbox.quota.model.HistoryEvolution; import org.apache.james.mailbox.quota.model.QuotaThresholdChange; import org.apache.james.util.ClassLoaderUtils; diff --git a/mailbox/plugin/quota-mailing-cassandra/src/test/java/org/apache/james/mailbox/quota/cassandra/listeners/CassandraQuotaMailingListenersIntegrationTest.java b/mailbox/plugin/quota-mailing-cassandra/src/test/java/org/apache/james/mailbox/quota/cassandra/listeners/CassandraQuotaMailingListenersIntegrationTest.java index 3c9b8a4e217..0464f95b75c 100644 --- a/mailbox/plugin/quota-mailing-cassandra/src/test/java/org/apache/james/mailbox/quota/cassandra/listeners/CassandraQuotaMailingListenersIntegrationTest.java +++ b/mailbox/plugin/quota-mailing-cassandra/src/test/java/org/apache/james/mailbox/quota/cassandra/listeners/CassandraQuotaMailingListenersIntegrationTest.java @@ -21,7 +21,7 @@ import org.apache.james.eventsourcing.eventstore.JsonEventSerializer; import org.apache.james.eventsourcing.eventstore.cassandra.CassandraEventStoreExtension; -import org.apache.james.mailbox.quota.cassandra.dto.QuotaEventDTOModules; +import org.apache.james.mailbox.quota.mailing.events.QuotaEventDTOModules; import org.apache.james.mailbox.quota.mailing.listeners.QuotaThresholdMailingIntegrationTest; import org.junit.jupiter.api.extension.RegisterExtension; diff --git a/mailbox/plugin/quota-mailing-cassandra/src/main/java/org/apache/james/mailbox/quota/cassandra/dto/HistoryEvolutionDTO.java b/mailbox/plugin/quota-mailing/src/main/java/org/apache/james/mailbox/quota/mailing/events/HistoryEvolutionDTO.java similarity index 66% rename from mailbox/plugin/quota-mailing-cassandra/src/main/java/org/apache/james/mailbox/quota/cassandra/dto/HistoryEvolutionDTO.java rename to mailbox/plugin/quota-mailing/src/main/java/org/apache/james/mailbox/quota/mailing/events/HistoryEvolutionDTO.java index 2d1dda1cc08..afeaab89b2f 100644 --- a/mailbox/plugin/quota-mailing-cassandra/src/main/java/org/apache/james/mailbox/quota/cassandra/dto/HistoryEvolutionDTO.java +++ b/mailbox/plugin/quota-mailing/src/main/java/org/apache/james/mailbox/quota/mailing/events/HistoryEvolutionDTO.java @@ -17,7 +17,7 @@ * under the License. * ****************************************************************/ -package org.apache.james.mailbox.quota.cassandra.dto; +package org.apache.james.mailbox.quota.mailing.events; import java.time.Instant; import java.util.Optional; @@ -26,14 +26,17 @@ import org.apache.james.mailbox.quota.model.QuotaThreshold; import org.apache.james.mailbox.quota.model.QuotaThresholdChange; -import com.fasterxml.jackson.annotation.JsonCreator; import com.fasterxml.jackson.annotation.JsonIgnore; import com.fasterxml.jackson.annotation.JsonProperty; import com.google.common.base.Preconditions; import com.google.common.primitives.Booleans; -class HistoryEvolutionDTO { +public record HistoryEvolutionDTO(@JsonProperty("change") HistoryEvolution.HistoryChangeType change, + @JsonProperty("recentness") Optional recentness, + @JsonProperty("threshold") Optional threshold, + @JsonProperty("instant") Optional instant) { + @JsonIgnore public static HistoryEvolutionDTO toDto(HistoryEvolution historyEvolution) { return new HistoryEvolutionDTO( historyEvolution.getThresholdHistoryChange(), @@ -46,43 +49,9 @@ public static HistoryEvolutionDTO toDto(HistoryEvolution historyEvolution) { .map(Instant::toEpochMilli)); } - private final HistoryEvolution.HistoryChangeType change; - private final Optional recentness; - private final Optional threshold; - private final Optional instant; - - @JsonCreator - public HistoryEvolutionDTO( - @JsonProperty("changeType") HistoryEvolution.HistoryChangeType change, - @JsonProperty("recentness") Optional recentness, - @JsonProperty("threshold") Optional threshold, - @JsonProperty("instant") Optional instant) { - this.change = change; - this.recentness = recentness; - this.threshold = threshold; - this.instant = instant; - } - - public HistoryEvolution.HistoryChangeType getChange() { - return change; - } - - public Optional getRecentness() { - return recentness; - } - - public Optional getThreshold() { - return threshold; - } - - public Optional getInstant() { - return instant; - } - @JsonIgnore public HistoryEvolution toHistoryEvolution() { - Preconditions.checkState(Booleans.countTrue( - threshold.isPresent(), instant.isPresent()) != 1, + Preconditions.checkState(Booleans.countTrue(threshold.isPresent(), instant.isPresent()) != 1, "threshold and instant needs to be both set, or both unset. Mixed states not allowed."); Optional quotaThresholdChange = threshold @@ -93,6 +62,5 @@ public HistoryEvolution toHistoryEvolution() { change, recentness, quotaThresholdChange); - } -} +} \ No newline at end of file diff --git a/mailbox/plugin/quota-mailing-cassandra/src/main/java/org/apache/james/mailbox/quota/cassandra/dto/QuotaDTO.java b/mailbox/plugin/quota-mailing/src/main/java/org/apache/james/mailbox/quota/mailing/events/QuotaDTO.java similarity index 82% rename from mailbox/plugin/quota-mailing-cassandra/src/main/java/org/apache/james/mailbox/quota/cassandra/dto/QuotaDTO.java rename to mailbox/plugin/quota-mailing/src/main/java/org/apache/james/mailbox/quota/mailing/events/QuotaDTO.java index 78e69cd5e8f..eff3a667e40 100644 --- a/mailbox/plugin/quota-mailing-cassandra/src/main/java/org/apache/james/mailbox/quota/cassandra/dto/QuotaDTO.java +++ b/mailbox/plugin/quota-mailing/src/main/java/org/apache/james/mailbox/quota/mailing/events/QuotaDTO.java @@ -17,7 +17,7 @@ * under the License. * ****************************************************************/ -package org.apache.james.mailbox.quota.cassandra.dto; +package org.apache.james.mailbox.quota.mailing.events; import java.util.Optional; @@ -27,11 +27,13 @@ import org.apache.james.core.quota.QuotaSizeUsage; import org.apache.james.mailbox.model.Quota; -import com.fasterxml.jackson.annotation.JsonCreator; import com.fasterxml.jackson.annotation.JsonIgnore; import com.fasterxml.jackson.annotation.JsonProperty; -class QuotaDTO { +public record QuotaDTO(@JsonProperty("used") long used, + @JsonProperty("limit") Optional limit) { + + @JsonIgnore public static QuotaDTO from(Quota quota) { if (quota.getLimit().isUnlimited()) { return new QuotaDTO(quota.getUsed().asLong(), Optional.empty()); @@ -39,24 +41,6 @@ public static QuotaDTO from(Quota quota) { return new QuotaDTO(quota.getUsed().asLong(), Optional.of(quota.getLimit().asLong())); } - private final long used; - private final Optional limit; - - @JsonCreator - private QuotaDTO(@JsonProperty("used") long used, - @JsonProperty("limit") Optional limit) { - this.used = used; - this.limit = limit; - } - - public long getUsed() { - return used; - } - - public Optional getLimit() { - return limit; - } - @JsonIgnore public Quota asSizeQuota() { return Quota.builder() @@ -72,4 +56,4 @@ public Quota asCountQuota() { .computedLimit(QuotaCountLimit.count(limit)) .build(); } -} +} \ No newline at end of file diff --git a/mailbox/plugin/quota-mailing-cassandra/src/main/java/org/apache/james/mailbox/quota/cassandra/dto/QuotaEventDTOModules.java b/mailbox/plugin/quota-mailing/src/main/java/org/apache/james/mailbox/quota/mailing/events/QuotaEventDTOModules.java similarity index 92% rename from mailbox/plugin/quota-mailing-cassandra/src/main/java/org/apache/james/mailbox/quota/cassandra/dto/QuotaEventDTOModules.java rename to mailbox/plugin/quota-mailing/src/main/java/org/apache/james/mailbox/quota/mailing/events/QuotaEventDTOModules.java index 1295411bf6b..5a0bb983c5c 100644 --- a/mailbox/plugin/quota-mailing-cassandra/src/main/java/org/apache/james/mailbox/quota/cassandra/dto/QuotaEventDTOModules.java +++ b/mailbox/plugin/quota-mailing/src/main/java/org/apache/james/mailbox/quota/mailing/events/QuotaEventDTOModules.java @@ -17,13 +17,11 @@ * under the License. * ****************************************************************/ -package org.apache.james.mailbox.quota.cassandra.dto; +package org.apache.james.mailbox.quota.mailing.events; import org.apache.james.eventsourcing.eventstore.dto.EventDTOModule; -import org.apache.james.mailbox.quota.mailing.events.QuotaThresholdChangedEvent; public interface QuotaEventDTOModules { - EventDTOModule QUOTA_THRESHOLD_CHANGE = EventDTOModule .forEvent(QuotaThresholdChangedEvent.class) diff --git a/mailbox/plugin/quota-mailing-cassandra/src/main/java/org/apache/james/mailbox/quota/cassandra/dto/QuotaThresholdChangedEventDTO.java b/mailbox/plugin/quota-mailing/src/main/java/org/apache/james/mailbox/quota/mailing/events/QuotaThresholdChangedEventDTO.java similarity index 57% rename from mailbox/plugin/quota-mailing-cassandra/src/main/java/org/apache/james/mailbox/quota/cassandra/dto/QuotaThresholdChangedEventDTO.java rename to mailbox/plugin/quota-mailing/src/main/java/org/apache/james/mailbox/quota/mailing/events/QuotaThresholdChangedEventDTO.java index 829feda3ae6..725a1f5ecdf 100644 --- a/mailbox/plugin/quota-mailing-cassandra/src/main/java/org/apache/james/mailbox/quota/cassandra/dto/QuotaThresholdChangedEventDTO.java +++ b/mailbox/plugin/quota-mailing/src/main/java/org/apache/james/mailbox/quota/mailing/events/QuotaThresholdChangedEventDTO.java @@ -17,23 +17,28 @@ * under the License. * ****************************************************************/ -package org.apache.james.mailbox.quota.cassandra.dto; +package org.apache.james.mailbox.quota.mailing.events; import org.apache.james.eventsourcing.EventId; import org.apache.james.eventsourcing.eventstore.dto.EventDTO; import org.apache.james.mailbox.quota.mailing.aggregates.UserQuotaThresholds; -import org.apache.james.mailbox.quota.mailing.events.QuotaThresholdChangedEvent; -import com.fasterxml.jackson.annotation.JsonCreator; import com.fasterxml.jackson.annotation.JsonIgnore; import com.fasterxml.jackson.annotation.JsonProperty; -class QuotaThresholdChangedEventDTO implements EventDTO { +public record QuotaThresholdChangedEventDTO(@JsonProperty("type") String type, + @JsonProperty("eventId") int eventId, + @JsonProperty("aggregateId") String aggregateId, + @JsonProperty("sizeQuota") QuotaDTO sizeQuota, + @JsonProperty("countQuota") QuotaDTO countQuota, + @JsonProperty("sizeEvolution") HistoryEvolutionDTO sizeEvolution, + @JsonProperty("countEvolution") HistoryEvolutionDTO countEvolution) implements EventDTO { @JsonIgnore public static QuotaThresholdChangedEventDTO from(QuotaThresholdChangedEvent event, String type) { return new QuotaThresholdChangedEventDTO( - type, event.eventId().serialize(), + type, + event.eventId().serialize(), event.getAggregateId().asAggregateKey(), QuotaDTO.from(event.getSizeQuota()), QuotaDTO.from(event.getCountQuota()), @@ -41,60 +46,6 @@ public static QuotaThresholdChangedEventDTO from(QuotaThresholdChangedEvent even HistoryEvolutionDTO.toDto(event.getCountHistoryEvolution())); } - private final String type; - private final int eventId; - private final String aggregateId; - private final QuotaDTO sizeQuota; - private final QuotaDTO countQuota; - private final HistoryEvolutionDTO sizeEvolution; - private final HistoryEvolutionDTO countEvolution; - - @JsonCreator - private QuotaThresholdChangedEventDTO( - @JsonProperty("type") String type, - @JsonProperty("eventId") int eventId, - @JsonProperty("aggregateId") String aggregateId, - @JsonProperty("sizeQuota") QuotaDTO sizeQuota, - @JsonProperty("countQuota") QuotaDTO countQuota, - @JsonProperty("sizeEvolution") HistoryEvolutionDTO sizeEvolution, - @JsonProperty("countEvolution") HistoryEvolutionDTO countEvolution) { - this.type = type; - this.eventId = eventId; - this.aggregateId = aggregateId; - this.sizeQuota = sizeQuota; - this.countQuota = countQuota; - this.sizeEvolution = sizeEvolution; - this.countEvolution = countEvolution; - } - - public String getType() { - return type; - } - - public long getEventId() { - return eventId; - } - - public String getAggregateId() { - return aggregateId; - } - - public QuotaDTO getSizeQuota() { - return sizeQuota; - } - - public QuotaDTO getCountQuota() { - return countQuota; - } - - public HistoryEvolutionDTO getSizeEvolution() { - return sizeEvolution; - } - - public HistoryEvolutionDTO getCountEvolution() { - return countEvolution; - } - @JsonIgnore public QuotaThresholdChangedEvent toEvent() { return new QuotaThresholdChangedEvent( @@ -105,4 +56,10 @@ public QuotaThresholdChangedEvent toEvent() { countQuota.asCountQuota(), UserQuotaThresholds.Id.fromKey(aggregateId)); } -} + + @Override + @JsonIgnore + public String getType() { + return type; + } +} \ No newline at end of file diff --git a/mailbox/pom.xml b/mailbox/pom.xml index e79c7da284d..0fe0776bc09 100644 --- a/mailbox/pom.xml +++ b/mailbox/pom.xml @@ -49,6 +49,7 @@ plugin/deleted-messages-vault plugin/deleted-messages-vault-cassandra + plugin/deleted-messages-vault-postgres plugin/quota-mailing plugin/quota-mailing-cassandra @@ -58,6 +59,8 @@ plugin/quota-search-opensearch plugin/quota-search-scanning + postgres + scanning-search spring store diff --git a/mailbox/postgres/pom.xml b/mailbox/postgres/pom.xml new file mode 100644 index 00000000000..96f13038906 --- /dev/null +++ b/mailbox/postgres/pom.xml @@ -0,0 +1,185 @@ + + + + 4.0.0 + + org.apache.james + apache-james-mailbox + 3.9.0-SNAPSHOT + ../pom.xml + + + apache-james-mailbox-postgres + Apache James :: Mailbox :: Postgres + + + 5.3.7 + + + + + ${james.groupId} + apache-james-backends-postgres + + + ${james.groupId} + apache-james-backends-postgres + test-jar + test + + + ${james.groupId} + apache-james-mailbox-api + + + ${james.groupId} + apache-james-mailbox-api + test-jar + test + + + ${james.groupId} + apache-james-mailbox-event-json + + + ${james.groupId} + apache-james-mailbox-store + + + ${james.groupId} + apache-james-mailbox-store + test-jar + test + + + ${james.groupId} + apache-james-mailbox-tools-quota-recompute + test + + + ${james.groupId} + apache-james-mailbox-tools-quota-recompute + test-jar + test + + + ${james.groupId} + blob-api + + + ${james.groupId} + blob-memory + test + + + ${james.groupId} + blob-memory + test + + + ${james.groupId} + blob-storage-strategy + test + + + ${james.groupId} + blob-storage-strategy + test + + + ${james.groupId} + event-bus-api + test-jar + test + + + ${james.groupId} + event-bus-in-vm + test + + + ${james.groupId} + james-json + test-jar + test + + + ${james.groupId} + james-server-data-postgres + test + + + ${james.groupId} + james-server-guice-common + test-jar + test + + + ${james.groupId} + james-server-testing + test + + + ${james.groupId} + james-server-util + + + ${james.groupId} + metrics-tests + test + + + ${james.groupId} + testing-base + test + + + com.fasterxml.jackson.datatype + jackson-datatype-jdk8 + + + com.github.f4b6a3 + uuid-creator + ${uuid-creator.version} + + + org.eclipse.angus + jakarta.mail + + + org.jasypt + jasypt + + + org.mockito + mockito-core + test + + + org.slf4j + slf4j-api + + + org.testcontainers + postgresql + test + + + diff --git a/mailbox/postgres/src/main/java/org/apache/james/mailbox/postgres/DeleteMessageListener.java b/mailbox/postgres/src/main/java/org/apache/james/mailbox/postgres/DeleteMessageListener.java new file mode 100644 index 00000000000..c6fc63c8684 --- /dev/null +++ b/mailbox/postgres/src/main/java/org/apache/james/mailbox/postgres/DeleteMessageListener.java @@ -0,0 +1,175 @@ +/**************************************************************** + * 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.mailbox.postgres; + +import java.util.Set; +import java.util.function.Function; + +import jakarta.inject.Inject; + +import org.apache.james.blob.api.BlobStore; +import org.apache.james.core.Username; +import org.apache.james.events.Event; +import org.apache.james.events.EventListener; +import org.apache.james.events.Group; +import org.apache.james.mailbox.events.MailboxEvents.Expunged; +import org.apache.james.mailbox.events.MailboxEvents.MailboxDeletion; +import org.apache.james.mailbox.model.MailboxId; +import org.apache.james.mailbox.model.MessageMetaData; +import org.apache.james.mailbox.postgres.mail.MessageRepresentation; +import org.apache.james.mailbox.postgres.mail.dao.PostgresAttachmentDAO; +import org.apache.james.mailbox.postgres.mail.dao.PostgresMailboxMessageDAO; +import org.apache.james.mailbox.postgres.mail.dao.PostgresMessageDAO; +import org.apache.james.mailbox.postgres.mail.dao.PostgresThreadDAO; +import org.apache.james.util.FunctionalUtils; +import org.apache.james.util.ReactorUtils; +import org.reactivestreams.Publisher; + +import reactor.core.publisher.Flux; +import reactor.core.publisher.Mono; + +public class DeleteMessageListener implements EventListener.ReactiveGroupEventListener { + @FunctionalInterface + public interface DeletionCallback { + Mono forMessage(MessageRepresentation messageRepresentation, MailboxId mailboxId, Username owner); + } + + public static class DeleteMessageListenerGroup extends Group { + } + + public static final int LOW_CONCURRENCY = 4; + + private final BlobStore blobStore; + private final Set deletionCallbackList; + + private final PostgresMessageDAO.Factory messageDAOFactory; + private final PostgresMailboxMessageDAO.Factory mailboxMessageDAOFactory; + private final PostgresAttachmentDAO.Factory attachmentDAOFactory; + private final PostgresThreadDAO.Factory threadDAOFactory; + + @Inject + public DeleteMessageListener(BlobStore blobStore, + PostgresMailboxMessageDAO.Factory mailboxMessageDAOFactory, + PostgresMessageDAO.Factory messageDAOFactory, + PostgresAttachmentDAO.Factory attachmentDAOFactory, + PostgresThreadDAO.Factory threadDAOFactory, + Set deletionCallbackList) { + this.messageDAOFactory = messageDAOFactory; + this.mailboxMessageDAOFactory = mailboxMessageDAOFactory; + this.blobStore = blobStore; + this.deletionCallbackList = deletionCallbackList; + this.attachmentDAOFactory = attachmentDAOFactory; + this.threadDAOFactory = threadDAOFactory; + } + + @Override + public Group getDefaultGroup() { + return new DeleteMessageListenerGroup(); + } + + @Override + public boolean isHandling(Event event) { + return event instanceof Expunged || event instanceof MailboxDeletion; + } + + @Override + public Publisher reactiveEvent(Event event) { + if (event instanceof Expunged) { + Expunged expunged = (Expunged) event; + return handleMessageDeletion(expunged); + } + if (event instanceof MailboxDeletion) { + MailboxDeletion mailboxDeletion = (MailboxDeletion) event; + return handleMailboxDeletion(mailboxDeletion); + } + return Mono.empty(); + } + + private Mono handleMailboxDeletion(MailboxDeletion event) { + PostgresMessageDAO postgresMessageDAO = messageDAOFactory.create(event.getUsername().getDomainPart()); + PostgresMailboxMessageDAO postgresMailboxMessageDAO = mailboxMessageDAOFactory.create(event.getUsername().getDomainPart()); + PostgresAttachmentDAO attachmentDAO = attachmentDAOFactory.create(event.getUsername().getDomainPart()); + PostgresThreadDAO threadDAO = threadDAOFactory.create(event.getUsername().getDomainPart()); + + return postgresMailboxMessageDAO.deleteByMailboxId((PostgresMailboxId) event.getMailboxId()) + .flatMap(msgId -> handleMessageDeletion(postgresMessageDAO, postgresMailboxMessageDAO, attachmentDAO, threadDAO, msgId, event.getMailboxId(), event.getMailboxPath().getUser()), + LOW_CONCURRENCY) + .then(); + } + + private Mono handleMessageDeletion(Expunged event) { + PostgresMessageDAO postgresMessageDAO = messageDAOFactory.create(event.getUsername().getDomainPart()); + PostgresMailboxMessageDAO postgresMailboxMessageDAO = mailboxMessageDAOFactory.create(event.getUsername().getDomainPart()); + PostgresAttachmentDAO attachmentDAO = attachmentDAOFactory.create(event.getUsername().getDomainPart()); + PostgresThreadDAO threadDAO = threadDAOFactory.create(event.getUsername().getDomainPart()); + + return Flux.fromIterable(event.getExpunged() + .values()) + .map(MessageMetaData::getMessageId) + .map(PostgresMessageId.class::cast) + .flatMap(msgId -> handleMessageDeletion(postgresMessageDAO, postgresMailboxMessageDAO, attachmentDAO, threadDAO, msgId, event.getMailboxId(), event.getMailboxPath().getUser()), LOW_CONCURRENCY) + .then(); + } + + private Mono handleMessageDeletion(PostgresMessageDAO postgresMessageDAO, + PostgresMailboxMessageDAO postgresMailboxMessageDAO, + PostgresAttachmentDAO attachmentDAO, + PostgresThreadDAO threadDAO, + PostgresMessageId messageId, + MailboxId mailboxId, + Username owner) { + return Mono.just(messageId) + .filterWhen(msgId -> isUnreferenced(msgId, postgresMailboxMessageDAO)) + .flatMap(msgId -> postgresMessageDAO.retrieveMessage(messageId) + .flatMap(executeDeletionCallbacks(mailboxId, owner)) + .then(deleteBodyBlob(msgId, postgresMessageDAO)) + .then(deleteAttachment(msgId, attachmentDAO)) + .then(threadDAO.deleteSome(owner, msgId)) + .then(postgresMessageDAO.deleteByMessageId(msgId))); + } + + private Function> executeDeletionCallbacks(MailboxId mailboxId, Username owner) { + return messageRepresentation -> Flux.fromIterable(deletionCallbackList) + .concatMap(callback -> callback.forMessage(messageRepresentation, mailboxId, owner)) + .then(); + } + + private Mono deleteBodyBlob(PostgresMessageId id, PostgresMessageDAO postgresMessageDAO) { + return postgresMessageDAO.getBodyBlobId(id) + .flatMap(blobId -> Mono.from(blobStore.delete(blobStore.getDefaultBucketName(), blobId)) + .then()); + } + + private Mono isUnreferenced(PostgresMessageId id, PostgresMailboxMessageDAO postgresMailboxMessageDAO) { + return postgresMailboxMessageDAO.existsByMessageId(id) + .map(FunctionalUtils.negate()); + } + + private Mono deleteAttachment(PostgresMessageId messageId, PostgresAttachmentDAO attachmentDAO) { + return deleteAttachmentBlobs(messageId, attachmentDAO) + .then(attachmentDAO.deleteByMessageId(messageId)); + } + + private Mono deleteAttachmentBlobs(PostgresMessageId messageId, PostgresAttachmentDAO attachmentDAO) { + return attachmentDAO.listBlobsByMessageId(messageId) + .flatMap(blobId -> Mono.from(blobStore.delete(blobStore.getDefaultBucketName(), blobId)), ReactorUtils.DEFAULT_CONCURRENCY) + .then(); + } +} diff --git a/mailbox/postgres/src/main/java/org/apache/james/mailbox/postgres/PostgresMailboxAggregateModule.java b/mailbox/postgres/src/main/java/org/apache/james/mailbox/postgres/PostgresMailboxAggregateModule.java new file mode 100644 index 00000000000..90a52df7c4b --- /dev/null +++ b/mailbox/postgres/src/main/java/org/apache/james/mailbox/postgres/PostgresMailboxAggregateModule.java @@ -0,0 +1,38 @@ +/**************************************************************** + * 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.mailbox.postgres; + +import org.apache.james.backends.postgres.PostgresModule; +import org.apache.james.mailbox.postgres.mail.PostgresAttachmentModule; +import org.apache.james.mailbox.postgres.mail.PostgresMailboxModule; +import org.apache.james.mailbox.postgres.mail.PostgresMessageModule; +import org.apache.james.mailbox.postgres.mail.dao.PostgresThreadModule; +import org.apache.james.mailbox.postgres.user.PostgresSubscriptionModule; + +public interface PostgresMailboxAggregateModule { + + PostgresModule MODULE = PostgresModule.aggregateModules( + PostgresMailboxModule.MODULE, + PostgresSubscriptionModule.MODULE, + PostgresMessageModule.MODULE, + PostgresMailboxAnnotationModule.MODULE, + PostgresAttachmentModule.MODULE, + PostgresThreadModule.MODULE); +} diff --git a/mailbox/postgres/src/main/java/org/apache/james/mailbox/postgres/PostgresMailboxAnnotationModule.java b/mailbox/postgres/src/main/java/org/apache/james/mailbox/postgres/PostgresMailboxAnnotationModule.java new file mode 100644 index 00000000000..64f0937ae81 --- /dev/null +++ b/mailbox/postgres/src/main/java/org/apache/james/mailbox/postgres/PostgresMailboxAnnotationModule.java @@ -0,0 +1,57 @@ +/**************************************************************** + * 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.mailbox.postgres; + +import static org.apache.james.mailbox.postgres.mail.PostgresMailboxModule.PostgresMailboxTable; + +import java.util.UUID; + +import org.apache.james.backends.postgres.PostgresModule; +import org.apache.james.backends.postgres.PostgresTable; +import org.jooq.Field; +import org.jooq.Record; +import org.jooq.Table; +import org.jooq.impl.DSL; +import org.jooq.impl.DefaultDataType; +import org.jooq.impl.SQLDataType; +import org.jooq.postgres.extensions.bindings.HstoreBinding; +import org.jooq.postgres.extensions.types.Hstore; + +public interface PostgresMailboxAnnotationModule { + interface PostgresMailboxAnnotationTable { + Table TABLE_NAME = DSL.table("mailbox_annotations"); + + Field MAILBOX_ID = DSL.field("mailbox_id", SQLDataType.UUID.notNull()); + Field ANNOTATIONS = DSL.field("annotations", DefaultDataType.getDefaultDataType("hstore").asConvertedDataType(new HstoreBinding()).notNull()); + + PostgresTable TABLE = PostgresTable.name(TABLE_NAME.getName()) + .createTableStep(((dsl, tableName) -> dsl.createTableIfNotExists(tableName) + .column(MAILBOX_ID) + .column(ANNOTATIONS) + .primaryKey(MAILBOX_ID) + .constraints(DSL.constraint().foreignKey(MAILBOX_ID).references(PostgresMailboxTable.TABLE_NAME, PostgresMailboxTable.MAILBOX_ID).onDeleteCascade()))) + .supportsRowLevelSecurity() + .build(); + } + + PostgresModule MODULE = PostgresModule.builder() + .addTable(PostgresMailboxAnnotationModule.PostgresMailboxAnnotationTable.TABLE) + .build(); +} diff --git a/mailbox/postgres/src/main/java/org/apache/james/mailbox/postgres/PostgresMailboxId.java b/mailbox/postgres/src/main/java/org/apache/james/mailbox/postgres/PostgresMailboxId.java new file mode 100644 index 00000000000..52111dd4cb6 --- /dev/null +++ b/mailbox/postgres/src/main/java/org/apache/james/mailbox/postgres/PostgresMailboxId.java @@ -0,0 +1,86 @@ +/**************************************************************** + * 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.mailbox.postgres; + +import java.io.Serializable; +import java.util.Objects; +import java.util.UUID; + +import org.apache.james.mailbox.model.MailboxId; + +import com.google.common.base.MoreObjects; + +public class PostgresMailboxId implements MailboxId, Serializable { + + public static class Factory implements MailboxId.Factory { + @Override + public PostgresMailboxId fromString(String serialized) { + return of(serialized); + } + } + + private final UUID id; + + public static PostgresMailboxId generate() { + return of(UUID.randomUUID()); + } + + public static PostgresMailboxId of(UUID id) { + return new PostgresMailboxId(id); + } + + public static PostgresMailboxId of(String serialized) { + return new PostgresMailboxId(UUID.fromString(serialized)); + } + + private PostgresMailboxId(UUID id) { + this.id = id; + } + + @Override + public String serialize() { + return id.toString(); + } + + public UUID asUuid() { + return id; + } + + @Override + public final boolean equals(Object o) { + if (o instanceof PostgresMailboxId) { + PostgresMailboxId other = (PostgresMailboxId) o; + return Objects.equals(id, other.id); + } + return false; + } + + @Override + public final int hashCode() { + return Objects.hash(id); + } + + @Override + public String toString() { + return MoreObjects.toStringHelper(this) + .add("id", id) + .toString(); + } + +} diff --git a/mailbox/postgres/src/main/java/org/apache/james/mailbox/postgres/PostgresMailboxManager.java b/mailbox/postgres/src/main/java/org/apache/james/mailbox/postgres/PostgresMailboxManager.java new file mode 100644 index 00000000000..bce2f957de9 --- /dev/null +++ b/mailbox/postgres/src/main/java/org/apache/james/mailbox/postgres/PostgresMailboxManager.java @@ -0,0 +1,102 @@ +/**************************************************************** + * 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.mailbox.postgres; + +import java.time.Clock; +import java.util.EnumSet; + +import jakarta.inject.Inject; + +import org.apache.james.events.EventBus; +import org.apache.james.mailbox.MailboxSession; +import org.apache.james.mailbox.SessionProvider; +import org.apache.james.mailbox.model.Mailbox; +import org.apache.james.mailbox.model.MessageId; +import org.apache.james.mailbox.store.MailboxManagerConfiguration; +import org.apache.james.mailbox.store.NoMailboxPathLocker; +import org.apache.james.mailbox.store.PreDeletionHooks; +import org.apache.james.mailbox.store.StoreMailboxAnnotationManager; +import org.apache.james.mailbox.store.StoreMailboxManager; +import org.apache.james.mailbox.store.StoreMessageManager; +import org.apache.james.mailbox.store.StoreRightManager; +import org.apache.james.mailbox.store.mail.ThreadIdGuessingAlgorithm; +import org.apache.james.mailbox.store.mail.model.impl.MessageParser; +import org.apache.james.mailbox.store.quota.QuotaComponents; +import org.apache.james.mailbox.store.search.MessageSearchIndex; + +public class PostgresMailboxManager extends StoreMailboxManager { + + public static final EnumSet MAILBOX_CAPABILITIES = EnumSet.of( + MailboxCapabilities.UserFlag, + MailboxCapabilities.Namespace, + MailboxCapabilities.Move, + MailboxCapabilities.Annotation, + MailboxCapabilities.ACL); + + private final PostgresMailboxSessionMapperFactory mapperFactory; + + @Inject + public PostgresMailboxManager(PostgresMailboxSessionMapperFactory mapperFactory, + SessionProvider sessionProvider, + MessageParser messageParser, + MessageId.Factory messageIdFactory, + EventBus eventBus, + StoreMailboxAnnotationManager annotationManager, + StoreRightManager storeRightManager, + QuotaComponents quotaComponents, + MessageSearchIndex index, + ThreadIdGuessingAlgorithm threadIdGuessingAlgorithm, + PreDeletionHooks preDeletionHooks, + Clock clock) { + super(mapperFactory, sessionProvider, new NoMailboxPathLocker(), + messageParser, messageIdFactory, annotationManager, + eventBus, storeRightManager, quotaComponents, + index, MailboxManagerConfiguration.DEFAULT, preDeletionHooks, threadIdGuessingAlgorithm, clock); + this.mapperFactory = mapperFactory; + } + + @Override + protected StoreMessageManager createMessageManager(Mailbox mailboxRow, MailboxSession session) { + return new PostgresMessageManager(mapperFactory, + getMessageSearchIndex(), + getEventBus(), + getLocker(), + mailboxRow, + getQuotaComponents().getQuotaManager(), + getQuotaComponents().getQuotaRootResolver(), + getMessageParser(), + getMessageIdFactory(), + configuration.getBatchSizes(), + getStoreRightManager(), + getThreadIdGuessingAlgorithm(), + getClock(), + getPreDeletionHooks()); + } + + @Override + public EnumSet getSupportedMailboxCapabilities() { + return MAILBOX_CAPABILITIES; + } + + @Override + public EnumSet getSupportedMessageCapabilities() { + return EnumSet.of(MessageCapabilities.UniqueID); + } +} diff --git a/mailbox/postgres/src/main/java/org/apache/james/mailbox/postgres/PostgresMailboxSessionMapperFactory.java b/mailbox/postgres/src/main/java/org/apache/james/mailbox/postgres/PostgresMailboxSessionMapperFactory.java new file mode 100644 index 00000000000..8e157c514b0 --- /dev/null +++ b/mailbox/postgres/src/main/java/org/apache/james/mailbox/postgres/PostgresMailboxSessionMapperFactory.java @@ -0,0 +1,152 @@ +/**************************************************************** + * 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.mailbox.postgres; + +import java.time.Clock; + +import jakarta.inject.Inject; + +import org.apache.james.backends.postgres.PostgresConfiguration; +import org.apache.james.backends.postgres.RowLevelSecurity; +import org.apache.james.backends.postgres.utils.PostgresExecutor; +import org.apache.james.blob.api.BlobId; +import org.apache.james.blob.api.BlobStore; +import org.apache.james.mailbox.MailboxSession; +import org.apache.james.mailbox.postgres.mail.PostgresAnnotationMapper; +import org.apache.james.mailbox.postgres.mail.PostgresAttachmentMapper; +import org.apache.james.mailbox.postgres.mail.PostgresMailboxMapper; +import org.apache.james.mailbox.postgres.mail.PostgresMailboxMemberDAO; +import org.apache.james.mailbox.postgres.mail.PostgresMessageIdMapper; +import org.apache.james.mailbox.postgres.mail.PostgresMessageMapper; +import org.apache.james.mailbox.postgres.mail.PostgresModSeqProvider; +import org.apache.james.mailbox.postgres.mail.PostgresUidProvider; +import org.apache.james.mailbox.postgres.mail.RLSSupportPostgresMailboxMapper; +import org.apache.james.mailbox.postgres.mail.dao.PostgresAttachmentDAO; +import org.apache.james.mailbox.postgres.mail.dao.PostgresMailboxAnnotationDAO; +import org.apache.james.mailbox.postgres.mail.dao.PostgresMailboxDAO; +import org.apache.james.mailbox.postgres.mail.dao.PostgresMailboxMessageDAO; +import org.apache.james.mailbox.postgres.mail.dao.PostgresMessageDAO; +import org.apache.james.mailbox.postgres.mail.dao.PostgresThreadDAO; +import org.apache.james.mailbox.postgres.user.PostgresSubscriptionDAO; +import org.apache.james.mailbox.postgres.user.PostgresSubscriptionMapper; +import org.apache.james.mailbox.store.MailboxSessionMapperFactory; +import org.apache.james.mailbox.store.mail.AnnotationMapper; +import org.apache.james.mailbox.store.mail.AttachmentMapperFactory; +import org.apache.james.mailbox.store.mail.MailboxMapper; +import org.apache.james.mailbox.store.mail.MessageIdMapper; +import org.apache.james.mailbox.store.mail.MessageMapper; +import org.apache.james.mailbox.store.user.SubscriptionMapper; + +import com.google.common.collect.ImmutableSet; + +public class PostgresMailboxSessionMapperFactory extends MailboxSessionMapperFactory implements AttachmentMapperFactory { + + private final PostgresExecutor.Factory executorFactory; + private final BlobStore blobStore; + private final BlobId.Factory blobIdFactory; + private final Clock clock; + private final RowLevelSecurity rowLevelSecurity; + + @Inject + public PostgresMailboxSessionMapperFactory(PostgresExecutor.Factory executorFactory, + Clock clock, + BlobStore blobStore, + BlobId.Factory blobIdFactory, + PostgresConfiguration postgresConfiguration) { + this.executorFactory = executorFactory; + this.blobStore = blobStore; + this.blobIdFactory = blobIdFactory; + this.clock = clock; + this.rowLevelSecurity = postgresConfiguration.getRowLevelSecurity(); + } + + @Override + public MailboxMapper createMailboxMapper(MailboxSession session) { + PostgresMailboxDAO mailboxDAO = new PostgresMailboxDAO(executorFactory.create(session.getUser().getDomainPart())); + if (rowLevelSecurity.isRowLevelSecurityEnabled()) { + return new RLSSupportPostgresMailboxMapper(mailboxDAO, + new PostgresMailboxMemberDAO(executorFactory.create(session.getUser().getDomainPart()))); + } else { + return new PostgresMailboxMapper(mailboxDAO); + } + } + + @Override + public MessageMapper createMessageMapper(MailboxSession session) { + return new PostgresMessageMapper(executorFactory.create(session.getUser().getDomainPart()), + getModSeqProvider(session), + getUidProvider(session), + blobStore, + clock, + blobIdFactory); + } + + @Override + public MessageIdMapper createMessageIdMapper(MailboxSession session) { + return new PostgresMessageIdMapper(new PostgresMailboxDAO(executorFactory.create(session.getUser().getDomainPart())), + new PostgresMessageDAO(executorFactory.create(session.getUser().getDomainPart()), blobIdFactory), + new PostgresMailboxMessageDAO(executorFactory.create(session.getUser().getDomainPart())), + getModSeqProvider(session), + getAttachmentMapper(session), + blobStore, + blobIdFactory, + clock); + } + + @Override + public SubscriptionMapper createSubscriptionMapper(MailboxSession session) { + return new PostgresSubscriptionMapper(new PostgresSubscriptionDAO(executorFactory.create(session.getUser().getDomainPart()))); + } + + @Override + public AnnotationMapper createAnnotationMapper(MailboxSession session) { + return new PostgresAnnotationMapper(new PostgresMailboxAnnotationDAO(executorFactory.create(session.getUser().getDomainPart()))); + } + + @Override + public PostgresUidProvider getUidProvider(MailboxSession session) { + return new PostgresUidProvider.Factory(executorFactory).create(session); + } + + @Override + public PostgresModSeqProvider getModSeqProvider(MailboxSession session) { + return new PostgresModSeqProvider.Factory(executorFactory).create(session); + } + + @Override + public PostgresAttachmentMapper createAttachmentMapper(MailboxSession session) { + PostgresAttachmentDAO postgresAttachmentDAO = new PostgresAttachmentDAO(executorFactory.create(session.getUser().getDomainPart()), blobIdFactory); + return new PostgresAttachmentMapper(postgresAttachmentDAO, blobStore); + } + + @Override + public PostgresAttachmentMapper getAttachmentMapper(MailboxSession session) { + return createAttachmentMapper(session); + } + + protected DeleteMessageListener deleteMessageListener() { + PostgresMessageDAO.Factory postgresMessageDAOFactory = new PostgresMessageDAO.Factory(blobIdFactory, executorFactory); + PostgresMailboxMessageDAO.Factory postgresMailboxMessageDAOFactory = new PostgresMailboxMessageDAO.Factory(executorFactory); + PostgresAttachmentDAO.Factory attachmentDAOFactory = new PostgresAttachmentDAO.Factory(executorFactory, blobIdFactory); + PostgresThreadDAO.Factory threadDAOFactory = new PostgresThreadDAO.Factory(executorFactory); + + return new DeleteMessageListener(blobStore, postgresMailboxMessageDAOFactory, postgresMessageDAOFactory, + attachmentDAOFactory, threadDAOFactory, ImmutableSet.of()); + } +} diff --git a/mailbox/postgres/src/main/java/org/apache/james/mailbox/postgres/PostgresMessageId.java b/mailbox/postgres/src/main/java/org/apache/james/mailbox/postgres/PostgresMessageId.java new file mode 100644 index 00000000000..57594b45987 --- /dev/null +++ b/mailbox/postgres/src/main/java/org/apache/james/mailbox/postgres/PostgresMessageId.java @@ -0,0 +1,89 @@ +/**************************************************************** + * 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.mailbox.postgres; + +import java.util.Objects; +import java.util.UUID; + +import org.apache.james.mailbox.model.MessageId; + +import com.github.f4b6a3.uuid.UuidCreator; +import com.google.common.base.MoreObjects; + +public class PostgresMessageId implements MessageId { + + public static class Factory implements MessageId.Factory { + + @Override + public PostgresMessageId generate() { + return of(UuidCreator.getTimeOrderedEpoch()); + } + + public static PostgresMessageId of(UUID uuid) { + return new PostgresMessageId(uuid); + } + + @Override + public PostgresMessageId fromString(String serialized) { + return of(UUID.fromString(serialized)); + } + } + + private final UUID uuid; + + private PostgresMessageId(UUID uuid) { + this.uuid = uuid; + } + + @Override + public String serialize() { + return uuid.toString(); + } + + public UUID asUuid() { + return uuid; + } + + @Override + public boolean isSerializable() { + return true; + } + + @Override + public final boolean equals(Object o) { + if (o instanceof PostgresMessageId) { + PostgresMessageId other = (PostgresMessageId) o; + return Objects.equals(uuid, other.uuid); + } + return false; + } + + @Override + public final int hashCode() { + return Objects.hash(uuid); + } + + @Override + public String toString() { + return MoreObjects.toStringHelper(this) + .add("uuid", uuid) + .toString(); + } +} diff --git a/mailbox/postgres/src/main/java/org/apache/james/mailbox/postgres/PostgresMessageManager.java b/mailbox/postgres/src/main/java/org/apache/james/mailbox/postgres/PostgresMessageManager.java new file mode 100644 index 00000000000..ad2621b4aaf --- /dev/null +++ b/mailbox/postgres/src/main/java/org/apache/james/mailbox/postgres/PostgresMessageManager.java @@ -0,0 +1,124 @@ +/**************************************************************** + * 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.mailbox.postgres; + +import java.time.Clock; +import java.util.EnumSet; +import java.util.List; +import java.util.Optional; + +import jakarta.mail.Flags; + +import org.apache.james.events.EventBus; +import org.apache.james.mailbox.MailboxPathLocker; +import org.apache.james.mailbox.MailboxSession; +import org.apache.james.mailbox.MessageUid; +import org.apache.james.mailbox.exception.MailboxException; +import org.apache.james.mailbox.model.Mailbox; +import org.apache.james.mailbox.model.MailboxACL; +import org.apache.james.mailbox.model.MailboxCounters; +import org.apache.james.mailbox.model.MessageId; +import org.apache.james.mailbox.postgres.mail.PostgresMailbox; +import org.apache.james.mailbox.quota.QuotaManager; +import org.apache.james.mailbox.quota.QuotaRootResolver; +import org.apache.james.mailbox.store.BatchSizes; +import org.apache.james.mailbox.store.MailboxSessionMapperFactory; +import org.apache.james.mailbox.store.MessageFactory; +import org.apache.james.mailbox.store.MessageStorer; +import org.apache.james.mailbox.store.PreDeletionHooks; +import org.apache.james.mailbox.store.StoreMailboxManager; +import org.apache.james.mailbox.store.StoreMessageManager; +import org.apache.james.mailbox.store.StoreRightManager; +import org.apache.james.mailbox.store.mail.MessageMapper; +import org.apache.james.mailbox.store.mail.ThreadIdGuessingAlgorithm; +import org.apache.james.mailbox.store.mail.model.impl.MessageParser; +import org.apache.james.mailbox.store.search.MessageSearchIndex; + +import reactor.core.publisher.Mono; +import reactor.util.function.Tuple2; + +public class PostgresMessageManager extends StoreMessageManager { + + private final MailboxSessionMapperFactory mapperFactory; + private final StoreRightManager storeRightManager; + private final Mailbox mailbox; + + public PostgresMessageManager(PostgresMailboxSessionMapperFactory mapperFactory, + MessageSearchIndex index, EventBus eventBus, + MailboxPathLocker locker, Mailbox mailbox, + QuotaManager quotaManager, QuotaRootResolver quotaRootResolver, + MessageParser messageParser, + MessageId.Factory messageIdFactory, BatchSizes batchSizes, + StoreRightManager storeRightManager, ThreadIdGuessingAlgorithm threadIdGuessingAlgorithm, + Clock clock, PreDeletionHooks preDeletionHooks) { + super(StoreMailboxManager.DEFAULT_NO_MESSAGE_CAPABILITIES, mapperFactory, index, eventBus, locker, mailbox, + quotaManager, quotaRootResolver, batchSizes, storeRightManager, preDeletionHooks, + new MessageStorer.WithAttachment(mapperFactory, messageIdFactory, new MessageFactory.StoreMessageFactory(), mapperFactory, messageParser, threadIdGuessingAlgorithm, clock)); + this.storeRightManager = storeRightManager; + this.mapperFactory = mapperFactory; + this.mailbox = mailbox; + } + + + @Override + public Flags getPermanentFlags(MailboxSession session) { + Flags flags = super.getPermanentFlags(session); + flags.add(Flags.Flag.USER); + return flags; + } + + public Mono getMetaDataReactive(MailboxMetaData.RecentMode recentMode, MailboxSession mailboxSession, EnumSet items) throws MailboxException { + if (!storeRightManager.hasRight(mailbox, MailboxACL.Right.Read, mailboxSession)) { + return Mono.just(MailboxMetaData.sensibleInformationFree(getResolvedAcl(mailboxSession), getMailboxEntity().getUidValidity(), isWriteable(mailboxSession))); + } + + Flags permanentFlags = getPermanentFlags(mailboxSession); + MessageMapper messageMapper = mapperFactory.getMessageMapper(mailboxSession); + + Mono postgresMailboxMetaDataPublisher = Mono.just(mapperFactory.getMailboxMapper(mailboxSession)) + .flatMap(postgresMailboxMapper -> postgresMailboxMapper.findMailboxById(getMailboxEntity().getMailboxId()) + .map(mailbox -> (PostgresMailbox) mailbox)); + + Mono, List>> firstUnseenAndRecentPublisher = Mono.zip(firstUnseen(messageMapper, items), recent(recentMode, mailboxSession)); + + return messageMapper.executeReactive(Mono.zip(postgresMailboxMetaDataPublisher, mailboxCounters(messageMapper, items)) + .flatMap(metadataAndCounter -> { + PostgresMailbox metadata = metadataAndCounter.getT1(); + MailboxCounters counters = metadataAndCounter.getT2(); + return firstUnseenAndRecentPublisher.map(firstUnseenAndRecent -> new MailboxMetaData( + firstUnseenAndRecent.getT2(), + permanentFlags, + metadata.getUidValidity(), + nextUid(metadata), + metadata.getHighestModSeq(), + counters.getCount(), + counters.getUnseen(), + firstUnseenAndRecent.getT1().orElse(null), + isWriteable(mailboxSession), + metadata.getACL())); + })); + } + + private MessageUid nextUid(PostgresMailbox mailboxMetaData) { + return Optional.ofNullable(mailboxMetaData.getLastUid()) + .map(MessageUid::next) + .orElse(MessageUid.MIN_VALUE); + } +} diff --git a/mailbox/postgres/src/main/java/org/apache/james/mailbox/postgres/PostgresThreadIdGuessingAlgorithm.java b/mailbox/postgres/src/main/java/org/apache/james/mailbox/postgres/PostgresThreadIdGuessingAlgorithm.java new file mode 100644 index 00000000000..77419e98fc1 --- /dev/null +++ b/mailbox/postgres/src/main/java/org/apache/james/mailbox/postgres/PostgresThreadIdGuessingAlgorithm.java @@ -0,0 +1,91 @@ +/**************************************************************** + * 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.mailbox.postgres; + +import java.util.HashSet; +import java.util.List; +import java.util.Optional; +import java.util.Set; +import java.util.stream.Collectors; + +import jakarta.inject.Inject; + +import org.apache.commons.lang3.tuple.Pair; +import org.apache.james.mailbox.MailboxSession; +import org.apache.james.mailbox.exception.ThreadNotFoundException; +import org.apache.james.mailbox.model.MessageId; +import org.apache.james.mailbox.model.ThreadId; +import org.apache.james.mailbox.postgres.mail.dao.PostgresThreadDAO; +import org.apache.james.mailbox.store.mail.ThreadIdGuessingAlgorithm; +import org.apache.james.mailbox.store.mail.model.MimeMessageId; +import org.apache.james.mailbox.store.mail.model.Subject; +import org.apache.james.mailbox.store.search.SearchUtil; + +import com.google.common.hash.Hashing; + +import reactor.core.publisher.Flux; +import reactor.core.publisher.Mono; + +public class PostgresThreadIdGuessingAlgorithm implements ThreadIdGuessingAlgorithm { + private final PostgresThreadDAO.Factory threadDAOFactory; + + @Inject + public PostgresThreadIdGuessingAlgorithm(PostgresThreadDAO.Factory threadDAOFactory) { + this.threadDAOFactory = threadDAOFactory; + } + + @Override + public Mono guessThreadIdReactive(MessageId messageId, Optional mimeMessageId, Optional inReplyTo, + Optional> references, Optional subject, MailboxSession session) { + PostgresThreadDAO threadDAO = threadDAOFactory.create(session.getUser().getDomainPart()); + + Set hashMimeMessageIds = buildMimeMessageIdSet(mimeMessageId, inReplyTo, references) + .stream() + .map(mimeMessageId1 -> Hashing.murmur3_32_fixed().hashBytes(mimeMessageId1.getValue().getBytes()).asInt()) + .collect(Collectors.toSet()); + + Optional hashBaseSubject = subject.map(value -> new Subject(SearchUtil.getBaseSubject(value.getValue()))) + .map(subject1 -> Hashing.murmur3_32_fixed().hashBytes(subject1.getValue().getBytes()).asInt()); + + return threadDAO.findThreads(session.getUser(), hashMimeMessageIds) + .filter(pair -> pair.getLeft().equals(hashBaseSubject)) + .next() + .map(Pair::getRight) + .switchIfEmpty(Mono.just(ThreadId.fromBaseMessageId(messageId))) + .flatMap(threadId -> threadDAO + .insertSome(session.getUser(), hashMimeMessageIds, PostgresMessageId.class.cast(messageId), threadId, hashBaseSubject) + .then(Mono.just(threadId))); + } + + @Override + public Flux getMessageIdsInThread(ThreadId threadId, MailboxSession session) { + PostgresThreadDAO threadDAO = threadDAOFactory.create(session.getUser().getDomainPart()); + return threadDAO.findMessageIds(threadId, session.getUser()) + .switchIfEmpty(Flux.error(new ThreadNotFoundException(threadId))); + } + + private Set buildMimeMessageIdSet(Optional mimeMessageId, Optional inReplyTo, Optional> references) { + Set mimeMessageIds = new HashSet<>(); + mimeMessageId.ifPresent(mimeMessageIds::add); + inReplyTo.ifPresent(mimeMessageIds::add); + references.ifPresent(mimeMessageIds::addAll); + return mimeMessageIds; + } +} diff --git a/mailbox/postgres/src/main/java/org/apache/james/mailbox/postgres/UnsupportAttachmentContentLoader.java b/mailbox/postgres/src/main/java/org/apache/james/mailbox/postgres/UnsupportAttachmentContentLoader.java new file mode 100644 index 00000000000..e4954999eca --- /dev/null +++ b/mailbox/postgres/src/main/java/org/apache/james/mailbox/postgres/UnsupportAttachmentContentLoader.java @@ -0,0 +1,34 @@ +/**************************************************************** + * 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.mailbox.postgres; + +import java.io.InputStream; + +import org.apache.commons.lang3.NotImplementedException; +import org.apache.james.mailbox.AttachmentContentLoader; +import org.apache.james.mailbox.MailboxSession; +import org.apache.james.mailbox.model.AttachmentMetadata; + +public class UnsupportAttachmentContentLoader implements AttachmentContentLoader { + @Override + public InputStream load(AttachmentMetadata attachment, MailboxSession mailboxSession) { + throw new NotImplementedException("Postgresql doesn't support loading attachment separately from Message"); + } +} diff --git a/mailbox/postgres/src/main/java/org/apache/james/mailbox/postgres/mail/MailboxDeleteDuringUpdateException.java b/mailbox/postgres/src/main/java/org/apache/james/mailbox/postgres/mail/MailboxDeleteDuringUpdateException.java new file mode 100644 index 00000000000..e738905441a --- /dev/null +++ b/mailbox/postgres/src/main/java/org/apache/james/mailbox/postgres/mail/MailboxDeleteDuringUpdateException.java @@ -0,0 +1,23 @@ +/**************************************************************** + * 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.mailbox.postgres.mail; + +public class MailboxDeleteDuringUpdateException extends Exception { +} diff --git a/mailbox/postgres/src/main/java/org/apache/james/mailbox/postgres/mail/MessageRepresentation.java b/mailbox/postgres/src/main/java/org/apache/james/mailbox/postgres/mail/MessageRepresentation.java new file mode 100644 index 00000000000..dd24c5dd60d --- /dev/null +++ b/mailbox/postgres/src/main/java/org/apache/james/mailbox/postgres/mail/MessageRepresentation.java @@ -0,0 +1,179 @@ +/**************************************************************** + * 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.mailbox.postgres.mail; + +import java.util.Date; +import java.util.List; +import java.util.Optional; +import java.util.stream.Collectors; + +import org.apache.james.blob.api.BlobId; +import org.apache.james.mailbox.model.AttachmentId; +import org.apache.james.mailbox.model.Cid; +import org.apache.james.mailbox.model.Content; +import org.apache.james.mailbox.model.MessageAttachmentMetadata; +import org.apache.james.mailbox.model.MessageId; + +import com.google.common.base.Preconditions; + +public class MessageRepresentation { + public static class AttachmentRepresentation { + public static AttachmentRepresentation from(MessageAttachmentMetadata messageAttachmentMetadata) { + return new AttachmentRepresentation( + messageAttachmentMetadata.getAttachment().getAttachmentId(), + messageAttachmentMetadata.getName(), + messageAttachmentMetadata.getCid(), + messageAttachmentMetadata.isInline()); + } + + public static List from(List messageAttachmentMetadata) { + return messageAttachmentMetadata.stream() + .map(AttachmentRepresentation::from) + .collect(Collectors.toList()); + } + + private final AttachmentId attachmentId; + private final Optional name; + private final Optional cid; + private final boolean isInline; + + public AttachmentRepresentation(AttachmentId attachmentId, Optional name, Optional cid, boolean isInline) { + Preconditions.checkNotNull(attachmentId, "attachmentId is required"); + this.attachmentId = attachmentId; + this.name = name; + this.cid = cid; + this.isInline = isInline; + } + + public AttachmentId getAttachmentId() { + return attachmentId; + } + + public Optional getName() { + return name; + } + + public Optional getCid() { + return cid; + } + + public boolean isInline() { + return isInline; + } + } + + public static MessageRepresentation.Builder builder() { + return new MessageRepresentation.Builder(); + } + + public static class Builder { + private MessageId messageId; + private Date internalDate; + private Long size; + private Content headerContent; + private BlobId bodyBlobId; + + private List attachments = List.of(); + + public MessageRepresentation.Builder messageId(MessageId messageId) { + this.messageId = messageId; + return this; + } + + public MessageRepresentation.Builder internalDate(Date internalDate) { + this.internalDate = internalDate; + return this; + } + + public MessageRepresentation.Builder size(long size) { + Preconditions.checkArgument(size >= 0, "size can not be negative"); + this.size = size; + return this; + } + + public MessageRepresentation.Builder headerContent(Content headerContent) { + this.headerContent = headerContent; + return this; + } + + public MessageRepresentation.Builder bodyBlobId(BlobId bodyBlobId) { + this.bodyBlobId = bodyBlobId; + return this; + } + + public MessageRepresentation.Builder attachments(List attachments) { + this.attachments = attachments; + return this; + } + + public MessageRepresentation build() { + Preconditions.checkNotNull(messageId, "messageId is required"); + Preconditions.checkNotNull(internalDate, "internalDate is required"); + Preconditions.checkNotNull(size, "size is required"); + Preconditions.checkNotNull(headerContent, "headerContent is required"); + Preconditions.checkNotNull(bodyBlobId, "mailboxId is required"); + + return new MessageRepresentation(messageId, internalDate, size, headerContent, bodyBlobId, attachments); + } + } + + private final MessageId messageId; + private final Date internalDate; + private final Long size; + private final Content headerContent; + private final BlobId bodyBlobId; + + private final List attachments; + + private MessageRepresentation(MessageId messageId, Date internalDate, Long size, + Content headerContent, BlobId bodyBlobId, + List attachments) { + this.messageId = messageId; + this.internalDate = internalDate; + this.size = size; + this.headerContent = headerContent; + this.bodyBlobId = bodyBlobId; + this.attachments = attachments; + } + + public Date getInternalDate() { + return internalDate; + } + + public Long getSize() { + return size; + } + + public MessageId getMessageId() { + return messageId; + } + + public Content getHeaderContent() { + return headerContent; + } + + public BlobId getBodyBlobId() { + return bodyBlobId; + } + + public List getAttachments() { + return attachments; + } +} diff --git a/mailbox/postgres/src/main/java/org/apache/james/mailbox/postgres/mail/PostgresACLUpsertException.java b/mailbox/postgres/src/main/java/org/apache/james/mailbox/postgres/mail/PostgresACLUpsertException.java new file mode 100644 index 00000000000..f87e8090ac7 --- /dev/null +++ b/mailbox/postgres/src/main/java/org/apache/james/mailbox/postgres/mail/PostgresACLUpsertException.java @@ -0,0 +1,26 @@ +/**************************************************************** + * 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.mailbox.postgres.mail; + +public class PostgresACLUpsertException extends RuntimeException { + public PostgresACLUpsertException(String message) { + super(message); + } +} diff --git a/mailbox/postgres/src/main/java/org/apache/james/mailbox/postgres/mail/PostgresAnnotationMapper.java b/mailbox/postgres/src/main/java/org/apache/james/mailbox/postgres/mail/PostgresAnnotationMapper.java new file mode 100644 index 00000000000..c58498be1f5 --- /dev/null +++ b/mailbox/postgres/src/main/java/org/apache/james/mailbox/postgres/mail/PostgresAnnotationMapper.java @@ -0,0 +1,140 @@ +/**************************************************************** + * 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.mailbox.postgres.mail; + +import java.util.List; +import java.util.Set; + +import jakarta.inject.Inject; + +import org.apache.james.mailbox.model.MailboxAnnotation; +import org.apache.james.mailbox.model.MailboxAnnotationKey; +import org.apache.james.mailbox.model.MailboxId; +import org.apache.james.mailbox.postgres.PostgresMailboxId; +import org.apache.james.mailbox.postgres.mail.dao.PostgresMailboxAnnotationDAO; +import org.apache.james.mailbox.store.mail.AnnotationMapper; + +import reactor.core.publisher.Flux; +import reactor.core.publisher.Mono; + +public class PostgresAnnotationMapper implements AnnotationMapper { + private final PostgresMailboxAnnotationDAO annotationDAO; + + @Inject + public PostgresAnnotationMapper(PostgresMailboxAnnotationDAO annotationDAO) { + this.annotationDAO = annotationDAO; + } + + @Override + public List getAllAnnotations(MailboxId mailboxId) { + return getAllAnnotationsReactive(mailboxId) + .collectList() + .block(); + } + + @Override + public Flux getAllAnnotationsReactive(MailboxId mailboxId) { + return annotationDAO.getAllAnnotations((PostgresMailboxId) mailboxId); + } + + @Override + public List getAnnotationsByKeys(MailboxId mailboxId, Set keys) { + return getAnnotationsByKeysReactive(mailboxId, keys) + .collectList() + .block(); + } + + @Override + public Flux getAnnotationsByKeysReactive(MailboxId mailboxId, Set keys) { + return annotationDAO.getAnnotationsByKeys((PostgresMailboxId) mailboxId, keys); + } + + @Override + public List getAnnotationsByKeysWithOneDepth(MailboxId mailboxId, Set keys) { + return getAnnotationsByKeysWithOneDepthReactive(mailboxId, keys) + .collectList() + .block(); + } + + @Override + public Flux getAnnotationsByKeysWithOneDepthReactive(MailboxId mailboxId, Set keys) { + return Flux.fromIterable(keys).flatMap(mailboxAnnotationKey -> + annotationDAO.getAnnotationsByKeyLike((PostgresMailboxId) mailboxId, mailboxAnnotationKey) + .filter(annotation -> mailboxAnnotationKey.isParentOrIsEqual(annotation.getKey()))); + } + + @Override + public List getAnnotationsByKeysWithAllDepth(MailboxId mailboxId, Set keys) { + return getAnnotationsByKeysWithAllDepthReactive(mailboxId, keys) + .collectList() + .block(); + } + + @Override + public Flux getAnnotationsByKeysWithAllDepthReactive(MailboxId mailboxId, Set keys) { + return Flux.fromIterable(keys).flatMap(mailboxAnnotationKey -> + annotationDAO.getAnnotationsByKeyLike((PostgresMailboxId) mailboxId, mailboxAnnotationKey) + .filter(annotation -> mailboxAnnotationKey.isAncestorOrIsEqual(annotation.getKey()))); + } + + @Override + public void deleteAnnotation(MailboxId mailboxId, MailboxAnnotationKey key) { + deleteAnnotationReactive(mailboxId, key) + .block(); + } + + @Override + public Mono deleteAnnotationReactive(MailboxId mailboxId, MailboxAnnotationKey key) { + return annotationDAO.deleteAnnotation((PostgresMailboxId) mailboxId, key); + } + + @Override + public void insertAnnotation(MailboxId mailboxId, MailboxAnnotation mailboxAnnotation) { + insertAnnotationReactive(mailboxId, mailboxAnnotation) + .block(); + } + + @Override + public Mono insertAnnotationReactive(MailboxId mailboxId, MailboxAnnotation mailboxAnnotation) { + return annotationDAO.insertAnnotation((PostgresMailboxId) mailboxId, mailboxAnnotation); + } + + @Override + public boolean exist(MailboxId mailboxId, MailboxAnnotation mailboxAnnotation) { + return existReactive(mailboxId, mailboxAnnotation) + .block(); + } + + @Override + public Mono existReactive(MailboxId mailboxId, MailboxAnnotation mailboxAnnotation) { + return annotationDAO.exist((PostgresMailboxId) mailboxId, mailboxAnnotation.getKey()); + } + + @Override + public int countAnnotations(MailboxId mailboxId) { + return countAnnotationsReactive(mailboxId) + .block(); + } + + @Override + public Mono countAnnotationsReactive(MailboxId mailboxId) { + return annotationDAO.countAnnotations((PostgresMailboxId) mailboxId); + } +} diff --git a/mailbox/postgres/src/main/java/org/apache/james/mailbox/postgres/mail/PostgresAttachmentBlobReferenceSource.java b/mailbox/postgres/src/main/java/org/apache/james/mailbox/postgres/mail/PostgresAttachmentBlobReferenceSource.java new file mode 100644 index 00000000000..3e64be72e31 --- /dev/null +++ b/mailbox/postgres/src/main/java/org/apache/james/mailbox/postgres/mail/PostgresAttachmentBlobReferenceSource.java @@ -0,0 +1,53 @@ +/**************************************************************** + * 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.mailbox.postgres.mail; + +import jakarta.inject.Inject; +import jakarta.inject.Named; +import jakarta.inject.Singleton; + +import org.apache.james.backends.postgres.utils.PostgresExecutor; +import org.apache.james.blob.api.BlobId; +import org.apache.james.blob.api.BlobReferenceSource; +import org.apache.james.mailbox.postgres.mail.dao.PostgresAttachmentDAO; + +import reactor.core.publisher.Flux; + +public class PostgresAttachmentBlobReferenceSource implements BlobReferenceSource { + + private final PostgresAttachmentDAO postgresAttachmentDAO; + + @Inject + @Singleton + public PostgresAttachmentBlobReferenceSource(@Named(PostgresExecutor.BY_PASS_RLS_INJECT) PostgresExecutor postgresExecutor, + BlobId.Factory bloIdFactory) { + this(new PostgresAttachmentDAO(postgresExecutor, bloIdFactory)); + } + + public PostgresAttachmentBlobReferenceSource(PostgresAttachmentDAO postgresAttachmentDAO) { + this.postgresAttachmentDAO = postgresAttachmentDAO; + } + + @Override + public Flux listReferencedBlobs() { + return postgresAttachmentDAO.listBlobs(); + } + +} diff --git a/mailbox/postgres/src/main/java/org/apache/james/mailbox/postgres/mail/PostgresAttachmentMapper.java b/mailbox/postgres/src/main/java/org/apache/james/mailbox/postgres/mail/PostgresAttachmentMapper.java new file mode 100644 index 00000000000..1be53fa3a64 --- /dev/null +++ b/mailbox/postgres/src/main/java/org/apache/james/mailbox/postgres/mail/PostgresAttachmentMapper.java @@ -0,0 +1,125 @@ +/**************************************************************** + * 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.mailbox.postgres.mail; + +import static org.apache.james.blob.api.BlobStore.StoragePolicy.LOW_COST; + +import java.io.InputStream; +import java.util.Collection; +import java.util.List; + +import org.apache.commons.lang3.tuple.Pair; +import org.apache.james.blob.api.BlobStore; +import org.apache.james.mailbox.exception.AttachmentNotFoundException; +import org.apache.james.mailbox.model.AttachmentId; +import org.apache.james.mailbox.model.AttachmentMetadata; +import org.apache.james.mailbox.model.MessageAttachmentMetadata; +import org.apache.james.mailbox.model.MessageId; +import org.apache.james.mailbox.model.ParsedAttachment; +import org.apache.james.mailbox.model.UuidBackedAttachmentId; +import org.apache.james.mailbox.postgres.mail.dao.PostgresAttachmentDAO; +import org.apache.james.mailbox.store.mail.AttachmentMapper; + +import com.github.fge.lambdas.Throwing; +import com.google.common.base.Preconditions; + +import reactor.core.publisher.Flux; +import reactor.core.publisher.Mono; + +public class PostgresAttachmentMapper implements AttachmentMapper { + + private final PostgresAttachmentDAO postgresAttachmentDAO; + private final BlobStore blobStore; + + public PostgresAttachmentMapper(PostgresAttachmentDAO postgresAttachmentDAO, BlobStore blobStore) { + this.postgresAttachmentDAO = postgresAttachmentDAO; + this.blobStore = blobStore; + } + + @Override + public InputStream loadAttachmentContent(AttachmentId attachmentId) { + return loadAttachmentContentReactive(attachmentId) + .block(); + } + + @Override + public Mono loadAttachmentContentReactive(AttachmentId attachmentId) { + return postgresAttachmentDAO.getAttachment(attachmentId) + .flatMap(pair -> Mono.from(blobStore.readReactive(blobStore.getDefaultBucketName(), pair.getRight(), LOW_COST))) + .switchIfEmpty(Mono.error(() -> new AttachmentNotFoundException(attachmentId.toString()))); + } + + @Override + public AttachmentMetadata getAttachment(AttachmentId attachmentId) throws AttachmentNotFoundException { + Preconditions.checkArgument(attachmentId != null); + return postgresAttachmentDAO.getAttachment(attachmentId) + .map(Pair::getLeft) + .blockOptional() + .orElseThrow(() -> new AttachmentNotFoundException(attachmentId.getId())); + } + + @Override + public Mono getAttachmentReactive(AttachmentId attachmentId) { + Preconditions.checkArgument(attachmentId != null); + return postgresAttachmentDAO.getAttachment(attachmentId) + .map(Pair::getLeft) + .switchIfEmpty(Mono.error(() -> new AttachmentNotFoundException(attachmentId.getId()))); + } + + public Flux getAttachmentsReactive(Collection attachmentIds) { + Preconditions.checkArgument(attachmentIds != null); + return postgresAttachmentDAO.getAttachments(attachmentIds); + } + + @Override + public List getAttachments(Collection attachmentIds) { + return getAttachmentsReactive(attachmentIds) + .collectList() + .block(); + } + + @Override + public List storeAttachments(Collection attachments, MessageId ownerMessageId) { + return storeAttachmentsReactive(attachments, ownerMessageId) + .block(); + } + + @Override + public Mono> storeAttachmentsReactive(Collection attachments, MessageId ownerMessageId) { + return Flux.fromIterable(attachments) + .concatMap(attachment -> storeAttachmentAsync(attachment, ownerMessageId)) + .collectList(); + } + + private Mono storeAttachmentAsync(ParsedAttachment parsedAttachment, MessageId ownerMessageId) { + return Mono.fromCallable(parsedAttachment::getContent) + .flatMap(content -> Mono.from(blobStore.save(blobStore.getDefaultBucketName(), parsedAttachment.getContent(), BlobStore.StoragePolicy.LOW_COST)) + .flatMap(blobId -> { + AttachmentId attachmentId = UuidBackedAttachmentId.random(); + return postgresAttachmentDAO.storeAttachment(AttachmentMetadata.builder() + .attachmentId(attachmentId) + .type(parsedAttachment.getContentType()) + .size(Throwing.supplier(content::size).get()) + .messageId(ownerMessageId) + .build(), blobId) + .thenReturn(Throwing.supplier(() -> parsedAttachment.asMessageAttachment(attachmentId, ownerMessageId)).get()); + })); + } +} diff --git a/mailbox/postgres/src/main/java/org/apache/james/mailbox/postgres/mail/PostgresAttachmentModule.java b/mailbox/postgres/src/main/java/org/apache/james/mailbox/postgres/mail/PostgresAttachmentModule.java new file mode 100644 index 00000000000..2bc4e0b16b2 --- /dev/null +++ b/mailbox/postgres/src/main/java/org/apache/james/mailbox/postgres/mail/PostgresAttachmentModule.java @@ -0,0 +1,64 @@ +/**************************************************************** + * 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.mailbox.postgres.mail; + +import java.util.UUID; + +import org.apache.james.backends.postgres.PostgresIndex; +import org.apache.james.backends.postgres.PostgresModule; +import org.apache.james.backends.postgres.PostgresTable; +import org.jooq.Field; +import org.jooq.Record; +import org.jooq.Table; +import org.jooq.impl.DSL; +import org.jooq.impl.SQLDataType; + +public interface PostgresAttachmentModule { + + interface PostgresAttachmentTable { + + Table TABLE_NAME = DSL.table("attachment"); + Field ID = DSL.field("id", SQLDataType.UUID.notNull()); + Field BLOB_ID = DSL.field("blob_id", SQLDataType.VARCHAR); + Field TYPE = DSL.field("type", SQLDataType.VARCHAR); + Field MESSAGE_ID = DSL.field("message_id", SQLDataType.UUID); + Field SIZE = DSL.field("size", SQLDataType.BIGINT); + + PostgresTable TABLE = PostgresTable.name(TABLE_NAME.getName()) + .createTableStep(((dsl, tableName) -> dsl.createTableIfNotExists(tableName) + .column(ID) + .column(BLOB_ID) + .column(TYPE) + .column(MESSAGE_ID) + .column(SIZE) + .constraint(DSL.primaryKey(ID)))) + .supportsRowLevelSecurity() + .build(); + + PostgresIndex MESSAGE_ID_INDEX = PostgresIndex.name("attachment_message_id_index") + .createIndexStep((dsl, indexName) -> dsl.createIndexIfNotExists(indexName) + .on(TABLE_NAME, MESSAGE_ID)); + } + + PostgresModule MODULE = PostgresModule.builder() + .addTable(PostgresAttachmentTable.TABLE) + .addIndex(PostgresAttachmentTable.MESSAGE_ID_INDEX) + .build(); +} diff --git a/mailbox/postgres/src/main/java/org/apache/james/mailbox/postgres/mail/PostgresMailbox.java b/mailbox/postgres/src/main/java/org/apache/james/mailbox/postgres/mail/PostgresMailbox.java new file mode 100644 index 00000000000..0485f5f49b9 --- /dev/null +++ b/mailbox/postgres/src/main/java/org/apache/james/mailbox/postgres/mail/PostgresMailbox.java @@ -0,0 +1,54 @@ +/**************************************************************** + * 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.mailbox.postgres.mail; + +import org.apache.james.mailbox.MessageUid; +import org.apache.james.mailbox.ModSeq; +import org.apache.james.mailbox.model.Mailbox; + +public class PostgresMailbox extends Mailbox { + private final ModSeq highestModSeq; + private final MessageUid lastUid; + + public PostgresMailbox(Mailbox mailbox, ModSeq highestModSeq, MessageUid lastUid) { + super(mailbox); + this.highestModSeq = highestModSeq; + this.lastUid = lastUid; + } + + + public ModSeq getHighestModSeq() { + return highestModSeq; + } + + public MessageUid getLastUid() { + return lastUid; + } + + @Override + public final boolean equals(Object o) { + return super.equals(o); + } + + @Override + public final int hashCode() { + return super.hashCode(); + } +} diff --git a/mailbox/postgres/src/main/java/org/apache/james/mailbox/postgres/mail/PostgresMailboxMapper.java b/mailbox/postgres/src/main/java/org/apache/james/mailbox/postgres/mail/PostgresMailboxMapper.java new file mode 100644 index 00000000000..e68bb3f1850 --- /dev/null +++ b/mailbox/postgres/src/main/java/org/apache/james/mailbox/postgres/mail/PostgresMailboxMapper.java @@ -0,0 +1,118 @@ +/**************************************************************** + * 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.mailbox.postgres.mail; + +import java.time.Duration; +import java.util.function.Function; + +import org.apache.james.core.Username; +import org.apache.james.mailbox.acl.ACLDiff; +import org.apache.james.mailbox.exception.UnsupportedRightException; +import org.apache.james.mailbox.model.Mailbox; +import org.apache.james.mailbox.model.MailboxACL; +import org.apache.james.mailbox.model.MailboxId; +import org.apache.james.mailbox.model.MailboxPath; +import org.apache.james.mailbox.model.UidValidity; +import org.apache.james.mailbox.model.search.MailboxQuery; +import org.apache.james.mailbox.postgres.mail.dao.PostgresMailboxDAO; +import org.apache.james.mailbox.store.mail.MailboxMapper; + +import reactor.core.publisher.Flux; +import reactor.core.publisher.Mono; +import reactor.util.retry.Retry; + +public class PostgresMailboxMapper implements MailboxMapper { + private final PostgresMailboxDAO postgresMailboxDAO; + + public PostgresMailboxMapper(PostgresMailboxDAO postgresMailboxDAO) { + this.postgresMailboxDAO = postgresMailboxDAO; + } + + @Override + public Mono create(MailboxPath mailboxPath, UidValidity uidValidity) { + return postgresMailboxDAO.create(mailboxPath,uidValidity); + } + + @Override + public Mono rename(Mailbox mailbox) { + return postgresMailboxDAO.rename(mailbox); + } + + @Override + public Mono delete(Mailbox mailbox) { + return postgresMailboxDAO.delete(mailbox.getMailboxId()); + } + + @Override + public Mono findMailboxByPath(MailboxPath mailboxName) { + return postgresMailboxDAO.findMailboxByPath(mailboxName) + .map(Function.identity()); + } + + @Override + public Mono findMailboxById(MailboxId mailboxId) { + return postgresMailboxDAO.findMailboxById(mailboxId) + .map(Function.identity()); + } + + @Override + public Flux findMailboxWithPathLike(MailboxQuery.UserBound query) { + return postgresMailboxDAO.findMailboxWithPathLike(query) + .map(Function.identity()); + } + + @Override + public Mono hasChildren(Mailbox mailbox, char delimiter) { + return postgresMailboxDAO.hasChildren(mailbox, delimiter); + } + + @Override + public Flux list() { + return postgresMailboxDAO.getAll() + .map(Function.identity()); + } + + public Flux findNonPersonalMailboxes(Username userName, MailboxACL.Right right) { + return postgresMailboxDAO.findMailboxesByUsername(userName) + .filter(postgresMailbox -> postgresMailbox.getACL().getEntries().get(MailboxACL.EntryKey.createUserEntryKey(userName)).contains(right)) + .map(Function.identity()); + } + + @Override + public Mono updateACL(Mailbox mailbox, MailboxACL.ACLCommand mailboxACLCommand) { + return postgresMailboxDAO.getACL(mailbox.getMailboxId()) + .flatMap(pairMailboxACLAndVersion -> { + try { + MailboxACL newACL = pairMailboxACLAndVersion.getLeft().apply(mailboxACLCommand); + return postgresMailboxDAO.upsertACL(mailbox.getMailboxId(), newACL, pairMailboxACLAndVersion.getRight()) + .thenReturn(ACLDiff.computeDiff(pairMailboxACLAndVersion.getLeft(), newACL)); + } catch (UnsupportedRightException e) { + throw new RuntimeException(e); + } + }).retryWhen(Retry.backoff(3, Duration.ofMillis(100)) + .filter(throwable -> throwable instanceof PostgresACLUpsertException)); + } + + @Override + public Mono setACL(Mailbox mailbox, MailboxACL mailboxACL) { + return postgresMailboxDAO.upsertACL(mailbox.getMailboxId(), mailboxACL) + .thenReturn(ACLDiff.computeDiff(mailbox.getACL(), mailboxACL)); + } +} diff --git a/mailbox/postgres/src/main/java/org/apache/james/mailbox/postgres/mail/PostgresMailboxMemberDAO.java b/mailbox/postgres/src/main/java/org/apache/james/mailbox/postgres/mail/PostgresMailboxMemberDAO.java new file mode 100644 index 00000000000..5cf73eb29f6 --- /dev/null +++ b/mailbox/postgres/src/main/java/org/apache/james/mailbox/postgres/mail/PostgresMailboxMemberDAO.java @@ -0,0 +1,65 @@ +/**************************************************************** + * 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.mailbox.postgres.mail; + +import static org.apache.james.mailbox.postgres.mail.PostgresMailboxMemberModule.PostgresMailboxMemberTable.MAILBOX_ID; +import static org.apache.james.mailbox.postgres.mail.PostgresMailboxMemberModule.PostgresMailboxMemberTable.TABLE_NAME; +import static org.apache.james.mailbox.postgres.mail.PostgresMailboxMemberModule.PostgresMailboxMemberTable.USER_NAME; + +import java.util.List; + +import org.apache.james.backends.postgres.utils.PostgresExecutor; +import org.apache.james.core.Username; +import org.apache.james.mailbox.postgres.PostgresMailboxId; + +import reactor.core.publisher.Flux; +import reactor.core.publisher.Mono; + +public class PostgresMailboxMemberDAO { + private final PostgresExecutor postgresExecutor; + + public PostgresMailboxMemberDAO(PostgresExecutor postgresExecutor) { + this.postgresExecutor = postgresExecutor; + } + + public Flux findMailboxIdByUsername(Username username) { + return postgresExecutor.executeRows(dslContext -> Flux.from(dslContext.select(MAILBOX_ID) + .from(TABLE_NAME) + .where(USER_NAME.eq(username.asString())))) + .map(record -> PostgresMailboxId.of(record.get(MAILBOX_ID))); + } + + public Mono insert(PostgresMailboxId mailboxId, List usernames) { + return postgresExecutor.executeVoid(dslContext -> Mono.from(dslContext.insertInto(TABLE_NAME, USER_NAME, MAILBOX_ID) + .valuesOfRecords(usernames.stream() + .map(username -> dslContext.newRecord(USER_NAME, MAILBOX_ID) + .value1(username.asString()) + .value2(mailboxId.asUuid())) + .toList()))); + } + + public Mono delete(PostgresMailboxId mailboxId, List usernames) { + return postgresExecutor.executeVoid(dslContext -> Mono.from(dslContext.batch(usernames.stream() + .map(username -> dslContext.deleteFrom(TABLE_NAME) + .where(USER_NAME.eq(username.asString()) + .and(MAILBOX_ID.eq(mailboxId.asUuid())))) + .toList()))); + } +} diff --git a/mailbox/postgres/src/main/java/org/apache/james/mailbox/postgres/mail/PostgresMailboxMemberModule.java b/mailbox/postgres/src/main/java/org/apache/james/mailbox/postgres/mail/PostgresMailboxMemberModule.java new file mode 100644 index 00000000000..abcd3bfde3e --- /dev/null +++ b/mailbox/postgres/src/main/java/org/apache/james/mailbox/postgres/mail/PostgresMailboxMemberModule.java @@ -0,0 +1,57 @@ +/**************************************************************** + * 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.mailbox.postgres.mail; + +import java.util.UUID; + +import org.apache.james.backends.postgres.PostgresIndex; +import org.apache.james.backends.postgres.PostgresModule; +import org.apache.james.backends.postgres.PostgresTable; +import org.jooq.Field; +import org.jooq.Record; +import org.jooq.Table; +import org.jooq.impl.DSL; +import org.jooq.impl.SQLDataType; + +public interface PostgresMailboxMemberModule { + interface PostgresMailboxMemberTable { + Table TABLE_NAME = DSL.table("mailbox_member"); + + Field USER_NAME = DSL.field("user_name", SQLDataType.VARCHAR(255)); + Field MAILBOX_ID = DSL.field("mailbox_id", SQLDataType.UUID.notNull()); + + PostgresTable TABLE = PostgresTable.name(TABLE_NAME.getName()) + .createTableStep(((dsl, tableName) -> dsl.createTableIfNotExists(tableName) + .column(USER_NAME) + .column(MAILBOX_ID) + .constraint(DSL.primaryKey(USER_NAME, MAILBOX_ID)))) + .supportsRowLevelSecurity() + .build(); + + PostgresIndex MAILBOX_MEMBER_USERNAME_INDEX = PostgresIndex.name("mailbox_member_username_index") + .createIndexStep((dsl, indexName) -> dsl.createIndexIfNotExists(indexName) + .on(TABLE_NAME, USER_NAME)); + } + + PostgresModule MODULE = PostgresModule.builder() + .addTable(PostgresMailboxMemberTable.TABLE) + .addIndex(PostgresMailboxMemberTable.MAILBOX_MEMBER_USERNAME_INDEX) + .build(); +} diff --git a/mailbox/postgres/src/main/java/org/apache/james/mailbox/postgres/mail/PostgresMailboxModule.java b/mailbox/postgres/src/main/java/org/apache/james/mailbox/postgres/mail/PostgresMailboxModule.java new file mode 100644 index 00000000000..0c302964a93 --- /dev/null +++ b/mailbox/postgres/src/main/java/org/apache/james/mailbox/postgres/mail/PostgresMailboxModule.java @@ -0,0 +1,79 @@ +/**************************************************************** + * 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.mailbox.postgres.mail; + +import static org.jooq.impl.SQLDataType.BIGINT; + +import java.util.UUID; + +import org.apache.james.backends.postgres.PostgresIndex; +import org.apache.james.backends.postgres.PostgresModule; +import org.apache.james.backends.postgres.PostgresTable; +import org.jooq.Field; +import org.jooq.Name; +import org.jooq.Record; +import org.jooq.Table; +import org.jooq.impl.DSL; +import org.jooq.impl.SQLDataType; +import org.jooq.postgres.extensions.bindings.HstoreBinding; +import org.jooq.postgres.extensions.types.Hstore; + +public interface PostgresMailboxModule { + interface PostgresMailboxTable { + Table TABLE_NAME = DSL.table("mailbox"); + + Field MAILBOX_ID = DSL.field("mailbox_id", SQLDataType.UUID.notNull()); + Field MAILBOX_NAME = DSL.field("mailbox_name", SQLDataType.VARCHAR(255).notNull()); + Field MAILBOX_UID_VALIDITY = DSL.field("mailbox_uid_validity", BIGINT.notNull()); + Field USER_NAME = DSL.field("user_name", SQLDataType.VARCHAR(255)); + Field MAILBOX_NAMESPACE = DSL.field("mailbox_namespace", SQLDataType.VARCHAR(255).notNull()); + Field MAILBOX_LAST_UID = DSL.field("mailbox_last_uid", BIGINT); + Field MAILBOX_HIGHEST_MODSEQ = DSL.field("mailbox_highest_modseq", BIGINT); + Field MAILBOX_ACL = DSL.field("mailbox_acl", org.jooq.impl.DefaultDataType.getDefaultDataType("hstore").asConvertedDataType(new HstoreBinding())); + Field MAILBOX_ACL_VERSION = DSL.field("mailbox_acl_version", BIGINT.notNull().defaultValue(DSL.field("0", BIGINT))); + + Name MAILBOX_NAME_USER_NAME_NAMESPACE_UNIQUE_CONSTRAINT = DSL.name("mailbox_mailbox_name_user_name_mailbox_namespace_key"); + + PostgresTable TABLE = PostgresTable.name(TABLE_NAME.getName()) + .createTableStep(((dsl, tableName) -> dsl.createTableIfNotExists(tableName) + .column(MAILBOX_ID, SQLDataType.UUID) + .column(MAILBOX_NAME) + .column(MAILBOX_UID_VALIDITY) + .column(USER_NAME) + .column(MAILBOX_NAMESPACE) + .column(MAILBOX_LAST_UID) + .column(MAILBOX_HIGHEST_MODSEQ) + .column(MAILBOX_ACL) + .column(MAILBOX_ACL_VERSION) + .constraint(DSL.primaryKey(MAILBOX_ID)) + .constraint(DSL.constraint(MAILBOX_NAME_USER_NAME_NAMESPACE_UNIQUE_CONSTRAINT).unique(MAILBOX_NAME, USER_NAME, MAILBOX_NAMESPACE)))) + .supportsRowLevelSecurity() + .addAdditionalAlterQueries(new PostgresTable.NonRLSOnlyAdditionalAlterQuery("CREATE INDEX mailbox_mailbox_acl_index ON " + TABLE_NAME.getName() + " USING GIN (" + MAILBOX_ACL.getName() + ")")) + .build(); + PostgresIndex MAILBOX_USERNAME_NAMESPACE_INDEX = PostgresIndex.name("mailbox_username_namespace_index") + .createIndexStep((dsl, indexName) -> dsl.createIndexIfNotExists(indexName) + .on(TABLE_NAME, USER_NAME, MAILBOX_NAMESPACE)); + } + + PostgresModule MODULE = PostgresModule.builder() + .addTable(PostgresMailboxTable.TABLE) + .addIndex(PostgresMailboxTable.MAILBOX_USERNAME_NAMESPACE_INDEX) + .build(); +} \ No newline at end of file diff --git a/mailbox/postgres/src/main/java/org/apache/james/mailbox/postgres/mail/PostgresMessageBlobReferenceSource.java b/mailbox/postgres/src/main/java/org/apache/james/mailbox/postgres/mail/PostgresMessageBlobReferenceSource.java new file mode 100644 index 00000000000..3ea9032b298 --- /dev/null +++ b/mailbox/postgres/src/main/java/org/apache/james/mailbox/postgres/mail/PostgresMessageBlobReferenceSource.java @@ -0,0 +1,42 @@ +/**************************************************************** + * 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.mailbox.postgres.mail; + +import jakarta.inject.Inject; + +import org.apache.james.blob.api.BlobId; +import org.apache.james.blob.api.BlobReferenceSource; +import org.apache.james.mailbox.postgres.mail.dao.PostgresMessageDAO; + +import reactor.core.publisher.Flux; + +public class PostgresMessageBlobReferenceSource implements BlobReferenceSource { + private PostgresMessageDAO postgresMessageDAO; + + @Inject + public PostgresMessageBlobReferenceSource(PostgresMessageDAO postgresMessageDAO) { + this.postgresMessageDAO = postgresMessageDAO; + } + + @Override + public Flux listReferencedBlobs() { + return postgresMessageDAO.listBlobs(); + } +} diff --git a/mailbox/postgres/src/main/java/org/apache/james/mailbox/postgres/mail/PostgresMessageIdMapper.java b/mailbox/postgres/src/main/java/org/apache/james/mailbox/postgres/mail/PostgresMessageIdMapper.java new file mode 100644 index 00000000000..961b51fb53b --- /dev/null +++ b/mailbox/postgres/src/main/java/org/apache/james/mailbox/postgres/mail/PostgresMessageIdMapper.java @@ -0,0 +1,257 @@ +/**************************************************************** + * 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.mailbox.postgres.mail; + +import static org.apache.james.blob.api.BlobStore.StoragePolicy.LOW_COST; + +import java.io.IOException; +import java.io.InputStream; +import java.time.Clock; +import java.util.Collection; +import java.util.Date; +import java.util.List; +import java.util.function.Function; + +import jakarta.mail.Flags; + +import org.apache.commons.lang3.tuple.Pair; +import org.apache.james.backends.postgres.utils.PostgresUtils; +import org.apache.james.blob.api.BlobId; +import org.apache.james.blob.api.BlobStore; +import org.apache.james.mailbox.MessageManager; +import org.apache.james.mailbox.MessageUid; +import org.apache.james.mailbox.ModSeq; +import org.apache.james.mailbox.exception.MailboxException; +import org.apache.james.mailbox.exception.MailboxNotFoundException; +import org.apache.james.mailbox.model.ComposedMessageIdWithMetaData; +import org.apache.james.mailbox.model.Mailbox; +import org.apache.james.mailbox.model.MailboxId; +import org.apache.james.mailbox.model.MessageId; +import org.apache.james.mailbox.model.UpdatedFlags; +import org.apache.james.mailbox.postgres.PostgresMailboxId; +import org.apache.james.mailbox.postgres.PostgresMessageId; +import org.apache.james.mailbox.postgres.mail.dao.PostgresMailboxDAO; +import org.apache.james.mailbox.postgres.mail.dao.PostgresMailboxMessageDAO; +import org.apache.james.mailbox.postgres.mail.dao.PostgresMessageDAO; +import org.apache.james.mailbox.store.FlagsUpdateCalculator; +import org.apache.james.mailbox.store.MailboxReactorUtils; +import org.apache.james.mailbox.store.mail.MessageIdMapper; +import org.apache.james.mailbox.store.mail.MessageMapper; +import org.apache.james.mailbox.store.mail.model.MailboxMessage; +import org.apache.james.mailbox.store.mail.model.impl.SimpleMailboxMessage; +import org.apache.james.util.ReactorUtils; +import org.jooq.Record; +import org.reactivestreams.Publisher; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + +import com.google.common.collect.ImmutableList; +import com.google.common.collect.ImmutableListMultimap; +import com.google.common.collect.Multimap; +import com.google.common.io.ByteSource; + +import reactor.core.publisher.Flux; +import reactor.core.publisher.Mono; + +public class PostgresMessageIdMapper implements MessageIdMapper { + private static final Function MESSAGE_BODY_CONTENT_LOADER = (mailboxMessage) -> new ByteSource() { + @Override + public InputStream openStream() { + try { + return mailboxMessage.getBodyContent(); + } catch (IOException e) { + throw new RuntimeException(e); + } + } + + @Override + public long size() { + return mailboxMessage.getBodyOctets(); + } + }; + + public static final int NUM_RETRIES = 5; + public static final Logger LOGGER = LoggerFactory.getLogger(PostgresMessageIdMapper.class); + + private final PostgresMailboxDAO mailboxDAO; + private final PostgresMessageDAO messageDAO; + private final PostgresMailboxMessageDAO mailboxMessageDAO; + private final PostgresModSeqProvider modSeqProvider; + private final BlobStore blobStore; + private final Clock clock; + private final PostgresMessageRetriever messageRetriever; + + public PostgresMessageIdMapper(PostgresMailboxDAO mailboxDAO, + PostgresMessageDAO messageDAO, + PostgresMailboxMessageDAO mailboxMessageDAO, + PostgresModSeqProvider modSeqProvider, + PostgresAttachmentMapper attachmentMapper, + BlobStore blobStore, + BlobId.Factory blobIdFactory, + Clock clock) { + this.mailboxDAO = mailboxDAO; + this.messageDAO = messageDAO; + this.mailboxMessageDAO = mailboxMessageDAO; + this.modSeqProvider = modSeqProvider; + this.blobStore = blobStore; + this.clock = clock; + this.messageRetriever = new PostgresMessageRetriever(blobStore, blobIdFactory, attachmentMapper); + } + + @Override + public List find(Collection messageIds, MessageMapper.FetchType fetchType) { + return findReactive(messageIds, fetchType) + .collectList() + .block(); + } + + @Override + public Publisher findMetadata(MessageId messageId) { + return mailboxMessageDAO.findMetadataByMessageId(PostgresMessageId.class.cast(messageId)); + } + + @Override + public Flux findReactive(Collection messageIds, MessageMapper.FetchType fetchType) { + Flux> fetchMessagePublisher = mailboxMessageDAO.findMessagesByMessageIds(messageIds.stream().map(PostgresMessageId.class::cast).collect(ImmutableList.toImmutableList()), fetchType); + return messageRetriever.get(fetchType, fetchMessagePublisher); + } + + @Override + public List findMailboxes(MessageId messageId) { + return mailboxMessageDAO.findMailboxes(PostgresMessageId.class.cast(messageId)) + .collect(ImmutableList.toImmutableList()) + .block(); + } + + @Override + public void save(MailboxMessage mailboxMessage) throws MailboxException { + PostgresMailboxId mailboxId = PostgresMailboxId.class.cast(mailboxMessage.getMailboxId()); + mailboxMessage.setSaveDate(Date.from(clock.instant())); + MailboxReactorUtils.block(mailboxDAO.findMailboxById(mailboxId) + .switchIfEmpty(Mono.error(() -> new MailboxNotFoundException(mailboxId))) + .then(saveBodyContent(mailboxMessage)) + .flatMap(blobId -> messageDAO.insert(mailboxMessage, blobId.asString()) + .onErrorResume(PostgresUtils.UNIQUE_CONSTRAINT_VIOLATION_PREDICATE, e -> Mono.empty())) + .then(mailboxMessageDAO.insert(mailboxMessage))); + } + + @Override + public void copyInMailbox(MailboxMessage mailboxMessage, Mailbox mailbox) throws MailboxException { + MailboxReactorUtils.block(copyInMailboxReactive(mailboxMessage, mailbox)); + } + + @Override + public Mono copyInMailboxReactive(MailboxMessage mailboxMessage, Mailbox mailbox) { + mailboxMessage.setSaveDate(Date.from(clock.instant())); + PostgresMailboxId mailboxId = (PostgresMailboxId) mailbox.getMailboxId(); + return mailboxMessageDAO.insert(mailboxMessage, mailboxId) + .onErrorResume(PostgresUtils.UNIQUE_CONSTRAINT_VIOLATION_PREDICATE, e -> Mono.empty()); + } + + @Override + public void delete(MessageId messageId) { + mailboxMessageDAO.deleteByMessageId((PostgresMessageId) messageId).block(); + } + + @Override + public void delete(MessageId messageId, Collection mailboxIds) { + mailboxMessageDAO.deleteByMessageIdAndMailboxIds((PostgresMessageId) messageId, + mailboxIds.stream().map(PostgresMailboxId.class::cast).collect(ImmutableList.toImmutableList())).block(); + } + + @Override + public Mono> setFlags(MessageId messageId, List mailboxIds, Flags newState, MessageManager.FlagsUpdateMode updateMode) { + return Flux.fromIterable(mailboxIds) + .distinct() + .map(PostgresMailboxId.class::cast) + .concatMap(mailboxId -> flagsUpdateWithRetry(newState, updateMode, mailboxId, messageId)) + .collect(ImmutableListMultimap.toImmutableListMultimap(Pair::getLeft, Pair::getRight)); + } + + private Flux> flagsUpdateWithRetry(Flags newState, MessageManager.FlagsUpdateMode updateMode, MailboxId mailboxId, MessageId messageId) { + return updateFlags(mailboxId, messageId, newState, updateMode) + .retry(NUM_RETRIES) + .onErrorResume(MailboxDeleteDuringUpdateException.class, e -> { + LOGGER.info("Mailbox {} was deleted during flag update", mailboxId); + return Mono.empty(); + }) + .flatMapIterable(Function.identity()) + .map(pair -> buildUpdatedFlags(pair.getRight(), pair.getLeft())); + } + + private Pair buildUpdatedFlags(ComposedMessageIdWithMetaData composedMessageIdWithMetaData, Flags oldFlags) { + return Pair.of(composedMessageIdWithMetaData.getComposedMessageId().getMailboxId(), + UpdatedFlags.builder() + .uid(composedMessageIdWithMetaData.getComposedMessageId().getUid()) + .messageId(composedMessageIdWithMetaData.getComposedMessageId().getMessageId()) + .modSeq(composedMessageIdWithMetaData.getModSeq()) + .oldFlags(oldFlags) + .newFlags(composedMessageIdWithMetaData.getFlags()) + .build()); + } + + private Mono>> updateFlags(MailboxId mailboxId, MessageId messageId, Flags newState, MessageManager.FlagsUpdateMode updateMode) { + PostgresMailboxId postgresMailboxId = (PostgresMailboxId) mailboxId; + PostgresMessageId postgresMessageId = (PostgresMessageId) messageId; + return mailboxMessageDAO.findMetadataByMessageId(postgresMessageId, postgresMailboxId) + .flatMap(oldComposedId -> updateFlags(newState, updateMode, postgresMailboxId, oldComposedId), ReactorUtils.DEFAULT_CONCURRENCY) + .switchIfEmpty(Mono.error(MailboxDeleteDuringUpdateException::new)) + .collectList(); + } + + private Mono> updateFlags(Flags newState, MessageManager.FlagsUpdateMode updateMode, PostgresMailboxId mailboxId, ComposedMessageIdWithMetaData oldComposedId) { + FlagsUpdateCalculator flagsUpdateCalculator = new FlagsUpdateCalculator(newState, updateMode); + Flags newFlags = flagsUpdateCalculator.buildNewFlags(oldComposedId.getFlags()); + if (identicalFlags(oldComposedId, newFlags)) { + return Mono.just(Pair.of(oldComposedId.getFlags(), oldComposedId)); + } else { + return modSeqProvider.nextModSeqReactive(mailboxId) + .flatMap(newModSeq -> updateFlags(mailboxId, flagsUpdateCalculator, newModSeq, oldComposedId.getComposedMessageId().getUid()) + .map(flags -> Pair.of(oldComposedId.getFlags(), new ComposedMessageIdWithMetaData( + oldComposedId.getComposedMessageId(), + flags, + newModSeq, + oldComposedId.getThreadId())))); + } + } + + private Mono updateFlags(PostgresMailboxId mailboxId, FlagsUpdateCalculator flagsUpdateCalculator, ModSeq newModSeq, MessageUid uid) { + + switch (flagsUpdateCalculator.getMode()) { + case ADD: + return mailboxMessageDAO.addFlags(mailboxId, uid, flagsUpdateCalculator.providedFlags(), newModSeq); + case REMOVE: + return mailboxMessageDAO.removeFlags(mailboxId, uid, flagsUpdateCalculator.providedFlags(), newModSeq); + case REPLACE: + return mailboxMessageDAO.replaceFlags(mailboxId, uid, flagsUpdateCalculator.providedFlags(), newModSeq); + default: + return Mono.error(() -> new RuntimeException("Unknown MessageRange type " + flagsUpdateCalculator.getMode())); + } + } + + private boolean identicalFlags(ComposedMessageIdWithMetaData oldComposedId, Flags newFlags) { + return oldComposedId.getFlags().equals(newFlags); + } + + private Mono saveBodyContent(MailboxMessage message) { + return Mono.fromCallable(() -> MESSAGE_BODY_CONTENT_LOADER.apply(message)) + .flatMap(bodyByteSource -> Mono.from(blobStore.save(blobStore.getDefaultBucketName(), bodyByteSource, LOW_COST))); + } +} diff --git a/mailbox/postgres/src/main/java/org/apache/james/mailbox/postgres/mail/PostgresMessageMapper.java b/mailbox/postgres/src/main/java/org/apache/james/mailbox/postgres/mail/PostgresMessageMapper.java new file mode 100644 index 00000000000..5112324b10f --- /dev/null +++ b/mailbox/postgres/src/main/java/org/apache/james/mailbox/postgres/mail/PostgresMessageMapper.java @@ -0,0 +1,446 @@ +/**************************************************************** + * 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.mailbox.postgres.mail; + +import static org.apache.james.blob.api.BlobStore.StoragePolicy.LOW_COST; +import static org.apache.james.util.ReactorUtils.DEFAULT_CONCURRENCY; + +import java.io.IOException; +import java.io.InputStream; +import java.time.Clock; +import java.util.Date; +import java.util.Iterator; +import java.util.List; +import java.util.Map; +import java.util.Optional; +import java.util.function.Function; + +import jakarta.mail.Flags; + +import org.apache.commons.lang3.tuple.Pair; +import org.apache.james.backends.postgres.utils.PostgresExecutor; +import org.apache.james.backends.postgres.utils.PostgresUtils; +import org.apache.james.blob.api.BlobId; +import org.apache.james.blob.api.BlobStore; +import org.apache.james.mailbox.ApplicableFlagBuilder; +import org.apache.james.mailbox.FlagsBuilder; +import org.apache.james.mailbox.MessageUid; +import org.apache.james.mailbox.ModSeq; +import org.apache.james.mailbox.exception.MailboxException; +import org.apache.james.mailbox.model.ComposedMessageId; +import org.apache.james.mailbox.model.ComposedMessageIdWithMetaData; +import org.apache.james.mailbox.model.Mailbox; +import org.apache.james.mailbox.model.MailboxCounters; +import org.apache.james.mailbox.model.MessageMetaData; +import org.apache.james.mailbox.model.MessageRange; +import org.apache.james.mailbox.model.UpdatedFlags; +import org.apache.james.mailbox.postgres.PostgresMailboxId; +import org.apache.james.mailbox.postgres.mail.dao.PostgresAttachmentDAO; +import org.apache.james.mailbox.postgres.mail.dao.PostgresMailboxDAO; +import org.apache.james.mailbox.postgres.mail.dao.PostgresMailboxMessageDAO; +import org.apache.james.mailbox.postgres.mail.dao.PostgresMessageDAO; +import org.apache.james.mailbox.store.FlagsUpdateCalculator; +import org.apache.james.mailbox.store.MailboxReactorUtils; +import org.apache.james.mailbox.store.mail.MessageMapper; +import org.apache.james.mailbox.store.mail.model.MailboxMessage; +import org.apache.james.mailbox.store.mail.model.impl.SimpleMailboxMessage; +import org.apache.james.util.streams.Limit; +import org.jooq.Record; + +import com.google.common.io.ByteSource; + +import reactor.core.publisher.Flux; +import reactor.core.publisher.Mono; + +public class PostgresMessageMapper implements MessageMapper { + + private static final Function MESSAGE_BODY_CONTENT_LOADER = (mailboxMessage) -> new ByteSource() { + @Override + public InputStream openStream() { + try { + return mailboxMessage.getBodyContent(); + } catch (IOException e) { + throw new RuntimeException(e); + } + } + + @Override + public long size() { + return mailboxMessage.getBodyOctets(); + } + }; + + + private final PostgresMessageDAO messageDAO; + private final PostgresMailboxMessageDAO mailboxMessageDAO; + private final PostgresMailboxDAO mailboxDAO; + private final PostgresModSeqProvider modSeqProvider; + private final PostgresUidProvider uidProvider; + private final BlobStore blobStore; + private final Clock clock; + private final PostgresMessageRetriever messageRetriever; + + public PostgresMessageMapper(PostgresExecutor postgresExecutor, + PostgresModSeqProvider modSeqProvider, + PostgresUidProvider uidProvider, + BlobStore blobStore, + Clock clock, + BlobId.Factory blobIdFactory) { + this.messageDAO = new PostgresMessageDAO(postgresExecutor, blobIdFactory); + this.mailboxMessageDAO = new PostgresMailboxMessageDAO(postgresExecutor); + this.mailboxDAO = new PostgresMailboxDAO(postgresExecutor); + this.modSeqProvider = modSeqProvider; + this.uidProvider = uidProvider; + this.blobStore = blobStore; + this.clock = clock; + PostgresAttachmentMapper attachmentMapper = new PostgresAttachmentMapper(new PostgresAttachmentDAO(postgresExecutor, blobIdFactory), blobStore); + this.messageRetriever = new PostgresMessageRetriever(blobStore, blobIdFactory, attachmentMapper); + } + + + @Override + public Iterator findInMailbox(Mailbox mailbox, MessageRange set, FetchType type, int limit) { + return findInMailboxReactive(mailbox, set, type, limit) + .toIterable() + .iterator(); + } + + @Override + public Flux listMessagesMetadata(Mailbox mailbox, MessageRange set) { + return mailboxMessageDAO.findMessagesMetadata((PostgresMailboxId) mailbox.getMailboxId(), set); + } + + @Override + public Flux findInMailboxReactive(Mailbox mailbox, MessageRange messageRange, FetchType fetchType, int limitAsInt) { + Flux> fetchMessagePublisher = fetchMessageWithoutFullContent(mailbox, messageRange, fetchType, limitAsInt); + return messageRetriever.get(fetchType, fetchMessagePublisher); + } + + private Flux> fetchMessageWithoutFullContent(Mailbox mailbox, MessageRange messageRange, FetchType fetchType, int limitAsInt) { + return Mono.just(messageRange) + .flatMapMany(range -> { + Limit limit = Limit.from(limitAsInt); + switch (messageRange.getType()) { + case ALL: + return mailboxMessageDAO.findMessagesByMailboxId((PostgresMailboxId) mailbox.getMailboxId(), limit, fetchType); + case FROM: + return mailboxMessageDAO.findMessagesByMailboxIdAndAfterUID((PostgresMailboxId) mailbox.getMailboxId(), range.getUidFrom(), limit, fetchType); + case ONE: + return mailboxMessageDAO.findMessageByMailboxIdAndUid((PostgresMailboxId) mailbox.getMailboxId(), range.getUidFrom(), fetchType) + .flatMapMany(Flux::just); + case RANGE: + return mailboxMessageDAO.findMessagesByMailboxIdAndBetweenUIDs((PostgresMailboxId) mailbox.getMailboxId(), range.getUidFrom(), range.getUidTo(), limit, fetchType); + default: + throw new RuntimeException("Unknown MessageRange range " + range.getType()); + } + }); + } + + @Override + public List retrieveMessagesMarkedForDeletion(Mailbox mailbox, MessageRange messageRange) { + return retrieveMessagesMarkedForDeletionReactive(mailbox, messageRange) + .collectList() + .block(); + } + + @Override + public Flux retrieveMessagesMarkedForDeletionReactive(Mailbox mailbox, MessageRange messageRange) { + return Mono.just(messageRange) + .flatMapMany(range -> { + switch (messageRange.getType()) { + case ALL: + return mailboxMessageDAO.findDeletedMessagesByMailboxId((PostgresMailboxId) mailbox.getMailboxId()); + case FROM: + return mailboxMessageDAO.findDeletedMessagesByMailboxIdAndAfterUID((PostgresMailboxId) mailbox.getMailboxId(), range.getUidFrom()); + case ONE: + return mailboxMessageDAO.findDeletedMessageByMailboxIdAndUid((PostgresMailboxId) mailbox.getMailboxId(), range.getUidFrom()) + .flatMapMany(Flux::just); + case RANGE: + return mailboxMessageDAO.findDeletedMessagesByMailboxIdAndBetweenUIDs((PostgresMailboxId) mailbox.getMailboxId(), range.getUidFrom(), range.getUidTo()); + default: + throw new RuntimeException("Unknown MessageRange type " + range.getType()); + } + }); + } + + @Override + public long countMessagesInMailbox(Mailbox mailbox) { + return mailboxMessageDAO.countTotalMessagesByMailboxId((PostgresMailboxId) mailbox.getMailboxId()) + .block(); + } + + @Override + public MailboxCounters getMailboxCounters(Mailbox mailbox) { + return getMailboxCountersReactive(mailbox).block(); + } + + @Override + public Mono getMailboxCountersReactive(Mailbox mailbox) { + return mailboxMessageDAO.countTotalAndUnseenMessagesByMailboxId((PostgresMailboxId) mailbox.getMailboxId()) + .map(pair -> MailboxCounters.builder() + .mailboxId(mailbox.getMailboxId()) + .count(pair.getLeft()) + .unseen(pair.getRight()) + .build()); + } + + @Override + public void delete(Mailbox mailbox, MailboxMessage message) throws MailboxException { + deleteMessages(mailbox, List.of(message.getUid())); + } + + @Override + public Map deleteMessages(Mailbox mailbox, List uids) { + return deleteMessagesReactive(mailbox, uids).block(); + } + + @Override + public Mono> deleteMessagesReactive(Mailbox mailbox, List uids) { + return mailboxMessageDAO.findMessagesByMailboxIdAndUIDs((PostgresMailboxId) mailbox.getMailboxId(), uids) + .map(SimpleMailboxMessage.Builder::build) + .collectMap(MailboxMessage::getUid, MailboxMessage::metaData) + .flatMap(map -> mailboxMessageDAO.deleteByMailboxIdAndMessageUids((PostgresMailboxId) mailbox.getMailboxId(), uids) + .then(Mono.just(map))); + } + + @Override + public MessageUid findFirstUnseenMessageUid(Mailbox mailbox) { + return mailboxMessageDAO.findFirstUnseenMessageUid((PostgresMailboxId) mailbox.getMailboxId()).block(); + } + + @Override + public Mono> findFirstUnseenMessageUidReactive(Mailbox mailbox) { + return mailboxMessageDAO.findFirstUnseenMessageUid((PostgresMailboxId) mailbox.getMailboxId()) + .map(Optional::of) + .switchIfEmpty(Mono.just(Optional.empty())); + } + + @Override + public List findRecentMessageUidsInMailbox(Mailbox mailbox) { + return findRecentMessageUidsInMailboxReactive(mailbox).block(); + } + + @Override + public Mono> findRecentMessageUidsInMailboxReactive(Mailbox mailbox) { + return mailboxMessageDAO.findAllRecentMessageUid((PostgresMailboxId) mailbox.getMailboxId()) + .collectList(); + } + + @Override + public MessageMetaData add(Mailbox mailbox, MailboxMessage message) throws MailboxException { + return addReactive(mailbox, message).block(); + } + + @Override + public Mono addReactive(Mailbox mailbox, MailboxMessage message) { + return Mono.fromCallable(() -> { + message.setSaveDate(Date.from(clock.instant())); + return message; + }) + .flatMap(this::setNewUidAndModSeq) + .then(saveBodyContent(message) + .flatMap(bodyBlobId -> messageDAO.insert(message, bodyBlobId.asString()) + .onErrorResume(PostgresUtils.UNIQUE_CONSTRAINT_VIOLATION_PREDICATE, e -> Mono.empty()))) + .then(Mono.defer(() -> mailboxMessageDAO.insert(message))) + .then(Mono.fromCallable(message::metaData)); + } + + private Mono saveBodyContent(MailboxMessage message) { + return Mono.fromCallable(() -> MESSAGE_BODY_CONTENT_LOADER.apply(message)) + .flatMap(bodyByteSource -> Mono.from(blobStore.save(blobStore.getDefaultBucketName(), bodyByteSource, LOW_COST))); + } + + @Override + public Iterator updateFlags(Mailbox mailbox, FlagsUpdateCalculator flagsUpdateCalculator, MessageRange range) { + return updateFlagsPublisher(mailbox, flagsUpdateCalculator, range) + .toIterable() + .iterator(); + } + + @Override + public Mono> updateFlagsReactive(Mailbox mailbox, FlagsUpdateCalculator flagsUpdateCalculator, MessageRange range) { + return updateFlagsPublisher(mailbox, flagsUpdateCalculator, range) + .collectList(); + } + + private Flux updateFlagsPublisher(Mailbox mailbox, FlagsUpdateCalculator flagsUpdateCalculator, MessageRange range) { + return mailboxMessageDAO.findMessagesMetadata((PostgresMailboxId) mailbox.getMailboxId(), range) + .collectList() + .flatMapMany(listMessagesMetadata -> updatedFlags(listMessagesMetadata, mailbox, flagsUpdateCalculator)); + } + + private Flux updatedFlags(List listMessagesMetaData, + Mailbox mailbox, + FlagsUpdateCalculator flagsUpdateCalculator) { + return modSeqProvider.nextModSeqReactive(mailbox.getMailboxId()) + .flatMapMany(newModSeq -> Flux.fromIterable(listMessagesMetaData) + .flatMapSequential(messageMetaData -> updateFlags(messageMetaData, flagsUpdateCalculator, newModSeq), DEFAULT_CONCURRENCY)); + } + + private Mono updateFlags(ComposedMessageIdWithMetaData currentMetaData, + FlagsUpdateCalculator flagsUpdateCalculator, + ModSeq newModSeq) { + Flags oldFlags = currentMetaData.getFlags(); + ComposedMessageId composedMessageId = currentMetaData.getComposedMessageId(); + + if (oldFlags.equals(flagsUpdateCalculator.buildNewFlags(oldFlags))) { + return Mono.just(UpdatedFlags.builder() + .messageId(composedMessageId.getMessageId()) + .oldFlags(oldFlags) + .newFlags(oldFlags) + .uid(composedMessageId.getUid()) + .modSeq(currentMetaData.getModSeq()) + .build()); + } else { + return Mono.just(flagsUpdateCalculator.getMode()) + .flatMap(mode -> { + switch (mode) { + case ADD: + return mailboxMessageDAO.addFlags((PostgresMailboxId) composedMessageId.getMailboxId(), composedMessageId.getUid(), flagsUpdateCalculator.providedFlags(), newModSeq); + case REMOVE: + return mailboxMessageDAO.removeFlags((PostgresMailboxId) composedMessageId.getMailboxId(), composedMessageId.getUid(), flagsUpdateCalculator.providedFlags(), newModSeq); + case REPLACE: + return mailboxMessageDAO.replaceFlags((PostgresMailboxId) composedMessageId.getMailboxId(), composedMessageId.getUid(), flagsUpdateCalculator.providedFlags(), newModSeq); + default: + return Mono.error(() -> new RuntimeException("Unknown MessageRange type " + mode)); + } + }).map(updatedFlags -> UpdatedFlags.builder() + .messageId(composedMessageId.getMessageId()) + .oldFlags(oldFlags) + .newFlags(updatedFlags) + .uid(composedMessageId.getUid()) + .modSeq(newModSeq) + .build()); + } + } + + @Override + public List resetRecent(Mailbox mailbox) { + return resetRecentReactive(mailbox).block(); + } + + @Override + public Mono> resetRecentReactive(Mailbox mailbox) { + return mailboxMessageDAO.findAllRecentMessageMetadata((PostgresMailboxId) mailbox.getMailboxId()) + .collectList() + .flatMapMany(mailboxMessageList -> resetRecentFlag((PostgresMailboxId) mailbox.getMailboxId(), mailboxMessageList)) + .collectList(); + } + + private Flux resetRecentFlag(PostgresMailboxId mailboxId, List messageIdWithMetaDataList) { + return Flux.fromIterable(messageIdWithMetaDataList) + .collectMap(m -> m.getComposedMessageId().getUid(), Function.identity()) + .flatMapMany(uidMapping -> modSeqProvider.nextModSeqReactive(mailboxId) + .flatMapMany(newModSeq -> mailboxMessageDAO.resetRecentFlag(mailboxId, List.copyOf(uidMapping.keySet()), newModSeq)) + .map(newMetaData -> UpdatedFlags.builder() + .messageId(newMetaData.getMessageId()) + .modSeq(newMetaData.getModSeq()) + .oldFlags(uidMapping.get(newMetaData.getUid()).getFlags()) + .newFlags(newMetaData.getFlags()) + .uid(newMetaData.getUid()) + .build())); + } + + @Override + public MessageMetaData copy(Mailbox mailbox, MailboxMessage original) throws MailboxException { + return copyReactive(mailbox, original).block(); + } + + private Mono setNewUidAndModSeq(MailboxMessage mailboxMessage) { + return mailboxDAO.incrementAndGetLastUidAndModSeq(mailboxMessage.getMailboxId()) + .defaultIfEmpty(Pair.of(MessageUid.MIN_VALUE, ModSeq.first())) + .map(pair -> { + mailboxMessage.setUid(pair.getLeft()); + mailboxMessage.setModSeq(pair.getRight()); + return pair; + }).then(); + } + + + @Override + public Mono copyReactive(Mailbox mailbox, MailboxMessage original) { + return Mono.fromCallable(() -> { + MailboxMessage copiedMessage = original.copy(mailbox); + copiedMessage.setFlags(new FlagsBuilder().add(original.createFlags()).add(Flags.Flag.RECENT).build()); + copiedMessage.setSaveDate(Date.from(clock.instant())); + return copiedMessage; + }) + .flatMap(copiedMessage -> setNewUidAndModSeq(copiedMessage) + .then(Mono.defer(() -> mailboxMessageDAO.insert(copiedMessage)) + .thenReturn(copiedMessage)) + .map(MailboxMessage::metaData)); + } + + + @Override + public MessageMetaData move(Mailbox mailbox, MailboxMessage original) { + return moveReactive(mailbox, original).block(); + } + + @Override + public List move(Mailbox mailbox, List original) throws MailboxException { + return MailboxReactorUtils.block(moveReactive(mailbox, original)); + } + + + @Override + public Mono moveReactive(Mailbox mailbox, MailboxMessage original) { + return copyReactive(mailbox, original) + .flatMap(copiedResult -> mailboxMessageDAO.deleteByMailboxIdAndMessageUid((PostgresMailboxId) original.getMailboxId(), original.getUid()) + .thenReturn(copiedResult)); + } + + @Override + public Optional getLastUid(Mailbox mailbox) { + return uidProvider.lastUid(mailbox); + } + + @Override + public Mono> getLastUidReactive(Mailbox mailbox) { + return uidProvider.lastUidReactive(mailbox); + } + + @Override + public ModSeq getHighestModSeq(Mailbox mailbox) { + return modSeqProvider.highestModSeq(mailbox); + } + + @Override + public Mono getHighestModSeqReactive(Mailbox mailbox) { + return modSeqProvider.highestModSeqReactive(mailbox); + } + + @Override + public Flags getApplicableFlag(Mailbox mailbox) { + return getApplicableFlagReactive(mailbox).block(); + } + + @Override + public Mono getApplicableFlagReactive(Mailbox mailbox) { + return mailboxMessageDAO.listDistinctUserFlags((PostgresMailboxId) mailbox.getMailboxId()) + .map(flags -> ApplicableFlagBuilder.builder().add(flags).build()); + } + + @Override + public Flux listAllMessageUids(Mailbox mailbox) { + return mailboxMessageDAO.listAllMessageUid((PostgresMailboxId) mailbox.getMailboxId()); + } + +} diff --git a/mailbox/postgres/src/main/java/org/apache/james/mailbox/postgres/mail/PostgresMessageModule.java b/mailbox/postgres/src/main/java/org/apache/james/mailbox/postgres/mail/PostgresMessageModule.java new file mode 100644 index 00000000000..87499f2ad84 --- /dev/null +++ b/mailbox/postgres/src/main/java/org/apache/james/mailbox/postgres/mail/PostgresMessageModule.java @@ -0,0 +1,189 @@ +/**************************************************************** + * 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.mailbox.postgres.mail; + +import static org.jooq.impl.DSL.foreignKey; + +import java.time.LocalDateTime; +import java.util.UUID; + +import org.apache.james.backends.postgres.PostgresCommons.DataTypes; +import org.apache.james.backends.postgres.PostgresIndex; +import org.apache.james.backends.postgres.PostgresModule; +import org.apache.james.backends.postgres.PostgresTable; +import org.apache.james.mailbox.postgres.mail.dto.AttachmentsDTO; +import org.jooq.Field; +import org.jooq.Record; +import org.jooq.Table; +import org.jooq.impl.DSL; +import org.jooq.impl.SQLDataType; +import org.jooq.postgres.extensions.types.Hstore; + +public interface PostgresMessageModule { + + Field MESSAGE_ID = DSL.field("message_id", SQLDataType.UUID.notNull()); + Field INTERNAL_DATE = DSL.field("internal_date", DataTypes.TIMESTAMP); + Field SIZE = DSL.field("size", SQLDataType.BIGINT.notNull()); + + interface MessageTable { + Table TABLE_NAME = DSL.table("message"); + Field MESSAGE_ID = PostgresMessageModule.MESSAGE_ID; + Field BODY_BLOB_ID = DSL.field("body_blob_id", SQLDataType.VARCHAR(200).notNull()); + Field MIME_TYPE = DSL.field("mime_type", SQLDataType.VARCHAR(200)); + Field MIME_SUBTYPE = DSL.field("mime_subtype", SQLDataType.VARCHAR(200)); + Field INTERNAL_DATE = PostgresMessageModule.INTERNAL_DATE; + Field SIZE = PostgresMessageModule.SIZE; + Field BODY_START_OCTET = DSL.field("body_start_octet", SQLDataType.INTEGER.notNull()); + Field HEADER_CONTENT = DSL.field("header_content", SQLDataType.BLOB.notNull()); + Field TEXTUAL_LINE_COUNT = DSL.field("textual_line_count", SQLDataType.INTEGER); + + Field CONTENT_DESCRIPTION = DSL.field("content_description", SQLDataType.VARCHAR(200)); + Field CONTENT_LOCATION = DSL.field("content_location", SQLDataType.VARCHAR(200)); + Field CONTENT_TRANSFER_ENCODING = DSL.field("content_transfer_encoding", SQLDataType.VARCHAR(200)); + Field CONTENT_DISPOSITION_TYPE = DSL.field("content_disposition_type", SQLDataType.VARCHAR(200)); + Field CONTENT_ID = DSL.field("content_id", SQLDataType.VARCHAR(200)); + Field CONTENT_MD5 = DSL.field("content_md5", SQLDataType.VARCHAR(200)); + Field CONTENT_LANGUAGE = DSL.field("content_language", DataTypes.STRING_ARRAY); + Field CONTENT_TYPE_PARAMETERS = DSL.field("content_type_parameters", DataTypes.HSTORE); + Field CONTENT_DISPOSITION_PARAMETERS = DSL.field("content_disposition_parameters", DataTypes.HSTORE); + Field ATTACHMENT_METADATA = DSL.field("attachment_metadata", + SQLDataType.JSONB + .asConvertedDataType(new AttachmentsDTO.AttachmentsDTOBinding())); + + + PostgresTable TABLE = PostgresTable.name(TABLE_NAME.getName()) + .createTableStep(((dsl, tableName) -> dsl.createTableIfNotExists(tableName) + .column(MESSAGE_ID) + .column(BODY_BLOB_ID) + .column(MIME_TYPE) + .column(MIME_SUBTYPE) + .column(INTERNAL_DATE) + .column(SIZE) + .column(BODY_START_OCTET) + .column(HEADER_CONTENT) + .column(TEXTUAL_LINE_COUNT) + .column(CONTENT_DESCRIPTION) + .column(CONTENT_LOCATION) + .column(CONTENT_TRANSFER_ENCODING) + .column(CONTENT_DISPOSITION_TYPE) + .column(CONTENT_ID) + .column(CONTENT_MD5) + .column(CONTENT_LANGUAGE) + .column(CONTENT_TYPE_PARAMETERS) + .column(CONTENT_DISPOSITION_PARAMETERS) + .column(ATTACHMENT_METADATA) + .constraint(DSL.primaryKey(MESSAGE_ID)) + .comment("Holds the metadata of a mail"))) + .supportsRowLevelSecurity() + .build(); + } + + interface MessageToMailboxTable { + Table TABLE_NAME = DSL.table("message_mailbox"); + Field MAILBOX_ID = DSL.field("mailbox_id", SQLDataType.UUID.notNull()); + Field MESSAGE_UID = DSL.field("message_uid", SQLDataType.BIGINT.notNull()); + Field MOD_SEQ = DSL.field("mod_seq", SQLDataType.BIGINT.notNull()); + Field MESSAGE_ID = PostgresMessageModule.MESSAGE_ID; + Field THREAD_ID = DSL.field("thread_id", SQLDataType.UUID); + Field INTERNAL_DATE = PostgresMessageModule.INTERNAL_DATE; + Field SIZE = PostgresMessageModule.SIZE; + Field IS_DELETED = DSL.field("is_deleted", SQLDataType.BOOLEAN.nullable(false) + .defaultValue(DSL.field("false", SQLDataType.BOOLEAN))); + Field IS_ANSWERED = DSL.field("is_answered", SQLDataType.BOOLEAN.nullable(false)); + Field IS_DRAFT = DSL.field("is_draft", SQLDataType.BOOLEAN.nullable(false)); + Field IS_FLAGGED = DSL.field("is_flagged", SQLDataType.BOOLEAN.nullable(false)); + Field IS_RECENT = DSL.field("is_recent", SQLDataType.BOOLEAN.nullable(false)); + Field IS_SEEN = DSL.field("is_seen", SQLDataType.BOOLEAN.nullable(false)); + Field USER_FLAGS = DSL.field("user_flags", DataTypes.STRING_ARRAY); + Field SAVE_DATE = DSL.field("save_date", DataTypes.TIMESTAMP); + + String REMOVE_ELEMENTS_FROM_ARRAY_FUNCTION_NAME = "remove_elements_from_array"; + String CREATE_ARRAY_REMOVE_JAMES_FUNCTION = + "CREATE OR REPLACE FUNCTION " + REMOVE_ELEMENTS_FROM_ARRAY_FUNCTION_NAME + "(\n" + + " source text[],\n" + + " elements_to_remove text[])\n" + + " RETURNS text[]\n" + + "AS\n" + + "$$\n" + + "DECLARE\n" + + " result text[];\n" + + "BEGIN\n" + + " select array_agg(elements) INTO result\n" + + " from (select unnest(source)\n" + + " except\n" + + " select unnest(elements_to_remove)) t (elements);\n" + + " RETURN result;\n" + + "END;\n" + + "$$ LANGUAGE plpgsql;"; + + PostgresTable TABLE = PostgresTable.name(TABLE_NAME.getName()) + .createTableStep(((dsl, tableName) -> dsl.createTableIfNotExists(tableName) + .column(MAILBOX_ID) + .column(MESSAGE_UID) + .column(MOD_SEQ) + .column(MESSAGE_ID) + .column(THREAD_ID) + .column(INTERNAL_DATE) + .column(SIZE) + .column(IS_DELETED) + .column(IS_ANSWERED) + .column(IS_DRAFT) + .column(IS_FLAGGED) + .column(IS_RECENT) + .column(IS_SEEN) + .column(USER_FLAGS) + .column(SAVE_DATE) + .constraints(DSL.primaryKey(MAILBOX_ID, MESSAGE_UID), + foreignKey(MESSAGE_ID).references(MessageTable.TABLE_NAME, MessageTable.MESSAGE_ID)) + .comment("Holds mailbox and flags for each message"))) + .supportsRowLevelSecurity() + .addAdditionalAlterQueries(CREATE_ARRAY_REMOVE_JAMES_FUNCTION) + .build(); + + PostgresIndex MESSAGE_ID_INDEX = PostgresIndex.name("message_mailbox_message_id_index") + .createIndexStep((dsl, indexName) -> dsl.createIndexIfNotExists(indexName) + .on(TABLE_NAME, MESSAGE_ID)); + + PostgresIndex MAILBOX_ID_MESSAGE_UID_INDEX = PostgresIndex.name("mailbox_id_mail_uid_index") + .createIndexStep((dsl, indexName) -> dsl.createIndexIfNotExists(indexName) + .on(TABLE_NAME, MAILBOX_ID, MESSAGE_UID.asc())); + PostgresIndex MAILBOX_ID_IS_SEEN_MESSAGE_UID_INDEX = PostgresIndex.name("mailbox_id_is_seen_mail_uid_index") + .createIndexStep((dsl, indexName) -> dsl.createIndexIfNotExists(indexName) + .on(TABLE_NAME, MAILBOX_ID, IS_SEEN, MESSAGE_UID.asc())); + PostgresIndex MAILBOX_ID_IS_RECENT_MESSAGE_UID_INDEX = PostgresIndex.name("mailbox_id_is_recent_mail_uid_index") + .createIndexStep((dsl, indexName) -> dsl.createIndexIfNotExists(indexName) + .on(TABLE_NAME, MAILBOX_ID, IS_RECENT, MESSAGE_UID.asc())); + PostgresIndex MAILBOX_ID_IS_DELETE_MESSAGE_UID_INDEX = PostgresIndex.name("mailbox_id_is_delete_mail_uid_index") + .createIndexStep((dsl, indexName) -> dsl.createIndexIfNotExists(indexName) + .on(TABLE_NAME, MAILBOX_ID, IS_DELETED, MESSAGE_UID.asc())); + + } + + PostgresModule MODULE = PostgresModule.builder() + .addTable(MessageTable.TABLE) + .addTable(MessageToMailboxTable.TABLE) + .addIndex(MessageToMailboxTable.MESSAGE_ID_INDEX) + .addIndex(MessageToMailboxTable.MAILBOX_ID_MESSAGE_UID_INDEX) + .addIndex(MessageToMailboxTable.MAILBOX_ID_IS_SEEN_MESSAGE_UID_INDEX) + .addIndex(MessageToMailboxTable.MAILBOX_ID_IS_RECENT_MESSAGE_UID_INDEX) + .addIndex(MessageToMailboxTable.MAILBOX_ID_IS_DELETE_MESSAGE_UID_INDEX) + .build(); + +} diff --git a/mailbox/postgres/src/main/java/org/apache/james/mailbox/postgres/mail/PostgresMessageRetriever.java b/mailbox/postgres/src/main/java/org/apache/james/mailbox/postgres/mail/PostgresMessageRetriever.java new file mode 100644 index 00000000000..14121cbfea2 --- /dev/null +++ b/mailbox/postgres/src/main/java/org/apache/james/mailbox/postgres/mail/PostgresMessageRetriever.java @@ -0,0 +1,142 @@ +/**************************************************************** + * 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.mailbox.postgres.mail; + +import static org.apache.james.blob.api.BlobStore.StoragePolicy.SIZE_BASED; +import static org.apache.james.mailbox.postgres.mail.PostgresMessageModule.MessageTable.ATTACHMENT_METADATA; +import static org.apache.james.mailbox.postgres.mail.PostgresMessageModule.MessageTable.BODY_BLOB_ID; +import static org.apache.james.mailbox.postgres.mail.PostgresMessageModule.MessageTable.HEADER_CONTENT; + +import java.util.List; +import java.util.Map; + +import org.apache.commons.lang3.tuple.Pair; +import org.apache.james.blob.api.BlobId; +import org.apache.james.blob.api.BlobStore; +import org.apache.james.mailbox.model.AttachmentId; +import org.apache.james.mailbox.model.AttachmentMetadata; +import org.apache.james.mailbox.model.Content; +import org.apache.james.mailbox.model.HeaderAndBodyByteContent; +import org.apache.james.mailbox.model.MessageAttachmentMetadata; +import org.apache.james.mailbox.postgres.mail.dto.AttachmentsDTO; +import org.apache.james.mailbox.store.mail.MessageMapper; +import org.apache.james.mailbox.store.mail.model.MailboxMessage; +import org.apache.james.mailbox.store.mail.model.impl.SimpleMailboxMessage; +import org.apache.james.util.ReactorUtils; +import org.jooq.Record; + +import com.google.common.collect.ImmutableMap; + +import reactor.core.publisher.Flux; +import reactor.core.publisher.Mono; + +public class PostgresMessageRetriever { + + interface PartRetriever { + + boolean isApplicable(MessageMapper.FetchType fetchType); + + Flux> doRetrieve(Flux> chain); + } + + class AttachmentPartRetriever implements PartRetriever { + + @Override + public boolean isApplicable(MessageMapper.FetchType fetchType) { + return fetchType == MessageMapper.FetchType.FULL || fetchType == MessageMapper.FetchType.ATTACHMENTS_METADATA; + } + + @Override + public Flux> doRetrieve(Flux> chain) { + return chain.collectList() // convert to list to avoid hanging the database connection with Jooq + .flatMapMany(list -> Flux.fromIterable(list) + .flatMapSequential(pair -> Mono.fromCallable(() -> toMap(pair.getRight().get(ATTACHMENT_METADATA))) + .flatMap(this::getAttachments) + .map(messageAttachmentMetadata -> { + pair.getLeft().addAttachments(messageAttachmentMetadata); + return pair; + }).switchIfEmpty(Mono.just(pair)))); + } + + private Map toMap(AttachmentsDTO attachmentRepresentations) { + return attachmentRepresentations.stream().collect(ImmutableMap.toImmutableMap(MessageRepresentation.AttachmentRepresentation::getAttachmentId, obj -> obj)); + } + + private Mono> getAttachments(Map mapAttachmentIdToAttachmentRepresentation) { + return Mono.fromCallable(mapAttachmentIdToAttachmentRepresentation::keySet) + .flatMapMany(attachmentMapper::getAttachmentsReactive) + .map(attachmentMetadata -> constructMessageAttachment(attachmentMetadata, mapAttachmentIdToAttachmentRepresentation.get(attachmentMetadata.getAttachmentId()))) + .collectList(); + } + + private MessageAttachmentMetadata constructMessageAttachment(AttachmentMetadata attachment, MessageRepresentation.AttachmentRepresentation messageAttachmentRepresentation) { + return MessageAttachmentMetadata.builder() + .attachment(attachment) + .name(messageAttachmentRepresentation.getName().orElse(null)) + .cid(messageAttachmentRepresentation.getCid()) + .isInline(messageAttachmentRepresentation.isInline()) + .build(); + } + } + + class BlobContentPartRetriever implements PartRetriever { + + @Override + public boolean isApplicable(MessageMapper.FetchType fetchType) { + return fetchType == MessageMapper.FetchType.FULL; + } + + @Override + public Flux> doRetrieve(Flux> chain) { + return chain + .flatMapSequential(pair -> retrieveFullContent(pair.getRight()) + .map(headerAndBodyContent -> Pair.of(pair.getLeft().content(headerAndBodyContent), pair.getRight())), + ReactorUtils.DEFAULT_CONCURRENCY); + } + + private Mono retrieveFullContent(Record messageRecord) { + return Mono.from(blobStore.readBytes(blobStore.getDefaultBucketName(), + blobIdFactory.parse(messageRecord.get(BODY_BLOB_ID)), + SIZE_BASED)) + .map(bodyBytes -> new HeaderAndBodyByteContent(messageRecord.get(HEADER_CONTENT), bodyBytes)); + } + } + + private final BlobStore blobStore; + private final BlobId.Factory blobIdFactory; + private final PostgresAttachmentMapper attachmentMapper; + private final List partRetrievers = List.of(new AttachmentPartRetriever(), new BlobContentPartRetriever()); + + public PostgresMessageRetriever(BlobStore blobStore, + BlobId.Factory blobIdFactory, + PostgresAttachmentMapper attachmentMapper) { + this.blobStore = blobStore; + this.blobIdFactory = blobIdFactory; + this.attachmentMapper = attachmentMapper; + } + + public Flux get(MessageMapper.FetchType fetchType, Flux> initialFlux) { + return Flux.fromIterable(partRetrievers) + .filter(partRetriever -> partRetriever.isApplicable(fetchType)) + .reduce(initialFlux, (flux, partRetriever) -> partRetriever.doRetrieve(flux)) + .flatMapMany(flux -> flux) + .map(pair -> pair.getLeft().build()); + } +} diff --git a/mailbox/postgres/src/main/java/org/apache/james/mailbox/postgres/mail/PostgresModSeqProvider.java b/mailbox/postgres/src/main/java/org/apache/james/mailbox/postgres/mail/PostgresModSeqProvider.java new file mode 100644 index 00000000000..23734e8138e --- /dev/null +++ b/mailbox/postgres/src/main/java/org/apache/james/mailbox/postgres/mail/PostgresModSeqProvider.java @@ -0,0 +1,92 @@ +/**************************************************************** + * 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.mailbox.postgres.mail; + +import org.apache.james.backends.postgres.utils.PostgresExecutor; +import org.apache.james.mailbox.MailboxSession; +import org.apache.james.mailbox.ModSeq; +import org.apache.james.mailbox.exception.MailboxException; +import org.apache.james.mailbox.model.Mailbox; +import org.apache.james.mailbox.model.MailboxId; +import org.apache.james.mailbox.postgres.mail.dao.PostgresMailboxDAO; +import org.apache.james.mailbox.store.mail.ModSeqProvider; + +import reactor.core.publisher.Mono; + +public class PostgresModSeqProvider implements ModSeqProvider { + + public static class Factory { + + private final PostgresExecutor.Factory executorFactory; + + public Factory(PostgresExecutor.Factory executorFactory) { + this.executorFactory = executorFactory; + } + + public PostgresModSeqProvider create(MailboxSession session) { + PostgresExecutor postgresExecutor = executorFactory.create(session.getUser().getDomainPart()); + return new PostgresModSeqProvider(new PostgresMailboxDAO(postgresExecutor)); + } + } + + private final PostgresMailboxDAO mailboxDAO; + + public PostgresModSeqProvider(PostgresMailboxDAO mailboxDAO) { + this.mailboxDAO = mailboxDAO; + } + + @Override + public ModSeq nextModSeq(Mailbox mailbox) throws MailboxException { + return nextModSeq(mailbox.getMailboxId()); + } + + @Override + public ModSeq nextModSeq(MailboxId mailboxId) throws MailboxException { + return nextModSeqReactive(mailboxId) + .blockOptional() + .orElseThrow(() -> new MailboxException("Can not retrieve modseq for " + mailboxId)); + } + + @Override + public ModSeq highestModSeq(Mailbox mailbox) { + return highestModSeqReactive(mailbox).block(); + } + + @Override + public Mono highestModSeqReactive(Mailbox mailbox) { + return getHighestModSeq(mailbox.getMailboxId()); + } + + private Mono getHighestModSeq(MailboxId mailboxId) { + return mailboxDAO.findHighestModSeqByMailboxId(mailboxId) + .defaultIfEmpty(ModSeq.first()); + } + + @Override + public ModSeq highestModSeq(MailboxId mailboxId) { + return getHighestModSeq(mailboxId).block(); + } + + @Override + public Mono nextModSeqReactive(MailboxId mailboxId) { + return mailboxDAO.incrementAndGetModSeq(mailboxId) + .defaultIfEmpty(ModSeq.first()); + } +} diff --git a/mailbox/postgres/src/main/java/org/apache/james/mailbox/postgres/mail/PostgresUidProvider.java b/mailbox/postgres/src/main/java/org/apache/james/mailbox/postgres/mail/PostgresUidProvider.java new file mode 100644 index 00000000000..8333fcbf036 --- /dev/null +++ b/mailbox/postgres/src/main/java/org/apache/james/mailbox/postgres/mail/PostgresUidProvider.java @@ -0,0 +1,106 @@ +/**************************************************************** + * 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.mailbox.postgres.mail; + +import java.util.List; +import java.util.Optional; +import java.util.stream.LongStream; + +import org.apache.james.backends.postgres.utils.PostgresExecutor; +import org.apache.james.mailbox.MailboxSession; +import org.apache.james.mailbox.MessageUid; +import org.apache.james.mailbox.exception.MailboxException; +import org.apache.james.mailbox.model.Mailbox; +import org.apache.james.mailbox.model.MailboxId; +import org.apache.james.mailbox.postgres.mail.dao.PostgresMailboxDAO; +import org.apache.james.mailbox.store.mail.UidProvider; + +import com.google.common.base.Preconditions; +import com.google.common.collect.ImmutableList; + +import reactor.core.publisher.Mono; + +public class PostgresUidProvider implements UidProvider { + + public static class Factory { + + private final PostgresExecutor.Factory executorFactory; + + public Factory(PostgresExecutor.Factory executorFactory) { + this.executorFactory = executorFactory; + } + + public PostgresUidProvider create(MailboxSession session) { + PostgresExecutor postgresExecutor = executorFactory.create(session.getUser().getDomainPart()); + return new PostgresUidProvider(new PostgresMailboxDAO(postgresExecutor)); + } + } + + private final PostgresMailboxDAO mailboxDAO; + + public PostgresUidProvider(PostgresMailboxDAO mailboxDAO) { + this.mailboxDAO = mailboxDAO; + } + + @Override + public MessageUid nextUid(Mailbox mailbox) throws MailboxException { + return nextUid(mailbox.getMailboxId()); + } + + @Override + public Optional lastUid(Mailbox mailbox) { + return lastUidReactive(mailbox).block(); + } + + @Override + public MessageUid nextUid(MailboxId mailboxId) throws MailboxException { + return nextUidReactive(mailboxId) + .blockOptional() + .orElseThrow(() -> new MailboxException("Error during Uid update")); + } + + @Override + public Mono> lastUidReactive(Mailbox mailbox) { + return mailboxDAO.findLastUidByMailboxId(mailbox.getMailboxId()) + .map(Optional::of) + .switchIfEmpty(Mono.just(Optional.empty())); + } + + @Override + public Mono nextUidReactive(MailboxId mailboxId) { + return mailboxDAO.incrementAndGetLastUid(mailboxId, 1) + .defaultIfEmpty(MessageUid.MIN_VALUE); + } + + @Override + public Mono> nextUids(MailboxId mailboxId, int count) { + Preconditions.checkArgument(count > 0, "Count need to be positive"); + Mono updateNewLastUid = mailboxDAO.incrementAndGetLastUid(mailboxId, count) + .defaultIfEmpty(MessageUid.MIN_VALUE); + return updateNewLastUid.map(lastUid -> range(lastUid, count)); + } + + private List range(MessageUid higherInclusive, int count) { + return LongStream.range(higherInclusive.asLong() - count + 1, higherInclusive.asLong() + 1) + .mapToObj(MessageUid::of) + .collect(ImmutableList.toImmutableList()); + } + +} diff --git a/mailbox/postgres/src/main/java/org/apache/james/mailbox/postgres/mail/RLSSupportPostgresMailboxMapper.java b/mailbox/postgres/src/main/java/org/apache/james/mailbox/postgres/mail/RLSSupportPostgresMailboxMapper.java new file mode 100644 index 00000000000..aa3db2b311a --- /dev/null +++ b/mailbox/postgres/src/main/java/org/apache/james/mailbox/postgres/mail/RLSSupportPostgresMailboxMapper.java @@ -0,0 +1,85 @@ +/**************************************************************** + * 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.mailbox.postgres.mail; + +import java.util.function.Function; + +import org.apache.james.core.Username; +import org.apache.james.mailbox.acl.ACLDiff; +import org.apache.james.mailbox.acl.PositiveUserACLDiff; +import org.apache.james.mailbox.model.Mailbox; +import org.apache.james.mailbox.model.MailboxACL; +import org.apache.james.mailbox.postgres.PostgresMailboxId; +import org.apache.james.mailbox.postgres.mail.dao.PostgresMailboxDAO; + +import com.github.fge.lambdas.Throwing; + +import reactor.core.publisher.Flux; +import reactor.core.publisher.Mono; + +public class RLSSupportPostgresMailboxMapper extends PostgresMailboxMapper { + private final PostgresMailboxDAO postgresMailboxDAO; + private final PostgresMailboxMemberDAO postgresMailboxMemberDAO; + + public RLSSupportPostgresMailboxMapper(PostgresMailboxDAO postgresMailboxDAO, PostgresMailboxMemberDAO postgresMailboxMemberDAO) { + super(postgresMailboxDAO); + this.postgresMailboxDAO = postgresMailboxDAO; + this.postgresMailboxMemberDAO = postgresMailboxMemberDAO; + } + + @Override + public Flux findNonPersonalMailboxes(Username userName, MailboxACL.Right right) { + return postgresMailboxMemberDAO.findMailboxIdByUsername(userName) + .collectList() + .filter(postgresMailboxIds -> !postgresMailboxIds.isEmpty()) + .flatMapMany(postgresMailboxDAO::findMailboxByIds) + .filter(postgresMailbox -> postgresMailbox.getACL().getEntries().get(MailboxACL.EntryKey.createUserEntryKey(userName)).contains(right)) + .map(Function.identity()); + } + + @Override + public Mono updateACL(Mailbox mailbox, MailboxACL.ACLCommand mailboxACLCommand) { + MailboxACL oldACL = mailbox.getACL(); + MailboxACL newACL = Throwing.supplier(() -> oldACL.apply(mailboxACLCommand)).get(); + ACLDiff aclDiff = ACLDiff.computeDiff(oldACL, newACL); + PositiveUserACLDiff userACLDiff = new PositiveUserACLDiff(aclDiff); + return upsertACL(mailbox, newACL, aclDiff, userACLDiff); + } + + @Override + public Mono setACL(Mailbox mailbox, MailboxACL mailboxACL) { + MailboxACL oldACL = mailbox.getACL(); + ACLDiff aclDiff = ACLDiff.computeDiff(oldACL, mailboxACL); + PositiveUserACLDiff userACLDiff = new PositiveUserACLDiff(aclDiff); + return upsertACL(mailbox, mailboxACL, aclDiff, userACLDiff); + } + + private Mono upsertACL(Mailbox mailbox, MailboxACL newACL, ACLDiff aclDiff, PositiveUserACLDiff userACLDiff) { + return postgresMailboxDAO.upsertACL(mailbox.getMailboxId(), newACL) + .then(postgresMailboxMemberDAO.delete(PostgresMailboxId.class.cast(mailbox.getMailboxId()), + userACLDiff.removedEntries().map(entry -> Username.of(entry.getKey().getName())).toList())) + .then(postgresMailboxMemberDAO.insert(PostgresMailboxId.class.cast(mailbox.getMailboxId()), + userACLDiff.addedEntries().map(entry -> Username.of(entry.getKey().getName())).toList())) + .then(Mono.fromCallable(() -> { + mailbox.setACL(newACL); + return aclDiff; + })); + } +} diff --git a/mailbox/postgres/src/main/java/org/apache/james/mailbox/postgres/mail/dao/PostgresAttachmentDAO.java b/mailbox/postgres/src/main/java/org/apache/james/mailbox/postgres/mail/dao/PostgresAttachmentDAO.java new file mode 100644 index 00000000000..4fd55d4eb7f --- /dev/null +++ b/mailbox/postgres/src/main/java/org/apache/james/mailbox/postgres/mail/dao/PostgresAttachmentDAO.java @@ -0,0 +1,127 @@ +/**************************************************************** + * 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.mailbox.postgres.mail.dao; + +import java.util.Collection; +import java.util.Optional; + +import jakarta.inject.Inject; +import jakarta.inject.Singleton; + +import org.apache.commons.lang3.tuple.Pair; +import org.apache.james.backends.postgres.utils.PostgresExecutor; +import org.apache.james.blob.api.BlobId; +import org.apache.james.core.Domain; +import org.apache.james.mailbox.model.AttachmentId; +import org.apache.james.mailbox.model.AttachmentMetadata; +import org.apache.james.mailbox.model.UuidBackedAttachmentId; +import org.apache.james.mailbox.postgres.PostgresMessageId; +import org.apache.james.mailbox.postgres.mail.PostgresAttachmentModule.PostgresAttachmentTable; + +import com.google.common.collect.ImmutableList; + +import reactor.core.publisher.Flux; +import reactor.core.publisher.Mono; + +public class PostgresAttachmentDAO { + + public static class Factory { + private final PostgresExecutor.Factory executorFactory; + private final BlobId.Factory blobIdFactory; + + @Inject + @Singleton + public Factory(PostgresExecutor.Factory executorFactory, BlobId.Factory blobIdFactory) { + this.executorFactory = executorFactory; + this.blobIdFactory = blobIdFactory; + } + + public PostgresAttachmentDAO create(Optional domain) { + return new PostgresAttachmentDAO(executorFactory.create(domain), blobIdFactory); + } + } + + private final PostgresExecutor postgresExecutor; + private final BlobId.Factory blobIdFactory; + + public PostgresAttachmentDAO(PostgresExecutor postgresExecutor, BlobId.Factory blobIdFactory) { + this.postgresExecutor = postgresExecutor; + this.blobIdFactory = blobIdFactory; + } + + public Mono> getAttachment(AttachmentId attachmentId) { + return postgresExecutor.executeRow(dslContext -> Mono.from(dslContext.select( + PostgresAttachmentTable.TYPE, + PostgresAttachmentTable.BLOB_ID, + PostgresAttachmentTable.MESSAGE_ID, + PostgresAttachmentTable.SIZE) + .from(PostgresAttachmentTable.TABLE_NAME) + .where(PostgresAttachmentTable.ID.eq(attachmentId.asUUID())))) + .map(row -> Pair.of( + AttachmentMetadata.builder() + .attachmentId(attachmentId) + .type(row.get(PostgresAttachmentTable.TYPE)) + .messageId(PostgresMessageId.Factory.of(row.get(PostgresAttachmentTable.MESSAGE_ID))) + .size(row.get(PostgresAttachmentTable.SIZE)) + .build(), + blobIdFactory.parse(row.get(PostgresAttachmentTable.BLOB_ID)))); + } + + public Flux getAttachments(Collection attachmentIds) { + if (attachmentIds.isEmpty()) { + return Flux.empty(); + } + return postgresExecutor.executeRows(dslContext -> Flux.from(dslContext.selectFrom(PostgresAttachmentTable.TABLE_NAME) + .where(PostgresAttachmentTable.ID.in(attachmentIds.stream().map(AttachmentId::getId).collect(ImmutableList.toImmutableList()))))) + .map(row -> AttachmentMetadata.builder() + .attachmentId(UuidBackedAttachmentId.from(row.get(PostgresAttachmentTable.ID))) + .type(row.get(PostgresAttachmentTable.TYPE)) + .messageId(PostgresMessageId.Factory.of(row.get(PostgresAttachmentTable.MESSAGE_ID))) + .size(row.get(PostgresAttachmentTable.SIZE)) + .build()); + } + + public Mono storeAttachment(AttachmentMetadata attachment, BlobId blobId) { + return postgresExecutor.executeVoid(dslContext -> Mono.from(dslContext.insertInto(PostgresAttachmentTable.TABLE_NAME) + .set(PostgresAttachmentTable.ID, attachment.getAttachmentId().asUUID()) + .set(PostgresAttachmentTable.BLOB_ID, blobId.asString()) + .set(PostgresAttachmentTable.TYPE, attachment.getType().asString()) + .set(PostgresAttachmentTable.MESSAGE_ID, ((PostgresMessageId) attachment.getMessageId()).asUuid()) + .set(PostgresAttachmentTable.SIZE, attachment.getSize()))); + } + + public Mono deleteByMessageId(PostgresMessageId messageId) { + return postgresExecutor.executeVoid(dslContext -> Mono.from(dslContext.deleteFrom(PostgresAttachmentTable.TABLE_NAME) + .where(PostgresAttachmentTable.MESSAGE_ID.eq(messageId.asUuid())))); + } + + public Flux listBlobsByMessageId(PostgresMessageId messageId) { + return postgresExecutor.executeRows(dslContext -> Flux.from(dslContext.select(PostgresAttachmentTable.BLOB_ID) + .from(PostgresAttachmentTable.TABLE_NAME) + .where(PostgresAttachmentTable.MESSAGE_ID.eq(messageId.asUuid())))) + .map(row -> blobIdFactory.parse(row.get(PostgresAttachmentTable.BLOB_ID))); + } + + public Flux listBlobs() { + return postgresExecutor.executeRows(dslContext -> Flux.from(dslContext.select(PostgresAttachmentTable.BLOB_ID) + .from(PostgresAttachmentTable.TABLE_NAME))) + .map(row -> blobIdFactory.parse(row.get(PostgresAttachmentTable.BLOB_ID))); + } +} \ No newline at end of file diff --git a/mailbox/postgres/src/main/java/org/apache/james/mailbox/postgres/mail/dao/PostgresMailboxAnnotationDAO.java b/mailbox/postgres/src/main/java/org/apache/james/mailbox/postgres/mail/dao/PostgresMailboxAnnotationDAO.java new file mode 100644 index 00000000000..60d29c6d1aa --- /dev/null +++ b/mailbox/postgres/src/main/java/org/apache/james/mailbox/postgres/mail/dao/PostgresMailboxAnnotationDAO.java @@ -0,0 +1,145 @@ +/**************************************************************** + * 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.mailbox.postgres.mail.dao; + +import static org.apache.james.mailbox.postgres.PostgresMailboxAnnotationModule.PostgresMailboxAnnotationTable.ANNOTATIONS; +import static org.apache.james.mailbox.postgres.PostgresMailboxAnnotationModule.PostgresMailboxAnnotationTable.MAILBOX_ID; +import static org.apache.james.mailbox.postgres.PostgresMailboxAnnotationModule.PostgresMailboxAnnotationTable.TABLE_NAME; + +import java.util.LinkedHashMap; +import java.util.List; +import java.util.Set; +import java.util.stream.Collectors; + +import org.apache.james.backends.postgres.utils.PostgresExecutor; +import org.apache.james.mailbox.model.MailboxAnnotation; +import org.apache.james.mailbox.model.MailboxAnnotationKey; +import org.apache.james.mailbox.postgres.PostgresMailboxId; +import org.jooq.impl.DSL; +import org.jooq.impl.DefaultDataType; +import org.jooq.postgres.extensions.types.Hstore; + +import com.google.common.base.Preconditions; +import com.google.common.collect.ImmutableMap; + +import reactor.core.publisher.Flux; +import reactor.core.publisher.Mono; + +public class PostgresMailboxAnnotationDAO { + private static final char SQL_WILDCARD_CHAR = '%'; + private static final String ANNOTATION_KEY_FIELD_NAME = "annotation_key"; + private static final String ANNOTATION_VALUE_FIELD_NAME = "annotation_value"; + private static final String EMPTY_ANNOTATION_VALUE = null; + + private final PostgresExecutor postgresExecutor; + + public PostgresMailboxAnnotationDAO(PostgresExecutor postgresExecutor) { + this.postgresExecutor = postgresExecutor; + } + + public Flux getAllAnnotations(PostgresMailboxId mailboxId) { + return postgresExecutor.executeRows(dslContext -> + Flux.from(dslContext.selectFrom(TABLE_NAME) + .where(MAILBOX_ID.eq(mailboxId.asUuid())))) + .singleOrEmpty() + .map(record -> record.get(ANNOTATIONS, LinkedHashMap.class)) + .flatMapIterable(this::hstoreToAnnotations); + } + + public Flux getAnnotationsByKeys(PostgresMailboxId mailboxId, Set keys) { + return postgresExecutor.executeRows(dslContext -> + Flux.from(dslContext.select(DSL.function("slice", + DefaultDataType.getDefaultDataType("hstore"), + ANNOTATIONS, + DSL.array(keys.stream().map(mailboxAnnotationKey -> DSL.val(mailboxAnnotationKey.asString())).collect(Collectors.toUnmodifiableList())))) + .from(TABLE_NAME) + .where(MAILBOX_ID.eq(mailboxId.asUuid())))) + .singleOrEmpty() + .map(record -> record.get(0, LinkedHashMap.class)) + .flatMapIterable(this::hstoreToAnnotations); + } + + public Mono exist(PostgresMailboxId mailboxId, MailboxAnnotationKey key) { + return postgresExecutor.executeRows(dslContext -> + Flux.from(dslContext.select(DSL.field(" exist(" + ANNOTATIONS.getName() + ",?)", key.asString())) + .from(TABLE_NAME) + .where(MAILBOX_ID.eq(mailboxId.asUuid())))) + .singleOrEmpty() + .map(record -> record.get(0, Boolean.class)) + .defaultIfEmpty(false); + } + + public Flux getAnnotationsByKeyLike(PostgresMailboxId mailboxId, MailboxAnnotationKey key) { + return postgresExecutor.executeRows(dslContext -> + Flux.from(dslContext.selectFrom( + dslContext.select(DSL.field("(each(annotations)).key").as(ANNOTATION_KEY_FIELD_NAME), + DSL.field("(each(annotations)).value").as(ANNOTATION_VALUE_FIELD_NAME)) + .from(TABLE_NAME) + .where(MAILBOX_ID.eq(mailboxId.asUuid())).asTable()) + .where(DSL.field(ANNOTATION_KEY_FIELD_NAME).like(key.asString() + SQL_WILDCARD_CHAR)))) + .map(record -> MailboxAnnotation.newInstance(new MailboxAnnotationKey(record.get(ANNOTATION_KEY_FIELD_NAME, String.class)), + record.get(ANNOTATION_VALUE_FIELD_NAME, String.class))); + } + + public Mono insertAnnotation(PostgresMailboxId mailboxId, MailboxAnnotation mailboxAnnotation) { + Preconditions.checkArgument(!mailboxAnnotation.isNil()); + + return postgresExecutor.executeVoid(dslContext -> + Mono.from(dslContext.insertInto(TABLE_NAME, MAILBOX_ID, ANNOTATIONS) + .values(mailboxId.asUuid(), annotationAsHstore(mailboxAnnotation)) + .onConflict(MAILBOX_ID) + .doUpdate() + .set(DSL.field(ANNOTATIONS.getName() + "[?]", + mailboxAnnotation.getKey().asString()), + mailboxAnnotation.getValue().orElse(EMPTY_ANNOTATION_VALUE)))); + } + + public Mono deleteAnnotation(PostgresMailboxId mailboxId, MailboxAnnotationKey key) { + return postgresExecutor.executeVoid(dslContext -> + Mono.from(dslContext.update(TABLE_NAME) + .set(DSL.field(ANNOTATIONS.getName()), + (Object) DSL.function("delete", + DefaultDataType.getDefaultDataType("hstore"), + ANNOTATIONS, + DSL.val(key.asString()))) + .where(MAILBOX_ID.eq(mailboxId.asUuid())))); + } + + public Mono countAnnotations(PostgresMailboxId mailboxId) { + return postgresExecutor.executeRows(dslContext -> + Flux.from(dslContext.select(DSL.field("array_length(akeys(" + ANNOTATIONS.getName() + "), 1)")) + .from(TABLE_NAME) + .where(MAILBOX_ID.eq(mailboxId.asUuid())))) + .singleOrEmpty() + .flatMap(record -> Mono.justOrEmpty(record.get(0, Integer.class))) + .defaultIfEmpty(0); + } + + private List hstoreToAnnotations(LinkedHashMap hstore) { + return hstore.entrySet() + .stream() + .map(entry -> MailboxAnnotation.newInstance(new MailboxAnnotationKey(entry.getKey()), entry.getValue())) + .collect(Collectors.toList()); + } + + private Hstore annotationAsHstore(MailboxAnnotation mailboxAnnotation) { + return Hstore.hstore(ImmutableMap.of(mailboxAnnotation.getKey().asString(), mailboxAnnotation.getValue().get())); + } +} diff --git a/mailbox/postgres/src/main/java/org/apache/james/mailbox/postgres/mail/dao/PostgresMailboxDAO.java b/mailbox/postgres/src/main/java/org/apache/james/mailbox/postgres/mail/dao/PostgresMailboxDAO.java new file mode 100644 index 00000000000..a8bd57b8ba7 --- /dev/null +++ b/mailbox/postgres/src/main/java/org/apache/james/mailbox/postgres/mail/dao/PostgresMailboxDAO.java @@ -0,0 +1,307 @@ +/**************************************************************** + * 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.mailbox.postgres.mail.dao; + +import static org.apache.james.mailbox.postgres.mail.PostgresMailboxModule.PostgresMailboxTable.MAILBOX_ACL; +import static org.apache.james.mailbox.postgres.mail.PostgresMailboxModule.PostgresMailboxTable.MAILBOX_ACL_VERSION; +import static org.apache.james.mailbox.postgres.mail.PostgresMailboxModule.PostgresMailboxTable.MAILBOX_HIGHEST_MODSEQ; +import static org.apache.james.mailbox.postgres.mail.PostgresMailboxModule.PostgresMailboxTable.MAILBOX_ID; +import static org.apache.james.mailbox.postgres.mail.PostgresMailboxModule.PostgresMailboxTable.MAILBOX_LAST_UID; +import static org.apache.james.mailbox.postgres.mail.PostgresMailboxModule.PostgresMailboxTable.MAILBOX_NAME; +import static org.apache.james.mailbox.postgres.mail.PostgresMailboxModule.PostgresMailboxTable.MAILBOX_NAMESPACE; +import static org.apache.james.mailbox.postgres.mail.PostgresMailboxModule.PostgresMailboxTable.MAILBOX_NAME_USER_NAME_NAMESPACE_UNIQUE_CONSTRAINT; +import static org.apache.james.mailbox.postgres.mail.PostgresMailboxModule.PostgresMailboxTable.MAILBOX_UID_VALIDITY; +import static org.apache.james.mailbox.postgres.mail.PostgresMailboxModule.PostgresMailboxTable.TABLE_NAME; +import static org.apache.james.mailbox.postgres.mail.PostgresMailboxModule.PostgresMailboxTable.USER_NAME; +import static org.jooq.impl.DSL.coalesce; + +import java.util.LinkedHashMap; +import java.util.List; +import java.util.Map; +import java.util.Optional; +import java.util.UUID; +import java.util.function.Function; +import java.util.stream.Collectors; + +import org.apache.commons.lang3.tuple.Pair; +import org.apache.james.backends.postgres.utils.PostgresExecutor; +import org.apache.james.core.Username; +import org.apache.james.mailbox.MessageUid; +import org.apache.james.mailbox.ModSeq; +import org.apache.james.mailbox.exception.MailboxExistsException; +import org.apache.james.mailbox.exception.MailboxNotFoundException; +import org.apache.james.mailbox.exception.UnsupportedRightException; +import org.apache.james.mailbox.model.Mailbox; +import org.apache.james.mailbox.model.MailboxACL; +import org.apache.james.mailbox.model.MailboxId; +import org.apache.james.mailbox.model.MailboxPath; +import org.apache.james.mailbox.model.UidValidity; +import org.apache.james.mailbox.model.search.MailboxQuery; +import org.apache.james.mailbox.model.search.Wildcard; +import org.apache.james.mailbox.postgres.PostgresMailboxId; +import org.apache.james.mailbox.postgres.mail.PostgresACLUpsertException; +import org.apache.james.mailbox.postgres.mail.PostgresMailbox; +import org.apache.james.mailbox.store.MailboxExpressionBackwardCompatibility; +import org.jooq.Condition; +import org.jooq.Record; +import org.jooq.impl.DSL; +import org.jooq.impl.DefaultDataType; +import org.jooq.postgres.extensions.types.Hstore; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + +import com.google.common.base.Preconditions; +import com.google.common.collect.ImmutableMap; + +import reactor.core.publisher.Flux; +import reactor.core.publisher.Mono; + +public class PostgresMailboxDAO { + private static final Logger LOGGER = LoggerFactory.getLogger(PostgresMailboxDAO.class); + private static final char SQL_WILDCARD_CHAR = '%'; + private static final Function MAILBOX_ACL_TO_HSTORE_FUNCTION = acl -> Hstore.hstore(acl.getEntries() + .entrySet() + .stream() + .collect(Collectors.toMap( + entry -> entry.getKey().serialize(), + entry -> entry.getValue().serialize()))); + + private static final Function HSTORE_TO_MAILBOX_ACL_FUNCTION = hstore -> new MailboxACL(hstore.data() + .entrySet() + .stream() + .map(entry -> deserializeMailboxACLEntry(entry.getKey(), entry.getValue())) + .filter(Optional::isPresent) + .map(Optional::get) + .collect(ImmutableMap.toImmutableMap(Map.Entry::getKey, Map.Entry::getValue))); + + private static final Function RECORD_TO_MAILBOX_FUNCTION = record -> { + Mailbox mailbox = new Mailbox(new MailboxPath(record.get(MAILBOX_NAMESPACE), Username.of(record.get(USER_NAME)), record.get(MAILBOX_NAME)), + UidValidity.of(record.get(MAILBOX_UID_VALIDITY)), PostgresMailboxId.of(record.get(MAILBOX_ID))); + mailbox.setACL(HSTORE_TO_MAILBOX_ACL_FUNCTION.apply(Hstore.hstore(record.get(MAILBOX_ACL, LinkedHashMap.class)))); + return mailbox; + }; + + private static final Function RECORD_TO_POSTGRES_MAILBOX_FUNCTION = record -> new PostgresMailbox(RECORD_TO_MAILBOX_FUNCTION.apply(record), + Optional.ofNullable(record.get(MAILBOX_HIGHEST_MODSEQ)).map(ModSeq::of).orElse(ModSeq.first()), + Optional.ofNullable(record.get(MAILBOX_LAST_UID)).map(MessageUid::of).orElse(null)); + + + private static Optional> deserializeMailboxACLEntry(String key, String value) { + try { + MailboxACL.EntryKey entryKey = MailboxACL.EntryKey.deserialize(key); + MailboxACL.Rfc4314Rights rfc4314Rights = MailboxACL.Rfc4314Rights.deserialize(value); + return Optional.of(Map.entry(entryKey, rfc4314Rights)); + } catch (UnsupportedRightException e) { + LOGGER.error("Error while deserializing mailbox ACL", e); + return Optional.empty(); + } + } + + private final PostgresExecutor postgresExecutor; + + public PostgresMailboxDAO(PostgresExecutor postgresExecutor) { + this.postgresExecutor = postgresExecutor; + } + + public Mono create(MailboxPath mailboxPath, UidValidity uidValidity) { + PostgresMailboxId mailboxId = PostgresMailboxId.generate(); + + return postgresExecutor.executeRow(dslContext -> + Mono.from(dslContext.insertInto(TABLE_NAME, MAILBOX_ID, MAILBOX_NAME, USER_NAME, MAILBOX_NAMESPACE, MAILBOX_UID_VALIDITY) + .values(mailboxId.asUuid(), mailboxPath.getName(), mailboxPath.getUser().asString(), mailboxPath.getNamespace(), uidValidity.asLong()) + .onConflictOnConstraint(MAILBOX_NAME_USER_NAME_NAMESPACE_UNIQUE_CONSTRAINT) + .doNothing() + .returning(MAILBOX_ID))) + .map(record -> new Mailbox(mailboxPath, uidValidity, PostgresMailboxId.of(record.get(MAILBOX_ID)))) + .switchIfEmpty(Mono.error(new MailboxExistsException(mailboxPath.getName()))); + } + + public Mono rename(Mailbox mailbox) { + Preconditions.checkNotNull(mailbox.getMailboxId(), "A mailbox we want to rename should have a defined mailboxId"); + + return findMailboxByPath(mailbox.generateAssociatedPath()) + .flatMap(m -> Mono.error(new MailboxExistsException(mailbox.getName()))) + .then(update(mailbox)); + } + + private Mono update(Mailbox mailbox) { + return postgresExecutor.executeRow(dslContext -> Mono.from(dslContext.update(TABLE_NAME) + .set(MAILBOX_NAME, mailbox.getName()) + .set(USER_NAME, mailbox.getUser().asString()) + .set(MAILBOX_NAMESPACE, mailbox.getNamespace()) + .where(MAILBOX_ID.eq(((PostgresMailboxId) mailbox.getMailboxId()).asUuid())) + .returning(MAILBOX_ID))) + .map(record -> mailbox.getMailboxId()) + .switchIfEmpty(Mono.error(new MailboxNotFoundException(mailbox.getMailboxId()))); + } + + public Mono upsertACL(MailboxId mailboxId, MailboxACL acl, Long currentAclVersion) { + return postgresExecutor.executeReturnAffectedRowsCount(dslContext -> Mono.from(dslContext.update(TABLE_NAME) + .set(MAILBOX_ACL, MAILBOX_ACL_TO_HSTORE_FUNCTION.apply(acl)) + .set(MAILBOX_ACL_VERSION, currentAclVersion + 1) + .where(MAILBOX_ID.eq(((PostgresMailboxId) mailboxId).asUuid())) + .and(MAILBOX_ACL_VERSION.eq(currentAclVersion)))) + .filter(count -> count > 0) + .switchIfEmpty(Mono.error(new PostgresACLUpsertException("Upsert mailbox acl failed with mailboxId " + mailboxId.serialize()))) + .then(); + } + + public Mono upsertACL(MailboxId mailboxId, MailboxACL acl) { + return postgresExecutor.executeReturnAffectedRowsCount(dslContext -> Mono.from(dslContext.update(TABLE_NAME) + .set(MAILBOX_ACL, MAILBOX_ACL_TO_HSTORE_FUNCTION.apply(acl)) + .set(DSL.field(MAILBOX_ACL_VERSION.getName()), (Object) DSL.field(MAILBOX_ACL_VERSION.getName() + " + 1")) + .where(MAILBOX_ID.eq(((PostgresMailboxId) mailboxId).asUuid())))) + .filter(count -> count > 0) + .switchIfEmpty(Mono.error(new RuntimeException("Upsert mailbox acl failed with mailboxId " + mailboxId.serialize()))) + .then(); + } + + public Mono> getACL(MailboxId mailboxId) { + return postgresExecutor.executeRow(dsl -> Mono.from(dsl.select(MAILBOX_ACL, MAILBOX_ACL_VERSION) + .from(TABLE_NAME) + .where(MAILBOX_ID.eq(((PostgresMailboxId) mailboxId).asUuid())))) + .map(record -> Pair.of(Optional.ofNullable(record.get(MAILBOX_ACL)).map(HSTORE_TO_MAILBOX_ACL_FUNCTION).orElse(new MailboxACL()), record.get(MAILBOX_ACL_VERSION))); + } + + public Flux findMailboxesByUsername(Username userName) { + return postgresExecutor.executeRows(dslContext -> Flux.from(dslContext.select(MAILBOX_ID, + MAILBOX_NAME, + MAILBOX_UID_VALIDITY, + USER_NAME, + MAILBOX_NAMESPACE, + MAILBOX_LAST_UID, + MAILBOX_HIGHEST_MODSEQ, + DSL.function("slice", + DefaultDataType.getDefaultDataType("hstore"), + MAILBOX_ACL, + DSL.array(DSL.val(userName.asString()))).as(MAILBOX_ACL) + ).from(TABLE_NAME) + .where(DSL.sql(MAILBOX_ACL.getName() + " ? '" + userName.asString() + "'")))) //TODO fix security vulnerability + .map(RECORD_TO_POSTGRES_MAILBOX_FUNCTION); + } + + public Mono delete(MailboxId mailboxId) { + return postgresExecutor.executeVoid(dslContext -> Mono.from(dslContext.deleteFrom(TABLE_NAME) + .where(MAILBOX_ID.eq(((PostgresMailboxId) mailboxId).asUuid())))); + } + + public Mono findMailboxByPath(MailboxPath mailboxPath) { + return postgresExecutor.executeRow(dsl -> Mono.from(dsl.selectFrom(TABLE_NAME) + .where(MAILBOX_NAME.eq(mailboxPath.getName()) + .and(USER_NAME.eq(mailboxPath.getUser().asString())) + .and(MAILBOX_NAMESPACE.eq(mailboxPath.getNamespace()))))) + .map(RECORD_TO_POSTGRES_MAILBOX_FUNCTION); + } + + public Mono findMailboxById(MailboxId id) { + return postgresExecutor.executeRow(dsl -> Mono.from(dsl.selectFrom(TABLE_NAME) + .where(MAILBOX_ID.eq(((PostgresMailboxId) id).asUuid())))) + .map(RECORD_TO_POSTGRES_MAILBOX_FUNCTION) + .switchIfEmpty(Mono.error(new MailboxNotFoundException(id))); + } + + public Flux findMailboxByIds(List mailboxIds) { + return postgresExecutor.executeRows(dsl -> Flux.from(dsl.selectFrom(TABLE_NAME) + .where(MAILBOX_ID.in(mailboxIds.stream().map(PostgresMailboxId::asUuid).toList())))) + .map(RECORD_TO_POSTGRES_MAILBOX_FUNCTION); + } + + public Flux findMailboxWithPathLike(MailboxQuery.UserBound query) { + String pathLike = MailboxExpressionBackwardCompatibility.getPathLike(query); + Function getQueryCondition = mailboxQuery -> { + Condition baseCondition = USER_NAME.eq(mailboxQuery.getFixedUser().asString()) + .and(MAILBOX_NAMESPACE.eq(mailboxQuery.getFixedNamespace())); + + if (Wildcard.INSTANCE.equals(mailboxQuery.getMailboxNameExpression())) { + return baseCondition; + } + return baseCondition.and(MAILBOX_NAME.like(pathLike)); + }; + + return postgresExecutor.executeRows(dsl -> Flux.from(dsl.selectFrom(TABLE_NAME) + .where(getQueryCondition.apply(query))), PostgresExecutor.EAGER_FETCH) + .map(RECORD_TO_POSTGRES_MAILBOX_FUNCTION) + .filter(query::matches); + } + + public Mono hasChildren(Mailbox mailbox, char delimiter) { + String name = mailbox.getName() + delimiter + SQL_WILDCARD_CHAR; + + return postgresExecutor.executeExists(dsl -> dsl.selectOne().from(TABLE_NAME) + .where(MAILBOX_NAME.like(name) + .and(USER_NAME.eq(mailbox.getUser().asString())) + .and(MAILBOX_NAMESPACE.eq(mailbox.getNamespace())))); + } + + public Flux getAll() { + return postgresExecutor.executeRows(dsl -> Flux.from(dsl.selectFrom(TABLE_NAME))) + .map(RECORD_TO_POSTGRES_MAILBOX_FUNCTION); + } + + private UUID asUUID(MailboxId mailboxId) { + return ((PostgresMailboxId) mailboxId).asUuid(); + } + + public Mono findLastUidByMailboxId(MailboxId mailboxId) { + return postgresExecutor.executeRow(dsl -> Mono.from(dsl.select(MAILBOX_LAST_UID) + .from(TABLE_NAME) + .where(MAILBOX_ID.eq(asUUID(mailboxId))))) + .flatMap(record -> Mono.justOrEmpty(record.get(MAILBOX_LAST_UID))) + .map(MessageUid::of); + } + + public Mono incrementAndGetLastUid(MailboxId mailboxId, int count) { + return postgresExecutor.executeRow(dsl -> Mono.from(dsl.update(TABLE_NAME) + .set(MAILBOX_LAST_UID, coalesce(MAILBOX_LAST_UID, 0L).add(count)) + .where(MAILBOX_ID.eq(asUUID(mailboxId))) + .returning(MAILBOX_LAST_UID))) + .map(record -> record.get(MAILBOX_LAST_UID)) + .map(MessageUid::of); + } + + + public Mono findHighestModSeqByMailboxId(MailboxId mailboxId) { + return postgresExecutor.executeRow(dsl -> Mono.from(dsl.select(MAILBOX_HIGHEST_MODSEQ) + .from(TABLE_NAME) + .where(MAILBOX_ID.eq(asUUID(mailboxId))))) + .flatMap(record -> Mono.justOrEmpty(record.get(MAILBOX_HIGHEST_MODSEQ))) + .map(ModSeq::of); + } + + public Mono incrementAndGetModSeq(MailboxId mailboxId) { + return postgresExecutor.executeRow(dsl -> Mono.from(dsl.update(TABLE_NAME) + .set(MAILBOX_HIGHEST_MODSEQ, coalesce(MAILBOX_HIGHEST_MODSEQ, 0L).add(1)) + .where(MAILBOX_ID.eq(asUUID(mailboxId))) + .returning(MAILBOX_HIGHEST_MODSEQ))) + .map(record -> record.get(MAILBOX_HIGHEST_MODSEQ)) + .map(ModSeq::of); + } + + public Mono> incrementAndGetLastUidAndModSeq(MailboxId mailboxId) { + int increment = 1; + return postgresExecutor.executeRow(dsl -> Mono.from(dsl.update(TABLE_NAME) + .set(MAILBOX_LAST_UID, coalesce(MAILBOX_LAST_UID, 0L).add(increment)) + .set(MAILBOX_HIGHEST_MODSEQ, coalesce(MAILBOX_HIGHEST_MODSEQ, 0L).add(increment)) + .where(MAILBOX_ID.eq(asUUID(mailboxId))) + .returning(MAILBOX_LAST_UID, MAILBOX_HIGHEST_MODSEQ))) + .map(record -> Pair.of(MessageUid.of(record.get(MAILBOX_LAST_UID)), ModSeq.of(record.get(MAILBOX_HIGHEST_MODSEQ)))); + } + +} diff --git a/mailbox/postgres/src/main/java/org/apache/james/mailbox/postgres/mail/dao/PostgresMailboxMessageDAO.java b/mailbox/postgres/src/main/java/org/apache/james/mailbox/postgres/mail/dao/PostgresMailboxMessageDAO.java new file mode 100644 index 00000000000..e2055a12978 --- /dev/null +++ b/mailbox/postgres/src/main/java/org/apache/james/mailbox/postgres/mail/dao/PostgresMailboxMessageDAO.java @@ -0,0 +1,676 @@ +/**************************************************************** + * 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.mailbox.postgres.mail.dao; + +import static org.apache.james.backends.postgres.PostgresCommons.DATE_TO_LOCAL_DATE_TIME; +import static org.apache.james.backends.postgres.PostgresCommons.IN_CLAUSE_MAX_SIZE; +import static org.apache.james.backends.postgres.PostgresCommons.UNNEST_FIELD; +import static org.apache.james.backends.postgres.PostgresCommons.tableField; +import static org.apache.james.backends.postgres.utils.PostgresExecutor.EAGER_FETCH; +import static org.apache.james.mailbox.postgres.mail.PostgresMessageModule.MessageTable.INTERNAL_DATE; +import static org.apache.james.mailbox.postgres.mail.PostgresMessageModule.MessageTable.SIZE; +import static org.apache.james.mailbox.postgres.mail.PostgresMessageModule.MessageToMailboxTable.IS_ANSWERED; +import static org.apache.james.mailbox.postgres.mail.PostgresMessageModule.MessageToMailboxTable.IS_DELETED; +import static org.apache.james.mailbox.postgres.mail.PostgresMessageModule.MessageToMailboxTable.IS_DRAFT; +import static org.apache.james.mailbox.postgres.mail.PostgresMessageModule.MessageToMailboxTable.IS_FLAGGED; +import static org.apache.james.mailbox.postgres.mail.PostgresMessageModule.MessageToMailboxTable.IS_RECENT; +import static org.apache.james.mailbox.postgres.mail.PostgresMessageModule.MessageToMailboxTable.IS_SEEN; +import static org.apache.james.mailbox.postgres.mail.PostgresMessageModule.MessageToMailboxTable.MAILBOX_ID; +import static org.apache.james.mailbox.postgres.mail.PostgresMessageModule.MessageToMailboxTable.MESSAGE_ID; +import static org.apache.james.mailbox.postgres.mail.PostgresMessageModule.MessageToMailboxTable.MESSAGE_UID; +import static org.apache.james.mailbox.postgres.mail.PostgresMessageModule.MessageToMailboxTable.MOD_SEQ; +import static org.apache.james.mailbox.postgres.mail.PostgresMessageModule.MessageToMailboxTable.REMOVE_ELEMENTS_FROM_ARRAY_FUNCTION_NAME; +import static org.apache.james.mailbox.postgres.mail.PostgresMessageModule.MessageToMailboxTable.SAVE_DATE; +import static org.apache.james.mailbox.postgres.mail.PostgresMessageModule.MessageToMailboxTable.TABLE_NAME; +import static org.apache.james.mailbox.postgres.mail.PostgresMessageModule.MessageToMailboxTable.THREAD_ID; +import static org.apache.james.mailbox.postgres.mail.PostgresMessageModule.MessageToMailboxTable.USER_FLAGS; +import static org.apache.james.mailbox.postgres.mail.dao.PostgresMailboxMessageDAOUtils.BOOLEAN_FLAGS_MAPPING; +import static org.apache.james.mailbox.postgres.mail.dao.PostgresMailboxMessageDAOUtils.FETCH_TYPE_TO_FETCH_STRATEGY; +import static org.apache.james.mailbox.postgres.mail.dao.PostgresMailboxMessageDAOUtils.MESSAGE_METADATA_FIELDS_REQUIRE; +import static org.apache.james.mailbox.postgres.mail.dao.PostgresMailboxMessageDAOUtils.RECORD_TO_COMPOSED_MESSAGE_ID_WITH_META_DATA_FUNCTION; +import static org.apache.james.mailbox.postgres.mail.dao.PostgresMailboxMessageDAOUtils.RECORD_TO_FLAGS_FUNCTION; +import static org.apache.james.mailbox.postgres.mail.dao.PostgresMailboxMessageDAOUtils.RECORD_TO_MESSAGE_METADATA_FUNCTION; +import static org.apache.james.mailbox.postgres.mail.dao.PostgresMailboxMessageDAOUtils.RECORD_TO_MESSAGE_UID_FUNCTION; + +import java.util.Collection; +import java.util.List; +import java.util.Optional; +import java.util.concurrent.atomic.AtomicReference; +import java.util.function.Function; + +import jakarta.inject.Inject; +import jakarta.inject.Singleton; +import jakarta.mail.Flags; + +import org.apache.commons.lang3.tuple.Pair; +import org.apache.james.backends.postgres.utils.PostgresExecutor; +import org.apache.james.backends.postgres.utils.PostgresUtils; +import org.apache.james.core.Domain; +import org.apache.james.mailbox.MessageUid; +import org.apache.james.mailbox.ModSeq; +import org.apache.james.mailbox.model.ComposedMessageIdWithMetaData; +import org.apache.james.mailbox.model.MailboxId; +import org.apache.james.mailbox.model.MessageMetaData; +import org.apache.james.mailbox.model.MessageRange; +import org.apache.james.mailbox.postgres.PostgresMailboxId; +import org.apache.james.mailbox.postgres.PostgresMessageId; +import org.apache.james.mailbox.postgres.mail.PostgresMessageModule.MessageTable; +import org.apache.james.mailbox.store.mail.MessageMapper; +import org.apache.james.mailbox.store.mail.MessageMapper.FetchType; +import org.apache.james.mailbox.store.mail.model.MailboxMessage; +import org.apache.james.mailbox.store.mail.model.impl.SimpleMailboxMessage; +import org.apache.james.util.streams.Limit; +import org.jooq.Condition; +import org.jooq.DSLContext; +import org.jooq.Name; +import org.jooq.Record; +import org.jooq.SortField; +import org.jooq.TableOnConditionStep; +import org.jooq.UpdateConditionStep; +import org.jooq.UpdateSetStep; +import org.jooq.impl.DSL; + +import com.google.common.collect.ImmutableList; +import com.google.common.collect.Iterables; + +import reactor.core.publisher.Flux; +import reactor.core.publisher.Mono; + +public class PostgresMailboxMessageDAO { + + public static class Factory { + private final PostgresExecutor.Factory executorFactory; + + @Inject + @Singleton + public Factory(PostgresExecutor.Factory executorFactory) { + this.executorFactory = executorFactory; + } + + public PostgresMailboxMessageDAO create(Optional domain) { + return new PostgresMailboxMessageDAO(executorFactory.create(domain)); + } + } + + private static final TableOnConditionStep MESSAGES_JOIN_MAILBOX_MESSAGES_CONDITION_STEP = TABLE_NAME.join(MessageTable.TABLE_NAME) + .on(tableField(TABLE_NAME, MESSAGE_ID).eq(tableField(MessageTable.TABLE_NAME, MessageTable.MESSAGE_ID))); + + public static final SortField DEFAULT_SORT_ORDER_BY = MESSAGE_UID.asc(); + + private static final int QUERY_BATCH_SIZE = PostgresUtils.QUERY_BATCH_SIZE; + + private final PostgresExecutor postgresExecutor; + + public PostgresMailboxMessageDAO(PostgresExecutor postgresExecutor) { + this.postgresExecutor = postgresExecutor; + } + + public Mono findFirstUnseenMessageUid(PostgresMailboxId mailboxId) { + return postgresExecutor.executeRow(dslContext -> Flux.from(dslContext.select(MESSAGE_UID) + .from(TABLE_NAME) + .where(MAILBOX_ID.eq((mailboxId.asUuid()))) + .and(IS_SEEN.eq(false)) + .orderBy(DEFAULT_SORT_ORDER_BY) + .limit(1))) + .map(RECORD_TO_MESSAGE_UID_FUNCTION); + } + + public Flux listUnseen(PostgresMailboxId mailboxId) { + return postgresExecutor.executeRows(dslContext -> Flux.from(dslContext.select(MESSAGE_UID) + .from(TABLE_NAME) + .where(MAILBOX_ID.eq((mailboxId.asUuid()))) + .and(IS_SEEN.eq(false)) + .orderBy(DEFAULT_SORT_ORDER_BY)), EAGER_FETCH) + .map(RECORD_TO_MESSAGE_UID_FUNCTION); + } + + public Flux listUnseen(PostgresMailboxId mailboxId, MessageRange range) { + return switch (range.getType()) { + case ALL -> listUnseen(mailboxId); + case FROM -> postgresExecutor.executeRows(dslContext -> Flux.from(dslContext.select(MESSAGE_UID) + .from(TABLE_NAME) + .where(MAILBOX_ID.eq(mailboxId.asUuid())) + .and(IS_SEEN.eq(false)) + .and(MESSAGE_UID.greaterOrEqual(range.getUidFrom().asLong())) + .orderBy(DEFAULT_SORT_ORDER_BY)), EAGER_FETCH) + .map(RECORD_TO_MESSAGE_UID_FUNCTION); + case RANGE -> postgresExecutor.executeRows(dslContext -> Flux.from(dslContext.select(MESSAGE_UID) + .from(TABLE_NAME) + .where(MAILBOX_ID.eq(mailboxId.asUuid())) + .and(IS_SEEN.eq(false)) + .and(MESSAGE_UID.greaterOrEqual(range.getUidFrom().asLong())) + .and(MESSAGE_UID.lessOrEqual(range.getUidTo().asLong())) + .orderBy(DEFAULT_SORT_ORDER_BY)), EAGER_FETCH) + .map(RECORD_TO_MESSAGE_UID_FUNCTION); + case ONE -> postgresExecutor.executeRows(dslContext -> Flux.from(dslContext.select(MESSAGE_UID) + .from(TABLE_NAME) + .where(MAILBOX_ID.eq(mailboxId.asUuid())) + .and(IS_SEEN.eq(false)) + .and(MESSAGE_UID.eq(range.getUidFrom().asLong())) + .orderBy(DEFAULT_SORT_ORDER_BY)), EAGER_FETCH) + .map(RECORD_TO_MESSAGE_UID_FUNCTION); + }; + } + + public Flux findAllRecentMessageUid(PostgresMailboxId mailboxId) { + return postgresExecutor.executeRows(dslContext -> Flux.from(dslContext.select(MESSAGE_UID) + .from(TABLE_NAME) + .where(MAILBOX_ID.eq((mailboxId.asUuid()))) + .and(IS_RECENT.eq(true)) + .orderBy(DEFAULT_SORT_ORDER_BY)), EAGER_FETCH) + .map(RECORD_TO_MESSAGE_UID_FUNCTION); + } + + public Flux listAllMessageUid(PostgresMailboxId mailboxId) { + return postgresExecutor.executeRows(dslContext -> Flux.from(dslContext.select(MESSAGE_UID) + .from(TABLE_NAME) + .where(MAILBOX_ID.eq((mailboxId.asUuid()))) + .orderBy(DEFAULT_SORT_ORDER_BY)), EAGER_FETCH) + .map(RECORD_TO_MESSAGE_UID_FUNCTION); + } + + public Flux listUids(PostgresMailboxId mailboxId, MessageRange range) { + if (range.getType() == MessageRange.Type.ALL) { + return listAllMessageUid(mailboxId); + } + return doListUids(mailboxId, range); + } + + private Flux doListUids(PostgresMailboxId mailboxId, MessageRange range) { + return postgresExecutor.executeRows(dslContext -> Flux.from(dslContext.select(MESSAGE_UID) + .from(TABLE_NAME) + .where(MAILBOX_ID.eq(mailboxId.asUuid())) + .and(MESSAGE_UID.greaterOrEqual(range.getUidFrom().asLong())) + .and(MESSAGE_UID.lessOrEqual(range.getUidTo().asLong())) + .orderBy(DEFAULT_SORT_ORDER_BY)), EAGER_FETCH) + .map(RECORD_TO_MESSAGE_UID_FUNCTION); + } + + public Mono deleteByMailboxIdAndMessageUid(PostgresMailboxId mailboxId, MessageUid messageUid) { + return postgresExecutor.executeRow(dslContext -> Mono.from(dslContext.deleteFrom(TABLE_NAME) + .where(MAILBOX_ID.eq(mailboxId.asUuid())) + .and(MESSAGE_UID.eq(messageUid.asLong())) + .returning(MESSAGE_METADATA_FIELDS_REQUIRE))) + .map(RECORD_TO_MESSAGE_METADATA_FUNCTION); + } + + public Flux deleteByMailboxIdAndMessageUids(PostgresMailboxId mailboxId, List uids) { + if (uids.isEmpty()) { + return Flux.empty(); + } + Function, Flux> deletePublisherFunction = uidsToDelete -> postgresExecutor.executeDeleteAndReturnList(dslContext -> dslContext.deleteFrom(TABLE_NAME) + .where(MAILBOX_ID.eq(mailboxId.asUuid())) + .and(MESSAGE_UID.in(uidsToDelete.stream().map(MessageUid::asLong).toArray(Long[]::new))) + .returning(MESSAGE_METADATA_FIELDS_REQUIRE)) + .map(RECORD_TO_MESSAGE_METADATA_FUNCTION); + + if (uids.size() <= IN_CLAUSE_MAX_SIZE) { + return deletePublisherFunction.apply(uids); + } else { + return Flux.fromIterable(Iterables.partition(uids, IN_CLAUSE_MAX_SIZE)) + .flatMap(deletePublisherFunction); + } + } + + public Flux deleteByMailboxId(PostgresMailboxId mailboxId) { + return postgresExecutor.executeDeleteAndReturnList(dslContext -> dslContext.deleteFrom(TABLE_NAME) + .where(MAILBOX_ID.eq(mailboxId.asUuid())) + .returning(MESSAGE_ID)) + .map(record -> PostgresMessageId.Factory.of(record.get(MESSAGE_ID))) + .collectList() + .flatMapMany(Flux::fromIterable); + } + + public Mono deleteByMessageIdAndMailboxIds(PostgresMessageId messageId, Collection mailboxIds) { + if (mailboxIds.isEmpty()) { + return Mono.empty(); + } + return postgresExecutor.executeVoid(dslContext -> Mono.from(dslContext.deleteFrom(TABLE_NAME) + .where(MESSAGE_ID.eq(messageId.asUuid())) + .and(MAILBOX_ID.in(mailboxIds.stream().map(PostgresMailboxId::asUuid).collect(ImmutableList.toImmutableList()))))); + } + + public Mono deleteByMessageId(PostgresMessageId messageId) { + return postgresExecutor.executeVoid(dslContext -> Mono.from(dslContext.deleteFrom(TABLE_NAME) + .where(MESSAGE_ID.eq(messageId.asUuid())))); + } + + public Mono countTotalMessagesByMailboxId(PostgresMailboxId mailboxId) { + return postgresExecutor.executeCount(dslContext -> Mono.from(dslContext.selectCount() + .from(TABLE_NAME) + .where(MAILBOX_ID.eq(mailboxId.asUuid())))); + } + + public Mono> countTotalAndUnseenMessagesByMailboxId(PostgresMailboxId mailboxId) { + Name totalCount = DSL.name("total_count"); + Name unSeenCount = DSL.name("unseen_count"); + return postgresExecutor.executeRow(dslContext -> Mono.from(dslContext.select( + DSL.count().as(totalCount), + DSL.count().filterWhere(IS_SEEN.eq(false)).as(unSeenCount)) + .from(TABLE_NAME) + .where(MAILBOX_ID.eq(mailboxId.asUuid())))) + .map(record -> Pair.of(record.get(totalCount, Integer.class), record.get(unSeenCount, Integer.class))); + } + + public Flux> findMessagesByMailboxId(PostgresMailboxId mailboxId, Limit limit, MessageMapper.FetchType fetchType) { + if (limit.isUnlimited()) { + return Flux.defer(() -> findMessagesByMailboxIdBatch(mailboxId, fetchType, Optional.empty(), QUERY_BATCH_SIZE)) + .expand(messages -> { + if (messages.isEmpty() || messages.size() < QUERY_BATCH_SIZE) { + return Mono.empty(); + } + return findMessagesByMailboxIdBatch(mailboxId, fetchType, Optional.of(messages.getLast().getRight().get(MESSAGE_UID)), QUERY_BATCH_SIZE); + }) + .flatMapIterable(Function.identity()); + } else { + return findMessagesByMailboxIdBatch(mailboxId, fetchType, Optional.empty(), limit.getLimit().get()) + .flatMapIterable(Function.identity()); + } + } + + private Mono>> findMessagesByMailboxIdBatch(PostgresMailboxId mailboxId, MessageMapper.FetchType fetchType, + Optional messageUidFrom, int batchSize) { + PostgresMailboxMessageFetchStrategy fetchStrategy = FETCH_TYPE_TO_FETCH_STRATEGY.apply(fetchType); + return postgresExecutor.executeRows(dslContext -> Flux.from(dslContext.select(fetchStrategy.fetchFields()) + .from(MESSAGES_JOIN_MAILBOX_MESSAGES_CONDITION_STEP) + .where(MAILBOX_ID.eq(mailboxId.asUuid())) + .and(messageUidFrom.map(MESSAGE_UID::greaterThan).orElseGet(DSL::noCondition)) + .orderBy(MESSAGE_UID.asc()) + .limit(batchSize))) + .map(record -> Pair.of(fetchStrategy.toMessageBuilder().apply(record), record)) + .collectList() + .switchIfEmpty(Mono.just(ImmutableList.of())); + } + + public Flux> findMessagesByMailboxIdAndBetweenUIDs(PostgresMailboxId mailboxId, MessageUid from, MessageUid to, Limit limit, FetchType fetchType) { + if (limit.isUnlimited()) { + return Flux.defer(() -> findMessagesByMailboxIdAndBetweenUIDsBatch(mailboxId, MESSAGE_UID.greaterOrEqual(from.asLong()), to, fetchType, QUERY_BATCH_SIZE)) + .expand(messages -> { + if (messages.isEmpty() || messages.size() < QUERY_BATCH_SIZE) { + return Mono.empty(); + } + MessageUid messageUidFrom = MessageUid.of(messages.getLast().getRight().get(MESSAGE_UID)); + return findMessagesByMailboxIdAndBetweenUIDsBatch(mailboxId, MESSAGE_UID.greaterThan(messageUidFrom.asLong()), to, fetchType, QUERY_BATCH_SIZE); + }) + .flatMapIterable(Function.identity()); + } else { + return findMessagesByMailboxIdAndBetweenUIDsBatch(mailboxId, MESSAGE_UID.greaterOrEqual(from.asLong()), to, fetchType, limit.getLimit().get()) + .flatMapIterable(Function.identity()); + } + } + + private Mono>> findMessagesByMailboxIdAndBetweenUIDsBatch(PostgresMailboxId mailboxId, Condition messageUidFromCondition, + MessageUid to, + FetchType fetchType, int batchSize) { + PostgresMailboxMessageFetchStrategy fetchStrategy = FETCH_TYPE_TO_FETCH_STRATEGY.apply(fetchType); + return postgresExecutor.executeRows(dslContext -> Flux.from(dslContext.select(fetchStrategy.fetchFields()) + .from(MESSAGES_JOIN_MAILBOX_MESSAGES_CONDITION_STEP) + .where(MAILBOX_ID.eq(mailboxId.asUuid())) + .and(messageUidFromCondition) + .and(MESSAGE_UID.lessOrEqual(to.asLong())) + .orderBy(MESSAGE_UID.asc()) + .limit(batchSize))) + .map(record -> Pair.of(fetchStrategy.toMessageBuilder().apply(record), record)) + .collectList() + .switchIfEmpty(Mono.just(ImmutableList.of())); + } + + public Mono> findMessageByMailboxIdAndUid(PostgresMailboxId mailboxId, MessageUid uid, FetchType fetchType) { + PostgresMailboxMessageFetchStrategy fetchStrategy = FETCH_TYPE_TO_FETCH_STRATEGY.apply(fetchType); + return postgresExecutor.executeRow(dslContext -> Mono.from(dslContext.select(fetchStrategy.fetchFields()) + .from(MESSAGES_JOIN_MAILBOX_MESSAGES_CONDITION_STEP) + .where(MAILBOX_ID.eq(mailboxId.asUuid())) + .and(MESSAGE_UID.eq(uid.asLong())))) + .map(record -> Pair.of(fetchStrategy.toMessageBuilder().apply(record), record)); + } + + public Flux> findMessagesByMailboxIdAndAfterUID(PostgresMailboxId mailboxId, MessageUid from, Limit limit, FetchType fetchType) { + if (limit.isUnlimited()) { + return Flux.defer(() -> findMessagesByMailboxIdAndAfterUIDBatch(mailboxId, MESSAGE_UID.greaterOrEqual(from.asLong()), fetchType, QUERY_BATCH_SIZE)) + .expand(messages -> { + if (messages.isEmpty() || messages.size() < QUERY_BATCH_SIZE) { + return Mono.empty(); + } + MessageUid messageUidFrom = MessageUid.of(messages.getLast().getRight().get(MESSAGE_UID)); + return findMessagesByMailboxIdAndAfterUIDBatch(mailboxId, MESSAGE_UID.greaterThan(messageUidFrom.asLong()), fetchType, QUERY_BATCH_SIZE); + }) + .flatMapIterable(Function.identity()); + } else { + return findMessagesByMailboxIdAndAfterUIDBatch(mailboxId, MESSAGE_UID.greaterOrEqual(from.asLong()), fetchType, limit.getLimit().get()) + .flatMapIterable(Function.identity()); + } + } + + private Mono>> findMessagesByMailboxIdAndAfterUIDBatch(PostgresMailboxId mailboxId, + Condition messageUidFromCondition, + FetchType fetchType, int batchSize) { + PostgresMailboxMessageFetchStrategy fetchStrategy = FETCH_TYPE_TO_FETCH_STRATEGY.apply(fetchType); + return postgresExecutor.executeRows(dslContext -> Flux.from(dslContext.select(fetchStrategy.fetchFields()) + .from(MESSAGES_JOIN_MAILBOX_MESSAGES_CONDITION_STEP) + .where(MAILBOX_ID.eq(mailboxId.asUuid())) + .and(messageUidFromCondition) + .orderBy(MESSAGE_UID.asc()) + .limit(batchSize))) + .map(record -> Pair.of(fetchStrategy.toMessageBuilder().apply(record), record)) + .collectList() + .switchIfEmpty(Mono.just(ImmutableList.of())); + } + + public Flux findMessagesByMailboxIdAndUIDs(PostgresMailboxId mailboxId, List uids) { + if (uids.isEmpty()) { + return Flux.empty(); + } + PostgresMailboxMessageFetchStrategy fetchStrategy = PostgresMailboxMessageFetchStrategy.METADATA; + Function, Flux> queryPublisherFunction = + uidsToFetch -> postgresExecutor.executeRows(dslContext -> Flux.from(dslContext.select(fetchStrategy.fetchFields()) + .from(MESSAGES_JOIN_MAILBOX_MESSAGES_CONDITION_STEP) + .where(MAILBOX_ID.eq(mailboxId.asUuid())) + .and(MESSAGE_UID.in(uidsToFetch.stream().map(MessageUid::asLong).toArray(Long[]::new))) + .orderBy(DEFAULT_SORT_ORDER_BY)), EAGER_FETCH) + .map(fetchStrategy.toMessageBuilder()); + + if (uids.size() <= IN_CLAUSE_MAX_SIZE) { + return queryPublisherFunction.apply(uids); + } else { + return Flux.fromIterable(Iterables.partition(uids, IN_CLAUSE_MAX_SIZE)) + .flatMap(queryPublisherFunction); + } + } + + public Flux findDeletedMessagesByMailboxId(PostgresMailboxId mailboxId) { + return postgresExecutor.executeRows(dslContext -> Flux.from(dslContext.select(MESSAGE_UID) + .from(TABLE_NAME) + .where(MAILBOX_ID.eq(mailboxId.asUuid())) + .and(IS_DELETED.eq(true)) + .orderBy(DEFAULT_SORT_ORDER_BY)), EAGER_FETCH) + .map(RECORD_TO_MESSAGE_UID_FUNCTION); + } + + public Flux findDeletedMessagesByMailboxIdAndBetweenUIDs(PostgresMailboxId mailboxId, MessageUid from, MessageUid to) { + return postgresExecutor.executeRows(dslContext -> Flux.from(dslContext.select(MESSAGE_UID) + .from(TABLE_NAME) + .where(MAILBOX_ID.eq(mailboxId.asUuid())) + .and(IS_DELETED.eq(true)) + .and(MESSAGE_UID.greaterOrEqual(from.asLong())) + .and(MESSAGE_UID.lessOrEqual(to.asLong())) + .orderBy(DEFAULT_SORT_ORDER_BY)), EAGER_FETCH) + .map(RECORD_TO_MESSAGE_UID_FUNCTION); + } + + public Flux findDeletedMessagesByMailboxIdAndAfterUID(PostgresMailboxId mailboxId, MessageUid from) { + return postgresExecutor.executeRows(dslContext -> Flux.from(dslContext.select(MESSAGE_UID) + .from(TABLE_NAME) + .where(MAILBOX_ID.eq(mailboxId.asUuid())) + .and(IS_DELETED.eq(true)) + .and(MESSAGE_UID.greaterOrEqual(from.asLong())) + .orderBy(DEFAULT_SORT_ORDER_BY)), EAGER_FETCH) + .map(RECORD_TO_MESSAGE_UID_FUNCTION); + } + + public Mono findDeletedMessageByMailboxIdAndUid(PostgresMailboxId mailboxId, MessageUid uid) { + return postgresExecutor.executeRow(dslContext -> Mono.from(dslContext.select(MESSAGE_UID) + .from(TABLE_NAME) + .where(MAILBOX_ID.eq(mailboxId.asUuid())) + .and(IS_DELETED.eq(true)) + .and(MESSAGE_UID.eq(uid.asLong())))) + .map(RECORD_TO_MESSAGE_UID_FUNCTION); + } + + public Flux listNotDeletedUids(PostgresMailboxId mailboxId, MessageRange range) { + return postgresExecutor.executeRows(dslContext -> Flux.from(dslContext.select(MESSAGE_UID, IS_DELETED) + .from(TABLE_NAME) + .where(MAILBOX_ID.eq(mailboxId.asUuid())) + .and(MESSAGE_UID.greaterOrEqual(range.getUidFrom().asLong())) + .and(MESSAGE_UID.lessOrEqual(range.getUidTo().asLong())) + .and(IS_DELETED.eq(false)) + .orderBy(DEFAULT_SORT_ORDER_BY)), EAGER_FETCH) + .map(RECORD_TO_MESSAGE_UID_FUNCTION); + } + + public Mono existsByMessageId(PostgresMessageId messageId) { + return postgresExecutor.executeExists(dslContext -> dslContext.selectOne() + .from(TABLE_NAME) + .where(MESSAGE_ID.eq(messageId.asUuid()))); + } + + public Flux findMessagesMetadata(PostgresMailboxId mailboxId, MessageRange range) { + return Flux.defer(() -> findMessagesMetadataBatch(mailboxId, range.getUidTo(), MESSAGE_UID.greaterOrEqual(range.getUidFrom().asLong()), QUERY_BATCH_SIZE)) + .expand(messages -> { + if (messages.isEmpty() || messages.size() < QUERY_BATCH_SIZE) { + return Mono.empty(); + } + MessageUid messageUidFrom = messages.getLast().getComposedMessageId().getUid(); + return findMessagesMetadataBatch(mailboxId, range.getUidTo(), MESSAGE_UID.greaterThan(messageUidFrom.asLong()), QUERY_BATCH_SIZE); + }) + .flatMapIterable(Function.identity()); + } + + private Mono> findMessagesMetadataBatch(PostgresMailboxId mailboxId, MessageUid messageUidTo, Condition messageUidFromCondition, int batchSize) { + return postgresExecutor.executeRows(dslContext -> Flux.from(dslContext.select() + .from(TABLE_NAME) + .where(MAILBOX_ID.eq(mailboxId.asUuid())) + .and(messageUidFromCondition) + .and(MESSAGE_UID.lessOrEqual(messageUidTo.asLong())) + .orderBy(MESSAGE_UID.asc()) + .limit(batchSize))) + .map(RECORD_TO_COMPOSED_MESSAGE_ID_WITH_META_DATA_FUNCTION) + .collectList() + .switchIfEmpty(Mono.just(ImmutableList.of())); + } + + public Flux findAllRecentMessageMetadata(PostgresMailboxId mailboxId) { + return Flux.defer(() -> findAllRecentMessageMetadataBatch(mailboxId, Optional.empty(), QUERY_BATCH_SIZE)) + .expand(messages -> { + if (messages.isEmpty() || messages.size() < QUERY_BATCH_SIZE) { + return Mono.empty(); + } + return findAllRecentMessageMetadataBatch(mailboxId, Optional.of(messages.getLast().getComposedMessageId().getUid()), QUERY_BATCH_SIZE); + }) + .flatMapIterable(Function.identity()); + } + + private Mono> findAllRecentMessageMetadataBatch(PostgresMailboxId mailboxId, Optional messageUidFrom, int batchSize) { + return postgresExecutor.executeRows(dslContext -> Flux.from(dslContext.select() + .from(TABLE_NAME) + .where(MAILBOX_ID.eq(mailboxId.asUuid())) + .and(IS_RECENT.eq(true)) + .and(messageUidFrom.map(messageUid -> MESSAGE_UID.greaterThan(messageUid.asLong())).orElseGet(DSL::noCondition)) + .orderBy(MESSAGE_UID.asc()) + .limit(batchSize))) + .map(RECORD_TO_COMPOSED_MESSAGE_ID_WITH_META_DATA_FUNCTION) + .collectList() + .switchIfEmpty(Mono.just(ImmutableList.of())); + } + + public Mono replaceFlags(PostgresMailboxId mailboxId, MessageUid uid, Flags newFlags, ModSeq newModSeq) { + return postgresExecutor.executeRow(dslContext -> Mono.from(buildReplaceFlagsStatement(dslContext, newFlags, mailboxId, uid, newModSeq) + .returning(MESSAGE_METADATA_FIELDS_REQUIRE))) + .map(RECORD_TO_FLAGS_FUNCTION); + } + + public Mono addFlags(PostgresMailboxId mailboxId, MessageUid uid, Flags appendFlags, ModSeq newModSeq) { + return postgresExecutor.executeRow(dslContext -> Mono.from(buildAddFlagsStatement(dslContext, appendFlags, mailboxId, uid, newModSeq) + .returning(MESSAGE_METADATA_FIELDS_REQUIRE))) + .map(RECORD_TO_FLAGS_FUNCTION); + } + + public Mono removeFlags(PostgresMailboxId mailboxId, MessageUid uid, Flags removeFlags, ModSeq newModSeq) { + return postgresExecutor.executeRow(dslContext -> Mono.from(buildRemoveFlagsStatement(dslContext, removeFlags, mailboxId, uid, newModSeq) + .returning(MESSAGE_METADATA_FIELDS_REQUIRE))) + .map(RECORD_TO_FLAGS_FUNCTION); + } + + private UpdateConditionStep buildAddFlagsStatement(DSLContext dslContext, Flags addFlags, + PostgresMailboxId mailboxId, MessageUid uid, ModSeq newModSeq) { + AtomicReference> updateStatement = new AtomicReference<>(dslContext.update(TABLE_NAME)); + + BOOLEAN_FLAGS_MAPPING.forEach((flagColumn, flagMapped) -> { + if (addFlags.contains(flagMapped)) { + updateStatement.getAndUpdate(currentStatement -> currentStatement.set(flagColumn, true)); + } + }); + + if (addFlags.getUserFlags() != null && addFlags.getUserFlags().length > 0) { + if (addFlags.getUserFlags().length == 1) { + updateStatement.getAndUpdate(currentStatement -> currentStatement.set(USER_FLAGS, DSL.arrayAppend(USER_FLAGS, addFlags.getUserFlags()[0]))); + } else { + updateStatement.getAndUpdate(currentStatement -> currentStatement.set(USER_FLAGS, DSL.arrayConcat(USER_FLAGS, addFlags.getUserFlags()))); + } + } + + return updateStatement.get() + .set(MOD_SEQ, newModSeq.asLong()) + .where(MAILBOX_ID.eq(mailboxId.asUuid())) + .and(MESSAGE_UID.eq(uid.asLong())); + } + + private UpdateConditionStep buildReplaceFlagsStatement(DSLContext dslContext, Flags newFlags, + PostgresMailboxId mailboxId, MessageUid uid, ModSeq newModSeq) { + AtomicReference> updateStatement = new AtomicReference<>(dslContext.update(TABLE_NAME)); + + BOOLEAN_FLAGS_MAPPING.forEach((flagColumn, flagMapped) -> { + updateStatement.getAndUpdate(currentStatement -> currentStatement.set(flagColumn, newFlags.contains(flagMapped))); + }); + + return updateStatement.get() + .set(USER_FLAGS, newFlags.getUserFlags()) + .set(MOD_SEQ, newModSeq.asLong()) + .where(MAILBOX_ID.eq(mailboxId.asUuid())) + .and(MESSAGE_UID.eq(uid.asLong())); + } + + private UpdateConditionStep buildRemoveFlagsStatement(DSLContext dslContext, Flags removeFlags, + PostgresMailboxId mailboxId, MessageUid uid, ModSeq newModSeq) { + AtomicReference> updateStatement = new AtomicReference<>(dslContext.update(TABLE_NAME)); + + BOOLEAN_FLAGS_MAPPING.forEach((flagColumn, flagMapped) -> { + if (removeFlags.contains(flagMapped)) { + updateStatement.getAndUpdate(currentStatement -> currentStatement.set(flagColumn, false)); + } + }); + + if (removeFlags.getUserFlags() != null && removeFlags.getUserFlags().length > 0) { + if (removeFlags.getUserFlags().length == 1) { + updateStatement.getAndUpdate(currentStatement -> currentStatement.set(USER_FLAGS, DSL.arrayRemove(USER_FLAGS, removeFlags.getUserFlags()[0]))); + } else { + updateStatement.getAndUpdate(currentStatement -> currentStatement.set(USER_FLAGS, DSL.function(REMOVE_ELEMENTS_FROM_ARRAY_FUNCTION_NAME, String[].class, + USER_FLAGS, + DSL.array(removeFlags.getUserFlags())))); + } + } + + return updateStatement.get() + .set(MOD_SEQ, newModSeq.asLong()) + .where(MAILBOX_ID.eq(mailboxId.asUuid())) + .and(MESSAGE_UID.eq(uid.asLong())); + } + + public Mono listDistinctUserFlags(PostgresMailboxId mailboxId) { + return postgresExecutor.executeRows(dslContext -> Flux.from(dslContext.selectDistinct(UNNEST_FIELD.apply(USER_FLAGS)) + .from(TABLE_NAME) + .where(MAILBOX_ID.eq(mailboxId.asUuid())))) + .map(record -> record.get(0, String.class)) + .collectList() + .map(flagList -> { + Flags flags = new Flags(); + flagList.forEach(flags::add); + return flags; + }); + } + + public Flux resetRecentFlag(PostgresMailboxId mailboxId, List uids, ModSeq newModSeq) { + if (uids.isEmpty()) { + return Flux.empty(); + } + Function, Flux> queryPublisherFunction = uidsMatching -> postgresExecutor.executeRows(dslContext -> Flux.from(dslContext.update(TABLE_NAME) + .set(IS_RECENT, false) + .set(MOD_SEQ, newModSeq.asLong()) + .where(MAILBOX_ID.eq(mailboxId.asUuid())) + .and(MESSAGE_UID.in(uidsMatching.stream().map(MessageUid::asLong).toArray(Long[]::new))) + .and(MOD_SEQ.notEqual(newModSeq.asLong())) + .returning(MESSAGE_METADATA_FIELDS_REQUIRE))) + .map(RECORD_TO_MESSAGE_METADATA_FUNCTION); + if (uids.size() <= IN_CLAUSE_MAX_SIZE) { + return queryPublisherFunction.apply(uids); + } else { + return Flux.fromIterable(Iterables.partition(uids, IN_CLAUSE_MAX_SIZE)) + .flatMap(queryPublisherFunction); + } + } + + public Mono insert(MailboxMessage mailboxMessage) { + return insert(mailboxMessage, (PostgresMailboxId) mailboxMessage.getMailboxId()); + } + + public Mono insert(MailboxMessage mailboxMessage, PostgresMailboxId mailboxId) { + return postgresExecutor.executeVoid(dslContext -> Mono.from(dslContext.insertInto(TABLE_NAME) + .set(MAILBOX_ID, mailboxId.asUuid()) + .set(MESSAGE_UID, mailboxMessage.getUid().asLong()) + .set(MOD_SEQ, mailboxMessage.getModSeq().asLong()) + .set(MESSAGE_ID, ((PostgresMessageId) mailboxMessage.getMessageId()).asUuid()) + .set(THREAD_ID, ((PostgresMessageId) mailboxMessage.getThreadId().getBaseMessageId()).asUuid()) + .set(INTERNAL_DATE, DATE_TO_LOCAL_DATE_TIME.apply(mailboxMessage.getInternalDate())) + .set(SIZE, mailboxMessage.getFullContentOctets()) + .set(IS_DELETED, mailboxMessage.isDeleted()) + .set(IS_ANSWERED, mailboxMessage.isAnswered()) + .set(IS_DRAFT, mailboxMessage.isDraft()) + .set(IS_FLAGGED, mailboxMessage.isFlagged()) + .set(IS_RECENT, mailboxMessage.isRecent()) + .set(IS_SEEN, mailboxMessage.isSeen()) + .set(USER_FLAGS, mailboxMessage.createFlags().getUserFlags()) + .set(SAVE_DATE, mailboxMessage.getSaveDate().map(DATE_TO_LOCAL_DATE_TIME).orElse(null)))); + } + + public Flux findMailboxes(PostgresMessageId messageId) { + return postgresExecutor.executeRows(dslContext -> Flux.from(dslContext.select(MAILBOX_ID) + .from(TABLE_NAME) + .where(MESSAGE_ID.eq(messageId.asUuid()))), EAGER_FETCH) + .map(record -> PostgresMailboxId.of(record.get(MAILBOX_ID))); + } + + public Flux> findMessagesByMessageIds(Collection messageIds, MessageMapper.FetchType fetchType) { + if (messageIds.isEmpty()) { + return Flux.empty(); + } + PostgresMailboxMessageFetchStrategy fetchStrategy = FETCH_TYPE_TO_FETCH_STRATEGY.apply(fetchType); + + return postgresExecutor.executeRows(dslContext -> Flux.from(dslContext.select(fetchStrategy.fetchFields()) + .from(MESSAGES_JOIN_MAILBOX_MESSAGES_CONDITION_STEP) + .where(DSL.field(TABLE_NAME.getName() + "." + MESSAGE_ID.getName()) + .in(messageIds.stream().map(PostgresMessageId::asUuid).collect(ImmutableList.toImmutableList())))), EAGER_FETCH) + .map(record -> Pair.of(fetchStrategy.toMessageBuilder().apply(record), record)); + } + + public Flux findMetadataByMessageId(PostgresMessageId messageId) { + return postgresExecutor.executeRows(dslContext -> Flux.from(dslContext.select() + .from(TABLE_NAME) + .where(MESSAGE_ID.eq(messageId.asUuid()))), EAGER_FETCH) + .map(RECORD_TO_COMPOSED_MESSAGE_ID_WITH_META_DATA_FUNCTION); + } + + public Flux findMetadataByMessageId(PostgresMessageId messageId, PostgresMailboxId mailboxId) { + return postgresExecutor.executeRows(dslContext -> Flux.from(dslContext.select() + .from(TABLE_NAME) + .where(MESSAGE_ID.eq(messageId.asUuid())) + .and(MAILBOX_ID.eq(mailboxId.asUuid()))), EAGER_FETCH) + .map(RECORD_TO_COMPOSED_MESSAGE_ID_WITH_META_DATA_FUNCTION); + } + +} diff --git a/mailbox/postgres/src/main/java/org/apache/james/mailbox/postgres/mail/dao/PostgresMailboxMessageDAOUtils.java b/mailbox/postgres/src/main/java/org/apache/james/mailbox/postgres/mail/dao/PostgresMailboxMessageDAOUtils.java new file mode 100644 index 00000000000..0649ddb686f --- /dev/null +++ b/mailbox/postgres/src/main/java/org/apache/james/mailbox/postgres/mail/dao/PostgresMailboxMessageDAOUtils.java @@ -0,0 +1,184 @@ +/**************************************************************** + * 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.mailbox.postgres.mail.dao; + +import static org.apache.james.backends.postgres.PostgresCommons.LOCAL_DATE_TIME_DATE_FUNCTION; +import static org.apache.james.mailbox.postgres.mail.PostgresMessageModule.MessageTable.CONTENT_DESCRIPTION; +import static org.apache.james.mailbox.postgres.mail.PostgresMessageModule.MessageTable.CONTENT_DISPOSITION_PARAMETERS; +import static org.apache.james.mailbox.postgres.mail.PostgresMessageModule.MessageTable.CONTENT_DISPOSITION_TYPE; +import static org.apache.james.mailbox.postgres.mail.PostgresMessageModule.MessageTable.CONTENT_ID; +import static org.apache.james.mailbox.postgres.mail.PostgresMessageModule.MessageTable.CONTENT_LANGUAGE; +import static org.apache.james.mailbox.postgres.mail.PostgresMessageModule.MessageTable.CONTENT_LOCATION; +import static org.apache.james.mailbox.postgres.mail.PostgresMessageModule.MessageTable.CONTENT_MD5; +import static org.apache.james.mailbox.postgres.mail.PostgresMessageModule.MessageTable.CONTENT_TRANSFER_ENCODING; +import static org.apache.james.mailbox.postgres.mail.PostgresMessageModule.MessageTable.CONTENT_TYPE_PARAMETERS; +import static org.apache.james.mailbox.postgres.mail.PostgresMessageModule.MessageTable.INTERNAL_DATE; +import static org.apache.james.mailbox.postgres.mail.PostgresMessageModule.MessageTable.SIZE; +import static org.apache.james.mailbox.postgres.mail.PostgresMessageModule.MessageToMailboxTable.IS_ANSWERED; +import static org.apache.james.mailbox.postgres.mail.PostgresMessageModule.MessageToMailboxTable.IS_DELETED; +import static org.apache.james.mailbox.postgres.mail.PostgresMessageModule.MessageToMailboxTable.IS_DRAFT; +import static org.apache.james.mailbox.postgres.mail.PostgresMessageModule.MessageToMailboxTable.IS_FLAGGED; +import static org.apache.james.mailbox.postgres.mail.PostgresMessageModule.MessageToMailboxTable.IS_RECENT; +import static org.apache.james.mailbox.postgres.mail.PostgresMessageModule.MessageToMailboxTable.IS_SEEN; +import static org.apache.james.mailbox.postgres.mail.PostgresMessageModule.MessageToMailboxTable.MAILBOX_ID; +import static org.apache.james.mailbox.postgres.mail.PostgresMessageModule.MessageToMailboxTable.MESSAGE_ID; +import static org.apache.james.mailbox.postgres.mail.PostgresMessageModule.MessageToMailboxTable.MESSAGE_UID; +import static org.apache.james.mailbox.postgres.mail.PostgresMessageModule.MessageToMailboxTable.MOD_SEQ; +import static org.apache.james.mailbox.postgres.mail.PostgresMessageModule.MessageToMailboxTable.SAVE_DATE; +import static org.apache.james.mailbox.postgres.mail.PostgresMessageModule.MessageToMailboxTable.THREAD_ID; +import static org.apache.james.mailbox.postgres.mail.PostgresMessageModule.MessageToMailboxTable.USER_FLAGS; + +import java.io.ByteArrayInputStream; +import java.io.InputStream; +import java.util.Arrays; +import java.util.List; +import java.util.Map; +import java.util.Optional; +import java.util.function.Function; + +import jakarta.mail.Flags; + +import org.apache.james.mailbox.MessageUid; +import org.apache.james.mailbox.ModSeq; +import org.apache.james.mailbox.model.ComposedMessageId; +import org.apache.james.mailbox.model.ComposedMessageIdWithMetaData; +import org.apache.james.mailbox.model.Content; +import org.apache.james.mailbox.model.MessageMetaData; +import org.apache.james.mailbox.model.ThreadId; +import org.apache.james.mailbox.postgres.PostgresMailboxId; +import org.apache.james.mailbox.postgres.PostgresMessageId; +import org.apache.james.mailbox.postgres.mail.PostgresMessageModule; +import org.apache.james.mailbox.store.mail.MessageMapper; +import org.apache.james.mailbox.store.mail.model.impl.Properties; +import org.apache.james.mailbox.store.mail.model.impl.PropertyBuilder; +import org.jooq.Field; +import org.jooq.Record; + +interface PostgresMailboxMessageDAOUtils { + Map, Flags.Flag> BOOLEAN_FLAGS_MAPPING = Map.of( + IS_ANSWERED, Flags.Flag.ANSWERED, + IS_DELETED, Flags.Flag.DELETED, + IS_DRAFT, Flags.Flag.DRAFT, + IS_FLAGGED, Flags.Flag.FLAGGED, + IS_RECENT, Flags.Flag.RECENT, + IS_SEEN, Flags.Flag.SEEN); + Function RECORD_TO_MESSAGE_UID_FUNCTION = record -> MessageUid.of(record.get(MESSAGE_UID)); + Function RECORD_TO_FLAGS_FUNCTION = record -> { + Flags flags = new Flags(); + BOOLEAN_FLAGS_MAPPING.forEach((flagColumn, flagMapped) -> { + if (record.get(flagColumn)) { + flags.add(flagMapped); + } + }); + + Optional.ofNullable(record.get(USER_FLAGS)).stream() + .flatMap(Arrays::stream) + .forEach(flags::add); + return flags; + }; + + Function RECORD_TO_THREAD_ID_FUNCTION = record -> Optional.ofNullable(record.get(THREAD_ID)) + .map(threadIdAsUuid -> ThreadId.fromBaseMessageId(PostgresMessageId.Factory.of(threadIdAsUuid))) + .orElse(ThreadId.fromBaseMessageId(PostgresMessageId.Factory.of(record.get(MESSAGE_ID)))); + + + Field[] MESSAGE_METADATA_FIELDS_REQUIRE = new Field[]{ + MESSAGE_UID, + MOD_SEQ, + SIZE, + INTERNAL_DATE, + SAVE_DATE, + MESSAGE_ID, + THREAD_ID, + IS_ANSWERED, + IS_DELETED, + IS_DRAFT, + IS_FLAGGED, + IS_RECENT, + IS_SEEN, + USER_FLAGS + }; + + Function RECORD_TO_MESSAGE_METADATA_FUNCTION = record -> + new MessageMetaData(MessageUid.of(record.get(MESSAGE_UID)), + ModSeq.of(record.get(MOD_SEQ)), + RECORD_TO_FLAGS_FUNCTION.apply(record), + record.get(SIZE), + LOCAL_DATE_TIME_DATE_FUNCTION.apply(record.get(INTERNAL_DATE)), + Optional.ofNullable(record.get(SAVE_DATE)).map(LOCAL_DATE_TIME_DATE_FUNCTION), + PostgresMessageId.Factory.of(record.get(MESSAGE_ID)), + RECORD_TO_THREAD_ID_FUNCTION.apply(record)); + + Function RECORD_TO_COMPOSED_MESSAGE_ID_WITH_META_DATA_FUNCTION = record -> ComposedMessageIdWithMetaData + .builder() + .composedMessageId(new ComposedMessageId(PostgresMailboxId.of(record.get(MAILBOX_ID)), + PostgresMessageId.Factory.of(record.get(MESSAGE_ID)), + MessageUid.of(record.get(MESSAGE_UID)))) + .threadId(RECORD_TO_THREAD_ID_FUNCTION.apply(record)) + .flags(RECORD_TO_FLAGS_FUNCTION.apply(record)) + .modSeq(ModSeq.of(record.get(MOD_SEQ))) + .build(); + + Function RECORD_TO_PROPERTIES_FUNCTION = record -> { + PropertyBuilder property = new PropertyBuilder(); + + property.setMediaType(record.get(PostgresMessageModule.MessageTable.MIME_TYPE)); + property.setSubType(record.get(PostgresMessageModule.MessageTable.MIME_SUBTYPE)); + property.setTextualLineCount(Optional.ofNullable(record.get(PostgresMessageModule.MessageTable.TEXTUAL_LINE_COUNT)) + .map(Long::valueOf) + .orElse(null)); + + property.setContentDescription(record.get(CONTENT_DESCRIPTION)); + property.setContentDispositionType(record.get(CONTENT_DISPOSITION_TYPE)); + property.setContentID(record.get(CONTENT_ID)); + property.setContentMD5(record.get(CONTENT_MD5)); + property.setContentTransferEncoding(record.get(CONTENT_TRANSFER_ENCODING)); + property.setContentLocation(record.get(CONTENT_LOCATION)); + property.setContentLanguage(Optional.ofNullable(record.get(CONTENT_LANGUAGE)).map(List::of).orElse(null)); + property.setContentDispositionParameters(record.get(CONTENT_DISPOSITION_PARAMETERS).data()); + property.setContentTypeParameters(record.get(CONTENT_TYPE_PARAMETERS).data()); + return property.build(); + }; + + Function BYTE_TO_CONTENT_FUNCTION = contentAsBytes -> new Content() { + @Override + public InputStream getInputStream() { + return new ByteArrayInputStream(contentAsBytes); + } + + @Override + public long size() { + return contentAsBytes.length; + } + }; + + Function FETCH_TYPE_TO_FETCH_STRATEGY = fetchType -> { + switch (fetchType) { + case METADATA: + case ATTACHMENTS_METADATA: + return PostgresMailboxMessageFetchStrategy.METADATA; + case HEADERS: + case FULL: + return PostgresMailboxMessageFetchStrategy.FULL; + default: + throw new RuntimeException("Unknown FetchType " + fetchType); + } + }; +} diff --git a/mailbox/postgres/src/main/java/org/apache/james/mailbox/postgres/mail/dao/PostgresMailboxMessageFetchStrategy.java b/mailbox/postgres/src/main/java/org/apache/james/mailbox/postgres/mail/dao/PostgresMailboxMessageFetchStrategy.java new file mode 100644 index 00000000000..eb2049d4575 --- /dev/null +++ b/mailbox/postgres/src/main/java/org/apache/james/mailbox/postgres/mail/dao/PostgresMailboxMessageFetchStrategy.java @@ -0,0 +1,148 @@ +/**************************************************************** + * 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.mailbox.postgres.mail.dao; + +import static org.apache.james.backends.postgres.PostgresCommons.LOCAL_DATE_TIME_DATE_FUNCTION; +import static org.apache.james.backends.postgres.PostgresCommons.tableField; +import static org.apache.james.mailbox.postgres.mail.PostgresMessageModule.MessageTable; +import static org.apache.james.mailbox.postgres.mail.PostgresMessageModule.MessageTable.BODY_START_OCTET; +import static org.apache.james.mailbox.postgres.mail.PostgresMessageModule.MessageTable.HEADER_CONTENT; +import static org.apache.james.mailbox.postgres.mail.PostgresMessageModule.MessageToMailboxTable.MAILBOX_ID; +import static org.apache.james.mailbox.postgres.mail.PostgresMessageModule.MessageToMailboxTable.MESSAGE_UID; +import static org.apache.james.mailbox.postgres.mail.PostgresMessageModule.MessageToMailboxTable.MOD_SEQ; +import static org.apache.james.mailbox.postgres.mail.PostgresMessageModule.MessageToMailboxTable.SAVE_DATE; +import static org.apache.james.mailbox.postgres.mail.dao.PostgresMailboxMessageDAOUtils.BYTE_TO_CONTENT_FUNCTION; +import static org.apache.james.mailbox.postgres.mail.dao.PostgresMailboxMessageDAOUtils.RECORD_TO_FLAGS_FUNCTION; +import static org.apache.james.mailbox.postgres.mail.dao.PostgresMailboxMessageDAOUtils.RECORD_TO_PROPERTIES_FUNCTION; +import static org.apache.james.mailbox.postgres.mail.dao.PostgresMailboxMessageDAOUtils.RECORD_TO_THREAD_ID_FUNCTION; + +import java.time.LocalDateTime; +import java.util.function.Function; + +import org.apache.commons.lang3.ArrayUtils; +import org.apache.james.mailbox.MessageUid; +import org.apache.james.mailbox.ModSeq; +import org.apache.james.mailbox.model.Content; +import org.apache.james.mailbox.postgres.PostgresMailboxId; +import org.apache.james.mailbox.postgres.PostgresMessageId; +import org.apache.james.mailbox.postgres.mail.PostgresMessageModule; +import org.apache.james.mailbox.postgres.mail.PostgresMessageModule.MessageToMailboxTable; +import org.apache.james.mailbox.store.mail.model.impl.PropertyBuilder; +import org.apache.james.mailbox.store.mail.model.impl.SimpleMailboxMessage; +import org.jooq.Field; +import org.jooq.Record; + +public interface PostgresMailboxMessageFetchStrategy { + PostgresMailboxMessageFetchStrategy METADATA = new MetaData(); + PostgresMailboxMessageFetchStrategy FULL = new Full(); + + Field[] fetchFields(); + + Function toMessageBuilder(); + + static Function toMessageBuilderMetadata() { + return record -> SimpleMailboxMessage.builder() + .messageId(PostgresMessageId.Factory.of(record.get(MessageTable.MESSAGE_ID))) + .mailboxId(PostgresMailboxId.of(record.get(MAILBOX_ID))) + .uid(MessageUid.of(record.get(MESSAGE_UID))) + .modseq(ModSeq.of(record.get(MOD_SEQ))) + .threadId(RECORD_TO_THREAD_ID_FUNCTION.apply(record)) + .internalDate(LOCAL_DATE_TIME_DATE_FUNCTION.apply(record.get(PostgresMessageModule.MessageTable.INTERNAL_DATE, LocalDateTime.class))) + .saveDate(LOCAL_DATE_TIME_DATE_FUNCTION.apply(record.get(SAVE_DATE, LocalDateTime.class))) + .flags(RECORD_TO_FLAGS_FUNCTION.apply(record)) + .size(record.get(PostgresMessageModule.MessageTable.SIZE)) + .bodyStartOctet(record.get(BODY_START_OCTET)); + } + + static Field[] fetchFieldsMetadata() { + return new Field[]{ + tableField(MessageTable.TABLE_NAME, MessageTable.MESSAGE_ID).as(MessageTable.MESSAGE_ID), + tableField(MessageTable.TABLE_NAME, MessageTable.INTERNAL_DATE).as(MessageTable.INTERNAL_DATE), + tableField(MessageTable.TABLE_NAME, MessageTable.SIZE).as(MessageTable.SIZE), + MessageTable.BODY_BLOB_ID, + MessageTable.MIME_TYPE, + MessageTable.MIME_SUBTYPE, + MessageTable.BODY_START_OCTET, + MessageTable.TEXTUAL_LINE_COUNT, + MessageTable.ATTACHMENT_METADATA, + MessageToMailboxTable.MAILBOX_ID, + MessageToMailboxTable.MESSAGE_UID, + MessageToMailboxTable.MOD_SEQ, + MessageToMailboxTable.THREAD_ID, + MessageToMailboxTable.IS_DELETED, + MessageToMailboxTable.IS_ANSWERED, + MessageToMailboxTable.IS_DRAFT, + MessageToMailboxTable.IS_FLAGGED, + MessageToMailboxTable.IS_RECENT, + MessageToMailboxTable.IS_SEEN, + MessageToMailboxTable.USER_FLAGS, + MessageToMailboxTable.SAVE_DATE}; + } + + class MetaData implements PostgresMailboxMessageFetchStrategy { + public static final Field[] FETCH_FIELDS = fetchFieldsMetadata(); + public static final Content EMPTY_CONTENT = BYTE_TO_CONTENT_FUNCTION.apply(new byte[0]); + public static final PropertyBuilder EMPTY_PROPERTY_BUILDER = new PropertyBuilder(); + + + @Override + public Field[] fetchFields() { + return FETCH_FIELDS; + } + + @Override + public Function toMessageBuilder() { + return record -> toMessageBuilderMetadata() + .apply(record) + .content(EMPTY_CONTENT) + .properties(EMPTY_PROPERTY_BUILDER); + } + } + + class Full implements PostgresMailboxMessageFetchStrategy { + + public static final Field[] FETCH_FIELDS = ArrayUtils.addAll(fetchFieldsMetadata(), + MessageTable.HEADER_CONTENT, + MessageTable.TEXTUAL_LINE_COUNT, + MessageTable.CONTENT_DESCRIPTION, + MessageTable.CONTENT_LOCATION, + MessageTable.CONTENT_TRANSFER_ENCODING, + MessageTable.CONTENT_DISPOSITION_TYPE, + MessageTable.CONTENT_ID, + MessageTable.CONTENT_MD5, + MessageTable.CONTENT_LANGUAGE, + MessageTable.CONTENT_TYPE_PARAMETERS, + MessageTable.CONTENT_DISPOSITION_PARAMETERS); + + @Override + public Field[] fetchFields() { + return FETCH_FIELDS; + } + + @Override + public Function toMessageBuilder() { + return record -> toMessageBuilderMetadata() + .apply(record) + .content(BYTE_TO_CONTENT_FUNCTION.apply(record.get(HEADER_CONTENT))) + .properties(RECORD_TO_PROPERTIES_FUNCTION.apply(record)); + } + } + +} diff --git a/mailbox/postgres/src/main/java/org/apache/james/mailbox/postgres/mail/dao/PostgresMessageDAO.java b/mailbox/postgres/src/main/java/org/apache/james/mailbox/postgres/mail/dao/PostgresMessageDAO.java new file mode 100644 index 00000000000..e85373e1d28 --- /dev/null +++ b/mailbox/postgres/src/main/java/org/apache/james/mailbox/postgres/mail/dao/PostgresMessageDAO.java @@ -0,0 +1,160 @@ +/**************************************************************** + * 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.mailbox.postgres.mail.dao; + +import static org.apache.james.backends.postgres.PostgresCommons.DATE_TO_LOCAL_DATE_TIME; +import static org.apache.james.backends.postgres.PostgresCommons.LOCAL_DATE_TIME_DATE_FUNCTION; +import static org.apache.james.mailbox.postgres.mail.PostgresMessageModule.MessageTable.ATTACHMENT_METADATA; +import static org.apache.james.mailbox.postgres.mail.PostgresMessageModule.MessageTable.BODY_BLOB_ID; +import static org.apache.james.mailbox.postgres.mail.PostgresMessageModule.MessageTable.BODY_START_OCTET; +import static org.apache.james.mailbox.postgres.mail.PostgresMessageModule.MessageTable.CONTENT_DESCRIPTION; +import static org.apache.james.mailbox.postgres.mail.PostgresMessageModule.MessageTable.CONTENT_DISPOSITION_PARAMETERS; +import static org.apache.james.mailbox.postgres.mail.PostgresMessageModule.MessageTable.CONTENT_DISPOSITION_TYPE; +import static org.apache.james.mailbox.postgres.mail.PostgresMessageModule.MessageTable.CONTENT_ID; +import static org.apache.james.mailbox.postgres.mail.PostgresMessageModule.MessageTable.CONTENT_LANGUAGE; +import static org.apache.james.mailbox.postgres.mail.PostgresMessageModule.MessageTable.CONTENT_LOCATION; +import static org.apache.james.mailbox.postgres.mail.PostgresMessageModule.MessageTable.CONTENT_MD5; +import static org.apache.james.mailbox.postgres.mail.PostgresMessageModule.MessageTable.CONTENT_TRANSFER_ENCODING; +import static org.apache.james.mailbox.postgres.mail.PostgresMessageModule.MessageTable.CONTENT_TYPE_PARAMETERS; +import static org.apache.james.mailbox.postgres.mail.PostgresMessageModule.MessageTable.HEADER_CONTENT; +import static org.apache.james.mailbox.postgres.mail.PostgresMessageModule.MessageTable.INTERNAL_DATE; +import static org.apache.james.mailbox.postgres.mail.PostgresMessageModule.MessageTable.MESSAGE_ID; +import static org.apache.james.mailbox.postgres.mail.PostgresMessageModule.MessageTable.MIME_SUBTYPE; +import static org.apache.james.mailbox.postgres.mail.PostgresMessageModule.MessageTable.MIME_TYPE; +import static org.apache.james.mailbox.postgres.mail.PostgresMessageModule.MessageTable.SIZE; +import static org.apache.james.mailbox.postgres.mail.PostgresMessageModule.MessageTable.TABLE_NAME; +import static org.apache.james.mailbox.postgres.mail.PostgresMessageModule.MessageTable.TEXTUAL_LINE_COUNT; +import static org.apache.james.mailbox.postgres.mail.dao.PostgresMailboxMessageDAOUtils.BYTE_TO_CONTENT_FUNCTION; + +import java.time.LocalDateTime; +import java.util.Optional; + +import jakarta.inject.Inject; +import jakarta.inject.Named; +import jakarta.inject.Singleton; + +import org.apache.commons.io.IOUtils; +import org.apache.james.backends.postgres.utils.PostgresExecutor; +import org.apache.james.blob.api.BlobId; +import org.apache.james.core.Domain; +import org.apache.james.mailbox.model.MessageId; +import org.apache.james.mailbox.postgres.PostgresMessageId; +import org.apache.james.mailbox.postgres.mail.MessageRepresentation; +import org.apache.james.mailbox.postgres.mail.PostgresMessageModule; +import org.apache.james.mailbox.postgres.mail.dto.AttachmentsDTO; +import org.apache.james.mailbox.store.mail.model.MailboxMessage; +import org.jooq.Record; +import org.jooq.postgres.extensions.types.Hstore; + +import reactor.core.publisher.Flux; +import reactor.core.publisher.Mono; +import reactor.core.scheduler.Schedulers; + +public class PostgresMessageDAO { + + public static class Factory { + private final BlobId.Factory blobIdFactory; + private final PostgresExecutor.Factory executorFactory; + + @Inject + @Singleton + public Factory(BlobId.Factory blobIdFactory, PostgresExecutor.Factory executorFactory) { + this.blobIdFactory = blobIdFactory; + this.executorFactory = executorFactory; + } + + public PostgresMessageDAO create(Optional domain) { + return new PostgresMessageDAO(executorFactory.create(domain), blobIdFactory); + } + } + + public static final long DEFAULT_LONG_VALUE = 0L; + private final PostgresExecutor postgresExecutor; + private final BlobId.Factory blobIdFactory; + + @Inject + public PostgresMessageDAO(@Named(PostgresExecutor.BY_PASS_RLS_INJECT) PostgresExecutor postgresExecutor, BlobId.Factory blobIdFactory) { + this.postgresExecutor = postgresExecutor; + this.blobIdFactory = blobIdFactory; + } + + public Mono insert(MailboxMessage message, String bodyBlobId) { + return Mono.fromCallable(() -> IOUtils.toByteArray(message.getHeaderContent(), message.getHeaderOctets())) + .subscribeOn(Schedulers.boundedElastic()) + .flatMap(headerContentAsByte -> postgresExecutor.executeVoid(dslContext -> Mono.from(dslContext.insertInto(TABLE_NAME) + .set(MESSAGE_ID, ((PostgresMessageId) message.getMessageId()).asUuid()) + .set(BODY_BLOB_ID, bodyBlobId) + .set(MIME_TYPE, message.getMediaType()) + .set(MIME_SUBTYPE, message.getSubType()) + .set(INTERNAL_DATE, DATE_TO_LOCAL_DATE_TIME.apply(message.getInternalDate())) + .set(SIZE, message.getFullContentOctets()) + .set(BODY_START_OCTET, (int) (message.getFullContentOctets() - message.getBodyOctets())) + .set(TEXTUAL_LINE_COUNT, Optional.ofNullable(message.getTextualLineCount()).orElse(DEFAULT_LONG_VALUE).intValue()) + .set(CONTENT_DESCRIPTION, message.getProperties().getContentDescription()) + .set(CONTENT_DISPOSITION_TYPE, message.getProperties().getContentDispositionType()) + .set(CONTENT_ID, message.getProperties().getContentID()) + .set(CONTENT_MD5, message.getProperties().getContentMD5()) + .set(CONTENT_LANGUAGE, message.getProperties().getContentLanguage().toArray(new String[0])) + .set(CONTENT_LOCATION, message.getProperties().getContentLocation()) + .set(CONTENT_TRANSFER_ENCODING, message.getProperties().getContentTransferEncoding()) + .set(CONTENT_TYPE_PARAMETERS, Hstore.hstore(message.getProperties().getContentTypeParameters())) + .set(CONTENT_DISPOSITION_PARAMETERS, Hstore.hstore(message.getProperties().getContentDispositionParameters())) + .set(ATTACHMENT_METADATA, AttachmentsDTO.from(message.getAttachments())) + .set(HEADER_CONTENT, headerContentAsByte)))); + } + + public Mono retrieveMessage(PostgresMessageId messageId) { + return postgresExecutor.executeRow(dslContext -> Mono.from(dslContext.select( + INTERNAL_DATE, SIZE, BODY_START_OCTET, HEADER_CONTENT, BODY_BLOB_ID, ATTACHMENT_METADATA) + .from(TABLE_NAME) + .where(MESSAGE_ID.eq(messageId.asUuid())))) + .map(record -> toMessageRepresentation(record, messageId)); + } + + private MessageRepresentation toMessageRepresentation(Record record, MessageId messageId) { + return MessageRepresentation.builder() + .messageId(messageId) + .internalDate(LOCAL_DATE_TIME_DATE_FUNCTION.apply(record.get(PostgresMessageModule.MessageTable.INTERNAL_DATE, LocalDateTime.class))) + .size(record.get(PostgresMessageModule.MessageTable.SIZE)) + .headerContent(BYTE_TO_CONTENT_FUNCTION.apply(record.get(HEADER_CONTENT))) + .bodyBlobId(blobIdFactory.parse(record.get(BODY_BLOB_ID))) + .attachments(record.get(ATTACHMENT_METADATA)) + .build(); + } + + public Mono deleteByMessageId(PostgresMessageId messageId) { + return postgresExecutor.executeVoid(dslContext -> Mono.from(dslContext.deleteFrom(TABLE_NAME) + .where(MESSAGE_ID.eq(messageId.asUuid())))); + } + + public Mono getBodyBlobId(PostgresMessageId messageId) { + return postgresExecutor.executeRow(dslContext -> Mono.from(dslContext.select(BODY_BLOB_ID) + .from(TABLE_NAME) + .where(MESSAGE_ID.eq(messageId.asUuid())))) + .map(record -> blobIdFactory.parse(record.get(BODY_BLOB_ID))); + } + + public Flux listBlobs() { + return postgresExecutor.executeRows(dslContext -> Flux.from(dslContext.select(BODY_BLOB_ID) + .from(TABLE_NAME))) + .map(record -> blobIdFactory.parse(record.get(BODY_BLOB_ID))); + } + +} diff --git a/mailbox/postgres/src/main/java/org/apache/james/mailbox/postgres/mail/dao/PostgresThreadDAO.java b/mailbox/postgres/src/main/java/org/apache/james/mailbox/postgres/mail/dao/PostgresThreadDAO.java new file mode 100644 index 00000000000..e561dc61941 --- /dev/null +++ b/mailbox/postgres/src/main/java/org/apache/james/mailbox/postgres/mail/dao/PostgresThreadDAO.java @@ -0,0 +1,123 @@ +/**************************************************************** + * 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.mailbox.postgres.mail.dao; + +import static org.apache.james.backends.postgres.PostgresCommons.IN_CLAUSE_MAX_SIZE; +import static org.apache.james.mailbox.postgres.mail.dao.PostgresThreadModule.PostgresThreadTable.HASH_BASE_SUBJECT; +import static org.apache.james.mailbox.postgres.mail.dao.PostgresThreadModule.PostgresThreadTable.HASH_MIME_MESSAGE_ID; +import static org.apache.james.mailbox.postgres.mail.dao.PostgresThreadModule.PostgresThreadTable.MESSAGE_ID; +import static org.apache.james.mailbox.postgres.mail.dao.PostgresThreadModule.PostgresThreadTable.TABLE_NAME; +import static org.apache.james.mailbox.postgres.mail.dao.PostgresThreadModule.PostgresThreadTable.THREAD_ID; +import static org.apache.james.mailbox.postgres.mail.dao.PostgresThreadModule.PostgresThreadTable.USERNAME; + +import java.util.Collection; +import java.util.Optional; +import java.util.Set; +import java.util.function.Function; + +import jakarta.inject.Inject; +import jakarta.inject.Singleton; + +import org.apache.commons.lang3.tuple.Pair; +import org.apache.james.backends.postgres.utils.PostgresExecutor; +import org.apache.james.core.Domain; +import org.apache.james.core.Username; +import org.apache.james.mailbox.model.MessageId; +import org.apache.james.mailbox.model.ThreadId; +import org.apache.james.mailbox.postgres.PostgresMessageId; +import org.jooq.Record; + +import com.google.common.collect.ImmutableList; +import com.google.common.collect.Iterables; + +import reactor.core.publisher.Flux; +import reactor.core.publisher.Mono; + +public class PostgresThreadDAO { + public static class Factory { + private final PostgresExecutor.Factory executorFactory; + + @Inject + @Singleton + public Factory(PostgresExecutor.Factory executorFactory) { + this.executorFactory = executorFactory; + } + + public PostgresThreadDAO create(Optional domain) { + return new PostgresThreadDAO(executorFactory.create(domain)); + } + } + + private final PostgresExecutor postgresExecutor; + + public PostgresThreadDAO(PostgresExecutor postgresExecutor) { + this.postgresExecutor = postgresExecutor; + } + + public Mono insertSome(Username username, Set hashMimeMessageIds, PostgresMessageId messageId, ThreadId threadId, Optional hashBaseSubject) { + return postgresExecutor.executeVoid(dslContext -> Mono.from(dslContext.batch( + hashMimeMessageIds.stream().map(hashMimeMessageId -> dslContext.insertInto(TABLE_NAME) + .set(USERNAME, username.asString()) + .set(HASH_MIME_MESSAGE_ID, hashMimeMessageId) + .set(MESSAGE_ID, messageId.asUuid()) + .set(THREAD_ID, ((PostgresMessageId) threadId.getBaseMessageId()).asUuid()) + .set(HASH_BASE_SUBJECT, hashBaseSubject.orElse(null))) + .collect(ImmutableList.toImmutableList())))); + } + + public Flux, ThreadId>> findThreads(Username username, Set hashMimeMessageIds) { + if (hashMimeMessageIds.isEmpty()) { + return Flux.empty(); + } + Function, Flux, ThreadId>>> function = hashMimeMessageIdSubSet -> + postgresExecutor.executeRows(dslContext -> Flux.from(dslContext.select(THREAD_ID, HASH_BASE_SUBJECT) + .from(TABLE_NAME) + .where(USERNAME.eq(username.asString())) + .and(HASH_MIME_MESSAGE_ID.in(hashMimeMessageIdSubSet)))) + .map(this::readRecord); + + if (hashMimeMessageIds.size() <= IN_CLAUSE_MAX_SIZE) { + return function.apply(hashMimeMessageIds); + } else { + return Flux.fromIterable(Iterables.partition(hashMimeMessageIds, IN_CLAUSE_MAX_SIZE)) + .flatMap(function); + } + } + + public Flux findMessageIds(ThreadId threadId, Username username) { + return postgresExecutor.executeRows(dslContext -> Flux.from(dslContext.selectDistinct(MESSAGE_ID) + .from(TABLE_NAME) + .where(USERNAME.eq(username.asString())) + .and(THREAD_ID.eq(PostgresMessageId.class.cast(threadId.getBaseMessageId()).asUuid())) + .orderBy(MESSAGE_ID))) + .map(record -> PostgresMessageId.Factory.of(record.get(MESSAGE_ID))); + } + + public Pair, ThreadId> readRecord(Record record) { + return Pair.of(Optional.ofNullable(record.get(HASH_BASE_SUBJECT)), + ThreadId.fromBaseMessageId(PostgresMessageId.Factory.of(record.get(THREAD_ID)))); + } + + public Mono deleteSome(Username username, PostgresMessageId messageId) { + return postgresExecutor.executeVoid(dslContext -> Mono.from(dslContext.deleteFrom(TABLE_NAME) + .where(USERNAME.eq(username.asString())) + .and(MESSAGE_ID.eq(messageId.asUuid())))); + } +} diff --git a/mailbox/postgres/src/main/java/org/apache/james/mailbox/postgres/mail/dao/PostgresThreadModule.java b/mailbox/postgres/src/main/java/org/apache/james/mailbox/postgres/mail/dao/PostgresThreadModule.java new file mode 100644 index 00000000000..046db43c82e --- /dev/null +++ b/mailbox/postgres/src/main/java/org/apache/james/mailbox/postgres/mail/dao/PostgresThreadModule.java @@ -0,0 +1,72 @@ +/**************************************************************** + * 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.mailbox.postgres.mail.dao; + +import static org.apache.james.mailbox.postgres.mail.dao.PostgresThreadModule.PostgresThreadTable.MESSAGE_ID_INDEX; +import static org.apache.james.mailbox.postgres.mail.dao.PostgresThreadModule.PostgresThreadTable.TABLE; +import static org.apache.james.mailbox.postgres.mail.dao.PostgresThreadModule.PostgresThreadTable.THREAD_ID_INDEX; + +import java.util.UUID; + +import org.apache.james.backends.postgres.PostgresIndex; +import org.apache.james.backends.postgres.PostgresModule; +import org.apache.james.backends.postgres.PostgresTable; +import org.jooq.Field; +import org.jooq.Record; +import org.jooq.Table; +import org.jooq.impl.DSL; +import org.jooq.impl.SQLDataType; + +public interface PostgresThreadModule { + interface PostgresThreadTable { + Table TABLE_NAME = DSL.table("thread"); + + Field USERNAME = DSL.field("username", SQLDataType.VARCHAR(255).notNull()); + Field HASH_MIME_MESSAGE_ID = DSL.field("hash_mime_message_id", SQLDataType.INTEGER.notNull()); + Field MESSAGE_ID = DSL.field("message_id", SQLDataType.UUID.notNull()); + Field THREAD_ID = DSL.field("thread_id", SQLDataType.UUID.notNull()); + Field HASH_BASE_SUBJECT = DSL.field("hash_base_subject", SQLDataType.INTEGER); + + PostgresTable TABLE = PostgresTable.name(TABLE_NAME.getName()) + .createTableStep(((dsl, tableName) -> dsl.createTableIfNotExists(tableName) + .column(USERNAME) + .column(HASH_MIME_MESSAGE_ID) + .column(MESSAGE_ID) + .column(THREAD_ID) + .column(HASH_BASE_SUBJECT) + .constraint(DSL.primaryKey(USERNAME, HASH_MIME_MESSAGE_ID, MESSAGE_ID)))) + .supportsRowLevelSecurity() + .build(); + + PostgresIndex MESSAGE_ID_INDEX = PostgresIndex.name("thread_message_id_index") + .createIndexStep((dsl, indexName) -> dsl.createIndexIfNotExists(indexName) + .on(TABLE_NAME, USERNAME, MESSAGE_ID)); + + PostgresIndex THREAD_ID_INDEX = PostgresIndex.name("thread_thread_id_index") + .createIndexStep((dsl, indexName) -> dsl.createIndexIfNotExists(indexName) + .on(TABLE_NAME, USERNAME, THREAD_ID)); + } + + PostgresModule MODULE = PostgresModule.builder() + .addTable(TABLE) + .addIndex(MESSAGE_ID_INDEX) + .addIndex(THREAD_ID_INDEX) + .build(); +} diff --git a/mailbox/postgres/src/main/java/org/apache/james/mailbox/postgres/mail/dto/AttachmentsDTO.java b/mailbox/postgres/src/main/java/org/apache/james/mailbox/postgres/mail/dto/AttachmentsDTO.java new file mode 100644 index 00000000000..a54f0e09727 --- /dev/null +++ b/mailbox/postgres/src/main/java/org/apache/james/mailbox/postgres/mail/dto/AttachmentsDTO.java @@ -0,0 +1,141 @@ +/**************************************************************** + * 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.mailbox.postgres.mail.dto; + +import java.io.Serializable; +import java.sql.SQLException; +import java.util.ArrayList; +import java.util.Collection; +import java.util.List; +import java.util.Map; +import java.util.Optional; +import java.util.stream.Collectors; +import java.util.stream.StreamSupport; + +import org.apache.james.mailbox.model.AttachmentId; +import org.apache.james.mailbox.model.Cid; +import org.apache.james.mailbox.model.MessageAttachmentMetadata; +import org.apache.james.mailbox.model.UuidBackedAttachmentId; +import org.apache.james.mailbox.postgres.mail.MessageRepresentation; +import org.jooq.BindingGetResultSetContext; +import org.jooq.BindingSetStatementContext; +import org.jooq.Converter; +import org.jooq.impl.AbstractConverter; +import org.jooq.postgres.extensions.bindings.AbstractPostgresBinding; + +import com.fasterxml.jackson.core.JsonProcessingException; +import com.fasterxml.jackson.databind.JsonNode; +import com.fasterxml.jackson.databind.ObjectMapper; +import com.fasterxml.jackson.datatype.jdk8.Jdk8Module; + +import io.r2dbc.postgresql.codec.Json; + +public class AttachmentsDTO extends ArrayList implements Serializable { + + public static class AttachmentsDTOConverter extends AbstractConverter { + private static final long serialVersionUID = 1L; + private static final String ATTACHMENT_ID_PROPERTY = "attachment_id"; + private static final String NAME_PROPERTY = "name"; + private static final String CID_PROPERTY = "cid"; + private static final String IN_LINE_PROPERTY = "in_line"; + private final ObjectMapper objectMapper; + + public AttachmentsDTOConverter() { + super(Object.class, AttachmentsDTO.class); + this.objectMapper = new ObjectMapper(); + this.objectMapper.registerModule(new Jdk8Module()); + } + + @Override + public AttachmentsDTO from(Object databaseObject) { + if (databaseObject instanceof Json) { + try { + JsonNode arrayNode = objectMapper.readTree(((Json) databaseObject).asArray()); + List collect = StreamSupport.stream(arrayNode.spliterator(), false) + .map(this::fromJsonNode) + .collect(Collectors.toList()); + return new AttachmentsDTO(collect); + } catch (Exception e) { + throw new RuntimeException("Error while deserializing attachment representation", e); + } + } + throw new RuntimeException("Error while deserializing attachment representation. Unknown type: " + databaseObject.getClass().getName()); + } + + @Override + public Object to(AttachmentsDTO userObject) { + try { + byte[] jsonAsByte = objectMapper.writeValueAsBytes(userObject + .stream().map(attachment -> Map.of( + ATTACHMENT_ID_PROPERTY, attachment.getAttachmentId().getId(), + NAME_PROPERTY, attachment.getName(), + CID_PROPERTY, attachment.getCid().map(Cid::getValue), + IN_LINE_PROPERTY, attachment.isInline())).collect(Collectors.toList())); + return Json.of(jsonAsByte); + } catch (JsonProcessingException e) { + throw new RuntimeException(e); + } + } + + private MessageRepresentation.AttachmentRepresentation fromJsonNode(JsonNode jsonNode) { + AttachmentId attachmentId = UuidBackedAttachmentId.from(jsonNode.get(ATTACHMENT_ID_PROPERTY).asText()); + Optional name = Optional.ofNullable(jsonNode.get(NAME_PROPERTY)).map(JsonNode::asText); + Optional cid = Optional.ofNullable(jsonNode.get(CID_PROPERTY)).map(JsonNode::asText).map(Cid::from); + boolean isInline = jsonNode.get(IN_LINE_PROPERTY).asBoolean(); + + return new MessageRepresentation.AttachmentRepresentation(attachmentId, name, cid, isInline); + } + } + + public static class AttachmentsDTOBinding extends AbstractPostgresBinding { + private static final long serialVersionUID = 1L; + private static final Converter CONVERTER = new AttachmentsDTOConverter(); + + @Override + public Converter converter() { + return CONVERTER; + } + + @Override + public void set(final BindingSetStatementContext ctx) throws SQLException { + Object value = ctx.convert(converter()).value(); + + ctx.statement().setObject(ctx.index(), value == null ? null : value); + } + + + @Override + public void get(final BindingGetResultSetContext ctx) throws SQLException { + ctx.convert(converter()).value((Json) ctx.resultSet().getObject(ctx.index())); + } + } + + public static AttachmentsDTO from(List messageAttachmentMetadata) { + return new AttachmentsDTO(MessageRepresentation.AttachmentRepresentation.from(messageAttachmentMetadata)); + } + + private static final long serialVersionUID = 1L; + + public AttachmentsDTO(Collection c) { + super(c); + } + + +} diff --git a/mailbox/postgres/src/main/java/org/apache/james/mailbox/postgres/mail/eventsourcing/acl/ACLModule.java b/mailbox/postgres/src/main/java/org/apache/james/mailbox/postgres/mail/eventsourcing/acl/ACLModule.java new file mode 100644 index 00000000000..7c99c08392e --- /dev/null +++ b/mailbox/postgres/src/main/java/org/apache/james/mailbox/postgres/mail/eventsourcing/acl/ACLModule.java @@ -0,0 +1,41 @@ +/**************************************************************** + * 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.mailbox.postgres.mail.eventsourcing.acl; + +import org.apache.james.event.acl.ACLUpdated; +import org.apache.james.event.acl.ACLUpdatedDTO; +import org.apache.james.eventsourcing.eventstore.dto.EventDTOModule; +import org.apache.james.json.DTOModule; +import org.apache.james.mailbox.model.MailboxId; +import org.apache.james.mailbox.postgres.PostgresMailboxId; + +public interface ACLModule { + String UPDATE_TYPE_NAME = "acl-updated"; + + MailboxId.Factory mailboxIdFactory = new PostgresMailboxId.Factory(); + + EventDTOModule ACL_UPDATE = + new DTOModule.Builder<>(ACLUpdated.class) + .convertToDTO(ACLUpdatedDTO.class) + .toDomainObjectConverter(dto -> dto.toEvent(mailboxIdFactory)) + .toDTOConverter(ACLUpdatedDTO::from) + .typeName(UPDATE_TYPE_NAME) + .withFactory(EventDTOModule::new); +} \ No newline at end of file diff --git a/mailbox/postgres/src/main/java/org/apache/james/mailbox/postgres/quota/PostgresCurrentQuotaManager.java b/mailbox/postgres/src/main/java/org/apache/james/mailbox/postgres/quota/PostgresCurrentQuotaManager.java new file mode 100644 index 00000000000..8617ca21318 --- /dev/null +++ b/mailbox/postgres/src/main/java/org/apache/james/mailbox/postgres/quota/PostgresCurrentQuotaManager.java @@ -0,0 +1,130 @@ +/**************************************************************** + * 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.mailbox.postgres.quota; + +import java.util.List; +import java.util.Optional; +import java.util.function.Predicate; + +import jakarta.inject.Inject; + +import org.apache.james.backends.postgres.quota.PostgresQuotaCurrentValueDAO; +import org.apache.james.core.quota.QuotaComponent; +import org.apache.james.core.quota.QuotaCountUsage; +import org.apache.james.core.quota.QuotaCurrentValue; +import org.apache.james.core.quota.QuotaSizeUsage; +import org.apache.james.core.quota.QuotaType; +import org.apache.james.mailbox.model.CurrentQuotas; +import org.apache.james.mailbox.model.QuotaOperation; +import org.apache.james.mailbox.model.QuotaRoot; +import org.apache.james.mailbox.quota.CurrentQuotaManager; + +import reactor.core.publisher.Mono; + +public class PostgresCurrentQuotaManager implements CurrentQuotaManager { + + private final PostgresQuotaCurrentValueDAO currentValueDao; + + @Inject + public PostgresCurrentQuotaManager(PostgresQuotaCurrentValueDAO currentValueDao) { + this.currentValueDao = currentValueDao; + } + + @Override + public Mono getCurrentMessageCount(QuotaRoot quotaRoot) { + return currentValueDao.getQuotaCurrentValue(asQuotaKeyCount(quotaRoot)) + .map(QuotaCurrentValue::getCurrentValue) + .map(QuotaCountUsage::count) + .defaultIfEmpty(QuotaCountUsage.count(0L)); + } + + @Override + public Mono getCurrentStorage(QuotaRoot quotaRoot) { + return currentValueDao.getQuotaCurrentValue(asQuotaKeySize(quotaRoot)) + .map(QuotaCurrentValue::getCurrentValue) + .map(QuotaSizeUsage::size) + .defaultIfEmpty(QuotaSizeUsage.size(0L)); + } + + @Override + public Mono getCurrentQuotas(QuotaRoot quotaRoot) { + return currentValueDao.getQuotaCurrentValues(QuotaComponent.MAILBOX, quotaRoot.asString()) + .collectList() + .map(this::buildCurrentQuotas); + } + + @Override + public Mono increase(QuotaOperation quotaOperation) { + return currentValueDao.increase(asQuotaKeyCount(quotaOperation.quotaRoot()), quotaOperation.count().asLong()) + .then(currentValueDao.increase(asQuotaKeySize(quotaOperation.quotaRoot()), quotaOperation.size().asLong())); + } + + @Override + public Mono decrease(QuotaOperation quotaOperation) { + return currentValueDao.decrease(asQuotaKeyCount(quotaOperation.quotaRoot()), quotaOperation.count().asLong()) + .then(currentValueDao.decrease(asQuotaKeySize(quotaOperation.quotaRoot()), quotaOperation.size().asLong())); + } + + @Override + public Mono setCurrentQuotas(QuotaOperation quotaOperation) { + return getCurrentQuotas(quotaOperation.quotaRoot()) + .filter(Predicate.not(Predicate.isEqual(CurrentQuotas.from(quotaOperation)))) + .flatMap(storedQuotas -> { + long count = quotaOperation.count().asLong() - storedQuotas.count().asLong(); + long size = quotaOperation.size().asLong() - storedQuotas.size().asLong(); + + return currentValueDao.increase(asQuotaKeyCount(quotaOperation.quotaRoot()), count) + .then(currentValueDao.increase(asQuotaKeySize(quotaOperation.quotaRoot()), size)); + }); + } + + private QuotaCurrentValue.Key asQuotaKeyCount(QuotaRoot quotaRoot) { + return asQuotaKey(quotaRoot, QuotaType.COUNT); + } + + private QuotaCurrentValue.Key asQuotaKeySize(QuotaRoot quotaRoot) { + return asQuotaKey(quotaRoot, QuotaType.SIZE); + } + + private QuotaCurrentValue.Key asQuotaKey(QuotaRoot quotaRoot, QuotaType quotaType) { + return QuotaCurrentValue.Key.of( + QuotaComponent.MAILBOX, + quotaRoot.asString(), + quotaType); + } + + private CurrentQuotas buildCurrentQuotas(List quotaCurrentValues) { + QuotaCountUsage count = extractQuotaByType(quotaCurrentValues, QuotaType.COUNT) + .map(value -> QuotaCountUsage.count(value.getCurrentValue())) + .orElse(QuotaCountUsage.count(0L)); + + QuotaSizeUsage size = extractQuotaByType(quotaCurrentValues, QuotaType.SIZE) + .map(value -> QuotaSizeUsage.size(value.getCurrentValue())) + .orElse(QuotaSizeUsage.size(0L)); + + return new CurrentQuotas(count, size); + } + + private Optional extractQuotaByType(List quotaCurrentValues, QuotaType quotaType) { + return quotaCurrentValues.stream() + .filter(quotaValue -> quotaValue.getQuotaType().equals(quotaType)) + .findAny(); + } +} diff --git a/mailbox/postgres/src/main/java/org/apache/james/mailbox/postgres/quota/PostgresPerUserMaxQuotaManager.java b/mailbox/postgres/src/main/java/org/apache/james/mailbox/postgres/quota/PostgresPerUserMaxQuotaManager.java new file mode 100644 index 00000000000..b8953b75e5e --- /dev/null +++ b/mailbox/postgres/src/main/java/org/apache/james/mailbox/postgres/quota/PostgresPerUserMaxQuotaManager.java @@ -0,0 +1,361 @@ +/**************************************************************** + * 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.mailbox.postgres.quota; + +import static org.apache.james.util.ReactorUtils.publishIfPresent; + +import java.util.Map; +import java.util.Optional; +import java.util.function.Function; +import java.util.stream.Collectors; +import java.util.stream.Stream; + +import jakarta.inject.Inject; + +import org.apache.commons.lang3.tuple.Pair; +import org.apache.james.backends.postgres.quota.PostgresQuotaLimitDAO; +import org.apache.james.core.Domain; +import org.apache.james.core.quota.QuotaComponent; +import org.apache.james.core.quota.QuotaCountLimit; +import org.apache.james.core.quota.QuotaLimit; +import org.apache.james.core.quota.QuotaScope; +import org.apache.james.core.quota.QuotaSizeLimit; +import org.apache.james.core.quota.QuotaType; +import org.apache.james.mailbox.model.Quota; +import org.apache.james.mailbox.model.QuotaRoot; +import org.apache.james.mailbox.quota.Limits; +import org.apache.james.mailbox.quota.MaxQuotaManager; +import org.apache.james.mailbox.quota.QuotaCodec; + +import com.google.common.collect.ImmutableMap; + +import reactor.core.publisher.Flux; +import reactor.core.publisher.Mono; + +public class PostgresPerUserMaxQuotaManager implements MaxQuotaManager { + private static final String GLOBAL_IDENTIFIER = "global"; + + private final PostgresQuotaLimitDAO postgresQuotaLimitDAO; + + @Inject + public PostgresPerUserMaxQuotaManager(PostgresQuotaLimitDAO postgresQuotaLimitDAO) { + this.postgresQuotaLimitDAO = postgresQuotaLimitDAO; + } + + @Override + public void setMaxStorage(QuotaRoot quotaRoot, QuotaSizeLimit maxStorageQuota) { + setMaxStorageReactive(quotaRoot, maxStorageQuota).block(); + } + + @Override + public Mono setMaxStorageReactive(QuotaRoot quotaRoot, QuotaSizeLimit maxStorageQuota) { + return postgresQuotaLimitDAO.setQuotaLimit(QuotaLimit.builder() + .quotaScope(QuotaScope.USER) + .identifier(quotaRoot.getValue()) + .quotaComponent(QuotaComponent.MAILBOX) + .quotaType(QuotaType.SIZE) + .quotaLimit(QuotaCodec.quotaValueToLong(maxStorageQuota)) + .build()); + } + + @Override + public void setMaxMessage(QuotaRoot quotaRoot, QuotaCountLimit maxMessageCount) { + setMaxMessageReactive(quotaRoot, maxMessageCount).block(); + } + + @Override + public Mono setMaxMessageReactive(QuotaRoot quotaRoot, QuotaCountLimit maxMessageCount) { + return postgresQuotaLimitDAO.setQuotaLimit(QuotaLimit.builder() + .quotaScope(QuotaScope.USER) + .identifier(quotaRoot.getValue()) + .quotaComponent(QuotaComponent.MAILBOX) + .quotaType(QuotaType.COUNT) + .quotaLimit(QuotaCodec.quotaValueToLong(maxMessageCount)) + .build()); + } + + @Override + public void setDomainMaxMessage(Domain domain, QuotaCountLimit count) { + setDomainMaxMessageReactive(domain, count).block(); + } + + @Override + public Mono setDomainMaxMessageReactive(Domain domain, QuotaCountLimit count) { + return postgresQuotaLimitDAO.setQuotaLimit(QuotaLimit.builder() + .quotaScope(QuotaScope.DOMAIN) + .identifier(domain.asString()) + .quotaComponent(QuotaComponent.MAILBOX) + .quotaType(QuotaType.COUNT) + .quotaLimit(QuotaCodec.quotaValueToLong(count)) + .build()); + } + + @Override + public void setDomainMaxStorage(Domain domain, QuotaSizeLimit size) { + setDomainMaxStorageReactive(domain, size).block(); + } + + @Override + public Mono setDomainMaxStorageReactive(Domain domain, QuotaSizeLimit size) { + return postgresQuotaLimitDAO.setQuotaLimit(QuotaLimit.builder() + .quotaScope(QuotaScope.DOMAIN) + .identifier(domain.asString()) + .quotaComponent(QuotaComponent.MAILBOX) + .quotaType(QuotaType.SIZE) + .quotaLimit(QuotaCodec.quotaValueToLong(size)) + .build()); + } + + @Override + public void removeDomainMaxMessage(Domain domain) { + removeDomainMaxMessageReactive(domain).block(); + } + + @Override + public Mono removeDomainMaxMessageReactive(Domain domain) { + return postgresQuotaLimitDAO.deleteQuotaLimit(QuotaLimit.QuotaLimitKey.of(QuotaComponent.MAILBOX, QuotaScope.DOMAIN, domain.asString(), QuotaType.COUNT)); + } + + @Override + public void removeDomainMaxStorage(Domain domain) { + removeDomainMaxStorageReactive(domain).block(); + } + + @Override + public Mono removeDomainMaxStorageReactive(Domain domain) { + return postgresQuotaLimitDAO.deleteQuotaLimit(QuotaLimit.QuotaLimitKey.of(QuotaComponent.MAILBOX, QuotaScope.DOMAIN, domain.asString(), QuotaType.SIZE)); + } + + @Override + public Optional getDomainMaxMessage(Domain domain) { + return getDomainMaxMessageReactive(domain).blockOptional(); + } + + @Override + public Mono getDomainMaxMessageReactive(Domain domain) { + return getMaxMessageReactive(QuotaScope.DOMAIN, domain.asString()); + } + + @Override + public Optional getDomainMaxStorage(Domain domain) { + return getDomainMaxStorageReactive(domain).blockOptional(); + } + + @Override + public Mono getDomainMaxStorageReactive(Domain domain) { + return getMaxStorageReactive(QuotaScope.DOMAIN, domain.asString()); + } + + @Override + public void removeMaxMessage(QuotaRoot quotaRoot) { + removeMaxMessageReactive(quotaRoot).block(); + } + + @Override + public Mono removeMaxMessageReactive(QuotaRoot quotaRoot) { + return postgresQuotaLimitDAO.deleteQuotaLimit(QuotaLimit.QuotaLimitKey.of(QuotaComponent.MAILBOX, QuotaScope.USER, quotaRoot.getValue(), QuotaType.COUNT)); + } + + @Override + public void removeMaxStorage(QuotaRoot quotaRoot) { + removeMaxStorageReactive(quotaRoot).block(); + } + + @Override + public Mono removeMaxStorageReactive(QuotaRoot quotaRoot) { + return postgresQuotaLimitDAO.deleteQuotaLimit(QuotaLimit.QuotaLimitKey.of(QuotaComponent.MAILBOX, QuotaScope.USER, quotaRoot.getValue(), QuotaType.SIZE)); + } + + @Override + public void setGlobalMaxStorage(QuotaSizeLimit globalMaxStorage) { + setGlobalMaxStorageReactive(globalMaxStorage).block(); + } + + @Override + public Mono setGlobalMaxStorageReactive(QuotaSizeLimit globalMaxStorage) { + return postgresQuotaLimitDAO.setQuotaLimit(QuotaLimit.builder() + .quotaScope(QuotaScope.GLOBAL).identifier(GLOBAL_IDENTIFIER) + .quotaComponent(QuotaComponent.MAILBOX) + .quotaType(QuotaType.SIZE) + .quotaLimit(QuotaCodec.quotaValueToLong(globalMaxStorage)) + .build()); + } + + @Override + public void removeGlobalMaxStorage() { + removeGlobalMaxStorageReactive().block(); + } + + @Override + public Mono removeGlobalMaxStorageReactive() { + return postgresQuotaLimitDAO.deleteQuotaLimit(QuotaLimit.QuotaLimitKey.of(QuotaComponent.MAILBOX, QuotaScope.GLOBAL, GLOBAL_IDENTIFIER, QuotaType.SIZE)); + } + + @Override + public void setGlobalMaxMessage(QuotaCountLimit globalMaxMessageCount) { + setGlobalMaxMessageReactive(globalMaxMessageCount).block(); + } + + @Override + public Mono setGlobalMaxMessageReactive(QuotaCountLimit globalMaxMessageCount) { + return postgresQuotaLimitDAO.setQuotaLimit(QuotaLimit.builder() + .quotaScope(QuotaScope.GLOBAL).identifier(GLOBAL_IDENTIFIER) + .quotaComponent(QuotaComponent.MAILBOX) + .quotaType(QuotaType.COUNT) + .quotaLimit(QuotaCodec.quotaValueToLong(globalMaxMessageCount)) + .build()); + } + + @Override + public void removeGlobalMaxMessage() { + removeGlobalMaxMessageReactive().block(); + } + + @Override + public Mono removeGlobalMaxMessageReactive() { + return postgresQuotaLimitDAO.deleteQuotaLimit(QuotaLimit.QuotaLimitKey.of(QuotaComponent.MAILBOX, QuotaScope.GLOBAL, GLOBAL_IDENTIFIER, QuotaType.COUNT)); + } + + @Override + public Optional getGlobalMaxStorage() { + return getGlobalMaxStorageReactive().blockOptional(); + } + + @Override + public Mono getGlobalMaxStorageReactive() { + return getMaxStorageReactive(QuotaScope.GLOBAL, GLOBAL_IDENTIFIER); + } + + @Override + public Optional getGlobalMaxMessage() { + return getGlobalMaxMessageReactive().blockOptional(); + } + + @Override + public Mono getGlobalMaxMessageReactive() { + return getMaxMessageReactive(QuotaScope.GLOBAL, GLOBAL_IDENTIFIER); + } + + @Override + public Map listMaxMessagesDetails(QuotaRoot quotaRoot) { + return listMaxMessagesDetailsReactive(quotaRoot).block(); + } + + @Override + public Mono> listMaxMessagesDetailsReactive(QuotaRoot quotaRoot) { + return Flux.merge( + getMaxMessageReactive(QuotaScope.USER, quotaRoot.getValue()) + .map(limit -> Pair.of(Quota.Scope.User, limit)), + Mono.justOrEmpty(quotaRoot.getDomain()) + .flatMap(domain -> getMaxMessageReactive(QuotaScope.DOMAIN, domain.asString())) + .map(limit -> Pair.of(Quota.Scope.Domain, limit)), + getGlobalMaxMessageReactive() + .map(limit -> Pair.of(Quota.Scope.Global, limit))) + .collect(ImmutableMap.toImmutableMap( + Pair::getKey, + Pair::getValue)); + } + + @Override + public Map listMaxStorageDetails(QuotaRoot quotaRoot) { + return listMaxStorageDetailsReactive(quotaRoot).block(); + } + + @Override + public Mono> listMaxStorageDetailsReactive(QuotaRoot quotaRoot) { + return Flux.merge( + getMaxStorageReactive(QuotaScope.USER, quotaRoot.getValue()) + .map(limit -> Pair.of(Quota.Scope.User, limit)), + Mono.justOrEmpty(quotaRoot.getDomain()) + .flatMap(domain -> getMaxStorageReactive(QuotaScope.DOMAIN, domain.asString())) + .map(limit -> Pair.of(Quota.Scope.Domain, limit)), + getGlobalMaxStorageReactive() + .map(limit -> Pair.of(Quota.Scope.Global, limit))) + .collect(ImmutableMap.toImmutableMap( + Pair::getKey, + Pair::getValue)); + } + + @Override + public QuotaDetails quotaDetails(QuotaRoot quotaRoot) { + return quotaDetailsReactive(quotaRoot) + .block(); + } + + @Override + public Mono quotaDetailsReactive(QuotaRoot quotaRoot) { + return Mono.zip( + getLimits(QuotaScope.USER, quotaRoot.getValue()), + Mono.justOrEmpty(quotaRoot.getDomain()).flatMap(domain -> getLimits(QuotaScope.DOMAIN, domain.asString())).switchIfEmpty(Mono.just(Limits.empty())), + getLimits(QuotaScope.GLOBAL, GLOBAL_IDENTIFIER)) + .map(tuple -> new QuotaDetails( + countDetails(tuple.getT1(), tuple.getT2(), tuple.getT3().getCountLimit()), + sizeDetails(tuple.getT1(), tuple.getT2(), tuple.getT3().getSizeLimit()))); + } + + private Mono getLimits(QuotaScope quotaScope, String identifier) { + return postgresQuotaLimitDAO.getQuotaLimits(QuotaComponent.MAILBOX, quotaScope, identifier) + .collectList() + .map(list -> { + Map> map = list.stream().collect(Collectors.toMap(QuotaLimit::getQuotaType, QuotaLimit::getQuotaLimit)); + return new Limits( + map.getOrDefault(QuotaType.SIZE, Optional.empty()).flatMap(QuotaCodec::longToQuotaSize), + map.getOrDefault(QuotaType.COUNT, Optional.empty()).flatMap(QuotaCodec::longToQuotaCount)); + }).switchIfEmpty(Mono.just(Limits.empty())); + } + + private Mono getMaxMessageReactive(QuotaScope quotaScope, String identifier) { + return postgresQuotaLimitDAO.getQuotaLimit(QuotaLimit.QuotaLimitKey.of(QuotaComponent.MAILBOX, quotaScope, identifier, QuotaType.COUNT)) + .map(QuotaLimit::getQuotaLimit) + .handle(publishIfPresent()) + .map(QuotaCodec::longToQuotaCount) + .handle(publishIfPresent()); + } + + public Mono getMaxStorageReactive(QuotaScope quotaScope, String identifier) { + return postgresQuotaLimitDAO.getQuotaLimit(QuotaLimit.QuotaLimitKey.of(QuotaComponent.MAILBOX, quotaScope, identifier, QuotaType.SIZE)) + .map(QuotaLimit::getQuotaLimit) + .handle(publishIfPresent()) + .map(QuotaCodec::longToQuotaSize) + .handle(publishIfPresent()); + } + + private Map sizeDetails(Limits userLimits, Limits domainLimits, Optional globalLimits) { + return Stream.of( + userLimits.getSizeLimit().stream().map(limit -> Pair.of(Quota.Scope.User, limit)), + domainLimits.getSizeLimit().stream().map(limit -> Pair.of(Quota.Scope.Domain, limit)), + globalLimits.stream().map(limit -> Pair.of(Quota.Scope.Global, limit))) + .flatMap(Function.identity()) + .collect(ImmutableMap.toImmutableMap( + Pair::getKey, + Pair::getValue)); + } + + private Map countDetails(Limits userLimits, Limits domainLimits, Optional globalLimits) { + return Stream.of( + userLimits.getCountLimit().stream().map(limit -> Pair.of(Quota.Scope.User, limit)), + domainLimits.getCountLimit().stream().map(limit -> Pair.of(Quota.Scope.Domain, limit)), + globalLimits.stream().map(limit -> Pair.of(Quota.Scope.Global, limit))) + .flatMap(Function.identity()) + .collect(ImmutableMap.toImmutableMap( + Pair::getKey, + Pair::getValue)); + } +} diff --git a/mailbox/postgres/src/main/java/org/apache/james/mailbox/postgres/search/AllSearchOverride.java b/mailbox/postgres/src/main/java/org/apache/james/mailbox/postgres/search/AllSearchOverride.java new file mode 100644 index 00000000000..8d0721805b2 --- /dev/null +++ b/mailbox/postgres/src/main/java/org/apache/james/mailbox/postgres/search/AllSearchOverride.java @@ -0,0 +1,73 @@ +/**************************************************************** + * 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.mailbox.postgres.search; + +import jakarta.inject.Inject; + +import org.apache.james.backends.postgres.utils.PostgresExecutor; +import org.apache.james.mailbox.MailboxSession; +import org.apache.james.mailbox.MessageUid; +import org.apache.james.mailbox.model.Mailbox; +import org.apache.james.mailbox.model.SearchQuery; +import org.apache.james.mailbox.postgres.PostgresMailboxId; +import org.apache.james.mailbox.postgres.mail.dao.PostgresMailboxMessageDAO; +import org.apache.james.mailbox.store.search.ListeningMessageSearchIndex; + +import reactor.core.publisher.Flux; +import reactor.core.publisher.Mono; + +public class AllSearchOverride implements ListeningMessageSearchIndex.SearchOverride { + private final PostgresExecutor.Factory executorFactory; + + @Inject + public AllSearchOverride(PostgresExecutor.Factory executorFactory) { + this.executorFactory = executorFactory; + } + + @Override + public boolean applicable(SearchQuery searchQuery, MailboxSession session) { + return isAll(searchQuery) + || isFromOne(searchQuery) + || isEmpty(searchQuery); + } + + private boolean isAll(SearchQuery searchQuery) { + return searchQuery.getCriteria().size() == 1 + && searchQuery.getCriteria().get(0).equals(SearchQuery.uid(new SearchQuery.UidRange(MessageUid.MIN_VALUE, MessageUid.MAX_VALUE))) + && searchQuery.getSorts().equals(SearchQuery.DEFAULT_SORTS); + } + + private boolean isFromOne(SearchQuery searchQuery) { + return searchQuery.getCriteria().size() == 1 + && searchQuery.getCriteria().get(0).equals(SearchQuery.all()) + && searchQuery.getSorts().equals(SearchQuery.DEFAULT_SORTS); + } + + private boolean isEmpty(SearchQuery searchQuery) { + return searchQuery.getCriteria().isEmpty() + && searchQuery.getSorts().equals(SearchQuery.DEFAULT_SORTS); + } + + @Override + public Flux search(MailboxSession session, Mailbox mailbox, SearchQuery searchQuery) { + return Mono.fromCallable(() -> new PostgresMailboxMessageDAO(executorFactory.create(session.getUser().getDomainPart()))) + .flatMapMany(dao -> dao.listAllMessageUid((PostgresMailboxId) mailbox.getMailboxId())); + } +} diff --git a/mailbox/postgres/src/main/java/org/apache/james/mailbox/postgres/search/DeletedSearchOverride.java b/mailbox/postgres/src/main/java/org/apache/james/mailbox/postgres/search/DeletedSearchOverride.java new file mode 100644 index 00000000000..5b1e1a47577 --- /dev/null +++ b/mailbox/postgres/src/main/java/org/apache/james/mailbox/postgres/search/DeletedSearchOverride.java @@ -0,0 +1,57 @@ +/**************************************************************** + * 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.mailbox.postgres.search; + +import jakarta.inject.Inject; +import jakarta.mail.Flags; + +import org.apache.james.backends.postgres.utils.PostgresExecutor; +import org.apache.james.mailbox.MailboxSession; +import org.apache.james.mailbox.MessageUid; +import org.apache.james.mailbox.model.Mailbox; +import org.apache.james.mailbox.model.SearchQuery; +import org.apache.james.mailbox.postgres.PostgresMailboxId; +import org.apache.james.mailbox.postgres.mail.dao.PostgresMailboxMessageDAO; +import org.apache.james.mailbox.store.search.ListeningMessageSearchIndex; + +import reactor.core.publisher.Flux; +import reactor.core.publisher.Mono; + +public class DeletedSearchOverride implements ListeningMessageSearchIndex.SearchOverride { + private final PostgresExecutor.Factory executorFactory; + + @Inject + public DeletedSearchOverride(PostgresExecutor.Factory executorFactory) { + this.executorFactory = executorFactory; + } + + @Override + public boolean applicable(SearchQuery searchQuery, MailboxSession session) { + return searchQuery.getCriteria().size() == 1 + && searchQuery.getCriteria().get(0).equals(SearchQuery.flagIsSet(Flags.Flag.DELETED)) + && searchQuery.getSorts().equals(SearchQuery.DEFAULT_SORTS); + } + + @Override + public Flux search(MailboxSession session, Mailbox mailbox, SearchQuery searchQuery) { + return Mono.fromCallable(() -> new PostgresMailboxMessageDAO(executorFactory.create(session.getUser().getDomainPart()))) + .flatMapMany(dao -> dao.findDeletedMessagesByMailboxId((PostgresMailboxId) mailbox.getMailboxId())); + } +} diff --git a/mailbox/postgres/src/main/java/org/apache/james/mailbox/postgres/search/DeletedWithRangeSearchOverride.java b/mailbox/postgres/src/main/java/org/apache/james/mailbox/postgres/search/DeletedWithRangeSearchOverride.java new file mode 100644 index 00000000000..cb9710c0b5a --- /dev/null +++ b/mailbox/postgres/src/main/java/org/apache/james/mailbox/postgres/search/DeletedWithRangeSearchOverride.java @@ -0,0 +1,71 @@ +/**************************************************************** + * 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.mailbox.postgres.search; + +import jakarta.inject.Inject; +import jakarta.mail.Flags; + +import org.apache.james.backends.postgres.utils.PostgresExecutor; +import org.apache.james.mailbox.MailboxSession; +import org.apache.james.mailbox.MessageUid; +import org.apache.james.mailbox.model.Mailbox; +import org.apache.james.mailbox.model.SearchQuery; +import org.apache.james.mailbox.postgres.PostgresMailboxId; +import org.apache.james.mailbox.postgres.mail.dao.PostgresMailboxMessageDAO; +import org.apache.james.mailbox.store.search.ListeningMessageSearchIndex; + +import com.google.common.collect.ImmutableList; + +import reactor.core.publisher.Flux; +import reactor.core.publisher.Mono; + +public class DeletedWithRangeSearchOverride implements ListeningMessageSearchIndex.SearchOverride { + private final PostgresExecutor.Factory executorFactory; + + @Inject + public DeletedWithRangeSearchOverride(PostgresExecutor.Factory executorFactory) { + this.executorFactory = executorFactory; + } + + @Override + public boolean applicable(SearchQuery searchQuery, MailboxSession session) { + return searchQuery.getCriteria().size() == 2 + && searchQuery.getCriteria().contains(SearchQuery.flagIsSet(Flags.Flag.DELETED)) + && searchQuery.getCriteria().stream() + .anyMatch(criterion -> criterion instanceof SearchQuery.UidCriterion) + && searchQuery.getSorts().equals(SearchQuery.DEFAULT_SORTS); + } + + @Override + public Flux search(MailboxSession session, Mailbox mailbox, SearchQuery searchQuery) { + SearchQuery.UidCriterion uidArgument = searchQuery.getCriteria().stream() + .filter(criterion -> criterion instanceof SearchQuery.UidCriterion) + .map(SearchQuery.UidCriterion.class::cast) + .findAny() + .orElseThrow(() -> new RuntimeException("Missing Uid argument")); + + SearchQuery.UidRange[] uidRanges = uidArgument.getOperator().getRange(); + + return Mono.fromCallable(() -> new PostgresMailboxMessageDAO(executorFactory.create(session.getUser().getDomainPart()))) + .flatMapMany(dao -> Flux.fromIterable(ImmutableList.copyOf(uidRanges)) + .concatMap(range -> dao.findDeletedMessagesByMailboxIdAndBetweenUIDs((PostgresMailboxId) mailbox.getMailboxId(), + range.getLowValue(), range.getHighValue()))); + } +} diff --git a/mailbox/postgres/src/main/java/org/apache/james/mailbox/postgres/search/NotDeletedWithRangeSearchOverride.java b/mailbox/postgres/src/main/java/org/apache/james/mailbox/postgres/search/NotDeletedWithRangeSearchOverride.java new file mode 100644 index 00000000000..cc752d53b75 --- /dev/null +++ b/mailbox/postgres/src/main/java/org/apache/james/mailbox/postgres/search/NotDeletedWithRangeSearchOverride.java @@ -0,0 +1,83 @@ +/**************************************************************** + * 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.mailbox.postgres.search; + +import jakarta.inject.Inject; +import jakarta.mail.Flags; + +import org.apache.james.backends.postgres.utils.PostgresExecutor; +import org.apache.james.mailbox.MailboxSession; +import org.apache.james.mailbox.MessageUid; +import org.apache.james.mailbox.model.Mailbox; +import org.apache.james.mailbox.model.MessageRange; +import org.apache.james.mailbox.model.SearchQuery; +import org.apache.james.mailbox.postgres.PostgresMailboxId; +import org.apache.james.mailbox.postgres.mail.dao.PostgresMailboxMessageDAO; +import org.apache.james.mailbox.store.search.ListeningMessageSearchIndex; + +import reactor.core.publisher.Flux; +import reactor.core.publisher.Mono; + +public class NotDeletedWithRangeSearchOverride implements ListeningMessageSearchIndex.SearchOverride { + + private final PostgresExecutor.Factory executorFactory; + + @Inject + public NotDeletedWithRangeSearchOverride(PostgresExecutor.Factory executorFactory) { + this.executorFactory = executorFactory; + } + + @Override + public boolean applicable(SearchQuery searchQuery, MailboxSession session) { + return isDeletedUnset(searchQuery) || isDeletedNotSet(searchQuery); + } + + private boolean isDeletedUnset(SearchQuery searchQuery) { + return searchQuery.getCriteria().size() == 2 + && searchQuery.getCriteria().contains(SearchQuery.flagIsUnSet(Flags.Flag.DELETED)) + && searchQuery.getCriteria().stream() + .anyMatch(criterion -> criterion instanceof SearchQuery.UidCriterion) + && searchQuery.getSorts().equals(SearchQuery.DEFAULT_SORTS); + } + + private boolean isDeletedNotSet(SearchQuery searchQuery) { + return searchQuery.getCriteria().size() == 2 + && searchQuery.getCriteria().contains(SearchQuery.not(SearchQuery.flagIsSet(Flags.Flag.DELETED))) + && searchQuery.getCriteria().stream() + .anyMatch(criterion -> criterion instanceof SearchQuery.UidCriterion) + && searchQuery.getSorts().equals(SearchQuery.DEFAULT_SORTS); + } + + @Override + public Flux search(MailboxSession session, Mailbox mailbox, SearchQuery searchQuery) { + SearchQuery.UidCriterion uidArgument = searchQuery.getCriteria().stream() + .filter(criterion -> criterion instanceof SearchQuery.UidCriterion) + .map(SearchQuery.UidCriterion.class::cast) + .findAny() + .orElseThrow(() -> new RuntimeException("Missing Uid argument")); + + SearchQuery.UidRange[] uidRanges = uidArgument.getOperator().getRange(); + + return Mono.fromCallable(() -> new PostgresMailboxMessageDAO(executorFactory.create(session.getUser().getDomainPart()))) + .flatMapMany(dao -> Flux.fromArray(uidRanges) + .concatMap(range -> dao.listNotDeletedUids((PostgresMailboxId) mailbox.getMailboxId(), + MessageRange.range(range.getLowValue(), range.getHighValue())))); + } +} diff --git a/mailbox/postgres/src/main/java/org/apache/james/mailbox/postgres/search/UidSearchOverride.java b/mailbox/postgres/src/main/java/org/apache/james/mailbox/postgres/search/UidSearchOverride.java new file mode 100644 index 00000000000..e2ea13e19d7 --- /dev/null +++ b/mailbox/postgres/src/main/java/org/apache/james/mailbox/postgres/search/UidSearchOverride.java @@ -0,0 +1,68 @@ +/**************************************************************** + * 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.mailbox.postgres.search; + +import jakarta.inject.Inject; + +import org.apache.james.backends.postgres.utils.PostgresExecutor; +import org.apache.james.mailbox.MailboxSession; +import org.apache.james.mailbox.MessageUid; +import org.apache.james.mailbox.model.Mailbox; +import org.apache.james.mailbox.model.MessageRange; +import org.apache.james.mailbox.model.SearchQuery; +import org.apache.james.mailbox.postgres.PostgresMailboxId; +import org.apache.james.mailbox.postgres.mail.dao.PostgresMailboxMessageDAO; +import org.apache.james.mailbox.store.search.ListeningMessageSearchIndex; + +import com.google.common.collect.ImmutableList; + +import reactor.core.publisher.Flux; +import reactor.core.publisher.Mono; + +public class UidSearchOverride implements ListeningMessageSearchIndex.SearchOverride { + private final PostgresExecutor.Factory executorFactory; + + @Inject + public UidSearchOverride(PostgresExecutor.Factory executorFactory) { + this.executorFactory = executorFactory; + } + + @Override + public boolean applicable(SearchQuery searchQuery, MailboxSession session) { + return searchQuery.getCriteria().size() == 1 + && searchQuery.getCriteria().get(0) instanceof SearchQuery.UidCriterion + && searchQuery.getSorts().equals(SearchQuery.DEFAULT_SORTS); + } + + @Override + public Flux search(MailboxSession session, Mailbox mailbox, SearchQuery searchQuery) { + SearchQuery.UidCriterion uidArgument = searchQuery.getCriteria().stream() + .filter(criterion -> criterion instanceof SearchQuery.UidCriterion) + .map(SearchQuery.UidCriterion.class::cast) + .findAny() + .orElseThrow(() -> new RuntimeException("Missing Uid argument")); + + SearchQuery.UidRange[] uidRanges = uidArgument.getOperator().getRange(); + return Mono.fromCallable(() -> new PostgresMailboxMessageDAO(executorFactory.create(session.getUser().getDomainPart()))) + .flatMapMany(dao -> Flux.fromIterable(ImmutableList.copyOf(uidRanges)) + .concatMap(range -> dao.listUids((PostgresMailboxId) mailbox.getMailboxId(), + MessageRange.range(range.getLowValue(), range.getHighValue())))); + } +} diff --git a/mailbox/postgres/src/main/java/org/apache/james/mailbox/postgres/search/UnseenSearchOverride.java b/mailbox/postgres/src/main/java/org/apache/james/mailbox/postgres/search/UnseenSearchOverride.java new file mode 100644 index 00000000000..1ad25baafdf --- /dev/null +++ b/mailbox/postgres/src/main/java/org/apache/james/mailbox/postgres/search/UnseenSearchOverride.java @@ -0,0 +1,97 @@ +/**************************************************************** + * 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.mailbox.postgres.search; + +import java.util.Optional; + +import jakarta.inject.Inject; +import jakarta.mail.Flags; + +import org.apache.james.backends.postgres.utils.PostgresExecutor; +import org.apache.james.mailbox.MailboxSession; +import org.apache.james.mailbox.MessageUid; +import org.apache.james.mailbox.model.Mailbox; +import org.apache.james.mailbox.model.MessageRange; +import org.apache.james.mailbox.model.SearchQuery; +import org.apache.james.mailbox.postgres.PostgresMailboxId; +import org.apache.james.mailbox.postgres.mail.dao.PostgresMailboxMessageDAO; +import org.apache.james.mailbox.store.search.ListeningMessageSearchIndex; + +import com.google.common.collect.ImmutableList; + +import reactor.core.publisher.Flux; +import reactor.core.publisher.Mono; + +public class UnseenSearchOverride implements ListeningMessageSearchIndex.SearchOverride { + + private final PostgresExecutor.Factory executorFactory; + + @Inject + public UnseenSearchOverride(PostgresExecutor.Factory executorFactory) { + this.executorFactory = executorFactory; + } + + @Override + public boolean applicable(SearchQuery searchQuery, MailboxSession session) { + return isUnseenWithAll(searchQuery) + || isNotSeenWithAll(searchQuery); + } + + private boolean isUnseenWithAll(SearchQuery searchQuery) { + return searchQuery.getCriteria().contains(SearchQuery.flagIsUnSet(Flags.Flag.SEEN)) + && allMessages(searchQuery) + && searchQuery.getSorts().equals(SearchQuery.DEFAULT_SORTS); + } + + private boolean isNotSeenWithAll(SearchQuery searchQuery) { + return searchQuery.getCriteria().contains(SearchQuery.not(SearchQuery.flagIsSet(Flags.Flag.SEEN))) + && allMessages(searchQuery) + && searchQuery.getSorts().equals(SearchQuery.DEFAULT_SORTS); + } + + private boolean allMessages(SearchQuery searchQuery) { + if (searchQuery.getCriteria().size() == 1) { + // Only the unseen critrion + return true; + } + if (searchQuery.getCriteria().size() == 2) { + return searchQuery.getCriteria().stream() + .anyMatch(criterion -> criterion instanceof SearchQuery.UidCriterion) || + searchQuery.getCriteria().stream() + .anyMatch(criterion -> criterion instanceof SearchQuery.AllCriterion); + } + return false; + } + + @Override + public Flux search(MailboxSession session, Mailbox mailbox, SearchQuery searchQuery) { + final Optional maybeUidCriterion = searchQuery.getCriteria().stream() + .filter(criterion -> criterion instanceof SearchQuery.UidCriterion) + .map(SearchQuery.UidCriterion.class::cast) + .findFirst(); + + return Mono.fromCallable(() -> new PostgresMailboxMessageDAO(executorFactory.create(session.getUser().getDomainPart()))) + .flatMapMany(dao -> maybeUidCriterion + .map(uidCriterion -> Flux.fromIterable(ImmutableList.copyOf(uidCriterion.getOperator().getRange())) + .concatMap(range -> dao.listUnseen((PostgresMailboxId) mailbox.getMailboxId(), + MessageRange.range(range.getLowValue(), range.getHighValue())))) + .orElseGet(() -> dao.listUnseen((PostgresMailboxId) mailbox.getMailboxId()))); + } +} diff --git a/mailbox/postgres/src/main/java/org/apache/james/mailbox/postgres/user/PostgresSubscriptionDAO.java b/mailbox/postgres/src/main/java/org/apache/james/mailbox/postgres/user/PostgresSubscriptionDAO.java new file mode 100644 index 00000000000..91b4baa2fe6 --- /dev/null +++ b/mailbox/postgres/src/main/java/org/apache/james/mailbox/postgres/user/PostgresSubscriptionDAO.java @@ -0,0 +1,57 @@ +/**************************************************************** + * 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.mailbox.postgres.user; + +import static org.apache.james.mailbox.postgres.user.PostgresSubscriptionModule.MAILBOX; +import static org.apache.james.mailbox.postgres.user.PostgresSubscriptionModule.TABLE_NAME; +import static org.apache.james.mailbox.postgres.user.PostgresSubscriptionModule.USER; + +import org.apache.james.backends.postgres.utils.PostgresExecutor; + +import reactor.core.publisher.Flux; +import reactor.core.publisher.Mono; + +public class PostgresSubscriptionDAO { + protected final PostgresExecutor executor; + + public PostgresSubscriptionDAO(PostgresExecutor executor) { + this.executor = executor; + } + + public Mono save(String username, String mailbox) { + return executor.executeVoid(dsl -> Mono.from(dsl.insertInto(TABLE_NAME, USER, MAILBOX) + .values(username, mailbox) + .onConflict(USER, MAILBOX) + .doNothing() + .returningResult(MAILBOX))); + } + + public Mono delete(String username, String mailbox) { + return executor.executeVoid(dsl -> Mono.from(dsl.deleteFrom(TABLE_NAME) + .where(USER.eq(username)) + .and(MAILBOX.eq(mailbox)))); + } + + public Flux findMailboxByUser(String username) { + return executor.executeRows(dsl -> Flux.from(dsl.selectFrom(TABLE_NAME) + .where(USER.eq(username)))) + .map(record -> record.get(MAILBOX)); + } +} diff --git a/mailbox/postgres/src/main/java/org/apache/james/mailbox/postgres/user/PostgresSubscriptionMapper.java b/mailbox/postgres/src/main/java/org/apache/james/mailbox/postgres/user/PostgresSubscriptionMapper.java new file mode 100644 index 00000000000..e9d06e16606 --- /dev/null +++ b/mailbox/postgres/src/main/java/org/apache/james/mailbox/postgres/user/PostgresSubscriptionMapper.java @@ -0,0 +1,69 @@ +/**************************************************************** + * 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.mailbox.postgres.user; + +import java.util.List; + +import org.apache.james.core.Username; +import org.apache.james.mailbox.store.user.SubscriptionMapper; +import org.apache.james.mailbox.store.user.model.Subscription; + +import reactor.core.publisher.Flux; +import reactor.core.publisher.Mono; + +public class PostgresSubscriptionMapper implements SubscriptionMapper { + + private final PostgresSubscriptionDAO subscriptionDAO; + + public PostgresSubscriptionMapper(PostgresSubscriptionDAO subscriptionDAO) { + this.subscriptionDAO = subscriptionDAO; + } + + @Override + public void save(Subscription subscription) { + saveReactive(subscription).block(); + } + + @Override + public List findSubscriptionsForUser(Username user) { + return findSubscriptionsForUserReactive(user).collectList().block(); + } + + @Override + public void delete(Subscription subscription) { + deleteReactive(subscription).block(); + } + + @Override + public Mono saveReactive(Subscription subscription) { + return subscriptionDAO.save(subscription.getUser().asString(), subscription.getMailbox()); + } + + @Override + public Flux findSubscriptionsForUserReactive(Username user) { + return subscriptionDAO.findMailboxByUser(user.asString()) + .map(mailbox -> new Subscription(user, mailbox)); + } + + @Override + public Mono deleteReactive(Subscription subscription) { + return subscriptionDAO.delete(subscription.getUser().asString(), subscription.getMailbox()); + } +} diff --git a/mailbox/postgres/src/main/java/org/apache/james/mailbox/postgres/user/PostgresSubscriptionModule.java b/mailbox/postgres/src/main/java/org/apache/james/mailbox/postgres/user/PostgresSubscriptionModule.java new file mode 100644 index 00000000000..43f35ca48a1 --- /dev/null +++ b/mailbox/postgres/src/main/java/org/apache/james/mailbox/postgres/user/PostgresSubscriptionModule.java @@ -0,0 +1,55 @@ +/**************************************************************** + * 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.mailbox.postgres.user; + +import org.apache.james.backends.postgres.PostgresIndex; +import org.apache.james.backends.postgres.PostgresModule; +import org.apache.james.backends.postgres.PostgresTable; +import org.jooq.Field; +import org.jooq.Record; +import org.jooq.Table; +import org.jooq.impl.DSL; +import org.jooq.impl.SQLDataType; + +public interface PostgresSubscriptionModule { + /** + * See {@link MailboxManager.MAX_MAILBOX_NAME_LENGTH} + */ + Field MAILBOX = DSL.field("mailbox", SQLDataType.VARCHAR(255).notNull()); + /** + * See {@link Username.MAXIMUM_MAIL_ADDRESS_LENGTH} + */ + Field USER = DSL.field("user_name", SQLDataType.VARCHAR(255).notNull()); + Table TABLE_NAME = DSL.table("subscription"); + PostgresTable TABLE = PostgresTable.name(TABLE_NAME.getName()) + .createTableStep(((dsl, tableName) -> dsl.createTableIfNotExists(tableName) + .column(MAILBOX) + .column(USER) + .constraint(DSL.unique(MAILBOX, USER)))) + .supportsRowLevelSecurity() + .build(); + PostgresIndex INDEX = PostgresIndex.name("subscription_user_index") + .createIndexStep((dsl, indexName) -> dsl.createIndexIfNotExists(indexName) + .on(TABLE_NAME, USER)); + PostgresModule MODULE = PostgresModule.builder() + .addTable(TABLE) + .addIndex(INDEX) + .build(); +} diff --git a/mailbox/postgres/src/test/java/org/apache/james/mailbox/postgres/DeleteMessageListenerContract.java b/mailbox/postgres/src/test/java/org/apache/james/mailbox/postgres/DeleteMessageListenerContract.java new file mode 100644 index 00000000000..719f47a732d --- /dev/null +++ b/mailbox/postgres/src/test/java/org/apache/james/mailbox/postgres/DeleteMessageListenerContract.java @@ -0,0 +1,321 @@ +/**************************************************************** + * 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.mailbox.postgres; + +import static org.assertj.core.api.Assertions.assertThat; +import static org.assertj.core.api.SoftAssertions.assertSoftly; +import static org.junit.jupiter.api.Assumptions.assumeTrue; + +import java.nio.charset.StandardCharsets; +import java.util.HashSet; +import java.util.List; +import java.util.Optional; +import java.util.Set; +import java.util.UUID; +import java.util.stream.Collectors; + +import org.apache.james.blob.api.BlobId; +import org.apache.james.blob.api.BlobStore; +import org.apache.james.blob.api.ObjectNotFoundException; +import org.apache.james.core.Username; +import org.apache.james.mailbox.MailboxSession; +import org.apache.james.mailbox.MessageManager; +import org.apache.james.mailbox.model.AttachmentId; +import org.apache.james.mailbox.model.MailboxId; +import org.apache.james.mailbox.model.MailboxPath; +import org.apache.james.mailbox.model.MessageRange; +import org.apache.james.mailbox.postgres.mail.dao.PostgresAttachmentDAO; +import org.apache.james.mailbox.postgres.mail.dao.PostgresMailboxMessageDAO; +import org.apache.james.mailbox.postgres.mail.dao.PostgresMessageDAO; +import org.apache.james.mailbox.postgres.mail.dao.PostgresThreadDAO; +import org.apache.james.mailbox.store.mail.model.MimeMessageId; +import org.apache.james.mime4j.dom.Message; +import org.apache.james.mime4j.stream.RawField; +import org.apache.james.server.blob.deduplication.DeDuplicationBlobStore; +import org.apache.james.util.ClassLoaderUtils; +import org.assertj.core.api.SoftAssertions; +import org.junit.jupiter.api.BeforeEach; +import org.junit.jupiter.api.Test; + +import com.github.fge.lambdas.Throwing; +import com.google.common.collect.ImmutableList; +import com.google.common.hash.Hashing; + +import reactor.core.publisher.Flux; +import reactor.core.publisher.Mono; + +public abstract class DeleteMessageListenerContract { + + private MailboxSession session; + private MailboxPath inbox; + private MessageManager inboxManager; + private MessageManager otherBoxManager; + private PostgresMailboxManager mailboxManager; + private PostgresMessageDAO postgresMessageDAO; + private PostgresMailboxMessageDAO postgresMailboxMessageDAO; + private PostgresThreadDAO postgresThreadDAO; + + private PostgresAttachmentDAO attachmentDAO; + private BlobStore blobStore; + + abstract PostgresMailboxManager provideMailboxManager(); + + abstract PostgresMessageDAO providePostgresMessageDAO(); + + abstract PostgresMailboxMessageDAO providePostgresMailboxMessageDAO(); + + abstract PostgresThreadDAO threadDAO(); + + abstract PostgresAttachmentDAO attachmentDAO(); + + abstract BlobStore blobStore(); + + @BeforeEach + void setUp() throws Exception { + mailboxManager = provideMailboxManager(); + Username username = getUsername(); + session = mailboxManager.createSystemSession(username); + inbox = MailboxPath.inbox(session); + MailboxPath newPath = MailboxPath.forUser(username, "specialMailbox"); + MailboxId inboxId = mailboxManager.createMailbox(inbox, session).get(); + inboxManager = mailboxManager.getMailbox(inboxId, session); + MailboxId otherId = mailboxManager.createMailbox(newPath, session).get(); + otherBoxManager = mailboxManager.getMailbox(otherId, session); + + postgresMessageDAO = providePostgresMessageDAO(); + postgresMailboxMessageDAO = providePostgresMailboxMessageDAO(); + postgresThreadDAO = threadDAO(); + attachmentDAO = attachmentDAO(); + blobStore = blobStore(); + } + + protected Username getUsername() { + return Username.of("user" + UUID.randomUUID()); + } + + @Test + void deleteMailboxShouldDeleteUnreferencedMessageMetadata() throws Exception { + MessageManager.AppendResult appendResult = inboxManager.appendMessage(MessageManager.AppendCommand.builder() + .build(ClassLoaderUtils.getSystemResourceAsByteArray("eml/emailWithOnlyAttachment.eml")), session); + + AttachmentId attachmentId = appendResult.getMessageAttachments().get(0).getAttachment().getAttachmentId(); + + mailboxManager.deleteMailbox(inbox, session); + + assertSoftly(softly -> { + PostgresMessageId messageId = (PostgresMessageId) appendResult.getId().getMessageId(); + PostgresMailboxId mailboxId = (PostgresMailboxId) appendResult.getId().getMailboxId(); + + softly.assertThat(postgresMessageDAO.getBodyBlobId(messageId).blockOptional()) + .isEmpty(); + + softly.assertThat(postgresMailboxMessageDAO.countTotalMessagesByMailboxId(mailboxId).block()) + .isEqualTo(0); + + softly.assertThat(attachmentDAO.getAttachment(attachmentId).blockOptional()) + .isEmpty(); + }); + } + + @Test + void deleteMailboxShouldNotDeleteReferencedMessageMetadata() throws Exception { + MessageManager.AppendResult appendResult = inboxManager.appendMessage(MessageManager.AppendCommand.builder() + .build(ClassLoaderUtils.getSystemResourceAsByteArray("eml/emailWithOnlyAttachment.eml")), session); + mailboxManager.copyMessages(MessageRange.all(), inboxManager.getId(), otherBoxManager.getId(), session); + AttachmentId attachmentId = appendResult.getMessageAttachments().get(0).getAttachment().getAttachmentId(); + + mailboxManager.deleteMailbox(inbox, session); + + assertSoftly(softly -> { + PostgresMessageId messageId = (PostgresMessageId) appendResult.getId().getMessageId(); + + softly.assertThat(postgresMessageDAO.getBodyBlobId(messageId).blockOptional()) + .isNotEmpty(); + + softly.assertThat(postgresMailboxMessageDAO.countTotalMessagesByMailboxId((PostgresMailboxId) otherBoxManager.getId()) + .block()) + .isEqualTo(1); + + softly.assertThat(attachmentDAO.getAttachment(attachmentId).blockOptional()) + .isNotEmpty(); + }); + } + + @Test + void deleteMessageInMailboxShouldDeleteUnreferencedMessageMetadata() throws Exception { + MessageManager.AppendResult appendResult = inboxManager.appendMessage(MessageManager.AppendCommand.builder() + .build(ClassLoaderUtils.getSystemResourceAsByteArray("eml/emailWithOnlyAttachment.eml")), session); + AttachmentId attachmentId = appendResult.getMessageAttachments().get(0).getAttachment().getAttachmentId(); + + inboxManager.delete(ImmutableList.of(appendResult.getId().getUid()), session); + + assertSoftly(softly -> { + PostgresMessageId messageId = (PostgresMessageId) appendResult.getId().getMessageId(); + + softly.assertThat(postgresMessageDAO.getBodyBlobId(messageId).blockOptional()) + .isEmpty(); + + softly.assertThat(attachmentDAO.getAttachment(attachmentId).blockOptional()) + .isEmpty(); + }); + } + + @Test + void deleteMessageInMailboxShouldNotDeleteReferencedMessageMetadata() throws Exception { + MessageManager.AppendResult appendResult = inboxManager.appendMessage(MessageManager.AppendCommand.builder() + .build(ClassLoaderUtils.getSystemResourceAsByteArray("eml/emailWithOnlyAttachment.eml")), session); + mailboxManager.copyMessages(MessageRange.all(), inboxManager.getId(), otherBoxManager.getId(), session); + AttachmentId attachmentId = appendResult.getMessageAttachments().get(0).getAttachment().getAttachmentId(); + + inboxManager.delete(ImmutableList.of(appendResult.getId().getUid()), session); + PostgresMessageId messageId = (PostgresMessageId) appendResult.getId().getMessageId(); + + assertSoftly(softly -> { + softly.assertThat(postgresMessageDAO.getBodyBlobId(messageId).blockOptional()) + .isNotEmpty(); + + softly.assertThat(postgresMailboxMessageDAO.countTotalMessagesByMailboxId((PostgresMailboxId) otherBoxManager.getId()) + .block()) + .isEqualTo(1); + + softly.assertThat(attachmentDAO.getAttachment(attachmentId).blockOptional()) + .isNotEmpty(); + }); + } + + @Test + void deleteMessageListenerShouldDeleteUnreferencedBlob() throws Exception { + assumeTrue(!(blobStore instanceof DeDuplicationBlobStore)); + + MessageManager.AppendResult appendResult = inboxManager.appendMessage(MessageManager.AppendCommand.builder() + .build(ClassLoaderUtils.getSystemResourceAsByteArray("eml/emailWithOnlyAttachment.eml")), session); + AttachmentId attachmentId = appendResult.getMessageAttachments().get(0).getAttachment().getAttachmentId(); + + BlobId attachmentBlobId = attachmentDAO.getAttachment(attachmentId).block().getRight(); + BlobId messageBodyBlobId = postgresMessageDAO.getBodyBlobId((PostgresMessageId) appendResult.getId().getMessageId()).block(); + + inboxManager.delete(ImmutableList.of(appendResult.getId().getUid()), session); + + assertSoftly(softly -> { + softly.assertThatThrownBy(() -> Mono.from(blobStore.readReactive(blobStore.getDefaultBucketName(), attachmentBlobId)).block()) + .isInstanceOf(ObjectNotFoundException.class); + softly.assertThatThrownBy(() -> Mono.from(blobStore.readReactive(blobStore.getDefaultBucketName(), messageBodyBlobId)).block()) + .isInstanceOf(ObjectNotFoundException.class); + }); + } + + @Test + void deleteMessageListenerShouldNotDeleteReferencedBlob() throws Exception { + assumeTrue(!(blobStore instanceof DeDuplicationBlobStore)); + + MessageManager.AppendResult appendResult = inboxManager.appendMessage(MessageManager.AppendCommand.builder() + .build(ClassLoaderUtils.getSystemResourceAsByteArray("eml/emailWithOnlyAttachment.eml")), session); + BlobId messageBodyBlobId = postgresMessageDAO.getBodyBlobId((PostgresMessageId) appendResult.getId().getMessageId()).block(); + mailboxManager.copyMessages(MessageRange.all(), inboxManager.getId(), otherBoxManager.getId(), session); + + AttachmentId attachmentId = appendResult.getMessageAttachments().get(0).getAttachment().getAttachmentId(); + BlobId attachmentBlobId = attachmentDAO.getAttachment(attachmentId).block().getRight(); + + inboxManager.delete(ImmutableList.of(appendResult.getId().getUid()), session); + + assertSoftly(softly -> { + assertThat(Mono.from(blobStore.readReactive(blobStore.getDefaultBucketName(), attachmentBlobId)).blockOptional()) + .isNotEmpty(); + assertThat(Mono.from(blobStore.readReactive(blobStore.getDefaultBucketName(), messageBodyBlobId)).blockOptional()) + .isNotEmpty(); + }); + } + + @Test + void deleteMessageListenerShouldSucceedWhenDeleteMailboxHasALotOfMessages() throws Exception { + List messageIdList = Flux.range(0, 50) + .map(i -> Throwing.supplier(() -> inboxManager.appendMessage(MessageManager.AppendCommand.builder() + .build(ClassLoaderUtils.getSystemResourceAsByteArray("eml/emailWithOnlyAttachment.eml")), session)).get()) + .map(appendResult -> (PostgresMessageId) appendResult.getId().getMessageId()) + .collectList() + .block(); + + mailboxManager.deleteMailbox(inbox, session); + + assertThat(Flux.fromIterable(messageIdList) + .flatMap(msgId -> postgresMessageDAO.getBodyBlobId(msgId)) + .collectList().block()).isEmpty(); + } + + @Test + void deleteMailboxShouldCleanUpThreadData() throws Exception { + // append a message + MessageManager.AppendResult message = inboxManager.appendMessage(MessageManager.AppendCommand.from(Message.Builder.of() + .setSubject("Test") + .setMessageId("Message-ID") + .setField(new RawField("In-Reply-To", "someInReplyTo")) + .addField(new RawField("References", "references1")) + .addField(new RawField("References", "references2")) + .setBody("testmail", StandardCharsets.UTF_8)), session); + + Set hashMimeMessageIds = buildMimeMessageIdSet(Optional.of(new MimeMessageId("Message-ID")), + Optional.of(new MimeMessageId("someInReplyTo")), + Optional.of(List.of(new MimeMessageId("references1"), new MimeMessageId("references2")))) + .stream() + .map(mimeMessageId1 -> Hashing.murmur3_32_fixed().hashBytes(mimeMessageId1.getValue().getBytes()).asInt()) + .collect(Collectors.toSet()); + + mailboxManager.deleteMailbox(inbox, session); + + SoftAssertions.assertSoftly(softly -> { + softly.assertThat(threadDAO().findThreads(session.getUser(), hashMimeMessageIds).collectList().block()) + .isEmpty(); + }); + } + + @Test + void deleteMessageShouldCleanUpThreadData() throws Exception { + // append a message + MessageManager.AppendResult message = inboxManager.appendMessage(MessageManager.AppendCommand.from(Message.Builder.of() + .setSubject("Test") + .setMessageId("Message-ID") + .setField(new RawField("In-Reply-To", "someInReplyTo")) + .addField(new RawField("References", "references1")) + .addField(new RawField("References", "references2")) + .setBody("testmail", StandardCharsets.UTF_8)), session); + + Set hashMimeMessageIds = buildMimeMessageIdSet(Optional.of(new MimeMessageId("Message-ID")), + Optional.of(new MimeMessageId("someInReplyTo")), + Optional.of(List.of(new MimeMessageId("references1"), new MimeMessageId("references2")))) + .stream() + .map(mimeMessageId1 -> Hashing.murmur3_32_fixed().hashBytes(mimeMessageId1.getValue().getBytes()).asInt()) + .collect(Collectors.toSet()); + + inboxManager.delete(ImmutableList.of(message.getId().getUid()), session); + + SoftAssertions.assertSoftly(softly -> { + softly.assertThat(threadDAO().findThreads(session.getUser(), hashMimeMessageIds).collectList().block()) + .isEmpty(); + }); + } + + private Set buildMimeMessageIdSet(Optional mimeMessageId, Optional inReplyTo, Optional> references) { + Set mimeMessageIds = new HashSet<>(); + mimeMessageId.ifPresent(mimeMessageIds::add); + inReplyTo.ifPresent(mimeMessageIds::add); + references.ifPresent(mimeMessageIds::addAll); + return mimeMessageIds; + } +} diff --git a/mailbox/postgres/src/test/java/org/apache/james/mailbox/postgres/DeleteMessageListenerTest.java b/mailbox/postgres/src/test/java/org/apache/james/mailbox/postgres/DeleteMessageListenerTest.java new file mode 100644 index 00000000000..678ae41cbb7 --- /dev/null +++ b/mailbox/postgres/src/test/java/org/apache/james/mailbox/postgres/DeleteMessageListenerTest.java @@ -0,0 +1,123 @@ +/**************************************************************** + * 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.mailbox.postgres; + +import static org.apache.james.mailbox.postgres.PostgresMailboxManagerProvider.BLOB_ID_FACTORY; + +import java.time.Clock; +import java.time.Instant; + +import org.apache.james.backends.postgres.PostgresConfiguration; +import org.apache.james.backends.postgres.PostgresExtension; +import org.apache.james.blob.api.BlobStore; +import org.apache.james.blob.api.BucketName; +import org.apache.james.blob.memory.MemoryBlobStoreDAO; +import org.apache.james.events.EventBusTestFixture; +import org.apache.james.events.InVMEventBus; +import org.apache.james.events.MemoryEventDeadLetters; +import org.apache.james.events.delivery.InVmEventDelivery; +import org.apache.james.mailbox.acl.MailboxACLResolver; +import org.apache.james.mailbox.acl.UnionMailboxACLResolver; +import org.apache.james.mailbox.postgres.mail.dao.PostgresAttachmentDAO; +import org.apache.james.mailbox.postgres.mail.dao.PostgresMailboxMessageDAO; +import org.apache.james.mailbox.postgres.mail.dao.PostgresMessageDAO; +import org.apache.james.mailbox.postgres.mail.dao.PostgresThreadDAO; +import org.apache.james.mailbox.store.PreDeletionHooks; +import org.apache.james.mailbox.store.SessionProviderImpl; +import org.apache.james.mailbox.store.StoreMailboxAnnotationManager; +import org.apache.james.mailbox.store.StoreRightManager; +import org.apache.james.mailbox.store.extractor.DefaultTextExtractor; +import org.apache.james.mailbox.store.mail.model.impl.MessageParser; +import org.apache.james.mailbox.store.quota.QuotaComponents; +import org.apache.james.mailbox.store.search.MessageSearchIndex; +import org.apache.james.mailbox.store.search.SimpleMessageSearchIndex; +import org.apache.james.metrics.tests.RecordingMetricFactory; +import org.apache.james.server.blob.deduplication.PassThroughBlobStore; +import org.apache.james.utils.UpdatableTickingClock; +import org.junit.jupiter.api.BeforeAll; +import org.junit.jupiter.api.extension.RegisterExtension; + +public class DeleteMessageListenerTest extends DeleteMessageListenerContract { + @RegisterExtension + static PostgresExtension postgresExtension = PostgresExtension.withoutRowLevelSecurity(PostgresMailboxAggregateModule.MODULE); + + private static PostgresMailboxManager mailboxManager; + private static BlobStore blobStore; + + @BeforeAll + static void beforeAll() { + blobStore = new PassThroughBlobStore(new MemoryBlobStoreDAO(), BucketName.DEFAULT, BLOB_ID_FACTORY); + + PostgresMailboxSessionMapperFactory mapperFactory = new PostgresMailboxSessionMapperFactory( + postgresExtension.getExecutorFactory(), + Clock.systemUTC(), + blobStore, + BLOB_ID_FACTORY, + PostgresConfiguration.builder().username("a").password("a").build()); + + MailboxACLResolver aclResolver = new UnionMailboxACLResolver(); + MessageParser messageParser = new MessageParser(); + + InVMEventBus eventBus = new InVMEventBus(new InVmEventDelivery(new RecordingMetricFactory()), EventBusTestFixture.RETRY_BACKOFF_CONFIGURATION, new MemoryEventDeadLetters()); + StoreRightManager storeRightManager = new StoreRightManager(mapperFactory, aclResolver, eventBus); + StoreMailboxAnnotationManager annotationManager = new StoreMailboxAnnotationManager(mapperFactory, storeRightManager, 3, 30); + SessionProviderImpl sessionProvider = new SessionProviderImpl(null, null); + QuotaComponents quotaComponents = QuotaComponents.disabled(sessionProvider, mapperFactory); + MessageSearchIndex index = new SimpleMessageSearchIndex(mapperFactory, mapperFactory, new DefaultTextExtractor(), new UnsupportAttachmentContentLoader()); + + eventBus.register(mapperFactory.deleteMessageListener()); + + mailboxManager = new PostgresMailboxManager(mapperFactory, sessionProvider, + messageParser, new PostgresMessageId.Factory(), + eventBus, annotationManager, + storeRightManager, quotaComponents, index, new PostgresThreadIdGuessingAlgorithm(new PostgresThreadDAO.Factory(postgresExtension.getExecutorFactory())), + PreDeletionHooks.NO_PRE_DELETION_HOOK, new UpdatableTickingClock(Instant.now())); + } + + @Override + PostgresMailboxManager provideMailboxManager() { + return mailboxManager; + } + + @Override + PostgresMessageDAO providePostgresMessageDAO() { + return new PostgresMessageDAO(postgresExtension.getDefaultPostgresExecutor(), BLOB_ID_FACTORY); + } + + @Override + PostgresMailboxMessageDAO providePostgresMailboxMessageDAO() { + return new PostgresMailboxMessageDAO(postgresExtension.getDefaultPostgresExecutor()); + } + + @Override + PostgresThreadDAO threadDAO() { + return new PostgresThreadDAO(postgresExtension.getDefaultPostgresExecutor()); + } + + @Override + PostgresAttachmentDAO attachmentDAO() { + return new PostgresAttachmentDAO(postgresExtension.getDefaultPostgresExecutor(), BLOB_ID_FACTORY); + } + + @Override + BlobStore blobStore() { + return blobStore; + } +} diff --git a/mailbox/postgres/src/test/java/org/apache/james/mailbox/postgres/DeleteMessageListenerWithRLSTest.java b/mailbox/postgres/src/test/java/org/apache/james/mailbox/postgres/DeleteMessageListenerWithRLSTest.java new file mode 100644 index 00000000000..9c0fd1d683f --- /dev/null +++ b/mailbox/postgres/src/test/java/org/apache/james/mailbox/postgres/DeleteMessageListenerWithRLSTest.java @@ -0,0 +1,134 @@ +/**************************************************************** + * 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.mailbox.postgres; + +import static org.apache.james.mailbox.postgres.PostgresMailboxManagerProvider.BLOB_ID_FACTORY; + +import java.time.Clock; +import java.time.Instant; +import java.util.UUID; + +import org.apache.james.backends.postgres.PostgresConfiguration; +import org.apache.james.backends.postgres.PostgresExtension; +import org.apache.james.blob.api.BlobId; +import org.apache.james.blob.api.BlobStore; +import org.apache.james.blob.api.BucketName; +import org.apache.james.blob.api.PlainBlobId; +import org.apache.james.blob.memory.MemoryBlobStoreDAO; +import org.apache.james.core.Username; +import org.apache.james.events.EventBusTestFixture; +import org.apache.james.events.InVMEventBus; +import org.apache.james.events.MemoryEventDeadLetters; +import org.apache.james.events.delivery.InVmEventDelivery; +import org.apache.james.mailbox.acl.MailboxACLResolver; +import org.apache.james.mailbox.acl.UnionMailboxACLResolver; +import org.apache.james.mailbox.postgres.mail.dao.PostgresAttachmentDAO; +import org.apache.james.mailbox.postgres.mail.dao.PostgresMailboxMessageDAO; +import org.apache.james.mailbox.postgres.mail.dao.PostgresMessageDAO; +import org.apache.james.mailbox.postgres.mail.dao.PostgresThreadDAO; +import org.apache.james.mailbox.store.PreDeletionHooks; +import org.apache.james.mailbox.store.SessionProviderImpl; +import org.apache.james.mailbox.store.StoreMailboxAnnotationManager; +import org.apache.james.mailbox.store.StoreRightManager; +import org.apache.james.mailbox.store.extractor.DefaultTextExtractor; +import org.apache.james.mailbox.store.mail.model.impl.MessageParser; +import org.apache.james.mailbox.store.quota.QuotaComponents; +import org.apache.james.mailbox.store.search.MessageSearchIndex; +import org.apache.james.mailbox.store.search.SimpleMessageSearchIndex; +import org.apache.james.metrics.tests.RecordingMetricFactory; +import org.apache.james.server.blob.deduplication.PassThroughBlobStore; +import org.apache.james.utils.UpdatableTickingClock; +import org.junit.jupiter.api.BeforeAll; +import org.junit.jupiter.api.extension.RegisterExtension; + +public class DeleteMessageListenerWithRLSTest extends DeleteMessageListenerContract { + + @RegisterExtension + static PostgresExtension postgresExtension = PostgresExtension.withRowLevelSecurity(PostgresMailboxAggregateModule.MODULE); + + private static PostgresMailboxManager mailboxManager; + private static BlobStore blobStore; + + @BeforeAll + static void beforeAll() { + blobStore = new PassThroughBlobStore(new MemoryBlobStoreDAO(), BucketName.DEFAULT, BLOB_ID_FACTORY); + BlobId.Factory blobIdFactory = new PlainBlobId.Factory(); + + PostgresMailboxSessionMapperFactory mapperFactory = new PostgresMailboxSessionMapperFactory( + postgresExtension.getExecutorFactory(), + Clock.systemUTC(), + blobStore, + blobIdFactory, + PostgresConfiguration.builder().username("a").password("a").build()); + + MailboxACLResolver aclResolver = new UnionMailboxACLResolver(); + MessageParser messageParser = new MessageParser(); + + InVMEventBus eventBus = new InVMEventBus(new InVmEventDelivery(new RecordingMetricFactory()), EventBusTestFixture.RETRY_BACKOFF_CONFIGURATION, new MemoryEventDeadLetters()); + StoreRightManager storeRightManager = new StoreRightManager(mapperFactory, aclResolver, eventBus); + StoreMailboxAnnotationManager annotationManager = new StoreMailboxAnnotationManager(mapperFactory, storeRightManager, 3, 30); + SessionProviderImpl sessionProvider = new SessionProviderImpl(null, null); + QuotaComponents quotaComponents = QuotaComponents.disabled(sessionProvider, mapperFactory); + MessageSearchIndex index = new SimpleMessageSearchIndex(mapperFactory, mapperFactory, new DefaultTextExtractor(), new UnsupportAttachmentContentLoader()); + + eventBus.register(mapperFactory.deleteMessageListener()); + + mailboxManager = new PostgresMailboxManager(mapperFactory, sessionProvider, + messageParser, new PostgresMessageId.Factory(), + eventBus, annotationManager, + storeRightManager, quotaComponents, index, new PostgresThreadIdGuessingAlgorithm(new PostgresThreadDAO.Factory(postgresExtension.getExecutorFactory())), + PreDeletionHooks.NO_PRE_DELETION_HOOK, new UpdatableTickingClock(Instant.now())); + } + + @Override + PostgresMailboxManager provideMailboxManager() { + return mailboxManager; + } + + @Override + PostgresMessageDAO providePostgresMessageDAO() { + return new PostgresMessageDAO(postgresExtension.getExecutorFactory().create(getUsername().getDomainPart()), BLOB_ID_FACTORY); + } + + @Override + PostgresMailboxMessageDAO providePostgresMailboxMessageDAO() { + return new PostgresMailboxMessageDAO(postgresExtension.getExecutorFactory().create(getUsername().getDomainPart())); + } + + @Override + PostgresThreadDAO threadDAO() { + return new PostgresThreadDAO.Factory(postgresExtension.getExecutorFactory()).create(getUsername().getDomainPart()); + } + + @Override + PostgresAttachmentDAO attachmentDAO() { + return new PostgresAttachmentDAO(postgresExtension.getExecutorFactory().create(getUsername().getDomainPart()), BLOB_ID_FACTORY); + } + + @Override + BlobStore blobStore() { + return blobStore; + } + + @Override + protected Username getUsername() { + return Username.of("userHasDomain" + UUID.randomUUID() + "@domain1.tld"); + } +} diff --git a/mailbox/postgres/src/test/java/org/apache/james/mailbox/postgres/PostgresCombinationManagerTest.java b/mailbox/postgres/src/test/java/org/apache/james/mailbox/postgres/PostgresCombinationManagerTest.java new file mode 100644 index 00000000000..b2bf09c39c1 --- /dev/null +++ b/mailbox/postgres/src/test/java/org/apache/james/mailbox/postgres/PostgresCombinationManagerTest.java @@ -0,0 +1,42 @@ +/**************************************************************** + * 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.mailbox.postgres; + +import org.apache.james.backends.postgres.PostgresExtension; +import org.apache.james.events.EventBusTestFixture; +import org.apache.james.events.InVMEventBus; +import org.apache.james.events.MemoryEventDeadLetters; +import org.apache.james.events.delivery.InVmEventDelivery; +import org.apache.james.mailbox.store.AbstractCombinationManagerTest; +import org.apache.james.mailbox.store.CombinationManagerTestSystem; +import org.apache.james.mailbox.store.quota.NoQuotaManager; +import org.apache.james.metrics.tests.RecordingMetricFactory; +import org.junit.jupiter.api.extension.RegisterExtension; + +public class PostgresCombinationManagerTest extends AbstractCombinationManagerTest { + @RegisterExtension + static PostgresExtension postgresExtension = PostgresExtension.withoutRowLevelSecurity(PostgresMailboxAggregateModule.MODULE); + + @Override + public CombinationManagerTestSystem createTestingData() { + InVMEventBus eventBus = new InVMEventBus(new InVmEventDelivery(new RecordingMetricFactory()), EventBusTestFixture.RETRY_BACKOFF_CONFIGURATION, new MemoryEventDeadLetters()); + return PostgresCombinationManagerTestSystem.createTestingData(postgresExtension, new NoQuotaManager(), eventBus); + } +} diff --git a/mailbox/postgres/src/test/java/org/apache/james/mailbox/postgres/PostgresCombinationManagerTestSystem.java b/mailbox/postgres/src/test/java/org/apache/james/mailbox/postgres/PostgresCombinationManagerTestSystem.java new file mode 100644 index 00000000000..d0421c9c4f2 --- /dev/null +++ b/mailbox/postgres/src/test/java/org/apache/james/mailbox/postgres/PostgresCombinationManagerTestSystem.java @@ -0,0 +1,66 @@ +/**************************************************************** + * 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.mailbox.postgres; + +import org.apache.james.backends.postgres.PostgresExtension; +import org.apache.james.events.EventBus; +import org.apache.james.mailbox.MailboxManager; +import org.apache.james.mailbox.MailboxSession; +import org.apache.james.mailbox.MessageIdManager; +import org.apache.james.mailbox.MessageManager; +import org.apache.james.mailbox.exception.MailboxException; +import org.apache.james.mailbox.exception.MailboxNotFoundException; +import org.apache.james.mailbox.model.Mailbox; +import org.apache.james.mailbox.model.MailboxPath; +import org.apache.james.mailbox.quota.QuotaManager; +import org.apache.james.mailbox.store.CombinationManagerTestSystem; +import org.apache.james.mailbox.store.PreDeletionHooks; + +public class PostgresCombinationManagerTestSystem extends CombinationManagerTestSystem { + private final PostgresMailboxSessionMapperFactory mapperFactory; + private final PostgresMailboxManager postgresMailboxManager; + + public static CombinationManagerTestSystem createTestingData(PostgresExtension postgresExtension, QuotaManager quotaManager, EventBus eventBus) { + PostgresMailboxSessionMapperFactory mapperFactory = PostgresTestSystemFixture.createMapperFactory(postgresExtension); + + return new PostgresCombinationManagerTestSystem(PostgresTestSystemFixture.createMessageIdManager(mapperFactory, quotaManager, eventBus, PreDeletionHooks.NO_PRE_DELETION_HOOK), + mapperFactory, + PostgresTestSystemFixture.createMailboxManager(mapperFactory)); + } + + private PostgresCombinationManagerTestSystem(MessageIdManager messageIdManager, PostgresMailboxSessionMapperFactory mapperFactory, MailboxManager postgresMailboxManager) { + super(postgresMailboxManager, messageIdManager); + this.mapperFactory = mapperFactory; + this.postgresMailboxManager = (PostgresMailboxManager) postgresMailboxManager; + } + + @Override + public Mailbox createMailbox(MailboxPath mailboxPath, MailboxSession session) throws MailboxException { + postgresMailboxManager.createMailbox(mailboxPath, session); + return mapperFactory.getMailboxMapper(session).findMailboxByPath(mailboxPath) + .blockOptional() + .orElseThrow(() -> new MailboxNotFoundException(mailboxPath)); + } + + @Override + public MessageManager createMessageManager(Mailbox mailbox, MailboxSession session) { + return postgresMailboxManager.createMessageManager(mailbox, session); + } +} diff --git a/mailbox/postgres/src/test/java/org/apache/james/mailbox/postgres/PostgresMailboxManagerAttachmentTest.java b/mailbox/postgres/src/test/java/org/apache/james/mailbox/postgres/PostgresMailboxManagerAttachmentTest.java new file mode 100644 index 00000000000..ed213baafeb --- /dev/null +++ b/mailbox/postgres/src/test/java/org/apache/james/mailbox/postgres/PostgresMailboxManagerAttachmentTest.java @@ -0,0 +1,153 @@ +/**************************************************************** + * 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.mailbox.postgres; + +import static org.mockito.ArgumentMatchers.any; +import static org.mockito.Mockito.mock; +import static org.mockito.Mockito.when; + +import java.io.InputStream; +import java.time.Clock; +import java.time.Instant; + +import org.apache.james.backends.postgres.PostgresConfiguration; +import org.apache.james.backends.postgres.PostgresExtension; +import org.apache.james.blob.api.BlobId; +import org.apache.james.blob.api.BucketName; +import org.apache.james.blob.api.PlainBlobId; +import org.apache.james.blob.memory.MemoryBlobStoreDAO; +import org.apache.james.events.EventBusTestFixture; +import org.apache.james.events.InVMEventBus; +import org.apache.james.events.MemoryEventDeadLetters; +import org.apache.james.events.delivery.InVmEventDelivery; +import org.apache.james.mailbox.MailboxManager; +import org.apache.james.mailbox.MessageIdManager; +import org.apache.james.mailbox.acl.MailboxACLResolver; +import org.apache.james.mailbox.acl.UnionMailboxACLResolver; +import org.apache.james.mailbox.postgres.mail.dao.PostgresAttachmentDAO; +import org.apache.james.mailbox.postgres.mail.dao.PostgresMailboxMessageDAO; +import org.apache.james.mailbox.postgres.mail.dao.PostgresMessageDAO; +import org.apache.james.mailbox.postgres.mail.dao.PostgresThreadDAO; +import org.apache.james.mailbox.quota.QuotaRootResolver; +import org.apache.james.mailbox.store.AbstractMailboxManagerAttachmentTest; +import org.apache.james.mailbox.store.MailboxSessionMapperFactory; +import org.apache.james.mailbox.store.PreDeletionHooks; +import org.apache.james.mailbox.store.SessionProviderImpl; +import org.apache.james.mailbox.store.StoreAttachmentManager; +import org.apache.james.mailbox.store.StoreMailboxAnnotationManager; +import org.apache.james.mailbox.store.StoreMessageIdManager; +import org.apache.james.mailbox.store.StoreRightManager; +import org.apache.james.mailbox.store.extractor.DefaultTextExtractor; +import org.apache.james.mailbox.store.mail.AttachmentMapperFactory; +import org.apache.james.mailbox.store.mail.NaiveThreadIdGuessingAlgorithm; +import org.apache.james.mailbox.store.mail.model.impl.MessageParser; +import org.apache.james.mailbox.store.quota.NoQuotaManager; +import org.apache.james.mailbox.store.quota.QuotaComponents; +import org.apache.james.mailbox.store.search.MessageSearchIndex; +import org.apache.james.mailbox.store.search.SimpleMessageSearchIndex; +import org.apache.james.metrics.tests.RecordingMetricFactory; +import org.apache.james.server.blob.deduplication.DeDuplicationBlobStore; +import org.apache.james.utils.UpdatableTickingClock; +import org.junit.jupiter.api.BeforeEach; +import org.junit.jupiter.api.extension.RegisterExtension; + +import com.google.common.collect.ImmutableSet; + +public class PostgresMailboxManagerAttachmentTest extends AbstractMailboxManagerAttachmentTest { + + @RegisterExtension + static PostgresExtension postgresExtension = PostgresExtension.withoutRowLevelSecurity(PostgresMailboxAggregateModule.MODULE); + private static PostgresMailboxManager mailboxManager; + private static PostgresMailboxManager parseFailingMailboxManager; + private static PostgresMailboxSessionMapperFactory mapperFactory; + + @BeforeEach + void beforeAll() throws Exception { + BlobId.Factory blobIdFactory = new PlainBlobId.Factory(); + DeDuplicationBlobStore blobStore = new DeDuplicationBlobStore(new MemoryBlobStoreDAO(), BucketName.DEFAULT, blobIdFactory); + mapperFactory = new PostgresMailboxSessionMapperFactory(postgresExtension.getExecutorFactory(), Clock.systemUTC(), blobStore, blobIdFactory, + PostgresConfiguration.builder().username("a").password("a").build()); + + MailboxACLResolver aclResolver = new UnionMailboxACLResolver(); + MessageParser messageParser = new MessageParser(); + + InVMEventBus eventBus = new InVMEventBus(new InVmEventDelivery(new RecordingMetricFactory()), EventBusTestFixture.RETRY_BACKOFF_CONFIGURATION, new MemoryEventDeadLetters()); + StoreRightManager storeRightManager = new StoreRightManager(mapperFactory, aclResolver, eventBus); + StoreMailboxAnnotationManager annotationManager = new StoreMailboxAnnotationManager(mapperFactory, storeRightManager, 3, 30); + SessionProviderImpl sessionProvider = new SessionProviderImpl(null, null); + QuotaComponents quotaComponents = QuotaComponents.disabled(sessionProvider, mapperFactory); + + MessageIdManager messageIdManager = new StoreMessageIdManager(storeRightManager, mapperFactory, + eventBus, new NoQuotaManager(), mock(QuotaRootResolver.class), PreDeletionHooks.NO_PRE_DELETION_HOOK); + + StoreAttachmentManager storeAttachmentManager = new StoreAttachmentManager(mapperFactory, messageIdManager); + + MessageSearchIndex index = new SimpleMessageSearchIndex(mapperFactory, mapperFactory, new DefaultTextExtractor(), storeAttachmentManager); + + PostgresMessageDAO.Factory postgresMessageDAOFactory = new PostgresMessageDAO.Factory(blobIdFactory, postgresExtension.getExecutorFactory()); + PostgresMailboxMessageDAO.Factory postgresMailboxMessageDAOFactory = new PostgresMailboxMessageDAO.Factory(postgresExtension.getExecutorFactory()); + PostgresAttachmentDAO.Factory attachmentDAOFactory = new PostgresAttachmentDAO.Factory(postgresExtension.getExecutorFactory(), blobIdFactory); + PostgresThreadDAO.Factory threadDAOFactory = new PostgresThreadDAO.Factory(postgresExtension.getExecutorFactory()); + + eventBus.register(new DeleteMessageListener(blobStore, postgresMailboxMessageDAOFactory, postgresMessageDAOFactory, + attachmentDAOFactory, threadDAOFactory, ImmutableSet.of())); + + mailboxManager = new PostgresMailboxManager(mapperFactory, sessionProvider, + messageParser, new PostgresMessageId.Factory(), + eventBus, annotationManager, + storeRightManager, quotaComponents, index, new NaiveThreadIdGuessingAlgorithm(), + PreDeletionHooks.NO_PRE_DELETION_HOOK, + new UpdatableTickingClock(Instant.now())); + + MessageParser failingMessageParser = mock(MessageParser.class); + when(failingMessageParser.retrieveAttachments(any(InputStream.class))) + .thenThrow(new RuntimeException("Message parser set to fail")); + + + parseFailingMailboxManager = new PostgresMailboxManager(mapperFactory, sessionProvider, + failingMessageParser, new PostgresMessageId.Factory(), + eventBus, annotationManager, + storeRightManager, quotaComponents, index, new NaiveThreadIdGuessingAlgorithm(), + PreDeletionHooks.NO_PRE_DELETION_HOOK, + new UpdatableTickingClock(Instant.now())); + + super.setUp(); + } + + @Override + protected MailboxManager getMailboxManager() { + return mailboxManager; + } + + @Override + protected MailboxManager getParseFailingMailboxManager() { + return parseFailingMailboxManager; + } + + @Override + protected MailboxSessionMapperFactory getMailboxSessionMapperFactory() { + return mapperFactory; + } + + @Override + protected AttachmentMapperFactory getAttachmentMapperFactory() { + return mapperFactory; + } +} diff --git a/mailbox/postgres/src/test/java/org/apache/james/mailbox/postgres/PostgresMailboxManagerProvider.java b/mailbox/postgres/src/test/java/org/apache/james/mailbox/postgres/PostgresMailboxManagerProvider.java new file mode 100644 index 00000000000..f83139a588b --- /dev/null +++ b/mailbox/postgres/src/test/java/org/apache/james/mailbox/postgres/PostgresMailboxManagerProvider.java @@ -0,0 +1,105 @@ +/**************************************************************** + * 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.mailbox.postgres; + +import java.time.Clock; +import java.time.Instant; + +import org.apache.james.backends.postgres.PostgresConfiguration; +import org.apache.james.backends.postgres.PostgresExtension; +import org.apache.james.blob.api.BlobId; +import org.apache.james.blob.api.BucketName; +import org.apache.james.blob.api.PlainBlobId; +import org.apache.james.blob.memory.MemoryBlobStoreDAO; +import org.apache.james.events.EventBusTestFixture; +import org.apache.james.events.InVMEventBus; +import org.apache.james.events.MemoryEventDeadLetters; +import org.apache.james.events.delivery.InVmEventDelivery; +import org.apache.james.mailbox.Authenticator; +import org.apache.james.mailbox.Authorizator; +import org.apache.james.mailbox.acl.MailboxACLResolver; +import org.apache.james.mailbox.acl.UnionMailboxACLResolver; +import org.apache.james.mailbox.store.PreDeletionHooks; +import org.apache.james.mailbox.store.SessionProviderImpl; +import org.apache.james.mailbox.store.StoreMailboxAnnotationManager; +import org.apache.james.mailbox.store.StoreRightManager; +import org.apache.james.mailbox.store.extractor.DefaultTextExtractor; +import org.apache.james.mailbox.store.mail.NaiveThreadIdGuessingAlgorithm; +import org.apache.james.mailbox.store.mail.model.impl.MessageParser; +import org.apache.james.mailbox.store.quota.QuotaComponents; +import org.apache.james.mailbox.store.search.MessageSearchIndex; +import org.apache.james.mailbox.store.search.SimpleMessageSearchIndex; +import org.apache.james.metrics.tests.RecordingMetricFactory; +import org.apache.james.server.blob.deduplication.DeDuplicationBlobStore; +import org.apache.james.utils.UpdatableTickingClock; + +public class PostgresMailboxManagerProvider { + + private static final int LIMIT_ANNOTATIONS = 3; + private static final int LIMIT_ANNOTATION_SIZE = 30; + + public static final BlobId.Factory BLOB_ID_FACTORY = new PlainBlobId.Factory(); + + public static PostgresMailboxManager provideMailboxManager(PostgresExtension postgresExtension, PreDeletionHooks preDeletionHooks) { + DeDuplicationBlobStore blobStore = new DeDuplicationBlobStore(new MemoryBlobStoreDAO(), BucketName.DEFAULT, BLOB_ID_FACTORY); + PostgresMailboxSessionMapperFactory mapperFactory = provideMailboxSessionMapperFactory(postgresExtension, BLOB_ID_FACTORY, blobStore); + + MailboxACLResolver aclResolver = new UnionMailboxACLResolver(); + MessageParser messageParser = new MessageParser(); + + Authenticator noAuthenticator = null; + Authorizator noAuthorizator = null; + + InVMEventBus eventBus = new InVMEventBus(new InVmEventDelivery(new RecordingMetricFactory()), EventBusTestFixture.RETRY_BACKOFF_CONFIGURATION, new MemoryEventDeadLetters()); + StoreRightManager storeRightManager = new StoreRightManager(mapperFactory, aclResolver, eventBus); + StoreMailboxAnnotationManager annotationManager = new StoreMailboxAnnotationManager(mapperFactory, storeRightManager, + LIMIT_ANNOTATIONS, LIMIT_ANNOTATION_SIZE); + SessionProviderImpl sessionProvider = new SessionProviderImpl(noAuthenticator, noAuthorizator); + QuotaComponents quotaComponents = QuotaComponents.disabled(sessionProvider, mapperFactory); + MessageSearchIndex index = new SimpleMessageSearchIndex(mapperFactory, mapperFactory, new DefaultTextExtractor(), new UnsupportAttachmentContentLoader()); + + eventBus.register(mapperFactory.deleteMessageListener()); + + return new PostgresMailboxManager(mapperFactory, sessionProvider, + messageParser, new PostgresMessageId.Factory(), + eventBus, annotationManager, + storeRightManager, quotaComponents, index, new NaiveThreadIdGuessingAlgorithm(), + preDeletionHooks, new UpdatableTickingClock(Instant.now())); + } + + public static PostgresMailboxSessionMapperFactory provideMailboxSessionMapperFactory(PostgresExtension postgresExtension) { + BlobId.Factory blobIdFactory = new PlainBlobId.Factory(); + DeDuplicationBlobStore blobStore = new DeDuplicationBlobStore(new MemoryBlobStoreDAO(), BucketName.DEFAULT, blobIdFactory); + + return provideMailboxSessionMapperFactory(postgresExtension, blobIdFactory, blobStore); + } + + public static PostgresMailboxSessionMapperFactory provideMailboxSessionMapperFactory(PostgresExtension postgresExtension, + BlobId.Factory blobIdFactory, + DeDuplicationBlobStore blobStore) { + return new PostgresMailboxSessionMapperFactory( + postgresExtension.getExecutorFactory(), + Clock.systemUTC(), + blobStore, + blobIdFactory, + PostgresConfiguration.builder().username("a").password("a").build()); + } + +} diff --git a/mailbox/postgres/src/test/java/org/apache/james/mailbox/postgres/PostgresMailboxManagerStressTest.java b/mailbox/postgres/src/test/java/org/apache/james/mailbox/postgres/PostgresMailboxManagerStressTest.java new file mode 100644 index 00000000000..46dd5731757 --- /dev/null +++ b/mailbox/postgres/src/test/java/org/apache/james/mailbox/postgres/PostgresMailboxManagerStressTest.java @@ -0,0 +1,56 @@ +/**************************************************************** + * 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.mailbox.postgres; + +import java.util.Optional; + +import org.apache.james.backends.postgres.PostgresExtension; +import org.apache.james.events.EventBus; +import org.apache.james.mailbox.MailboxManagerStressContract; +import org.apache.james.mailbox.store.PreDeletionHooks; +import org.junit.jupiter.api.BeforeEach; +import org.junit.jupiter.api.extension.RegisterExtension; + +class PostgresMailboxManagerStressTest implements MailboxManagerStressContract { + + @RegisterExtension + static PostgresExtension postgresExtension = PostgresExtension.withoutRowLevelSecurity(PostgresMailboxAggregateModule.MODULE); + + Optional mailboxManager = Optional.empty(); + + @Override + public PostgresMailboxManager getManager() { + return mailboxManager.get(); + } + + @Override + public EventBus retrieveEventBus() { + return getManager().getEventBus(); + } + + @BeforeEach + void setUp() { + if (mailboxManager.isEmpty()) { + mailboxManager = Optional.of(PostgresMailboxManagerProvider.provideMailboxManager(postgresExtension, + PreDeletionHooks.NO_PRE_DELETION_HOOK)); + } + } + +} diff --git a/mailbox/postgres/src/test/java/org/apache/james/mailbox/postgres/PostgresMailboxManagerTest.java b/mailbox/postgres/src/test/java/org/apache/james/mailbox/postgres/PostgresMailboxManagerTest.java new file mode 100644 index 00000000000..3acf0faf7f9 --- /dev/null +++ b/mailbox/postgres/src/test/java/org/apache/james/mailbox/postgres/PostgresMailboxManagerTest.java @@ -0,0 +1,98 @@ +/**************************************************************** + * 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.mailbox.postgres; + +import static org.assertj.core.api.AssertionsForInterfaceTypes.assertThat; + +import java.nio.charset.StandardCharsets; +import java.util.List; +import java.util.Optional; + +import jakarta.mail.Flags; + +import org.apache.james.backends.postgres.PostgresExtension; +import org.apache.james.core.Username; +import org.apache.james.events.EventBus; +import org.apache.james.mailbox.MailboxManagerTest; +import org.apache.james.mailbox.MailboxSession; +import org.apache.james.mailbox.MessageManager; +import org.apache.james.mailbox.MessageUid; +import org.apache.james.mailbox.SubscriptionManager; +import org.apache.james.mailbox.model.MailboxPath; +import org.apache.james.mailbox.model.MessageRange; +import org.apache.james.mailbox.store.PreDeletionHooks; +import org.apache.james.mailbox.store.StoreSubscriptionManager; +import org.apache.james.metrics.tests.RecordingMetricFactory; +import org.apache.james.mime4j.dom.Message; +import org.junit.jupiter.api.Test; +import org.junit.jupiter.api.extension.RegisterExtension; + +import reactor.core.publisher.Flux; +import reactor.core.publisher.Mono; + +class PostgresMailboxManagerTest extends MailboxManagerTest { + @RegisterExtension + static PostgresExtension postgresExtension = PostgresExtension.withoutRowLevelSecurity(PostgresMailboxAggregateModule.MODULE); + + Optional mailboxManager = Optional.empty(); + + @Override + protected PostgresMailboxManager provideMailboxManager() { + if (mailboxManager.isEmpty()) { + mailboxManager = Optional.of(PostgresMailboxManagerProvider.provideMailboxManager(postgresExtension, + new PreDeletionHooks(preDeletionHooks(), new RecordingMetricFactory()))); + } + return mailboxManager.get(); + } + + @Override + protected SubscriptionManager provideSubscriptionManager() { + return new StoreSubscriptionManager(provideMailboxManager().getMapperFactory(), provideMailboxManager().getMapperFactory(), provideMailboxManager().getEventBus()); + } + + @Override + protected EventBus retrieveEventBus(PostgresMailboxManager mailboxManager) { + return mailboxManager.getEventBus(); + } + + @Test + void expungeMessageShouldCorrectWhenALotOfMessages() throws Exception { + // Given a mailbox with 6000 messages + Username username = Username.of("tung"); + PostgresMailboxManager postgresMailboxManager = mailboxManager.get(); + MailboxSession session = postgresMailboxManager.createSystemSession(username); + postgresMailboxManager.createMailbox(MailboxPath.inbox(username), session).get(); + MessageManager inboxManager = postgresMailboxManager.getMailbox(MailboxPath.inbox(session), session); + + int totalMessages = 6000; + Flux.range(0, totalMessages) + .flatMap(i -> Mono.fromCallable(() -> inboxManager.appendMessage(MessageManager.AppendCommand.builder().build(Message.Builder.of() + .setSubject("test" + i) + .setBody("testmail" + i, StandardCharsets.UTF_8)), session)), 100) + .collectList().block(); + // When expunge all messages + inboxManager.setFlags(new Flags(Flags.Flag.DELETED), MessageManager.FlagsUpdateMode.ADD, MessageRange.all(), session); + + List expungeList = inboxManager.expungeReactive(MessageRange.all(), session) + .collectList().block(); + + // Then all messages are expunged + assertThat(expungeList).hasSize(totalMessages); + } +} diff --git a/mailbox/postgres/src/test/java/org/apache/james/mailbox/postgres/PostgresMessageIdManagerQuotaTest.java b/mailbox/postgres/src/test/java/org/apache/james/mailbox/postgres/PostgresMessageIdManagerQuotaTest.java new file mode 100644 index 00000000000..40bb250da09 --- /dev/null +++ b/mailbox/postgres/src/test/java/org/apache/james/mailbox/postgres/PostgresMessageIdManagerQuotaTest.java @@ -0,0 +1,58 @@ +/**************************************************************** + * 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.mailbox.postgres; + +import org.apache.james.backends.postgres.PostgresExtension; +import org.apache.james.backends.postgres.PostgresModule; +import org.apache.james.backends.postgres.quota.PostgresQuotaModule; +import org.apache.james.mailbox.quota.CurrentQuotaManager; +import org.apache.james.mailbox.quota.MaxQuotaManager; +import org.apache.james.mailbox.quota.QuotaManager; +import org.apache.james.mailbox.store.AbstractMessageIdManagerQuotaTest; +import org.apache.james.mailbox.store.MessageIdManagerTestSystem; +import org.apache.james.mailbox.store.quota.StoreQuotaManager; +import org.junit.jupiter.api.extension.RegisterExtension; + +public class PostgresMessageIdManagerQuotaTest extends AbstractMessageIdManagerQuotaTest { + @RegisterExtension + static PostgresExtension postgresExtension = PostgresExtension.withoutRowLevelSecurity(PostgresModule.aggregateModules( + PostgresMailboxAggregateModule.MODULE, + PostgresQuotaModule.MODULE)); + + @Override + protected MessageIdManagerTestSystem createTestSystem(QuotaManager quotaManager, CurrentQuotaManager currentQuotaManager) throws Exception { + return PostgresMessageIdManagerTestSystem.createTestingDataWithQuota(postgresExtension, quotaManager, currentQuotaManager); + } + + @Override + protected MaxQuotaManager createMaxQuotaManager() { + return PostgresTestSystemFixture.createMaxQuotaManager(postgresExtension); + } + + @Override + protected QuotaManager createQuotaManager(MaxQuotaManager maxQuotaManager, CurrentQuotaManager currentQuotaManager) { + return new StoreQuotaManager(currentQuotaManager, maxQuotaManager); + } + + @Override + protected CurrentQuotaManager createCurrentQuotaManager() { + return PostgresTestSystemFixture.createCurrentQuotaManager(postgresExtension); + } +} diff --git a/mailbox/postgres/src/test/java/org/apache/james/mailbox/postgres/PostgresMessageIdManagerSideEffectTest.java b/mailbox/postgres/src/test/java/org/apache/james/mailbox/postgres/PostgresMessageIdManagerSideEffectTest.java new file mode 100644 index 00000000000..35824217c7b --- /dev/null +++ b/mailbox/postgres/src/test/java/org/apache/james/mailbox/postgres/PostgresMessageIdManagerSideEffectTest.java @@ -0,0 +1,40 @@ +/**************************************************************** + * 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.mailbox.postgres; + +import java.util.Set; + +import org.apache.james.backends.postgres.PostgresExtension; +import org.apache.james.events.EventBus; +import org.apache.james.mailbox.extension.PreDeletionHook; +import org.apache.james.mailbox.quota.QuotaManager; +import org.apache.james.mailbox.store.AbstractMessageIdManagerSideEffectTest; +import org.apache.james.mailbox.store.MessageIdManagerTestSystem; +import org.junit.jupiter.api.extension.RegisterExtension; + +public class PostgresMessageIdManagerSideEffectTest extends AbstractMessageIdManagerSideEffectTest { + @RegisterExtension + static PostgresExtension postgresExtension = PostgresExtension.withoutRowLevelSecurity(PostgresMailboxAggregateModule.MODULE); + + @Override + protected MessageIdManagerTestSystem createTestSystem(QuotaManager quotaManager, EventBus eventBus, Set preDeletionHooks) { + return PostgresMessageIdManagerTestSystem.createTestingData(postgresExtension, quotaManager, eventBus, preDeletionHooks); + } +} diff --git a/mailbox/postgres/src/test/java/org/apache/james/mailbox/postgres/PostgresMessageIdManagerStorageTest.java b/mailbox/postgres/src/test/java/org/apache/james/mailbox/postgres/PostgresMessageIdManagerStorageTest.java new file mode 100644 index 00000000000..180a3780393 --- /dev/null +++ b/mailbox/postgres/src/test/java/org/apache/james/mailbox/postgres/PostgresMessageIdManagerStorageTest.java @@ -0,0 +1,43 @@ +/**************************************************************** + * 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.mailbox.postgres; + +import org.apache.james.backends.postgres.PostgresExtension; +import org.apache.james.events.EventBusTestFixture; +import org.apache.james.events.InVMEventBus; +import org.apache.james.events.MemoryEventDeadLetters; +import org.apache.james.events.delivery.InVmEventDelivery; +import org.apache.james.mailbox.extension.PreDeletionHook; +import org.apache.james.mailbox.store.AbstractMessageIdManagerStorageTest; +import org.apache.james.mailbox.store.MessageIdManagerTestSystem; +import org.apache.james.mailbox.store.quota.NoQuotaManager; +import org.apache.james.metrics.tests.RecordingMetricFactory; +import org.junit.jupiter.api.extension.RegisterExtension; + +public class PostgresMessageIdManagerStorageTest extends AbstractMessageIdManagerStorageTest { + @RegisterExtension + static PostgresExtension postgresExtension = PostgresExtension.withoutRowLevelSecurity(PostgresMailboxAggregateModule.MODULE); + + @Override + protected MessageIdManagerTestSystem createTestingData() { + InVMEventBus eventBus = new InVMEventBus(new InVmEventDelivery(new RecordingMetricFactory()), EventBusTestFixture.RETRY_BACKOFF_CONFIGURATION, new MemoryEventDeadLetters()); + return PostgresMessageIdManagerTestSystem.createTestingData(postgresExtension, new NoQuotaManager(), eventBus, PreDeletionHook.NO_PRE_DELETION_HOOK); + } +} diff --git a/mailbox/postgres/src/test/java/org/apache/james/mailbox/postgres/PostgresMessageIdManagerTestSystem.java b/mailbox/postgres/src/test/java/org/apache/james/mailbox/postgres/PostgresMessageIdManagerTestSystem.java new file mode 100644 index 00000000000..1e04f94ce0a --- /dev/null +++ b/mailbox/postgres/src/test/java/org/apache/james/mailbox/postgres/PostgresMessageIdManagerTestSystem.java @@ -0,0 +1,60 @@ +/**************************************************************** + * 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.mailbox.postgres; + +import java.util.Set; + +import org.apache.james.backends.postgres.PostgresExtension; +import org.apache.james.events.EventBus; +import org.apache.james.mailbox.extension.PreDeletionHook; +import org.apache.james.mailbox.quota.CurrentQuotaManager; +import org.apache.james.mailbox.quota.QuotaManager; +import org.apache.james.mailbox.store.MessageIdManagerTestSystem; +import org.apache.james.mailbox.store.PreDeletionHooks; +import org.apache.james.mailbox.store.quota.ListeningCurrentQuotaUpdater; +import org.apache.james.metrics.tests.RecordingMetricFactory; + +public class PostgresMessageIdManagerTestSystem { + static MessageIdManagerTestSystem createTestingData(PostgresExtension postgresExtension, QuotaManager quotaManager, EventBus eventBus, + Set preDeletionHooks) { + PostgresMailboxSessionMapperFactory mapperFactory = PostgresTestSystemFixture.createMapperFactory(postgresExtension); + + return new MessageIdManagerTestSystem(PostgresTestSystemFixture.createMessageIdManager(mapperFactory, quotaManager, eventBus, new PreDeletionHooks(preDeletionHooks, new RecordingMetricFactory())), + new PostgresMessageId.Factory(), + mapperFactory, + PostgresTestSystemFixture.createMailboxManager(mapperFactory)) { + }; + } + + static MessageIdManagerTestSystem createTestingDataWithQuota(PostgresExtension postgresExtension, QuotaManager quotaManager, CurrentQuotaManager currentQuotaManager) { + PostgresMailboxSessionMapperFactory mapperFactory = PostgresTestSystemFixture.createMapperFactory(postgresExtension); + + PostgresMailboxManager mailboxManager = PostgresTestSystemFixture.createMailboxManager(mapperFactory); + ListeningCurrentQuotaUpdater listeningCurrentQuotaUpdater = new ListeningCurrentQuotaUpdater( + currentQuotaManager, + mailboxManager.getQuotaComponents().getQuotaRootResolver(), mailboxManager.getEventBus(), quotaManager); + mailboxManager.getEventBus().register(listeningCurrentQuotaUpdater); + return new MessageIdManagerTestSystem(PostgresTestSystemFixture.createMessageIdManager(mapperFactory, quotaManager, mailboxManager.getEventBus(), + PreDeletionHooks.NO_PRE_DELETION_HOOK), + new PostgresMessageId.Factory(), + mapperFactory, + mailboxManager); + } +} diff --git a/mailbox/postgres/src/test/java/org/apache/james/mailbox/postgres/PostgresSubscriptionManagerTest.java b/mailbox/postgres/src/test/java/org/apache/james/mailbox/postgres/PostgresSubscriptionManagerTest.java new file mode 100644 index 00000000000..356f08ef1c4 --- /dev/null +++ b/mailbox/postgres/src/test/java/org/apache/james/mailbox/postgres/PostgresSubscriptionManagerTest.java @@ -0,0 +1,54 @@ +/**************************************************************** + * 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.mailbox.postgres; + +import org.apache.james.backends.postgres.PostgresExtension; +import org.apache.james.events.EventBusTestFixture; +import org.apache.james.events.InVMEventBus; +import org.apache.james.events.MemoryEventDeadLetters; +import org.apache.james.events.delivery.InVmEventDelivery; +import org.apache.james.mailbox.SubscriptionManager; +import org.apache.james.mailbox.SubscriptionManagerContract; +import org.apache.james.mailbox.store.MailboxSessionMapperFactory; +import org.apache.james.mailbox.store.StoreSubscriptionManager; +import org.apache.james.metrics.tests.RecordingMetricFactory; +import org.junit.jupiter.api.BeforeEach; +import org.junit.jupiter.api.extension.RegisterExtension; + +class PostgresSubscriptionManagerTest implements SubscriptionManagerContract { + + @RegisterExtension + static PostgresExtension postgresExtension = PostgresExtension.withoutRowLevelSecurity(PostgresMailboxAggregateModule.MODULE); + + SubscriptionManager subscriptionManager; + + @Override + public SubscriptionManager getSubscriptionManager() { + return subscriptionManager; + } + + @BeforeEach + void setUp() { + MailboxSessionMapperFactory mapperFactory = PostgresMailboxManagerProvider.provideMailboxSessionMapperFactory(postgresExtension); + InVMEventBus eventBus = new InVMEventBus(new InVmEventDelivery(new RecordingMetricFactory()), EventBusTestFixture.RETRY_BACKOFF_CONFIGURATION, new MemoryEventDeadLetters()); + subscriptionManager = new StoreSubscriptionManager(mapperFactory, mapperFactory, eventBus); + } + + +} diff --git a/mailbox/postgres/src/test/java/org/apache/james/mailbox/postgres/PostgresTestSystemFixture.java b/mailbox/postgres/src/test/java/org/apache/james/mailbox/postgres/PostgresTestSystemFixture.java new file mode 100644 index 00000000000..338a3be00a1 --- /dev/null +++ b/mailbox/postgres/src/test/java/org/apache/james/mailbox/postgres/PostgresTestSystemFixture.java @@ -0,0 +1,115 @@ +/**************************************************************** + * 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.mailbox.postgres; + +import static org.mockito.Mockito.mock; + +import java.time.Clock; +import java.time.Instant; + +import org.apache.james.backends.postgres.PostgresConfiguration; +import org.apache.james.backends.postgres.PostgresExtension; +import org.apache.james.backends.postgres.quota.PostgresQuotaCurrentValueDAO; +import org.apache.james.backends.postgres.quota.PostgresQuotaLimitDAO; +import org.apache.james.blob.api.BlobId; +import org.apache.james.blob.api.BucketName; +import org.apache.james.blob.api.PlainBlobId; +import org.apache.james.blob.memory.MemoryBlobStoreDAO; +import org.apache.james.events.EventBus; +import org.apache.james.events.EventBusTestFixture; +import org.apache.james.events.InVMEventBus; +import org.apache.james.events.MemoryEventDeadLetters; +import org.apache.james.events.delivery.InVmEventDelivery; +import org.apache.james.mailbox.AttachmentContentLoader; +import org.apache.james.mailbox.Authenticator; +import org.apache.james.mailbox.Authorizator; +import org.apache.james.mailbox.acl.UnionMailboxACLResolver; +import org.apache.james.mailbox.postgres.quota.PostgresCurrentQuotaManager; +import org.apache.james.mailbox.postgres.quota.PostgresPerUserMaxQuotaManager; +import org.apache.james.mailbox.quota.CurrentQuotaManager; +import org.apache.james.mailbox.quota.MaxQuotaManager; +import org.apache.james.mailbox.quota.QuotaManager; +import org.apache.james.mailbox.store.PreDeletionHooks; +import org.apache.james.mailbox.store.SessionProviderImpl; +import org.apache.james.mailbox.store.StoreMailboxAnnotationManager; +import org.apache.james.mailbox.store.StoreMessageIdManager; +import org.apache.james.mailbox.store.StoreRightManager; +import org.apache.james.mailbox.store.event.MailboxAnnotationListener; +import org.apache.james.mailbox.store.extractor.DefaultTextExtractor; +import org.apache.james.mailbox.store.mail.NaiveThreadIdGuessingAlgorithm; +import org.apache.james.mailbox.store.mail.model.impl.MessageParser; +import org.apache.james.mailbox.store.quota.DefaultUserQuotaRootResolver; +import org.apache.james.mailbox.store.quota.QuotaComponents; +import org.apache.james.mailbox.store.search.MessageSearchIndex; +import org.apache.james.mailbox.store.search.SimpleMessageSearchIndex; +import org.apache.james.metrics.tests.RecordingMetricFactory; +import org.apache.james.server.blob.deduplication.DeDuplicationBlobStore; +import org.apache.james.utils.UpdatableTickingClock; + +public class PostgresTestSystemFixture { + public static PostgresMailboxSessionMapperFactory createMapperFactory(PostgresExtension postgresExtension) { + BlobId.Factory blobIdFactory = new PlainBlobId.Factory(); + DeDuplicationBlobStore blobStore = new DeDuplicationBlobStore(new MemoryBlobStoreDAO(), BucketName.DEFAULT, blobIdFactory); + + return new PostgresMailboxSessionMapperFactory(postgresExtension.getExecutorFactory(), Clock.systemUTC(), blobStore, blobIdFactory, + PostgresConfiguration.builder().username("a").password("a").build()); + } + + public static PostgresMailboxManager createMailboxManager(PostgresMailboxSessionMapperFactory mapperFactory) { + InVMEventBus eventBus = new InVMEventBus(new InVmEventDelivery(new RecordingMetricFactory()), EventBusTestFixture.RETRY_BACKOFF_CONFIGURATION, new MemoryEventDeadLetters()); + StoreRightManager storeRightManager = new StoreRightManager(mapperFactory, new UnionMailboxACLResolver(), eventBus); + StoreMailboxAnnotationManager annotationManager = new StoreMailboxAnnotationManager(mapperFactory, storeRightManager); + + SessionProviderImpl sessionProvider = new SessionProviderImpl(mock(Authenticator.class), mock(Authorizator.class)); + + QuotaComponents quotaComponents = QuotaComponents.disabled(sessionProvider, mapperFactory); + AttachmentContentLoader attachmentContentLoader = null; + MessageSearchIndex index = new SimpleMessageSearchIndex(mapperFactory, mapperFactory, new DefaultTextExtractor(), attachmentContentLoader); + PostgresMailboxManager postgresMailboxManager = new PostgresMailboxManager(mapperFactory, sessionProvider, + new MessageParser(), new PostgresMessageId.Factory(), + eventBus, annotationManager, storeRightManager, quotaComponents, index, new NaiveThreadIdGuessingAlgorithm(), PreDeletionHooks.NO_PRE_DELETION_HOOK, + new UpdatableTickingClock(Instant.now())); + + eventBus.register(new MailboxAnnotationListener(mapperFactory, sessionProvider)); + eventBus.register(mapperFactory.deleteMessageListener()); + + return postgresMailboxManager; + } + + static StoreMessageIdManager createMessageIdManager(PostgresMailboxSessionMapperFactory mapperFactory, QuotaManager quotaManager, EventBus eventBus, + PreDeletionHooks preDeletionHooks) { + PostgresMailboxManager mailboxManager = createMailboxManager(mapperFactory); + return new StoreMessageIdManager( + mailboxManager, + mapperFactory, + eventBus, + quotaManager, + new DefaultUserQuotaRootResolver(mailboxManager.getSessionProvider(), mapperFactory), + preDeletionHooks); + } + + static MaxQuotaManager createMaxQuotaManager(PostgresExtension postgresExtension) { + return new PostgresPerUserMaxQuotaManager(new PostgresQuotaLimitDAO(postgresExtension.getDefaultPostgresExecutor())); + } + + public static CurrentQuotaManager createCurrentQuotaManager(PostgresExtension postgresExtension) { + return new PostgresCurrentQuotaManager(new PostgresQuotaCurrentValueDAO(postgresExtension.getDefaultPostgresExecutor())); + } +} diff --git a/mailbox/postgres/src/test/java/org/apache/james/mailbox/postgres/PostgresThreadIdGuessingAlgorithmTest.java b/mailbox/postgres/src/test/java/org/apache/james/mailbox/postgres/PostgresThreadIdGuessingAlgorithmTest.java new file mode 100644 index 00000000000..d9e5067c71a --- /dev/null +++ b/mailbox/postgres/src/test/java/org/apache/james/mailbox/postgres/PostgresThreadIdGuessingAlgorithmTest.java @@ -0,0 +1,166 @@ +/**************************************************************** + * 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.mailbox.postgres; + +import static org.assertj.core.api.Assertions.assertThat; + +import java.util.List; +import java.util.Optional; +import java.util.Set; +import java.util.stream.Collectors; + +import org.apache.james.backends.postgres.PostgresExtension; +import org.apache.james.core.Username; +import org.apache.james.events.EventBusTestFixture; +import org.apache.james.events.InVMEventBus; +import org.apache.james.events.MemoryEventDeadLetters; +import org.apache.james.events.delivery.InVmEventDelivery; +import org.apache.james.mailbox.MailboxSession; +import org.apache.james.mailbox.exception.MailboxException; +import org.apache.james.mailbox.model.MessageId; +import org.apache.james.mailbox.model.ThreadId; +import org.apache.james.mailbox.postgres.mail.dao.PostgresThreadDAO; +import org.apache.james.mailbox.store.CombinationManagerTestSystem; +import org.apache.james.mailbox.store.ThreadIdGuessingAlgorithmContract; +import org.apache.james.mailbox.store.mail.MessageMapper; +import org.apache.james.mailbox.store.mail.ThreadIdGuessingAlgorithm; +import org.apache.james.mailbox.store.mail.model.MimeMessageId; +import org.apache.james.mailbox.store.mail.model.Subject; +import org.apache.james.mailbox.store.quota.NoQuotaManager; +import org.apache.james.metrics.tests.RecordingMetricFactory; +import org.junit.jupiter.api.Test; +import org.junit.jupiter.api.extension.RegisterExtension; + +import com.google.common.collect.ImmutableList; +import com.google.common.collect.ImmutableSet; +import com.google.common.hash.Hashing; + +import reactor.core.publisher.Flux; + +public class PostgresThreadIdGuessingAlgorithmTest extends ThreadIdGuessingAlgorithmContract { + @RegisterExtension + static PostgresExtension postgresExtension = PostgresExtension.withoutRowLevelSecurity(PostgresMailboxAggregateModule.MODULE); + + private PostgresMailboxManager mailboxManager; + private PostgresThreadDAO.Factory threadDAOFactory; + + @Override + protected CombinationManagerTestSystem createTestingData() { + eventBus = new InVMEventBus(new InVmEventDelivery(new RecordingMetricFactory()), EventBusTestFixture.RETRY_BACKOFF_CONFIGURATION, new MemoryEventDeadLetters()); + PostgresCombinationManagerTestSystem testSystem = (PostgresCombinationManagerTestSystem) PostgresCombinationManagerTestSystem.createTestingData(postgresExtension, new NoQuotaManager(), eventBus); + mailboxManager = (PostgresMailboxManager) testSystem.getMailboxManager(); + messageIdFactory = new PostgresMessageId.Factory(); + return testSystem; + } + + @Override + protected ThreadIdGuessingAlgorithm initThreadIdGuessingAlgorithm(CombinationManagerTestSystem testingData) { + threadDAOFactory = new PostgresThreadDAO.Factory(postgresExtension.getExecutorFactory()); + return new PostgresThreadIdGuessingAlgorithm(threadDAOFactory); + } + + @Override + protected MessageMapper createMessageMapper(MailboxSession mailboxSession) { + return mailboxManager.getMapperFactory().createMessageMapper(mailboxSession); + } + + @Override + protected MessageId initNewBasedMessageId() { + return messageIdFactory.generate(); + } + + @Override + protected MessageId initOtherBasedMessageId() { + return messageIdFactory.generate(); + } + + @Override + protected void saveThreadData(Username username, Set mimeMessageIds, MessageId messageId, ThreadId threadId, Optional baseSubject) { + PostgresThreadDAO threadDAO = threadDAOFactory.create(username.getDomainPart()); + threadDAO.insertSome(username, hashMimeMessagesIds(mimeMessageIds), PostgresMessageId.class.cast(messageId), threadId, hashSubject(baseSubject)).block(); + } + + @Test + void givenAMailInAThreadThenGetThreadShouldReturnAListWithOnlyOneMessageIdInThatThread() throws MailboxException { + Set mimeMessageIds = buildMimeMessageIdSet(Optional.of(new MimeMessageId("Message-ID")), + Optional.of(new MimeMessageId("someInReplyTo")), + Optional.of(List.of(new MimeMessageId("references1"), new MimeMessageId("references2")))); + + MessageId messageId = initNewBasedMessageId(); + ThreadId threadId = ThreadId.fromBaseMessageId(newBasedMessageId); + saveThreadData(mailboxSession.getUser(), mimeMessageIds, messageId, threadId, Optional.of(new Subject("Test"))); + + Flux messageIds = testee.getMessageIdsInThread(threadId, mailboxSession); + + assertThat(messageIds.collectList().block()) + .containsOnly(messageId); + } + + @Test + void givenTwoDistinctThreadsThenGetThreadShouldNotReturnUnrelatedMails() throws MailboxException { + Set mimeMessageIds = buildMimeMessageIdSet(Optional.of(new MimeMessageId("Message-ID")), + Optional.of(new MimeMessageId("someInReplyTo")), + Optional.of(List.of(new MimeMessageId("references1"), new MimeMessageId("references2")))); + + MessageId messageId1 = initNewBasedMessageId(); + MessageId messageId2 = initNewBasedMessageId(); + MessageId messageId3 = initNewBasedMessageId(); + ThreadId threadId1 = ThreadId.fromBaseMessageId(newBasedMessageId); + ThreadId threadId2 = ThreadId.fromBaseMessageId(otherBasedMessageId); + + saveThreadData(mailboxSession.getUser(), mimeMessageIds, messageId1, threadId1, Optional.of(new Subject("Test"))); + saveThreadData(mailboxSession.getUser(), mimeMessageIds, messageId2, threadId1, Optional.of(new Subject("Test"))); + saveThreadData(mailboxSession.getUser(), mimeMessageIds, messageId3, threadId2, Optional.of(new Subject("Test"))); + + Flux messageIds = testee.getMessageIdsInThread(ThreadId.fromBaseMessageId(otherBasedMessageId), mailboxSession); + + assertThat(messageIds.collectList().block()) + .doesNotContain(messageId1, messageId2); + } + + @Test + void givenThreeMailsInAThreadThenGetThreadShouldReturnAListWithThreeMessageIdsSortedByArrivalDate() { + Set mimeMessageIds = ImmutableSet.of(new MimeMessageId("Message-ID")); + + MessageId messageId1 = initNewBasedMessageId(); + MessageId messageId2 = initNewBasedMessageId(); + MessageId messageId3 = initNewBasedMessageId(); + ThreadId threadId1 = ThreadId.fromBaseMessageId(newBasedMessageId); + + saveThreadData(mailboxSession.getUser(), mimeMessageIds, messageId1, threadId1, Optional.of(new Subject("Test1"))); + saveThreadData(mailboxSession.getUser(), mimeMessageIds, messageId2, threadId1, Optional.of(new Subject("Test2"))); + saveThreadData(mailboxSession.getUser(), mimeMessageIds, messageId3, threadId1, Optional.of(new Subject("Test3"))); + + Flux messageIds = testee.getMessageIdsInThread(ThreadId.fromBaseMessageId(newBasedMessageId), mailboxSession); + + assertThat(messageIds.collectList().block()) + .isEqualTo(ImmutableList.of(messageId1, messageId2, messageId3)); + } + + private Set hashMimeMessagesIds(Set mimeMessageIds) { + return mimeMessageIds.stream() + .map(mimeMessageId -> Hashing.murmur3_32_fixed().hashBytes(mimeMessageId.getValue().getBytes()).asInt()) + .collect(Collectors.toSet()); + } + + private Optional hashSubject(Optional baseSubjectOptional) { + return baseSubjectOptional.map(baseSubject -> Hashing.murmur3_32_fixed().hashBytes(baseSubject.getValue().getBytes()).asInt()); + } +} diff --git a/mailbox/postgres/src/test/java/org/apache/james/mailbox/postgres/mail/PostgresAnnotationMapperRowLevelSecurityTest.java b/mailbox/postgres/src/test/java/org/apache/james/mailbox/postgres/mail/PostgresAnnotationMapperRowLevelSecurityTest.java new file mode 100644 index 00000000000..dcdd0c2cc1b --- /dev/null +++ b/mailbox/postgres/src/test/java/org/apache/james/mailbox/postgres/mail/PostgresAnnotationMapperRowLevelSecurityTest.java @@ -0,0 +1,98 @@ +/**************************************************************** + * 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.mailbox.postgres.mail; + +import static org.assertj.core.api.Assertions.assertThat; + +import java.time.Instant; + +import org.apache.james.backends.postgres.PostgresConfiguration; +import org.apache.james.backends.postgres.PostgresExtension; +import org.apache.james.backends.postgres.utils.PostgresExecutor; +import org.apache.james.blob.api.BlobId; +import org.apache.james.blob.api.BucketName; +import org.apache.james.blob.api.PlainBlobId; +import org.apache.james.blob.memory.MemoryBlobStoreDAO; +import org.apache.james.core.Username; +import org.apache.james.mailbox.MailboxSession; +import org.apache.james.mailbox.MailboxSessionUtil; +import org.apache.james.mailbox.model.MailboxAnnotation; +import org.apache.james.mailbox.model.MailboxAnnotationKey; +import org.apache.james.mailbox.model.MailboxId; +import org.apache.james.mailbox.model.MailboxPath; +import org.apache.james.mailbox.model.UidValidity; +import org.apache.james.mailbox.postgres.PostgresMailboxAggregateModule; +import org.apache.james.mailbox.postgres.PostgresMailboxSessionMapperFactory; +import org.apache.james.mailbox.postgres.mail.dao.PostgresMailboxDAO; +import org.apache.james.mailbox.store.mail.MailboxMapper; +import org.apache.james.server.blob.deduplication.DeDuplicationBlobStore; +import org.apache.james.utils.UpdatableTickingClock; +import org.junit.jupiter.api.BeforeEach; +import org.junit.jupiter.api.Test; +import org.junit.jupiter.api.extension.RegisterExtension; + +public class PostgresAnnotationMapperRowLevelSecurityTest { + private static final UidValidity UID_VALIDITY = UidValidity.of(42); + private static final Username BENWA = Username.of("benwa@localhost"); + protected static final MailboxPath benwaInboxPath = MailboxPath.forUser(BENWA, "INBOX"); + private static final MailboxSession aliceSession = MailboxSessionUtil.create(Username.of("alice@domain1")); + private static final MailboxSession bobSession = MailboxSessionUtil.create(Username.of("bob@domain1")); + private static final MailboxSession bobDomain2Session = MailboxSessionUtil.create(Username.of("bob@domain2")); + private static final MailboxAnnotationKey PRIVATE_KEY = new MailboxAnnotationKey("/private/comment"); + private static final MailboxAnnotation PRIVATE_ANNOTATION = MailboxAnnotation.newInstance(PRIVATE_KEY, "My private comment"); + + @RegisterExtension + static PostgresExtension postgresExtension = PostgresExtension.withRowLevelSecurity(PostgresMailboxAggregateModule.MODULE); + + private PostgresMailboxSessionMapperFactory postgresMailboxSessionMapperFactory; + private MailboxId mailboxId; + + private MailboxId generateMailboxId() { + PostgresExecutor postgresExecutor = postgresExtension.getExecutorFactory().create(BENWA.getDomainPart()); + MailboxMapper mailboxMapper = new PostgresMailboxMapper(new PostgresMailboxDAO(postgresExecutor)); + return mailboxMapper.create(benwaInboxPath, UID_VALIDITY).block().getMailboxId(); + } + + @BeforeEach + public void setUp() { + BlobId.Factory blobIdFactory = new PlainBlobId.Factory(); + postgresMailboxSessionMapperFactory = new PostgresMailboxSessionMapperFactory(postgresExtension.getExecutorFactory(), + new UpdatableTickingClock(Instant.now()), + new DeDuplicationBlobStore(new MemoryBlobStoreDAO(), BucketName.DEFAULT, blobIdFactory), + blobIdFactory, + PostgresConfiguration.builder().username("a").password("a").build()); + + mailboxId = generateMailboxId(); + } + + @Test + void annotationsCanBeAccessedAtTheDataLevelByMembersOfTheSameDomain() { + postgresMailboxSessionMapperFactory.getAnnotationMapper(aliceSession).insertAnnotation(mailboxId, PRIVATE_ANNOTATION); + + assertThat(postgresMailboxSessionMapperFactory.getAnnotationMapper(bobSession).getAllAnnotations(mailboxId)).isNotEmpty(); + } + + @Test + void annotationsShouldBeIsolatedByDomain() { + postgresMailboxSessionMapperFactory.getAnnotationMapper(aliceSession).insertAnnotation(mailboxId, PRIVATE_ANNOTATION); + + assertThat(postgresMailboxSessionMapperFactory.getAnnotationMapper(bobDomain2Session).getAllAnnotations(mailboxId)).isEmpty(); + } +} diff --git a/mailbox/postgres/src/test/java/org/apache/james/mailbox/postgres/mail/PostgresAnnotationMapperTest.java b/mailbox/postgres/src/test/java/org/apache/james/mailbox/postgres/mail/PostgresAnnotationMapperTest.java new file mode 100644 index 00000000000..4bb1495356f --- /dev/null +++ b/mailbox/postgres/src/test/java/org/apache/james/mailbox/postgres/mail/PostgresAnnotationMapperTest.java @@ -0,0 +1,53 @@ +/**************************************************************** + * 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.mailbox.postgres.mail; + +import org.apache.james.backends.postgres.PostgresExtension; +import org.apache.james.core.Username; +import org.apache.james.mailbox.model.MailboxId; +import org.apache.james.mailbox.model.MailboxPath; +import org.apache.james.mailbox.model.UidValidity; +import org.apache.james.mailbox.postgres.PostgresMailboxAggregateModule; +import org.apache.james.mailbox.postgres.mail.dao.PostgresMailboxAnnotationDAO; +import org.apache.james.mailbox.postgres.mail.dao.PostgresMailboxDAO; +import org.apache.james.mailbox.store.mail.AnnotationMapper; +import org.apache.james.mailbox.store.mail.MailboxMapper; +import org.apache.james.mailbox.store.mail.model.AnnotationMapperTest; +import org.junit.jupiter.api.extension.RegisterExtension; + +public class PostgresAnnotationMapperTest extends AnnotationMapperTest { + private static final UidValidity UID_VALIDITY = UidValidity.of(42); + private static final Username BENWA = Username.of("benwa"); + protected static final MailboxPath benwaInboxPath = MailboxPath.forUser(BENWA, "INBOX"); + + @RegisterExtension + static PostgresExtension postgresExtension = PostgresExtension.withoutRowLevelSecurity(PostgresMailboxAggregateModule.MODULE); + + @Override + protected AnnotationMapper createAnnotationMapper() { + return new PostgresAnnotationMapper(new PostgresMailboxAnnotationDAO(postgresExtension.getDefaultPostgresExecutor())); + } + + @Override + protected MailboxId generateMailboxId() { + MailboxMapper mailboxMapper = new PostgresMailboxMapper(new PostgresMailboxDAO(postgresExtension.getDefaultPostgresExecutor())); + return mailboxMapper.create(benwaInboxPath, UID_VALIDITY).block().getMailboxId(); + } +} diff --git a/mailbox/postgres/src/test/java/org/apache/james/mailbox/postgres/mail/PostgresAttachmentBlobReferenceSourceTest.java b/mailbox/postgres/src/test/java/org/apache/james/mailbox/postgres/mail/PostgresAttachmentBlobReferenceSourceTest.java new file mode 100644 index 00000000000..8c169cc8fd0 --- /dev/null +++ b/mailbox/postgres/src/test/java/org/apache/james/mailbox/postgres/mail/PostgresAttachmentBlobReferenceSourceTest.java @@ -0,0 +1,112 @@ +/**************************************************************** + * 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.mailbox.postgres.mail; + +import static org.assertj.core.api.Assertions.assertThat; + +import org.apache.james.backends.postgres.PostgresExtension; +import org.apache.james.blob.api.BlobId; +import org.apache.james.blob.api.PlainBlobId; +import org.apache.james.mailbox.model.AttachmentId; +import org.apache.james.mailbox.model.AttachmentMetadata; +import org.apache.james.mailbox.model.UuidBackedAttachmentId; +import org.apache.james.mailbox.postgres.PostgresMailboxAggregateModule; +import org.apache.james.mailbox.postgres.PostgresMessageId; +import org.apache.james.mailbox.postgres.mail.dao.PostgresAttachmentDAO; +import org.junit.jupiter.api.BeforeEach; +import org.junit.jupiter.api.Test; +import org.junit.jupiter.api.extension.RegisterExtension; + +class PostgresAttachmentBlobReferenceSourceTest { + + private static final AttachmentId ATTACHMENT_ID = UuidBackedAttachmentId.random(); + private static final AttachmentId ATTACHMENT_ID_2 = UuidBackedAttachmentId.random(); + private static final BlobId.Factory BLOB_ID_FACTORY = new PlainBlobId.Factory(); + + @RegisterExtension + static PostgresExtension postgresExtension = PostgresExtension.withoutRowLevelSecurity(PostgresMailboxAggregateModule.MODULE); + + private PostgresAttachmentBlobReferenceSource testee; + + private PostgresAttachmentDAO postgresAttachmentDAO; + + @BeforeEach + void beforeEach() { + BlobId.Factory blobIdFactory = new PlainBlobId.Factory(); + postgresAttachmentDAO = new PostgresAttachmentDAO(postgresExtension.getDefaultPostgresExecutor(), + blobIdFactory); + testee = new PostgresAttachmentBlobReferenceSource(postgresAttachmentDAO); + } + + @Test + void blobReferencesShouldBeEmptyByDefault() { + assertThat(testee.listReferencedBlobs().collectList().block()) + .isEmpty(); + } + + @Test + void blobReferencesShouldReturnAllValues() { + AttachmentMetadata attachment1 = AttachmentMetadata.builder() + .attachmentId(ATTACHMENT_ID) + .messageId(new PostgresMessageId.Factory().generate()) + .type("application/json") + .size(36) + .build(); + BlobId blobId1 = BLOB_ID_FACTORY.parse("blobId"); + + postgresAttachmentDAO.storeAttachment(attachment1, blobId1).block(); + + AttachmentMetadata attachment2 = AttachmentMetadata.builder() + .attachmentId(ATTACHMENT_ID_2) + .messageId(new PostgresMessageId.Factory().generate()) + .type("application/json") + .size(36) + .build(); + BlobId blobId2 = BLOB_ID_FACTORY.parse("blobId"); + postgresAttachmentDAO.storeAttachment(attachment2, blobId2).block(); + + assertThat(testee.listReferencedBlobs().collectList().block()) + .containsOnly(blobId1, blobId2); + } + + @Test + void blobReferencesShouldReturnDuplicates() { + AttachmentMetadata attachment1 = AttachmentMetadata.builder() + .attachmentId(ATTACHMENT_ID) + .messageId(new PostgresMessageId.Factory().generate()) + .type("application/json") + .size(36) + .build(); + BlobId blobId = BLOB_ID_FACTORY.parse("blobId"); + postgresAttachmentDAO.storeAttachment(attachment1, blobId).block(); + + AttachmentMetadata attachment2 = AttachmentMetadata.builder() + .attachmentId(ATTACHMENT_ID_2) + .messageId(new PostgresMessageId.Factory().generate()) + .type("application/json") + .size(36) + .build(); + postgresAttachmentDAO.storeAttachment(attachment2, blobId).block(); + + assertThat(testee.listReferencedBlobs().collectList().block()) + .hasSize(2) + .containsOnly(blobId); + } +} diff --git a/mailbox/postgres/src/test/java/org/apache/james/mailbox/postgres/mail/PostgresAttachmentMapperTest.java b/mailbox/postgres/src/test/java/org/apache/james/mailbox/postgres/mail/PostgresAttachmentMapperTest.java new file mode 100644 index 00000000000..6ccf2a7a9db --- /dev/null +++ b/mailbox/postgres/src/test/java/org/apache/james/mailbox/postgres/mail/PostgresAttachmentMapperTest.java @@ -0,0 +1,54 @@ +/**************************************************************** + * 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.mailbox.postgres.mail; + +import org.apache.james.backends.postgres.PostgresExtension; +import org.apache.james.blob.api.BlobId; +import org.apache.james.blob.api.BlobStore; +import org.apache.james.blob.api.BucketName; +import org.apache.james.blob.api.PlainBlobId; +import org.apache.james.blob.memory.MemoryBlobStoreDAO; +import org.apache.james.mailbox.model.MessageId; +import org.apache.james.mailbox.postgres.PostgresMessageId; +import org.apache.james.mailbox.postgres.mail.dao.PostgresAttachmentDAO; +import org.apache.james.mailbox.store.mail.AttachmentMapper; +import org.apache.james.mailbox.store.mail.model.AttachmentMapperTest; +import org.apache.james.server.blob.deduplication.DeDuplicationBlobStore; +import org.junit.jupiter.api.extension.RegisterExtension; + +class PostgresAttachmentMapperTest extends AttachmentMapperTest { + + @RegisterExtension + static PostgresExtension postgresExtension = PostgresExtension.withoutRowLevelSecurity(PostgresAttachmentModule.MODULE); + + static BlobId.Factory BLOB_ID_FACTORY = new PlainBlobId.Factory(); + + @Override + protected AttachmentMapper createAttachmentMapper() { + PostgresAttachmentDAO postgresAttachmentDAO = new PostgresAttachmentDAO(postgresExtension.getDefaultPostgresExecutor(), BLOB_ID_FACTORY); + BlobStore blobStore = new DeDuplicationBlobStore(new MemoryBlobStoreDAO(), BucketName.DEFAULT, BLOB_ID_FACTORY); + return new PostgresAttachmentMapper(postgresAttachmentDAO, blobStore); + } + + @Override + protected MessageId generateMessageId() { + return new PostgresMessageId.Factory().generate(); + } +} diff --git a/mailbox/postgres/src/test/java/org/apache/james/mailbox/postgres/mail/PostgresMailboxMapperACLTest.java b/mailbox/postgres/src/test/java/org/apache/james/mailbox/postgres/mail/PostgresMailboxMapperACLTest.java new file mode 100644 index 00000000000..67f44e77eda --- /dev/null +++ b/mailbox/postgres/src/test/java/org/apache/james/mailbox/postgres/mail/PostgresMailboxMapperACLTest.java @@ -0,0 +1,81 @@ +/**************************************************************** + * 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.mailbox.postgres.mail; + +import static org.assertj.core.api.Assertions.assertThat; + +import java.time.Duration; +import java.util.concurrent.ExecutionException; +import java.util.stream.IntStream; + +import org.apache.james.backends.postgres.PostgresExtension; +import org.apache.james.mailbox.model.MailboxACL; +import org.apache.james.mailbox.postgres.mail.dao.PostgresMailboxDAO; +import org.apache.james.mailbox.store.mail.MailboxMapper; +import org.apache.james.mailbox.store.mail.model.MailboxMapperACLTest; +import org.apache.james.util.concurrency.ConcurrentTestRunner; +import org.junit.jupiter.api.Test; +import org.junit.jupiter.api.extension.RegisterExtension; + +import com.google.common.collect.ImmutableMap; + +class PostgresMailboxMapperACLTest extends MailboxMapperACLTest { + @RegisterExtension + static PostgresExtension postgresExtension = PostgresExtension.withoutRowLevelSecurity(PostgresMailboxModule.MODULE); + + private PostgresMailboxMapper mailboxMapper; + + @Override + protected MailboxMapper createMailboxMapper() { + mailboxMapper = new PostgresMailboxMapper(new PostgresMailboxDAO(postgresExtension.getDefaultPostgresExecutor())); + return mailboxMapper; + } + + @Test + protected void updateAclShouldWorkWellInMultiThreadEnv() throws ExecutionException, InterruptedException { + MailboxACL.Rfc4314Rights rights = new MailboxACL.Rfc4314Rights(MailboxACL.Right.Administer, MailboxACL.Right.Write); + MailboxACL.Rfc4314Rights newRights = new MailboxACL.Rfc4314Rights(MailboxACL.Right.Write); + + ConcurrentTestRunner.builder() + .reactorOperation((threadNumber, step) -> { + int userNumber = threadNumber / 2; + MailboxACL.EntryKey key = MailboxACL.EntryKey.createUserEntryKey("user" + userNumber); + if (threadNumber % 2 == 0) { + return mailboxMapper.updateACL(benwaInboxMailbox, MailboxACL.command().key(key).rights(rights).asReplacement()) + .then(); + } else { + return mailboxMapper.updateACL(benwaInboxMailbox, MailboxACL.command().key(key).rights(newRights).asAddition()) + .then(); + } + }) + .threadCount(10) + .operationCount(1) + .runSuccessfullyWithin(Duration.ofMinutes(1)); + + MailboxACL expectedMailboxACL = new MailboxACL(IntStream.range(0, 5).boxed() + .collect(ImmutableMap.toImmutableMap(userNumber -> MailboxACL.EntryKey.createUserEntryKey("user" + userNumber), userNumber -> rights))); + + assertThat( + mailboxMapper.findMailboxById(benwaInboxMailbox.getMailboxId()) + .block() + .getACL()) + .isEqualTo(expectedMailboxACL); + } +} diff --git a/mailbox/postgres/src/test/java/org/apache/james/mailbox/postgres/mail/PostgresMailboxMapperRowLevelSecurityTest.java b/mailbox/postgres/src/test/java/org/apache/james/mailbox/postgres/mail/PostgresMailboxMapperRowLevelSecurityTest.java new file mode 100644 index 00000000000..0d841de5782 --- /dev/null +++ b/mailbox/postgres/src/test/java/org/apache/james/mailbox/postgres/mail/PostgresMailboxMapperRowLevelSecurityTest.java @@ -0,0 +1,83 @@ +/**************************************************************** + * 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.mailbox.postgres.mail; + +import static org.assertj.core.api.Assertions.assertThat; + +import org.apache.james.backends.postgres.PostgresExtension; +import org.apache.james.backends.postgres.utils.PostgresExecutor; +import org.apache.james.core.Username; +import org.apache.james.mailbox.MailboxSession; +import org.apache.james.mailbox.MailboxSessionUtil; +import org.apache.james.mailbox.model.MailboxPath; +import org.apache.james.mailbox.model.UidValidity; +import org.apache.james.mailbox.postgres.mail.dao.PostgresMailboxDAO; +import org.apache.james.mailbox.store.mail.MailboxMapperFactory; +import org.junit.jupiter.api.BeforeEach; +import org.junit.jupiter.api.Test; +import org.junit.jupiter.api.extension.RegisterExtension; + +public class PostgresMailboxMapperRowLevelSecurityTest { + + @RegisterExtension + static PostgresExtension postgresExtension = PostgresExtension.withRowLevelSecurity(PostgresMailboxModule.MODULE); + + private MailboxMapperFactory mailboxMapperFactory; + + @BeforeEach + public void setUp() { + PostgresExecutor.Factory executorFactory = postgresExtension.getExecutorFactory(); + mailboxMapperFactory = session -> new PostgresMailboxMapper(new PostgresMailboxDAO(executorFactory.create(session.getUser().getDomainPart()))); + } + + @Test + void mailboxesCanBeAccessedAtTheDataLevelByMembersOfTheSameDomain() throws Exception { + Username username = Username.of("alice@domain1"); + Username username2 = Username.of("bob@domain1"); + + MailboxSession session = MailboxSessionUtil.create(username); + MailboxSession session2 = MailboxSessionUtil.create(username2); + + mailboxMapperFactory.getMailboxMapper(session) + .create(MailboxPath.forUser(username, "INBOX"), UidValidity.of(1L)) + .block(); + + assertThat(mailboxMapperFactory.getMailboxMapper(session2) + .findMailboxByPath(MailboxPath.forUser(username, "INBOX")).block()) + .isNotNull(); + } + + @Test + void mailboxesShouldBeIsolatedByDomain() throws Exception { + Username username = Username.of("alice@domain1"); + Username username2 = Username.of("bob@domain2"); + + MailboxSession session = MailboxSessionUtil.create(username); + MailboxSession session2 = MailboxSessionUtil.create(username2); + + mailboxMapperFactory.getMailboxMapper(session) + .create(MailboxPath.forUser(username, "INBOX"), UidValidity.of(1L)) + .block(); + + assertThat(mailboxMapperFactory.getMailboxMapper(session2) + .findMailboxByPath(MailboxPath.forUser(username, "INBOX")).block()) + .isNull(); + } +} diff --git a/mailbox/postgres/src/test/java/org/apache/james/mailbox/postgres/mail/PostgresMailboxMapperTest.java b/mailbox/postgres/src/test/java/org/apache/james/mailbox/postgres/mail/PostgresMailboxMapperTest.java new file mode 100644 index 00000000000..e49a33f7c24 --- /dev/null +++ b/mailbox/postgres/src/test/java/org/apache/james/mailbox/postgres/mail/PostgresMailboxMapperTest.java @@ -0,0 +1,85 @@ +/**************************************************************** + * 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.mailbox.postgres.mail; + +import static org.assertj.core.api.Assertions.assertThat; + +import org.apache.james.backends.postgres.PostgresExtension; +import org.apache.james.core.Username; +import org.apache.james.mailbox.MailboxSessionUtil; +import org.apache.james.mailbox.MessageUid; +import org.apache.james.mailbox.ModSeq; +import org.apache.james.mailbox.model.Mailbox; +import org.apache.james.mailbox.model.MailboxId; +import org.apache.james.mailbox.model.MailboxPath; +import org.apache.james.mailbox.model.UidValidity; +import org.apache.james.mailbox.postgres.PostgresMailboxId; +import org.apache.james.mailbox.postgres.mail.dao.PostgresMailboxDAO; +import org.apache.james.mailbox.store.mail.MailboxMapper; +import org.apache.james.mailbox.store.mail.model.MailboxMapperTest; +import org.junit.jupiter.api.Test; +import org.junit.jupiter.api.extension.RegisterExtension; + +public class PostgresMailboxMapperTest extends MailboxMapperTest { + @RegisterExtension + static PostgresExtension postgresExtension = PostgresExtension.withoutRowLevelSecurity(PostgresMailboxModule.MODULE); + + @Override + protected MailboxMapper createMailboxMapper() { + return new PostgresMailboxMapper(new PostgresMailboxDAO(postgresExtension.getDefaultPostgresExecutor())); + } + + @Override + protected MailboxId generateId() { + return PostgresMailboxId.generate(); + } + + @Test + void retrieveMailboxShouldReturnCorrectHighestModSeqAndLastUidWhenDefault() { + Mailbox mailbox = mailboxMapper.create(benwaInboxPath, UidValidity.of(43)).block(); + + PostgresMailbox metaData = (PostgresMailbox) mailboxMapper.findMailboxById(mailbox.getMailboxId()).block(); + + assertThat(metaData.getHighestModSeq()).isEqualTo(ModSeq.first()); + assertThat(metaData.getLastUid()).isEqualTo(null); + } + + @Test + void retrieveMailboxShouldReturnCorrectHighestModSeqAndLastUid() { + Username benwa = Username.of("benwa"); + MailboxPath benwaInboxPath = MailboxPath.forUser(benwa, "INBOX"); + + Mailbox mailbox = mailboxMapper.create(benwaInboxPath, UidValidity.of(43)).block(); + + // increase modSeq + ModSeq nextModSeq = new PostgresModSeqProvider.Factory(postgresExtension.getExecutorFactory()).create(MailboxSessionUtil.create(benwa)) + .nextModSeqReactive(mailbox.getMailboxId()).block(); + + // increase lastUid + MessageUid nextUid = new PostgresUidProvider.Factory(postgresExtension.getExecutorFactory()).create(MailboxSessionUtil.create(benwa)) + .nextUidReactive(mailbox.getMailboxId()).block(); + + PostgresMailbox metaData = (PostgresMailbox) mailboxMapper.findMailboxById(mailbox.getMailboxId()).block(); + + assertThat(metaData.getHighestModSeq()).isEqualTo(nextModSeq); + assertThat(metaData.getLastUid()).isEqualTo(nextUid); + } + +} diff --git a/mailbox/postgres/src/test/java/org/apache/james/mailbox/postgres/mail/PostgresMapperProvider.java b/mailbox/postgres/src/test/java/org/apache/james/mailbox/postgres/mail/PostgresMapperProvider.java new file mode 100644 index 00000000000..91e98eb91b6 --- /dev/null +++ b/mailbox/postgres/src/test/java/org/apache/james/mailbox/postgres/mail/PostgresMapperProvider.java @@ -0,0 +1,161 @@ +/**************************************************************** + * 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.mailbox.postgres.mail; + +import java.time.Instant; +import java.util.List; + +import org.apache.commons.lang3.NotImplementedException; +import org.apache.james.backends.postgres.PostgresExtension; +import org.apache.james.blob.api.BlobId; +import org.apache.james.blob.api.BlobStore; +import org.apache.james.blob.api.BucketName; +import org.apache.james.blob.api.PlainBlobId; +import org.apache.james.blob.memory.MemoryBlobStoreDAO; +import org.apache.james.mailbox.MessageUid; +import org.apache.james.mailbox.ModSeq; +import org.apache.james.mailbox.exception.MailboxException; +import org.apache.james.mailbox.model.Mailbox; +import org.apache.james.mailbox.model.MailboxId; +import org.apache.james.mailbox.model.MessageId; +import org.apache.james.mailbox.postgres.PostgresMailboxId; +import org.apache.james.mailbox.postgres.PostgresMessageId; +import org.apache.james.mailbox.postgres.mail.dao.PostgresAttachmentDAO; +import org.apache.james.mailbox.postgres.mail.dao.PostgresMailboxDAO; +import org.apache.james.mailbox.postgres.mail.dao.PostgresMailboxMessageDAO; +import org.apache.james.mailbox.postgres.mail.dao.PostgresMessageDAO; +import org.apache.james.mailbox.store.mail.AttachmentMapper; +import org.apache.james.mailbox.store.mail.MailboxMapper; +import org.apache.james.mailbox.store.mail.MessageIdMapper; +import org.apache.james.mailbox.store.mail.MessageMapper; +import org.apache.james.mailbox.store.mail.UidProvider; +import org.apache.james.mailbox.store.mail.model.MapperProvider; +import org.apache.james.server.blob.deduplication.DeDuplicationBlobStore; +import org.apache.james.utils.UpdatableTickingClock; + +import com.google.common.collect.ImmutableList; + +public class PostgresMapperProvider implements MapperProvider { + + private final PostgresMessageId.Factory messageIdFactory; + private final PostgresExtension postgresExtension; + private final UpdatableTickingClock updatableTickingClock; + private final BlobStore blobStore; + private final BlobId.Factory blobIdFactory; + private final UidProvider messageUidProvider; + + public PostgresMapperProvider(PostgresExtension postgresExtension) { + this.postgresExtension = postgresExtension; + this.updatableTickingClock = new UpdatableTickingClock(Instant.now()); + this.messageIdFactory = new PostgresMessageId.Factory(); + this.blobIdFactory = new PlainBlobId.Factory(); + this.blobStore = new DeDuplicationBlobStore(new MemoryBlobStoreDAO(), BucketName.DEFAULT, blobIdFactory); + this.messageUidProvider = new PostgresUidProvider(new PostgresMailboxDAO(postgresExtension.getDefaultPostgresExecutor())); + } + + @Override + public List getSupportedCapabilities() { + return ImmutableList.of(Capabilities.ANNOTATION, Capabilities.MAILBOX, Capabilities.MESSAGE, Capabilities.MOVE, + Capabilities.ATTACHMENT, Capabilities.THREAD_SAFE_FLAGS_UPDATE, Capabilities.UNIQUE_MESSAGE_ID); + } + + @Override + public MailboxMapper createMailboxMapper() { + return new PostgresMailboxMapper(new PostgresMailboxDAO(postgresExtension.getDefaultPostgresExecutor())); + } + + @Override + public MessageMapper createMessageMapper() { + PostgresMailboxDAO mailboxDAO = new PostgresMailboxDAO(postgresExtension.getDefaultPostgresExecutor()); + + PostgresModSeqProvider modSeqProvider = new PostgresModSeqProvider(mailboxDAO); + PostgresUidProvider uidProvider = new PostgresUidProvider(mailboxDAO); + + return new PostgresMessageMapper( + postgresExtension.getDefaultPostgresExecutor(), + modSeqProvider, + uidProvider, + blobStore, + updatableTickingClock, + blobIdFactory); + } + + @Override + public MessageIdMapper createMessageIdMapper() { + PostgresMailboxDAO mailboxDAO = new PostgresMailboxDAO(postgresExtension.getDefaultPostgresExecutor()); + return new PostgresMessageIdMapper(mailboxDAO, + new PostgresMessageDAO(postgresExtension.getDefaultPostgresExecutor(), blobIdFactory), + new PostgresMailboxMessageDAO(postgresExtension.getDefaultPostgresExecutor()), + new PostgresModSeqProvider(mailboxDAO), + new PostgresAttachmentMapper(new PostgresAttachmentDAO(postgresExtension.getDefaultPostgresExecutor(), blobIdFactory), blobStore), + blobStore, + blobIdFactory, + updatableTickingClock); + } + + @Override + public AttachmentMapper createAttachmentMapper() { + throw new NotImplementedException("not implemented"); + } + + @Override + public MailboxId generateId() { + return PostgresMailboxId.generate(); + } + + @Override + public MessageUid generateMessageUid(Mailbox mailbox) { + try { + return messageUidProvider.nextUid(mailbox); + } catch (MailboxException e) { + throw new RuntimeException(e); + } + } + + @Override + public ModSeq generateModSeq(Mailbox mailbox) { + try { + return new PostgresModSeqProvider(new PostgresMailboxDAO(postgresExtension.getDefaultPostgresExecutor())) + .nextModSeq(mailbox); + } catch (MailboxException e) { + throw new RuntimeException(e); + } + } + + @Override + public ModSeq highestModSeq(Mailbox mailbox) { + return new PostgresModSeqProvider(new PostgresMailboxDAO(postgresExtension.getDefaultPostgresExecutor())) + .highestModSeq(mailbox); + } + + @Override + public boolean supportPartialAttachmentFetch() { + return false; + } + + @Override + public MessageId generateMessageId() { + return messageIdFactory.generate(); + } + + public UpdatableTickingClock getUpdatableTickingClock() { + return updatableTickingClock; + } +} \ No newline at end of file diff --git a/mailbox/postgres/src/test/java/org/apache/james/mailbox/postgres/mail/PostgresMessageBlobReferenceSourceTest.java b/mailbox/postgres/src/test/java/org/apache/james/mailbox/postgres/mail/PostgresMessageBlobReferenceSourceTest.java new file mode 100644 index 00000000000..05462b06a30 --- /dev/null +++ b/mailbox/postgres/src/test/java/org/apache/james/mailbox/postgres/mail/PostgresMessageBlobReferenceSourceTest.java @@ -0,0 +1,100 @@ +/**************************************************************** + * 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.mailbox.postgres.mail; + +import static org.assertj.core.api.Assertions.assertThat; + +import java.nio.charset.StandardCharsets; +import java.util.Date; +import java.util.UUID; + +import jakarta.mail.Flags; + +import org.apache.james.backends.postgres.PostgresExtension; +import org.apache.james.blob.api.PlainBlobId; +import org.apache.james.mailbox.MessageUid; +import org.apache.james.mailbox.model.ByteContent; +import org.apache.james.mailbox.model.MessageId; +import org.apache.james.mailbox.model.ThreadId; +import org.apache.james.mailbox.postgres.PostgresMailboxAggregateModule; +import org.apache.james.mailbox.postgres.PostgresMailboxId; +import org.apache.james.mailbox.postgres.PostgresMessageId; +import org.apache.james.mailbox.postgres.mail.dao.PostgresMessageDAO; +import org.apache.james.mailbox.store.mail.model.MailboxMessage; +import org.apache.james.mailbox.store.mail.model.impl.PropertyBuilder; +import org.apache.james.mailbox.store.mail.model.impl.SimpleMailboxMessage; +import org.junit.jupiter.api.BeforeEach; +import org.junit.jupiter.api.Test; +import org.junit.jupiter.api.extension.RegisterExtension; + +public class PostgresMessageBlobReferenceSourceTest { + private static final int BODY_START = 16; + private static final PostgresMailboxId MAILBOX_ID = PostgresMailboxId.generate(); + private static final String CONTENT = "Subject: Test7 \n\nBody7\n.\n"; + private static final String CONTENT_2 = "Subject: Test3 \n\nBody23\n.\n"; + private static final MessageUid MESSAGE_UID = MessageUid.of(1); + + @RegisterExtension + static PostgresExtension postgresExtension = PostgresExtension.withoutRowLevelSecurity(PostgresMailboxAggregateModule.MODULE); + + PostgresMessageBlobReferenceSource blobReferenceSource; + PostgresMessageDAO postgresMessageDAO; + + @BeforeEach + void beforeEach() { + postgresMessageDAO = new PostgresMessageDAO(postgresExtension.getDefaultPostgresExecutor(), new PlainBlobId.Factory()); + blobReferenceSource = new PostgresMessageBlobReferenceSource(postgresMessageDAO); + } + + @Test + void blobReferencesShouldBeEmptyByDefault() { + assertThat(blobReferenceSource.listReferencedBlobs().collectList().block()) + .isEmpty(); + } + + @Test + void blobReferencesShouldReturnAllBlobs() { + MessageId messageId1 = PostgresMessageId.Factory.of(UUID.randomUUID()); + SimpleMailboxMessage message = createMessage(messageId1, ThreadId.fromBaseMessageId(messageId1), CONTENT, BODY_START, new PropertyBuilder()); + MessageId messageId2 = PostgresMessageId.Factory.of(UUID.randomUUID()); + MailboxMessage message2 = createMessage(messageId2, ThreadId.fromBaseMessageId(messageId2), CONTENT_2, BODY_START, new PropertyBuilder()); + postgresMessageDAO.insert(message, "1").block(); + postgresMessageDAO.insert(message2, "2").block(); + + assertThat(blobReferenceSource.listReferencedBlobs().collectList().block()) + .hasSize(2); + } + + private SimpleMailboxMessage createMessage(MessageId messageId, ThreadId threadId, String content, int bodyStart, PropertyBuilder propertyBuilder) { + return SimpleMailboxMessage.builder() + .messageId(messageId) + .threadId(threadId) + .mailboxId(MAILBOX_ID) + .uid(MESSAGE_UID) + .internalDate(new Date()) + .bodyStartOctet(bodyStart) + .size(content.length()) + .content(new ByteContent(content.getBytes(StandardCharsets.UTF_8))) + .flags(new Flags()) + .properties(propertyBuilder) + .build(); + } + +} diff --git a/mailbox/postgres/src/test/java/org/apache/james/mailbox/postgres/mail/PostgresMessageIdMapperTest.java b/mailbox/postgres/src/test/java/org/apache/james/mailbox/postgres/mail/PostgresMessageIdMapperTest.java new file mode 100644 index 00000000000..873e7b66332 --- /dev/null +++ b/mailbox/postgres/src/test/java/org/apache/james/mailbox/postgres/mail/PostgresMessageIdMapperTest.java @@ -0,0 +1,45 @@ +/**************************************************************** + * 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.mailbox.postgres.mail; + +import org.apache.james.backends.postgres.PostgresExtension; +import org.apache.james.mailbox.postgres.PostgresMailboxAggregateModule; +import org.apache.james.mailbox.store.mail.model.MapperProvider; +import org.apache.james.mailbox.store.mail.model.MessageIdMapperTest; +import org.apache.james.utils.UpdatableTickingClock; +import org.junit.jupiter.api.extension.RegisterExtension; + +public class PostgresMessageIdMapperTest extends MessageIdMapperTest { + @RegisterExtension + static PostgresExtension postgresExtension = PostgresExtension.withoutRowLevelSecurity(PostgresMailboxAggregateModule.MODULE); + + private PostgresMapperProvider postgresMapperProvider; + + @Override + protected MapperProvider provideMapper() { + postgresMapperProvider = new PostgresMapperProvider(postgresExtension); + return postgresMapperProvider; + } + + @Override + protected UpdatableTickingClock updatableTickingClock() { + return postgresMapperProvider.getUpdatableTickingClock(); + } +} diff --git a/mailbox/postgres/src/test/java/org/apache/james/mailbox/postgres/mail/PostgresMessageMapperRowLevelSecurityTest.java b/mailbox/postgres/src/test/java/org/apache/james/mailbox/postgres/mail/PostgresMessageMapperRowLevelSecurityTest.java new file mode 100644 index 00000000000..3023cd717a1 --- /dev/null +++ b/mailbox/postgres/src/test/java/org/apache/james/mailbox/postgres/mail/PostgresMessageMapperRowLevelSecurityTest.java @@ -0,0 +1,111 @@ +/**************************************************************** + * 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.mailbox.postgres.mail; + +import static org.assertj.core.api.Assertions.assertThat; + +import java.time.Instant; +import java.util.Date; + +import jakarta.mail.Flags; + +import org.apache.james.backends.postgres.PostgresConfiguration; +import org.apache.james.backends.postgres.PostgresExtension; +import org.apache.james.blob.api.BlobId; +import org.apache.james.blob.api.BucketName; +import org.apache.james.blob.api.PlainBlobId; +import org.apache.james.blob.memory.MemoryBlobStoreDAO; +import org.apache.james.core.Username; +import org.apache.james.mailbox.MailboxSession; +import org.apache.james.mailbox.MailboxSessionUtil; +import org.apache.james.mailbox.model.ByteContent; +import org.apache.james.mailbox.model.Mailbox; +import org.apache.james.mailbox.model.MailboxPath; +import org.apache.james.mailbox.model.MessageId; +import org.apache.james.mailbox.model.ThreadId; +import org.apache.james.mailbox.model.UidValidity; +import org.apache.james.mailbox.postgres.PostgresMailboxAggregateModule; +import org.apache.james.mailbox.postgres.PostgresMailboxSessionMapperFactory; +import org.apache.james.mailbox.postgres.PostgresMessageId; +import org.apache.james.mailbox.postgres.mail.dao.PostgresMailboxDAO; +import org.apache.james.mailbox.store.mail.MailboxMapper; +import org.apache.james.mailbox.store.mail.model.MailboxMessage; +import org.apache.james.mailbox.store.mail.model.impl.PropertyBuilder; +import org.apache.james.mailbox.store.mail.model.impl.SimpleMailboxMessage; +import org.apache.james.server.blob.deduplication.DeDuplicationBlobStore; +import org.apache.james.utils.UpdatableTickingClock; +import org.junit.jupiter.api.BeforeEach; +import org.junit.jupiter.api.Test; +import org.junit.jupiter.api.extension.RegisterExtension; + +public class PostgresMessageMapperRowLevelSecurityTest { + private static final int BODY_START = 16; + private static final UidValidity UID_VALIDITY = UidValidity.of(42); + private static final Username BENWA = Username.of("benwa@domain.org"); + protected static final MailboxPath benwaInboxPath = MailboxPath.forUser(BENWA, "INBOX"); + private static final MailboxSession aliceSession = MailboxSessionUtil.create(Username.of("alice@domain1")); + private static final MailboxSession bobSession = MailboxSessionUtil.create(Username.of("bob@domain1")); + private static final MailboxSession bobDomain2Session = MailboxSessionUtil.create(Username.of("bob@domain2")); + + @RegisterExtension + static PostgresExtension postgresExtension = PostgresExtension.withRowLevelSecurity(PostgresMailboxAggregateModule.MODULE); + + private PostgresMailboxSessionMapperFactory postgresMailboxSessionMapperFactory; + private Mailbox mailbox; + + private Mailbox generateMailbox() { + MailboxMapper mailboxMapper = new PostgresMailboxMapper(new PostgresMailboxDAO(postgresExtension.getDefaultPostgresExecutor())); + return mailboxMapper.create(benwaInboxPath, UID_VALIDITY).block(); + } + + @BeforeEach + public void setUp() { + BlobId.Factory blobIdFactory = new PlainBlobId.Factory(); + postgresMailboxSessionMapperFactory = new PostgresMailboxSessionMapperFactory(postgresExtension.getExecutorFactory(), + new UpdatableTickingClock(Instant.now()), + new DeDuplicationBlobStore(new MemoryBlobStoreDAO(), BucketName.DEFAULT, blobIdFactory), + blobIdFactory, + PostgresConfiguration.builder().username("a").password("a").build()); + + mailbox = generateMailbox(); + } + + @Test + void messagesCanBeAccessedAtTheDataLevelByMembersOfTheSameDomain() throws Exception { + postgresMailboxSessionMapperFactory.getMessageMapper(aliceSession).add(mailbox, createMessage()); + + assertThat(postgresMailboxSessionMapperFactory.getMessageMapper(bobSession).countMessagesInMailbox(mailbox)).isEqualTo(1L); + } + + @Test + void messagesShouldBeIsolatedByDomain() throws Exception { + postgresMailboxSessionMapperFactory.getMessageMapper(aliceSession).add(mailbox, createMessage()); + + assertThat(postgresMailboxSessionMapperFactory.getMessageMapper(bobDomain2Session).countMessagesInMailbox(mailbox)).isEqualTo(0L); + } + + private MailboxMessage createMessage() { + return createMessage(mailbox, new PostgresMessageId.Factory().generate(), "Subject: Test1 \n\nBody1\n.\n", BODY_START, new PropertyBuilder()); + } + + private MailboxMessage createMessage(Mailbox mailbox, MessageId messageId, String content, int bodyStart, PropertyBuilder propertyBuilder) { + return new SimpleMailboxMessage(messageId, ThreadId.fromBaseMessageId(messageId), new Date(), content.length(), bodyStart, new ByteContent(content.getBytes()), new Flags(), propertyBuilder.build(), mailbox.getMailboxId()); + } +} diff --git a/mailbox/postgres/src/test/java/org/apache/james/mailbox/postgres/mail/PostgresMessageMapperTest.java b/mailbox/postgres/src/test/java/org/apache/james/mailbox/postgres/mail/PostgresMessageMapperTest.java new file mode 100644 index 00000000000..f41d5561075 --- /dev/null +++ b/mailbox/postgres/src/test/java/org/apache/james/mailbox/postgres/mail/PostgresMessageMapperTest.java @@ -0,0 +1,47 @@ +/**************************************************************** + * 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.mailbox.postgres.mail; + +import org.apache.james.backends.postgres.PostgresExtension; +import org.apache.james.mailbox.postgres.PostgresMailboxAggregateModule; +import org.apache.james.mailbox.store.mail.model.MapperProvider; +import org.apache.james.mailbox.store.mail.model.MessageMapperTest; +import org.apache.james.utils.UpdatableTickingClock; +import org.junit.jupiter.api.extension.RegisterExtension; + +public class PostgresMessageMapperTest extends MessageMapperTest { + + @RegisterExtension + static PostgresExtension postgresExtension = PostgresExtension.withoutRowLevelSecurity(PostgresMailboxAggregateModule.MODULE); + + private PostgresMapperProvider postgresMapperProvider; + + @Override + protected MapperProvider createMapperProvider() { + postgresMapperProvider = new PostgresMapperProvider(postgresExtension); + return postgresMapperProvider; + } + + @Override + protected UpdatableTickingClock updatableTickingClock() { + return postgresMapperProvider.getUpdatableTickingClock(); + } + +} diff --git a/mailbox/postgres/src/test/java/org/apache/james/mailbox/postgres/mail/PostgresMessageMoveTest.java b/mailbox/postgres/src/test/java/org/apache/james/mailbox/postgres/mail/PostgresMessageMoveTest.java new file mode 100644 index 00000000000..bb524ca6c68 --- /dev/null +++ b/mailbox/postgres/src/test/java/org/apache/james/mailbox/postgres/mail/PostgresMessageMoveTest.java @@ -0,0 +1,37 @@ +/**************************************************************** + * 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.mailbox.postgres.mail; + +import org.apache.james.backends.postgres.PostgresExtension; +import org.apache.james.mailbox.postgres.PostgresMailboxAggregateModule; +import org.apache.james.mailbox.store.mail.model.MapperProvider; +import org.apache.james.mailbox.store.mail.model.MessageMoveTest; +import org.junit.jupiter.api.extension.RegisterExtension; + +class PostgresMessageMoveTest extends MessageMoveTest { + + @RegisterExtension + static PostgresExtension postgresExtension = PostgresExtension.withoutRowLevelSecurity(PostgresMailboxAggregateModule.MODULE); + + @Override + protected MapperProvider createMapperProvider() { + return new PostgresMapperProvider(postgresExtension); + } +} diff --git a/mailbox/postgres/src/test/java/org/apache/james/mailbox/postgres/mail/PostgresModSeqProviderTest.java b/mailbox/postgres/src/test/java/org/apache/james/mailbox/postgres/mail/PostgresModSeqProviderTest.java new file mode 100644 index 00000000000..cd7e59cca08 --- /dev/null +++ b/mailbox/postgres/src/test/java/org/apache/james/mailbox/postgres/mail/PostgresModSeqProviderTest.java @@ -0,0 +1,104 @@ +/**************************************************************** + * 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.mailbox.postgres.mail; + +import static org.assertj.core.api.Assertions.assertThat; + +import java.time.Duration; +import java.util.UUID; +import java.util.concurrent.ConcurrentSkipListSet; +import java.util.concurrent.ExecutionException; +import java.util.stream.LongStream; + +import org.apache.james.backends.postgres.PostgresExtension; +import org.apache.james.core.Username; +import org.apache.james.mailbox.ModSeq; +import org.apache.james.mailbox.exception.MailboxException; +import org.apache.james.mailbox.model.Mailbox; +import org.apache.james.mailbox.model.MailboxPath; +import org.apache.james.mailbox.model.UidValidity; +import org.apache.james.mailbox.postgres.mail.dao.PostgresMailboxDAO; +import org.apache.james.mailbox.store.mail.ModSeqProvider; +import org.apache.james.util.concurrency.ConcurrentTestRunner; +import org.junit.jupiter.api.BeforeEach; +import org.junit.jupiter.api.Test; +import org.junit.jupiter.api.extension.RegisterExtension; + +import com.github.fge.lambdas.Throwing; + +public class PostgresModSeqProviderTest { + + @RegisterExtension + static PostgresExtension postgresExtension = PostgresExtension.withoutRowLevelSecurity(PostgresMailboxModule.MODULE); + + private ModSeqProvider modSeqProvider; + + private Mailbox mailbox; + + @BeforeEach + void setup() { + PostgresMailboxDAO mailboxDAO = new PostgresMailboxDAO(postgresExtension.getDefaultPostgresExecutor()); + modSeqProvider = new PostgresModSeqProvider(mailboxDAO); + MailboxPath mailboxPath = new MailboxPath("gsoc", Username.of("ieugen" + UUID.randomUUID()), "INBOX"); + UidValidity uidValidity = UidValidity.of(1234); + mailbox = mailboxDAO.create(mailboxPath, uidValidity).block(); + } + + @Test + void highestModSeqShouldRetrieveValueStoredNextModSeq() throws Exception { + int nbEntries = 100; + ModSeq result = modSeqProvider.highestModSeq(mailbox); + assertThat(result).isEqualTo(ModSeq.first()); + LongStream.range(0, nbEntries) + .forEach(Throwing.longConsumer(value -> { + ModSeq modSeq = modSeqProvider.nextModSeq(mailbox); + assertThat(modSeq).isEqualTo(modSeqProvider.highestModSeq(mailbox)); + }) + ); + } + + @Test + void nextModSeqShouldIncrementValueByOne() throws Exception { + int nbEntries = 100; + ModSeq lastModSeq = modSeqProvider.highestModSeq(mailbox); + LongStream.range(lastModSeq.asLong() + 1, lastModSeq.asLong() + nbEntries) + .forEach(Throwing.longConsumer(value -> { + ModSeq result = modSeqProvider.nextModSeq(mailbox); + assertThat(result.asLong()).isEqualTo(value); + })); + } + + @Test + void nextModSeqShouldGenerateUniqueValuesWhenParallelCalls() throws ExecutionException, InterruptedException, MailboxException { + modSeqProvider.nextModSeq(mailbox); + + ConcurrentSkipListSet modSeqs = new ConcurrentSkipListSet<>(); + int nbEntries = 10; + + ConcurrentTestRunner.builder() + .operation( + (threadNumber, step) -> modSeqs.add(modSeqProvider.nextModSeq(mailbox))) + .threadCount(10) + .operationCount(nbEntries) + .runSuccessfullyWithin(Duration.ofMinutes(1)); + + assertThat(modSeqs).hasSize(100); + } +} diff --git a/mailbox/postgres/src/test/java/org/apache/james/mailbox/postgres/mail/PostgresUidProviderTest.java b/mailbox/postgres/src/test/java/org/apache/james/mailbox/postgres/mail/PostgresUidProviderTest.java new file mode 100644 index 00000000000..df8277fb6c7 --- /dev/null +++ b/mailbox/postgres/src/test/java/org/apache/james/mailbox/postgres/mail/PostgresUidProviderTest.java @@ -0,0 +1,140 @@ +/**************************************************************** + * 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.mailbox.postgres.mail; + +import static org.assertj.core.api.Assertions.assertThat; + +import java.time.Duration; +import java.util.List; +import java.util.Optional; +import java.util.UUID; +import java.util.concurrent.ConcurrentSkipListSet; +import java.util.concurrent.ExecutionException; +import java.util.stream.LongStream; + +import org.apache.james.backends.postgres.PostgresExtension; +import org.apache.james.core.Username; +import org.apache.james.mailbox.MessageUid; +import org.apache.james.mailbox.exception.MailboxException; +import org.apache.james.mailbox.model.Mailbox; +import org.apache.james.mailbox.model.MailboxPath; +import org.apache.james.mailbox.model.UidValidity; +import org.apache.james.mailbox.postgres.mail.dao.PostgresMailboxDAO; +import org.apache.james.mailbox.store.mail.UidProvider; +import org.apache.james.util.concurrency.ConcurrentTestRunner; +import org.junit.jupiter.api.BeforeEach; +import org.junit.jupiter.api.Test; +import org.junit.jupiter.api.extension.RegisterExtension; + +import com.github.fge.lambdas.Throwing; + +public class PostgresUidProviderTest { + + @RegisterExtension + static PostgresExtension postgresExtension = PostgresExtension.withoutRowLevelSecurity(PostgresMailboxModule.MODULE); + + private UidProvider uidProvider; + + private Mailbox mailbox; + + @BeforeEach + void setup() { + PostgresMailboxDAO mailboxDAO = new PostgresMailboxDAO(postgresExtension.getDefaultPostgresExecutor()); + uidProvider = new PostgresUidProvider(mailboxDAO); + MailboxPath mailboxPath = new MailboxPath("gsoc", Username.of("ieugen" + UUID.randomUUID()), "INBOX"); + UidValidity uidValidity = UidValidity.of(1234); + mailbox = mailboxDAO.create(mailboxPath, uidValidity).block(); + } + + @Test + void lastUidShouldRetrieveValueStoredByNextUid() throws Exception { + int nbEntries = 100; + Optional result = uidProvider.lastUid(mailbox); + assertThat(result).isEmpty(); + LongStream.range(0, nbEntries) + .forEach(Throwing.longConsumer(value -> { + MessageUid uid = uidProvider.nextUid(mailbox); + assertThat(uid).isEqualTo(uidProvider.lastUid(mailbox).get()); + }) + ); + } + + @Test + void nextUidShouldIncrementValueByOne() { + int nbEntries = 100; + LongStream.range(1, nbEntries) + .forEach(Throwing.longConsumer(value -> { + MessageUid result = uidProvider.nextUid(mailbox); + assertThat(value).isEqualTo(result.asLong()); + })); + } + + @Test + void nextUidShouldGenerateUniqueValuesWhenParallelCalls() throws ExecutionException, InterruptedException, MailboxException { + uidProvider.nextUid(mailbox); + int threadCount = 10; + int nbEntries = 100; + + ConcurrentSkipListSet messageUids = new ConcurrentSkipListSet<>(); + ConcurrentTestRunner.builder() + .operation((threadNumber, step) -> messageUids.add(uidProvider.nextUid(mailbox))) + .threadCount(threadCount) + .operationCount(nbEntries / threadCount) + .runSuccessfullyWithin(Duration.ofMinutes(1)); + + assertThat(messageUids).hasSize(nbEntries); + } + + @Test + void nextUidsShouldGenerateUniqueValuesWhenParallelCalls() throws ExecutionException, InterruptedException, MailboxException { + uidProvider.nextUid(mailbox); + + int threadCount = 10; + int nbOperations = 100; + + ConcurrentSkipListSet messageUids = new ConcurrentSkipListSet<>(); + ConcurrentTestRunner.builder() + .operation((threadNumber, step) -> messageUids.addAll(uidProvider.nextUids(mailbox.getMailboxId(), 10).block())) + .threadCount(threadCount) + .operationCount(nbOperations / threadCount) + .runSuccessfullyWithin(Duration.ofMinutes(1)); + + assertThat(messageUids).hasSize(nbOperations * 10); + } + + @Test + void nextUidWithCountShouldReturnCorrectUids() { + int count = 10; + List messageUids = uidProvider.nextUids(mailbox.getMailboxId(), count).block(); + assertThat(messageUids).hasSize(count) + .containsExactlyInAnyOrder( + MessageUid.of(1), + MessageUid.of(2), + MessageUid.of(3), + MessageUid.of(4), + MessageUid.of(5), + MessageUid.of(6), + MessageUid.of(7), + MessageUid.of(8), + MessageUid.of(9), + MessageUid.of(10)); + } + +} diff --git a/mailbox/postgres/src/test/java/org/apache/james/mailbox/postgres/mail/RLSSupportPostgresMailboxMapperACLTest.java b/mailbox/postgres/src/test/java/org/apache/james/mailbox/postgres/mail/RLSSupportPostgresMailboxMapperACLTest.java new file mode 100644 index 00000000000..1352f0b74e1 --- /dev/null +++ b/mailbox/postgres/src/test/java/org/apache/james/mailbox/postgres/mail/RLSSupportPostgresMailboxMapperACLTest.java @@ -0,0 +1,39 @@ +/**************************************************************** + * 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.mailbox.postgres.mail; + +import org.apache.james.backends.postgres.PostgresExtension; +import org.apache.james.backends.postgres.PostgresModule; +import org.apache.james.mailbox.postgres.mail.dao.PostgresMailboxDAO; +import org.apache.james.mailbox.store.mail.MailboxMapper; +import org.apache.james.mailbox.store.mail.model.MailboxMapperACLTest; +import org.junit.jupiter.api.extension.RegisterExtension; + +class RLSSupportPostgresMailboxMapperACLTest extends MailboxMapperACLTest { + @RegisterExtension + static PostgresExtension postgresExtension = PostgresExtension.withoutRowLevelSecurity(PostgresModule.aggregateModules(PostgresMailboxModule.MODULE, + PostgresMailboxMemberModule.MODULE)); + + @Override + protected MailboxMapper createMailboxMapper() { + return new RLSSupportPostgresMailboxMapper(new PostgresMailboxDAO(postgresExtension.getDefaultPostgresExecutor()), + new PostgresMailboxMemberDAO(postgresExtension.getDefaultPostgresExecutor())); + } +} diff --git a/mailbox/postgres/src/test/java/org/apache/james/mailbox/postgres/mail/dao/PostgresMailboxMessageDAOTest.java b/mailbox/postgres/src/test/java/org/apache/james/mailbox/postgres/mail/dao/PostgresMailboxMessageDAOTest.java new file mode 100644 index 00000000000..36785cc3d6c --- /dev/null +++ b/mailbox/postgres/src/test/java/org/apache/james/mailbox/postgres/mail/dao/PostgresMailboxMessageDAOTest.java @@ -0,0 +1,215 @@ +/**************************************************************** + * 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.mailbox.postgres.mail.dao; + +import static org.apache.james.mailbox.store.mail.MessageMapper.FetchType.METADATA; +import static org.assertj.core.api.Assertions.assertThat; + +import java.nio.charset.StandardCharsets; +import java.util.ArrayList; +import java.util.Date; +import java.util.List; +import java.util.UUID; + +import jakarta.mail.Flags; + +import org.apache.james.backends.postgres.PostgresExtension; +import org.apache.james.backends.postgres.PostgresModule; +import org.apache.james.blob.api.BlobId; +import org.apache.james.blob.api.PlainBlobId; +import org.apache.james.mailbox.MessageUid; +import org.apache.james.mailbox.ModSeq; +import org.apache.james.mailbox.model.ByteContent; +import org.apache.james.mailbox.model.ComposedMessageIdWithMetaData; +import org.apache.james.mailbox.model.MessageId; +import org.apache.james.mailbox.model.MessageRange; +import org.apache.james.mailbox.model.ThreadId; +import org.apache.james.mailbox.postgres.PostgresMailboxId; +import org.apache.james.mailbox.postgres.PostgresMessageId; +import org.apache.james.mailbox.postgres.mail.PostgresMailboxModule; +import org.apache.james.mailbox.postgres.mail.PostgresMessageModule; +import org.apache.james.mailbox.store.mail.model.DelegatingMailboxMessage; +import org.apache.james.mailbox.store.mail.model.impl.PropertyBuilder; +import org.apache.james.mailbox.store.mail.model.impl.SimpleMailboxMessage; +import org.apache.james.util.ReactorUtils; +import org.apache.james.util.streams.Limit; +import org.jetbrains.annotations.NotNull; +import org.junit.jupiter.api.AfterAll; +import org.junit.jupiter.api.BeforeAll; +import org.junit.jupiter.api.BeforeEach; +import org.junit.jupiter.api.Test; +import org.junit.jupiter.api.extension.RegisterExtension; + +import reactor.core.publisher.Flux; + +public class PostgresMailboxMessageDAOTest { + @RegisterExtension + static PostgresExtension postgresExtension = PostgresExtension.withoutRowLevelSecurity( + PostgresModule.aggregateModules(PostgresMailboxModule.MODULE, + PostgresMessageModule.MODULE)); + + private final MessageId.Factory messageIdFactory = new PostgresMessageId.Factory(); + private final BlobId.Factory blobIdFactory = new PlainBlobId.Factory(); + + private PostgresMailboxMessageDAO testee; + private PostgresMessageDAO messageDAO; + + @BeforeAll + static void setUpClass() { + // We set the batch size to 10 to test the batching + System.setProperty("james.postgresql.query.batch.size", "10"); + } + + @AfterAll + static void tearDownClass() { + System.clearProperty("james.postgresql.query.batch.size"); + } + + @BeforeEach + void setUp() { + testee = new PostgresMailboxMessageDAO(postgresExtension.getDefaultPostgresExecutor()); + messageDAO = new PostgresMessageDAO(postgresExtension.getDefaultPostgresExecutor(), blobIdFactory); + } + + @Test + void findAllRecentMessageMetadataShouldReturnAllMatchingEntryWhenBatchSizeIsSmallerThanAllEntries() { + // Given 100 entries + int sampleSize = 100; + PostgresMailboxId mailboxId = PostgresMailboxId.generate(); + ArrayList messageIds = provisionMailboxMessage(sampleSize, mailboxId); + + // When retrieve all entries + List listResult = testee.findAllRecentMessageMetadata(mailboxId) + .collectList().block(); + + // Then return all entries + assertThat(listResult).hasSize(sampleSize); + + assertThat(listResult.stream().map(metaData -> metaData.getComposedMessageId().getMessageId()).toList()) + .containsExactly(messageIds.toArray(MessageId[]::new)); + } + + private @NotNull ArrayList provisionMailboxMessage(int sampleSize, PostgresMailboxId mailboxId) { + ArrayList messageIds = new ArrayList<>(); + Flux.range(1, sampleSize) + .map(index -> { + SimpleMailboxMessage mailboxMessage = generateSimpleMailboxMessage(index, mailboxId); + messageIds.add(mailboxMessage.getMessageId()); + return mailboxMessage; + }) + .flatMap(message -> messageDAO.insert(message, UUID.randomUUID().toString()) + .then(testee.insert(message)), ReactorUtils.DEFAULT_CONCURRENCY) + .then().block(); + return messageIds; + } + + @Test + void findMessagesByMailboxIdShouldReturnAllMatchingEntryWhenBatchSizeIsSmallerThanAllEntries() { + // Given 100 entries + int sampleSize = 100; + PostgresMailboxId mailboxId = PostgresMailboxId.generate(); + ArrayList messageIds = provisionMailboxMessage(sampleSize, mailboxId); + + // When retrieve all entries + List listResult = testee.findMessagesByMailboxId(mailboxId, Limit.unlimited(), METADATA) + .map(e -> e.getKey().build()) + .collectList().block(); + + // Then return all entries + assertThat(listResult).hasSize(sampleSize); + + assertThat(listResult.stream().map(message -> message.getMessageId()).toList()) + .containsExactly(messageIds.toArray(MessageId[]::new)); + } + + @Test + void findMessagesByMailboxIdAndBetweenUIDsShouldReturnAllMatchingEntryWhenBatchSizeIsSmallerThanAllEntries() { + // Given 100 entries + int sampleSize = 100; + PostgresMailboxId mailboxId = PostgresMailboxId.generate(); + ArrayList messageIds = provisionMailboxMessage(sampleSize, mailboxId); + + // When retrieve all entries + List listResult = testee.findMessagesByMailboxIdAndBetweenUIDs(mailboxId, MessageUid.of(0), MessageUid.of(sampleSize + 1), Limit.unlimited(), METADATA) + .map(e -> e.getKey().build()) + .collectList().block(); + + // Then return all entries + assertThat(listResult).hasSize(sampleSize); + + assertThat(listResult.stream().map(DelegatingMailboxMessage::getMessageId).toList()) + .containsExactly(messageIds.toArray(MessageId[]::new)); + } + + @Test + void findMessagesByMailboxIdAndAfterUIDShouldReturnAllMatchingEntryWhenBatchSizeIsSmallerThanAllEntries() { + // Given 100 entries + int sampleSize = 100; + PostgresMailboxId mailboxId = PostgresMailboxId.generate(); + ArrayList messageIds = provisionMailboxMessage(sampleSize, mailboxId); + + // When retrieve all entries + List listResult = testee.findMessagesByMailboxIdAndAfterUID(mailboxId, MessageUid.of(0), Limit.unlimited(), METADATA) + .map(e -> e.getKey().build()) + .collectList().block(); + + // Then return all entries + assertThat(listResult).hasSize(sampleSize); + + assertThat(listResult.stream().map(DelegatingMailboxMessage::getMessageId).toList()) + .containsExactly(messageIds.toArray(MessageId[]::new)); + } + + @Test + void findMessagesMetadataShouldReturnAllMatchingEntryWhenBatchSizeIsSmallerThanAllEntries() { + // Given 100 entries + int sampleSize = 100; + PostgresMailboxId mailboxId = PostgresMailboxId.generate(); + ArrayList messageIds = provisionMailboxMessage(sampleSize, mailboxId); + + // When retrieve all entries + List listResult = testee.findMessagesMetadata(mailboxId, MessageRange.all()) + .collectList().block(); + + // Then return all entries + assertThat(listResult).hasSize(sampleSize); + + assertThat(listResult.stream().map(metaData -> metaData.getComposedMessageId().getMessageId()).toList()) + .containsExactly(messageIds.toArray(MessageId[]::new)); + } + + private SimpleMailboxMessage generateSimpleMailboxMessage(int index, PostgresMailboxId mailboxId) { + MessageId messageId = messageIdFactory.generate(); + String messageContent = "Simple message content" + index; + return SimpleMailboxMessage.builder() + .messageId(messageId) + .threadId(ThreadId.fromBaseMessageId(messageId)) + .uid(MessageUid.of(index)) + .content(new ByteContent((messageContent.getBytes(StandardCharsets.UTF_8)))) + .size(messageContent.length()) + .internalDate(new Date()) + .bodyStartOctet(0) + .flags(new Flags(Flags.Flag.RECENT)) + .properties(new PropertyBuilder()) + .mailboxId(mailboxId) + .modseq(ModSeq.of(index)) + .build(); + } +} diff --git a/mailbox/postgres/src/test/java/org/apache/james/mailbox/postgres/mail/task/PostgresRecomputeCurrentQuotasServiceTest.java b/mailbox/postgres/src/test/java/org/apache/james/mailbox/postgres/mail/task/PostgresRecomputeCurrentQuotasServiceTest.java new file mode 100644 index 00000000000..5055c77abc9 --- /dev/null +++ b/mailbox/postgres/src/test/java/org/apache/james/mailbox/postgres/mail/task/PostgresRecomputeCurrentQuotasServiceTest.java @@ -0,0 +1,128 @@ +/**************************************************************** + * 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.mailbox.postgres.mail.task; + +import org.apache.commons.configuration2.BaseHierarchicalConfiguration; +import org.apache.james.backends.postgres.PostgresExtension; +import org.apache.james.backends.postgres.PostgresModule; +import org.apache.james.backends.postgres.quota.PostgresQuotaCurrentValueDAO; +import org.apache.james.backends.postgres.quota.PostgresQuotaModule; +import org.apache.james.domainlist.api.DomainList; +import org.apache.james.mailbox.MailboxManager; +import org.apache.james.mailbox.SessionProvider; +import org.apache.james.mailbox.postgres.PostgresMailboxAggregateModule; +import org.apache.james.mailbox.postgres.PostgresMailboxManagerProvider; +import org.apache.james.mailbox.postgres.quota.PostgresCurrentQuotaManager; +import org.apache.james.mailbox.quota.CurrentQuotaManager; +import org.apache.james.mailbox.quota.UserQuotaRootResolver; +import org.apache.james.mailbox.quota.task.RecomputeCurrentQuotasService; +import org.apache.james.mailbox.quota.task.RecomputeCurrentQuotasServiceContract; +import org.apache.james.mailbox.quota.task.RecomputeMailboxCurrentQuotasService; +import org.apache.james.mailbox.store.MailboxSessionMapperFactory; +import org.apache.james.mailbox.store.PreDeletionHooks; +import org.apache.james.mailbox.store.StoreMailboxManager; +import org.apache.james.mailbox.store.quota.CurrentQuotaCalculator; +import org.apache.james.mailbox.store.quota.DefaultUserQuotaRootResolver; +import org.apache.james.user.api.UsersRepository; +import org.apache.james.user.postgres.PostgresUserModule; +import org.apache.james.user.postgres.PostgresUsersDAO; +import org.apache.james.user.postgres.PostgresUsersRepository; +import org.apache.james.user.postgres.PostgresUsersRepositoryConfiguration; +import org.junit.jupiter.api.BeforeEach; +import org.junit.jupiter.api.extension.RegisterExtension; + +import com.google.common.collect.ImmutableSet; + +class PostgresRecomputeCurrentQuotasServiceTest implements RecomputeCurrentQuotasServiceContract { + + @RegisterExtension + static PostgresExtension postgresExtension = PostgresExtension.withoutRowLevelSecurity(PostgresModule.aggregateModules( + PostgresMailboxAggregateModule.MODULE, + PostgresQuotaModule.MODULE, + PostgresUserModule.MODULE)); + + static final DomainList NO_DOMAIN_LIST = null; + + PostgresUsersRepository usersRepository; + StoreMailboxManager mailboxManager; + SessionProvider sessionProvider; + CurrentQuotaManager currentQuotaManager; + UserQuotaRootResolver userQuotaRootResolver; + RecomputeCurrentQuotasService testee; + + @BeforeEach + void setUp() throws Exception { + MailboxSessionMapperFactory mapperFactory = PostgresMailboxManagerProvider.provideMailboxSessionMapperFactory(postgresExtension); + + PostgresUsersDAO usersDAO = new PostgresUsersDAO(postgresExtension.getDefaultPostgresExecutor(), + PostgresUsersRepositoryConfiguration.DEFAULT); + + usersRepository = new PostgresUsersRepository(NO_DOMAIN_LIST, usersDAO); + BaseHierarchicalConfiguration configuration = new BaseHierarchicalConfiguration(); + configuration.addProperty("enableVirtualHosting", "false"); + usersRepository.configure(configuration); + + mailboxManager = PostgresMailboxManagerProvider.provideMailboxManager(postgresExtension, PreDeletionHooks.NO_PRE_DELETION_HOOK); + sessionProvider = mailboxManager.getSessionProvider(); + currentQuotaManager = new PostgresCurrentQuotaManager(new PostgresQuotaCurrentValueDAO(postgresExtension.getDefaultPostgresExecutor())); + + userQuotaRootResolver = new DefaultUserQuotaRootResolver(sessionProvider, mapperFactory); + + CurrentQuotaCalculator currentQuotaCalculator = new CurrentQuotaCalculator(mapperFactory, userQuotaRootResolver); + + testee = new RecomputeCurrentQuotasService(usersRepository, + ImmutableSet.of(new RecomputeMailboxCurrentQuotasService(currentQuotaManager, + currentQuotaCalculator, + userQuotaRootResolver, + sessionProvider, + mailboxManager), + RECOMPUTE_JMAP_UPLOAD_CURRENT_QUOTAS_SERVICE)); + } + + @Override + public UsersRepository usersRepository() { + return usersRepository; + } + + @Override + public SessionProvider sessionProvider() { + return sessionProvider; + } + + @Override + public MailboxManager mailboxManager() { + return mailboxManager; + } + + @Override + public CurrentQuotaManager currentQuotaManager() { + return currentQuotaManager; + } + + @Override + public UserQuotaRootResolver userQuotaRootResolver() { + return userQuotaRootResolver; + } + + @Override + public RecomputeCurrentQuotasService testee() { + return testee; + } +} diff --git a/mailbox/postgres/src/test/java/org/apache/james/mailbox/postgres/quota/PostgresCurrentQuotaManagerTest.java b/mailbox/postgres/src/test/java/org/apache/james/mailbox/postgres/quota/PostgresCurrentQuotaManagerTest.java new file mode 100644 index 00000000000..3402e281894 --- /dev/null +++ b/mailbox/postgres/src/test/java/org/apache/james/mailbox/postgres/quota/PostgresCurrentQuotaManagerTest.java @@ -0,0 +1,46 @@ +/**************************************************************** + * 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.mailbox.postgres.quota; + +import org.apache.james.backends.postgres.PostgresExtension; +import org.apache.james.backends.postgres.quota.PostgresQuotaCurrentValueDAO; +import org.apache.james.backends.postgres.quota.PostgresQuotaModule; +import org.apache.james.mailbox.quota.CurrentQuotaManager; +import org.apache.james.mailbox.store.quota.CurrentQuotaManagerContract; +import org.junit.jupiter.api.BeforeEach; +import org.junit.jupiter.api.extension.RegisterExtension; + +class PostgresCurrentQuotaManagerTest implements CurrentQuotaManagerContract { + + @RegisterExtension + static PostgresExtension postgresExtension = PostgresExtension.withoutRowLevelSecurity(PostgresQuotaModule.MODULE); + + private PostgresCurrentQuotaManager currentQuotaManager; + + @BeforeEach + void setup() { + currentQuotaManager = new PostgresCurrentQuotaManager(new PostgresQuotaCurrentValueDAO(postgresExtension.getDefaultPostgresExecutor())); + } + + @Override + public CurrentQuotaManager testee() { + return currentQuotaManager; + } +} diff --git a/mailbox/postgres/src/test/java/org/apache/james/mailbox/postgres/quota/PostgresPerUserMaxQuotaManagerTest.java b/mailbox/postgres/src/test/java/org/apache/james/mailbox/postgres/quota/PostgresPerUserMaxQuotaManagerTest.java new file mode 100644 index 00000000000..1d4eb2d14c8 --- /dev/null +++ b/mailbox/postgres/src/test/java/org/apache/james/mailbox/postgres/quota/PostgresPerUserMaxQuotaManagerTest.java @@ -0,0 +1,37 @@ +/**************************************************************** + * 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.mailbox.postgres.quota; + +import org.apache.james.backends.postgres.PostgresExtension; +import org.apache.james.backends.postgres.quota.PostgresQuotaLimitDAO; +import org.apache.james.backends.postgres.quota.PostgresQuotaModule; +import org.apache.james.mailbox.quota.MaxQuotaManager; +import org.apache.james.mailbox.store.quota.GenericMaxQuotaManagerTest; +import org.junit.jupiter.api.extension.RegisterExtension; + +public class PostgresPerUserMaxQuotaManagerTest extends GenericMaxQuotaManagerTest { + @RegisterExtension + static PostgresExtension postgresExtension = PostgresExtension.withoutRowLevelSecurity(PostgresQuotaModule.MODULE); + + @Override + protected MaxQuotaManager provideMaxQuotaManager() { + return new PostgresPerUserMaxQuotaManager(new PostgresQuotaLimitDAO(postgresExtension.getDefaultPostgresExecutor())); + } +} diff --git a/mailbox/postgres/src/test/java/org/apache/james/mailbox/postgres/search/AllSearchOverrideTest.java b/mailbox/postgres/src/test/java/org/apache/james/mailbox/postgres/search/AllSearchOverrideTest.java new file mode 100644 index 00000000000..f5c67e248e5 --- /dev/null +++ b/mailbox/postgres/src/test/java/org/apache/james/mailbox/postgres/search/AllSearchOverrideTest.java @@ -0,0 +1,132 @@ +/**************************************************************** + * 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.mailbox.postgres.search; + +import static org.apache.james.mailbox.postgres.search.SearchOverrideFixture.BLOB_ID; +import static org.apache.james.mailbox.postgres.search.SearchOverrideFixture.MAILBOX; +import static org.apache.james.mailbox.postgres.search.SearchOverrideFixture.MAILBOX_SESSION; +import static org.assertj.core.api.Assertions.assertThat; + +import jakarta.mail.Flags; + +import org.apache.james.backends.postgres.PostgresExtension; +import org.apache.james.blob.api.PlainBlobId; +import org.apache.james.mailbox.MessageUid; +import org.apache.james.mailbox.model.MailboxId; +import org.apache.james.mailbox.model.SearchQuery; +import org.apache.james.mailbox.postgres.PostgresMailboxAggregateModule; +import org.apache.james.mailbox.postgres.PostgresMailboxId; +import org.apache.james.mailbox.postgres.mail.dao.PostgresMailboxMessageDAO; +import org.apache.james.mailbox.postgres.mail.dao.PostgresMessageDAO; +import org.apache.james.mailbox.store.mail.model.MailboxMessage; +import org.junit.jupiter.api.BeforeEach; +import org.junit.jupiter.api.Test; +import org.junit.jupiter.api.extension.RegisterExtension; + +public class AllSearchOverrideTest { + @RegisterExtension + static PostgresExtension postgresExtension = PostgresExtension.withoutRowLevelSecurity(PostgresMailboxAggregateModule.MODULE); + + private PostgresMailboxMessageDAO postgresMailboxMessageDAO; + private PostgresMessageDAO postgresMessageDAO; + private AllSearchOverride testee; + + @BeforeEach + void setUp() { + postgresMessageDAO = new PostgresMessageDAO(postgresExtension.getDefaultPostgresExecutor(), new PlainBlobId.Factory()); + postgresMailboxMessageDAO = new PostgresMailboxMessageDAO(postgresExtension.getDefaultPostgresExecutor()); + testee = new AllSearchOverride(postgresExtension.getExecutorFactory()); + } + + @Test + void emptyQueryShouldBeApplicable() { + assertThat(testee.applicable( + SearchQuery.builder() + .sorts(SearchQuery.DEFAULT_SORTS) + .build(), + MAILBOX_SESSION)) + .isTrue(); + } + + @Test + void allQueryShouldBeApplicable() { + assertThat(testee.applicable( + SearchQuery.builder() + .andCriteria(SearchQuery.all()) + .sorts(SearchQuery.DEFAULT_SORTS) + .build(), + MAILBOX_SESSION)) + .isTrue(); + } + + @Test + void fromOneQueryShouldBeApplicable() { + assertThat(testee.applicable( + SearchQuery.builder() + .andCriteria(SearchQuery.uid(new SearchQuery.UidRange(MessageUid.MIN_VALUE, MessageUid.MAX_VALUE))) + .sorts(SearchQuery.DEFAULT_SORTS) + .build(), + MAILBOX_SESSION)) + .isTrue(); + } + + @Test + void sizeQueryShouldNotBeApplicable() { + assertThat(testee.applicable( + SearchQuery.builder() + .andCriteria(SearchQuery.sizeEquals(12)) + .build(), + MAILBOX_SESSION)) + .isFalse(); + } + + @Test + void searchShouldReturnEmptyByDefault() { + assertThat(testee.search(MAILBOX_SESSION, MAILBOX, + SearchQuery.builder() + .andCriteria(SearchQuery.all()) + .build()).collectList().block()) + .isEmpty(); + } + + @Test + void searchShouldReturnMailboxEntries() { + MessageUid messageUid = MessageUid.of(1); + insert(messageUid, MAILBOX.getMailboxId()); + + MessageUid messageUid2 = MessageUid.of(2); + insert(messageUid2, MAILBOX.getMailboxId()); + + MessageUid messageUid3 = MessageUid.of(3); + insert(messageUid3, PostgresMailboxId.generate()); + + assertThat(testee.search(MAILBOX_SESSION, MAILBOX, + SearchQuery.builder() + .andCriteria(SearchQuery.all()) + .build()).collectList().block()) + .containsOnly(messageUid, messageUid2); + } + + private void insert(MessageUid messageUid, MailboxId mailboxId) { + MailboxMessage message = SearchOverrideFixture.createMessage(messageUid, mailboxId, new Flags()); + postgresMessageDAO.insert(message, BLOB_ID).block(); + postgresMailboxMessageDAO.insert(message).block(); + } +} diff --git a/mailbox/postgres/src/test/java/org/apache/james/mailbox/postgres/search/DeletedSearchOverrideTest.java b/mailbox/postgres/src/test/java/org/apache/james/mailbox/postgres/search/DeletedSearchOverrideTest.java new file mode 100644 index 00000000000..a62d64b5b05 --- /dev/null +++ b/mailbox/postgres/src/test/java/org/apache/james/mailbox/postgres/search/DeletedSearchOverrideTest.java @@ -0,0 +1,112 @@ +/**************************************************************** + * 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.mailbox.postgres.search; + +import static jakarta.mail.Flags.Flag.DELETED; +import static jakarta.mail.Flags.Flag.SEEN; +import static org.apache.james.mailbox.postgres.search.SearchOverrideFixture.BLOB_ID; +import static org.apache.james.mailbox.postgres.search.SearchOverrideFixture.MAILBOX; +import static org.apache.james.mailbox.postgres.search.SearchOverrideFixture.MAILBOX_SESSION; +import static org.assertj.core.api.Assertions.assertThat; + +import jakarta.mail.Flags; + +import org.apache.james.backends.postgres.PostgresExtension; +import org.apache.james.blob.api.PlainBlobId; +import org.apache.james.mailbox.MessageUid; +import org.apache.james.mailbox.model.MailboxId; +import org.apache.james.mailbox.model.SearchQuery; +import org.apache.james.mailbox.postgres.PostgresMailboxAggregateModule; +import org.apache.james.mailbox.postgres.mail.dao.PostgresMailboxMessageDAO; +import org.apache.james.mailbox.postgres.mail.dao.PostgresMessageDAO; +import org.apache.james.mailbox.store.mail.model.MailboxMessage; +import org.junit.jupiter.api.BeforeEach; +import org.junit.jupiter.api.Test; +import org.junit.jupiter.api.extension.RegisterExtension; + +public class DeletedSearchOverrideTest { + @RegisterExtension + static PostgresExtension postgresExtension = PostgresExtension.withoutRowLevelSecurity(PostgresMailboxAggregateModule.MODULE); + + private PostgresMailboxMessageDAO postgresMailboxMessageDAO; + private PostgresMessageDAO postgresMessageDAO; + private DeletedSearchOverride testee; + + @BeforeEach + void setUp() { + postgresMessageDAO = new PostgresMessageDAO(postgresExtension.getDefaultPostgresExecutor(), new PlainBlobId.Factory()); + postgresMailboxMessageDAO = new PostgresMailboxMessageDAO(postgresExtension.getDefaultPostgresExecutor()); + testee = new DeletedSearchOverride(postgresExtension.getExecutorFactory()); + } + + @Test + void deletedQueryShouldBeApplicable() { + assertThat(testee.applicable( + SearchQuery.builder() + .andCriteria(SearchQuery.flagIsSet(DELETED)) + .sorts(SearchQuery.DEFAULT_SORTS) + .build(), + MAILBOX_SESSION)) + .isTrue(); + } + + @Test + void sizeQueryShouldNotBeApplicable() { + assertThat(testee.applicable( + SearchQuery.builder() + .andCriteria(SearchQuery.sizeEquals(12)) + .build(), + MAILBOX_SESSION)) + .isFalse(); + } + + @Test + void searchShouldReturnEmptyByDefault() { + assertThat(testee.search(MAILBOX_SESSION, MAILBOX, + SearchQuery.builder() + .andCriteria(SearchQuery.flagIsUnSet(SEEN)) + .build()).collectList().block()) + .isEmpty(); + } + + @Test + void searchShouldReturnMailboxEntries() { + MessageUid messageUid = MessageUid.of(1); + insert(messageUid, MAILBOX.getMailboxId(), new Flags(DELETED)); + + MessageUid messageUid2 = MessageUid.of(2); + insert(messageUid2, MAILBOX.getMailboxId(), new Flags(DELETED)); + + MessageUid messageUid3 = MessageUid.of(3); + insert(messageUid3, MAILBOX.getMailboxId(), new Flags()); + + assertThat(testee.search(MAILBOX_SESSION, MAILBOX, + SearchQuery.builder() + .andCriteria(SearchQuery.flagIsSet(DELETED)) + .build()).collectList().block()) + .containsOnly(messageUid, messageUid2); + } + + private void insert(MessageUid messageUid, MailboxId mailboxId, Flags flags) { + MailboxMessage message = SearchOverrideFixture.createMessage(messageUid, mailboxId, flags); + postgresMessageDAO.insert(message, BLOB_ID).block(); + postgresMailboxMessageDAO.insert(message).block(); + } +} diff --git a/mailbox/postgres/src/test/java/org/apache/james/mailbox/postgres/search/DeletedWithRangeSearchOverrideTest.java b/mailbox/postgres/src/test/java/org/apache/james/mailbox/postgres/search/DeletedWithRangeSearchOverrideTest.java new file mode 100644 index 00000000000..2ce69251d62 --- /dev/null +++ b/mailbox/postgres/src/test/java/org/apache/james/mailbox/postgres/search/DeletedWithRangeSearchOverrideTest.java @@ -0,0 +1,131 @@ +/**************************************************************** + * 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.mailbox.postgres.search; + +import static jakarta.mail.Flags.Flag.DELETED; +import static jakarta.mail.Flags.Flag.SEEN; +import static org.apache.james.mailbox.postgres.search.SearchOverrideFixture.BLOB_ID; +import static org.apache.james.mailbox.postgres.search.SearchOverrideFixture.MAILBOX; +import static org.apache.james.mailbox.postgres.search.SearchOverrideFixture.MAILBOX_SESSION; +import static org.assertj.core.api.Assertions.assertThat; + +import jakarta.mail.Flags; + +import org.apache.james.backends.postgres.PostgresExtension; +import org.apache.james.blob.api.PlainBlobId; +import org.apache.james.mailbox.MessageUid; +import org.apache.james.mailbox.model.MailboxId; +import org.apache.james.mailbox.model.SearchQuery; +import org.apache.james.mailbox.postgres.PostgresMailboxAggregateModule; +import org.apache.james.mailbox.postgres.mail.dao.PostgresMailboxMessageDAO; +import org.apache.james.mailbox.postgres.mail.dao.PostgresMessageDAO; +import org.apache.james.mailbox.store.mail.model.MailboxMessage; +import org.junit.jupiter.api.BeforeEach; +import org.junit.jupiter.api.Test; +import org.junit.jupiter.api.extension.RegisterExtension; + +public class DeletedWithRangeSearchOverrideTest { + @RegisterExtension + static PostgresExtension postgresExtension = PostgresExtension.withoutRowLevelSecurity(PostgresMailboxAggregateModule.MODULE); + + private PostgresMailboxMessageDAO postgresMailboxMessageDAO; + private PostgresMessageDAO postgresMessageDAO; + private DeletedWithRangeSearchOverride testee; + + @BeforeEach + void setUp() { + postgresMessageDAO = new PostgresMessageDAO(postgresExtension.getDefaultPostgresExecutor(), new PlainBlobId.Factory()); + postgresMailboxMessageDAO = new PostgresMailboxMessageDAO(postgresExtension.getDefaultPostgresExecutor()); + testee = new DeletedWithRangeSearchOverride(postgresExtension.getExecutorFactory()); + } + + @Test + void deletedWithRangeQueryShouldBeApplicable() { + assertThat(testee.applicable( + SearchQuery.builder() + .andCriteria(SearchQuery.flagIsSet(DELETED)) + .andCriteria(SearchQuery.uid(new SearchQuery.UidRange(MessageUid.of(4), MessageUid.of(45)))) + .sorts(SearchQuery.DEFAULT_SORTS) + .build(), + MAILBOX_SESSION)) + .isTrue(); + } + + @Test + void deletedQueryShouldNotBeApplicable() { + assertThat(testee.applicable( + SearchQuery.builder() + .andCriteria(SearchQuery.flagIsSet(DELETED)) + .build(), + MAILBOX_SESSION)) + .isFalse(); + } + + @Test + void sizeQueryShouldNotBeApplicable() { + assertThat(testee.applicable( + SearchQuery.builder() + .andCriteria(SearchQuery.sizeEquals(12)) + .build(), + MAILBOX_SESSION)) + .isFalse(); + } + + @Test + void searchShouldReturnEmptyByDefault() { + assertThat(testee.search(MAILBOX_SESSION, MAILBOX, + SearchQuery.builder() + .andCriteria(SearchQuery.flagIsUnSet(SEEN)) + .andCriteria(SearchQuery.uid(new SearchQuery.UidRange(MessageUid.MIN_VALUE, MessageUid.of(45)))) + .build()).collectList().block()) + .isEmpty(); + } + + @Test + void searchShouldReturnMailboxEntries() { + MessageUid messageUid = MessageUid.of(1); + insert(messageUid, MAILBOX.getMailboxId(), new Flags(DELETED)); + + MessageUid messageUid2 = MessageUid.of(2); + insert(messageUid2, MAILBOX.getMailboxId(), new Flags(DELETED)); + + MessageUid messageUid3 = MessageUid.of(3); + insert(messageUid3, MAILBOX.getMailboxId(), new Flags()); + + MessageUid messageUid4 = MessageUid.of(4); + insert(messageUid4, MAILBOX.getMailboxId(), new Flags(DELETED)); + + MessageUid messageUid5 = MessageUid.of(5); + insert(messageUid5, MAILBOX.getMailboxId(), new Flags(DELETED)); + + assertThat(testee.search(MAILBOX_SESSION, MAILBOX, + SearchQuery.builder() + .andCriteria(SearchQuery.flagIsSet(DELETED)) + .andCriteria(SearchQuery.uid(new SearchQuery.UidRange(messageUid2, messageUid4))) + .build()).collectList().block()) + .containsOnly(messageUid2, messageUid4); + } + + private void insert(MessageUid messageUid, MailboxId mailboxId, Flags flags) { + MailboxMessage message = SearchOverrideFixture.createMessage(messageUid, mailboxId, flags); + postgresMessageDAO.insert(message, BLOB_ID).block(); + postgresMailboxMessageDAO.insert(message).block(); + } +} diff --git a/mailbox/postgres/src/test/java/org/apache/james/mailbox/postgres/search/NotDeletedWithRangeSearchOverrideTest.java b/mailbox/postgres/src/test/java/org/apache/james/mailbox/postgres/search/NotDeletedWithRangeSearchOverrideTest.java new file mode 100644 index 00000000000..194d2b20c94 --- /dev/null +++ b/mailbox/postgres/src/test/java/org/apache/james/mailbox/postgres/search/NotDeletedWithRangeSearchOverrideTest.java @@ -0,0 +1,136 @@ +/**************************************************************** + * 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.mailbox.postgres.search; + +import static jakarta.mail.Flags.Flag.DELETED; +import static org.apache.james.mailbox.postgres.search.SearchOverrideFixture.BLOB_ID; +import static org.apache.james.mailbox.postgres.search.SearchOverrideFixture.MAILBOX; +import static org.apache.james.mailbox.postgres.search.SearchOverrideFixture.MAILBOX_SESSION; +import static org.assertj.core.api.Assertions.assertThat; + +import jakarta.mail.Flags; + +import org.apache.james.backends.postgres.PostgresExtension; +import org.apache.james.blob.api.PlainBlobId; +import org.apache.james.mailbox.MessageUid; +import org.apache.james.mailbox.model.MailboxId; +import org.apache.james.mailbox.model.SearchQuery; +import org.apache.james.mailbox.postgres.PostgresMailboxAggregateModule; +import org.apache.james.mailbox.postgres.PostgresMailboxId; +import org.apache.james.mailbox.postgres.mail.dao.PostgresMailboxMessageDAO; +import org.apache.james.mailbox.postgres.mail.dao.PostgresMessageDAO; +import org.apache.james.mailbox.store.mail.model.MailboxMessage; +import org.junit.jupiter.api.BeforeEach; +import org.junit.jupiter.api.Test; +import org.junit.jupiter.api.extension.RegisterExtension; + +public class NotDeletedWithRangeSearchOverrideTest { + @RegisterExtension + static PostgresExtension postgresExtension = PostgresExtension.withoutRowLevelSecurity(PostgresMailboxAggregateModule.MODULE); + + private PostgresMailboxMessageDAO postgresMailboxMessageDAO; + private PostgresMessageDAO postgresMessageDAO; + private NotDeletedWithRangeSearchOverride testee; + + @BeforeEach + void setUp() { + postgresMessageDAO = new PostgresMessageDAO(postgresExtension.getDefaultPostgresExecutor(), new PlainBlobId.Factory()); + postgresMailboxMessageDAO = new PostgresMailboxMessageDAO(postgresExtension.getDefaultPostgresExecutor()); + testee = new NotDeletedWithRangeSearchOverride(postgresExtension.getExecutorFactory()); + } + + @Test + void undeletedRangeQueryShouldBeApplicable() { + assertThat(testee.applicable( + SearchQuery.builder() + .andCriteria(SearchQuery.flagIsUnSet(DELETED)) + .andCriteria(SearchQuery.uid(new SearchQuery.UidRange(MessageUid.of(4), MessageUid.of(45)))) + .sorts(SearchQuery.DEFAULT_SORTS) + .build(), + MAILBOX_SESSION)) + .isTrue(); + } + + @Test + void notDeletedRangeQueryShouldBeApplicable() { + assertThat(testee.applicable( + SearchQuery.builder() + .andCriteria(SearchQuery.not(SearchQuery.flagIsSet(DELETED))) + .andCriteria(SearchQuery.uid(new SearchQuery.UidRange(MessageUid.of(4), MessageUid.of(45)))) + .sorts(SearchQuery.DEFAULT_SORTS) + .build(), + MAILBOX_SESSION)) + .isTrue(); + } + + @Test + void sizeQueryShouldNotBeApplicable() { + assertThat(testee.applicable( + SearchQuery.builder() + .andCriteria(SearchQuery.sizeEquals(12)) + .build(), + MAILBOX_SESSION)) + .isFalse(); + } + + @Test + void searchShouldReturnEmptyByDefault() { + assertThat(testee.search(MAILBOX_SESSION, MAILBOX, + SearchQuery.builder() + .andCriteria(SearchQuery.not(SearchQuery.flagIsSet(DELETED))) + .andCriteria(SearchQuery.uid(new SearchQuery.UidRange(MessageUid.of(34), MessageUid.of(345)))) + .build()).collectList().block()) + .isEmpty(); + } + + @Test + void searchShouldReturnMailboxEntries() { + MessageUid messageUid = MessageUid.of(1); + insert(messageUid, MAILBOX.getMailboxId(), new Flags()); + + MessageUid messageUid2 = MessageUid.of(2); + insert(messageUid2, MAILBOX.getMailboxId(), new Flags()); + + MessageUid messageUid3 = MessageUid.of(3); + insert(messageUid3, MAILBOX.getMailboxId(), new Flags(DELETED)); + + MessageUid messageUid4 = MessageUid.of(4); + insert(messageUid4, MAILBOX.getMailboxId(), new Flags()); + + MessageUid messageUid5 = MessageUid.of(5); + insert(messageUid5, MAILBOX.getMailboxId(), new Flags()); + + MessageUid messageUid6 = MessageUid.of(6); + insert(messageUid6, PostgresMailboxId.generate(), new Flags(DELETED)); + + assertThat(testee.search(MAILBOX_SESSION, MAILBOX, + SearchQuery.builder() + .andCriteria(SearchQuery.not(SearchQuery.flagIsSet(DELETED))) + .andCriteria(SearchQuery.uid(new SearchQuery.UidRange(messageUid2, messageUid4))) + .build()).collectList().block()) + .containsOnly(messageUid2, messageUid4); + } + + private void insert(MessageUid messageUid, MailboxId mailboxId, Flags flags) { + MailboxMessage message = SearchOverrideFixture.createMessage(messageUid, mailboxId, flags); + postgresMessageDAO.insert(message, BLOB_ID).block(); + postgresMailboxMessageDAO.insert(message).block(); + } +} diff --git a/mailbox/postgres/src/test/java/org/apache/james/mailbox/postgres/search/SearchOverrideFixture.java b/mailbox/postgres/src/test/java/org/apache/james/mailbox/postgres/search/SearchOverrideFixture.java new file mode 100644 index 00000000000..41f8e957dfd --- /dev/null +++ b/mailbox/postgres/src/test/java/org/apache/james/mailbox/postgres/search/SearchOverrideFixture.java @@ -0,0 +1,71 @@ +/**************************************************************** + * 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.mailbox.postgres.search; + +import java.nio.charset.Charset; +import java.nio.charset.StandardCharsets; +import java.util.Date; + +import jakarta.mail.Flags; + +import org.apache.james.core.Username; +import org.apache.james.mailbox.MailboxSession; +import org.apache.james.mailbox.MailboxSessionUtil; +import org.apache.james.mailbox.MessageUid; +import org.apache.james.mailbox.ModSeq; +import org.apache.james.mailbox.model.ByteContent; +import org.apache.james.mailbox.model.Mailbox; +import org.apache.james.mailbox.model.MailboxId; +import org.apache.james.mailbox.model.MailboxPath; +import org.apache.james.mailbox.model.ThreadId; +import org.apache.james.mailbox.model.UidValidity; +import org.apache.james.mailbox.postgres.PostgresMailboxId; +import org.apache.james.mailbox.postgres.PostgresMessageId; +import org.apache.james.mailbox.store.mail.model.MailboxMessage; +import org.apache.james.mailbox.store.mail.model.impl.PropertyBuilder; +import org.apache.james.mailbox.store.mail.model.impl.SimpleMailboxMessage; + +interface SearchOverrideFixture { + MailboxSession MAILBOX_SESSION = MailboxSessionUtil.create(Username.of("benwa")); + Mailbox MAILBOX = new Mailbox(MailboxPath.inbox(MAILBOX_SESSION), UidValidity.of(12), PostgresMailboxId.generate()); + String BLOB_ID = "abc"; + Charset MESSAGE_CHARSET = StandardCharsets.UTF_8; + String MESSAGE_CONTENT = "Simple message content"; + byte[] MESSAGE_CONTENT_BYTES = MESSAGE_CONTENT.getBytes(MESSAGE_CHARSET); + ByteContent CONTENT_STREAM = new ByteContent(MESSAGE_CONTENT_BYTES); + long SIZE = MESSAGE_CONTENT_BYTES.length; + + static MailboxMessage createMessage(MessageUid messageUid, MailboxId mailboxId, Flags flags) { + PostgresMessageId messageId = new PostgresMessageId.Factory().generate(); + return SimpleMailboxMessage.builder() + .messageId(messageId) + .threadId(ThreadId.fromBaseMessageId(messageId)) + .uid(messageUid) + .content(CONTENT_STREAM) + .size(SIZE) + .internalDate(new Date()) + .bodyStartOctet(18) + .flags(flags) + .properties(new PropertyBuilder()) + .mailboxId(mailboxId) + .modseq(ModSeq.of(1)) + .build(); + } +} diff --git a/mailbox/postgres/src/test/java/org/apache/james/mailbox/postgres/search/UidSearchOverrideTest.java b/mailbox/postgres/src/test/java/org/apache/james/mailbox/postgres/search/UidSearchOverrideTest.java new file mode 100644 index 00000000000..3c2649ae9b5 --- /dev/null +++ b/mailbox/postgres/src/test/java/org/apache/james/mailbox/postgres/search/UidSearchOverrideTest.java @@ -0,0 +1,120 @@ +/**************************************************************** + * 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.mailbox.postgres.search; + +import static org.apache.james.mailbox.postgres.search.SearchOverrideFixture.BLOB_ID; +import static org.apache.james.mailbox.postgres.search.SearchOverrideFixture.MAILBOX; +import static org.apache.james.mailbox.postgres.search.SearchOverrideFixture.MAILBOX_SESSION; +import static org.assertj.core.api.Assertions.assertThat; + +import jakarta.mail.Flags; + +import org.apache.james.backends.postgres.PostgresExtension; +import org.apache.james.blob.api.PlainBlobId; +import org.apache.james.mailbox.MessageUid; +import org.apache.james.mailbox.model.MailboxId; +import org.apache.james.mailbox.model.SearchQuery; +import org.apache.james.mailbox.postgres.PostgresMailboxAggregateModule; +import org.apache.james.mailbox.postgres.PostgresMailboxId; +import org.apache.james.mailbox.postgres.mail.dao.PostgresMailboxMessageDAO; +import org.apache.james.mailbox.postgres.mail.dao.PostgresMessageDAO; +import org.apache.james.mailbox.store.mail.model.MailboxMessage; +import org.junit.jupiter.api.BeforeEach; +import org.junit.jupiter.api.Test; +import org.junit.jupiter.api.extension.RegisterExtension; + +public class UidSearchOverrideTest { + @RegisterExtension + static PostgresExtension postgresExtension = PostgresExtension.withoutRowLevelSecurity(PostgresMailboxAggregateModule.MODULE); + + private PostgresMailboxMessageDAO postgresMailboxMessageDAO; + private PostgresMessageDAO postgresMessageDAO; + private UidSearchOverride testee; + + @BeforeEach + void setUp() { + postgresMessageDAO = new PostgresMessageDAO(postgresExtension.getDefaultPostgresExecutor(), new PlainBlobId.Factory()); + postgresMailboxMessageDAO = new PostgresMailboxMessageDAO(postgresExtension.getDefaultPostgresExecutor()); + testee = new UidSearchOverride(postgresExtension.getExecutorFactory()); + } + + @Test + void rangeQueryShouldBeApplicable() { + assertThat(testee.applicable( + SearchQuery.builder() + .andCriteria(SearchQuery.uid(new SearchQuery.UidRange(MessageUid.of(4), MessageUid.of(45)))) + .sorts(SearchQuery.DEFAULT_SORTS) + .build(), + MAILBOX_SESSION)) + .isTrue(); + } + + @Test + void sizeQueryShouldNotBeApplicable() { + assertThat(testee.applicable( + SearchQuery.builder() + .andCriteria(SearchQuery.sizeEquals(12)) + .build(), + MAILBOX_SESSION)) + .isFalse(); + } + + @Test + void searchShouldReturnEmptyByDefault() { + assertThat(testee.search(MAILBOX_SESSION, MAILBOX, + SearchQuery.builder() + .andCriteria(SearchQuery.uid(new SearchQuery.UidRange(MessageUid.of(34), MessageUid.of(345)))) + .build()).collectList().block()) + .isEmpty(); + } + + @Test + void searchShouldReturnMailboxEntries() { + MessageUid messageUid = MessageUid.of(1); + insert(messageUid, MAILBOX.getMailboxId()); + + MessageUid messageUid2 = MessageUid.of(2); + insert(messageUid2, MAILBOX.getMailboxId()); + + MessageUid messageUid3 = MessageUid.of(3); + insert(messageUid3, MAILBOX.getMailboxId()); + + MessageUid messageUid4 = MessageUid.of(4); + insert(messageUid4, MAILBOX.getMailboxId()); + + MessageUid messageUid5 = MessageUid.of(5); + insert(messageUid5, MAILBOX.getMailboxId()); + + MessageUid messageUid6 = MessageUid.of(6); + insert(messageUid6, PostgresMailboxId.generate()); + + assertThat(testee.search(MAILBOX_SESSION, MAILBOX, + SearchQuery.builder() + .andCriteria(SearchQuery.uid(new SearchQuery.UidRange(messageUid2, messageUid4))) + .build()).collectList().block()) + .containsOnly(messageUid2, messageUid3, messageUid4); + } + + private void insert(MessageUid messageUid, MailboxId mailboxId) { + MailboxMessage message = SearchOverrideFixture.createMessage(messageUid, mailboxId, new Flags()); + postgresMessageDAO.insert(message, BLOB_ID).block(); + postgresMailboxMessageDAO.insert(message).block(); + } +} diff --git a/mailbox/postgres/src/test/java/org/apache/james/mailbox/postgres/search/UnseenSearchOverrideTest.java b/mailbox/postgres/src/test/java/org/apache/james/mailbox/postgres/search/UnseenSearchOverrideTest.java new file mode 100644 index 00000000000..248f0a5c2a1 --- /dev/null +++ b/mailbox/postgres/src/test/java/org/apache/james/mailbox/postgres/search/UnseenSearchOverrideTest.java @@ -0,0 +1,190 @@ +/**************************************************************** + * 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.mailbox.postgres.search; + +import static jakarta.mail.Flags.Flag.SEEN; +import static org.apache.james.mailbox.postgres.search.SearchOverrideFixture.BLOB_ID; +import static org.apache.james.mailbox.postgres.search.SearchOverrideFixture.MAILBOX; +import static org.apache.james.mailbox.postgres.search.SearchOverrideFixture.MAILBOX_SESSION; +import static org.assertj.core.api.Assertions.assertThat; + +import jakarta.mail.Flags; + +import org.apache.james.backends.postgres.PostgresExtension; +import org.apache.james.blob.api.PlainBlobId; +import org.apache.james.mailbox.MessageUid; +import org.apache.james.mailbox.model.MailboxId; +import org.apache.james.mailbox.model.SearchQuery; +import org.apache.james.mailbox.postgres.PostgresMailboxAggregateModule; +import org.apache.james.mailbox.postgres.mail.dao.PostgresMailboxMessageDAO; +import org.apache.james.mailbox.postgres.mail.dao.PostgresMessageDAO; +import org.apache.james.mailbox.store.mail.model.MailboxMessage; +import org.junit.jupiter.api.BeforeEach; +import org.junit.jupiter.api.Test; +import org.junit.jupiter.api.extension.RegisterExtension; + +public class UnseenSearchOverrideTest { + @RegisterExtension + static PostgresExtension postgresExtension = PostgresExtension.withoutRowLevelSecurity(PostgresMailboxAggregateModule.MODULE); + + private PostgresMailboxMessageDAO postgresMailboxMessageDAO; + private PostgresMessageDAO postgresMessageDAO; + private UnseenSearchOverride testee; + + @BeforeEach + void setUp() { + postgresMessageDAO = new PostgresMessageDAO(postgresExtension.getDefaultPostgresExecutor(), new PlainBlobId.Factory()); + postgresMailboxMessageDAO = new PostgresMailboxMessageDAO(postgresExtension.getDefaultPostgresExecutor()); + testee = new UnseenSearchOverride(postgresExtension.getExecutorFactory()); + } + + @Test + void unseenQueryShouldBeApplicable() { + assertThat(testee.applicable( + SearchQuery.builder() + .andCriteria(SearchQuery.flagIsUnSet(SEEN)) + .sorts(SearchQuery.DEFAULT_SORTS) + .build(), + MAILBOX_SESSION)) + .isTrue(); + } + + @Test + void notSeenQueryShouldBeApplicable() { + assertThat(testee.applicable( + SearchQuery.builder() + .andCriteria(SearchQuery.not(SearchQuery.flagIsSet(SEEN))) + .sorts(SearchQuery.DEFAULT_SORTS) + .build(), + MAILBOX_SESSION)) + .isTrue(); + } + + @Test + void unseenAndAllQueryShouldBeApplicable() { + assertThat(testee.applicable( + SearchQuery.builder() + .andCriteria(SearchQuery.flagIsUnSet(SEEN)) + .andCriteria(SearchQuery.all()) + .sorts(SearchQuery.DEFAULT_SORTS) + .build(), + MAILBOX_SESSION)) + .isTrue(); + } + + @Test + void notSeenAndAllQueryShouldBeApplicable() { + assertThat(testee.applicable( + SearchQuery.builder() + .andCriteria(SearchQuery.not(SearchQuery.flagIsSet(SEEN))) + .andCriteria(SearchQuery.all()) + .sorts(SearchQuery.DEFAULT_SORTS) + .build(), + MAILBOX_SESSION)) + .isTrue(); + } + + @Test + void unseenAndFromOneQueryShouldBeApplicable() { + assertThat(testee.applicable( + SearchQuery.builder() + .andCriteria(SearchQuery.flagIsUnSet(SEEN)) + .andCriteria(SearchQuery.uid(new SearchQuery.UidRange(MessageUid.MIN_VALUE, MessageUid.MAX_VALUE))) + .sorts(SearchQuery.DEFAULT_SORTS) + .build(), + MAILBOX_SESSION)) + .isTrue(); + } + + @Test + void notSeenFromOneQueryShouldBeApplicable() { + assertThat(testee.applicable( + SearchQuery.builder() + .andCriteria(SearchQuery.not(SearchQuery.flagIsSet(SEEN))) + .andCriteria(SearchQuery.uid(new SearchQuery.UidRange(MessageUid.MIN_VALUE, MessageUid.MAX_VALUE))) + .sorts(SearchQuery.DEFAULT_SORTS) + .build(), + MAILBOX_SESSION)) + .isTrue(); + } + + @Test + void sizeQueryShouldNotBeApplicable() { + assertThat(testee.applicable( + SearchQuery.builder() + .andCriteria(SearchQuery.sizeEquals(12)) + .build(), + MAILBOX_SESSION)) + .isFalse(); + } + + @Test + void searchShouldReturnEmptyByDefault() { + assertThat(testee.search(MAILBOX_SESSION, MAILBOX, + SearchQuery.builder() + .andCriteria(SearchQuery.flagIsUnSet(SEEN)) + .build()).collectList().block()) + .isEmpty(); + } + + @Test + void searchShouldReturnMailboxEntries() { + MessageUid messageUid = MessageUid.of(1); + insert(messageUid, MAILBOX.getMailboxId(), new Flags()); + MessageUid messageUid2 = MessageUid.of(2); + insert(messageUid2, MAILBOX.getMailboxId(), new Flags()); + MessageUid messageUid3 = MessageUid.of(3); + insert(messageUid3, MAILBOX.getMailboxId(), new Flags(SEEN)); + + assertThat(testee.search(MAILBOX_SESSION, MAILBOX, + SearchQuery.builder() + .andCriteria(SearchQuery.flagIsUnSet(SEEN)) + .build()).collectList().block()) + .containsOnly(messageUid, messageUid2); + } + + @Test + void searchShouldSupportRanges() { + MessageUid messageUid = MessageUid.of(1); + insert(messageUid, MAILBOX.getMailboxId(), new Flags()); + + MessageUid messageUid2 = MessageUid.of(2); + insert(messageUid2, MAILBOX.getMailboxId(), new Flags()); + + MessageUid messageUid3 = MessageUid.of(3); + insert(messageUid3, MAILBOX.getMailboxId(), new Flags(SEEN)); + + MessageUid messageUid4 = MessageUid.of(4); + insert(messageUid4, MAILBOX.getMailboxId(), new Flags()); + + assertThat(testee.search(MAILBOX_SESSION, MAILBOX, + SearchQuery.builder() + .andCriteria(SearchQuery.flagIsUnSet(SEEN)) + .andCriterion(SearchQuery.uid(new SearchQuery.UidRange(MessageUid.of(2), MessageUid.of(4)))) + .build()).collectList().block()) + .containsOnly(messageUid2, messageUid4); + } + + private void insert(MessageUid messageUid, MailboxId mailboxId, Flags flags) { + MailboxMessage message = SearchOverrideFixture.createMessage(messageUid, mailboxId, flags); + postgresMessageDAO.insert(message, BLOB_ID).block(); + postgresMailboxMessageDAO.insert(message).block(); + } +} diff --git a/mailbox/postgres/src/test/java/org/apache/james/mailbox/postgres/user/PostgresSubscriptionMapperRowLevelSecurityTest.java b/mailbox/postgres/src/test/java/org/apache/james/mailbox/postgres/user/PostgresSubscriptionMapperRowLevelSecurityTest.java new file mode 100644 index 00000000000..a1db7adcd14 --- /dev/null +++ b/mailbox/postgres/src/test/java/org/apache/james/mailbox/postgres/user/PostgresSubscriptionMapperRowLevelSecurityTest.java @@ -0,0 +1,78 @@ +/**************************************************************** + * 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.mailbox.postgres.user; + +import static org.assertj.core.api.Assertions.assertThat; + +import org.apache.james.backends.postgres.PostgresExtension; +import org.apache.james.backends.postgres.utils.PostgresExecutor; +import org.apache.james.core.Username; +import org.apache.james.mailbox.MailboxSession; +import org.apache.james.mailbox.MailboxSessionUtil; +import org.apache.james.mailbox.store.user.SubscriptionMapperFactory; +import org.apache.james.mailbox.store.user.model.Subscription; +import org.junit.jupiter.api.BeforeEach; +import org.junit.jupiter.api.Test; +import org.junit.jupiter.api.extension.RegisterExtension; + +public class PostgresSubscriptionMapperRowLevelSecurityTest { + @RegisterExtension + static PostgresExtension postgresExtension = PostgresExtension.withRowLevelSecurity(PostgresSubscriptionModule.MODULE); + + private SubscriptionMapperFactory subscriptionMapperFactory; + + @BeforeEach + public void setUp() { + PostgresExecutor.Factory executorFactory = postgresExtension.getExecutorFactory(); + subscriptionMapperFactory = session -> new PostgresSubscriptionMapper(new PostgresSubscriptionDAO(executorFactory.create(session.getUser().getDomainPart()))); + } + + @Test + void subscriptionsCanBeAccessedAtTheDataLevelByMembersOfTheSameDomain() throws Exception { + Username username = Username.of("bob@domain1"); + Username username2 = Username.of("alice@domain1"); + MailboxSession session = MailboxSessionUtil.create(username); + MailboxSession session2 = MailboxSessionUtil.create(username2); + + Subscription subscription = new Subscription(username, "mailbox1"); + subscriptionMapperFactory.getSubscriptionMapper(session) + .save(subscription); + + assertThat(subscriptionMapperFactory.getSubscriptionMapper(session2) + .findSubscriptionsForUser(username)) + .containsOnly(subscription); + } + + @Test + void subscriptionsShouldBeIsolatedByDomain() throws Exception { + Username username = Username.of("bob@domain1"); + Username username2 = Username.of("alice@domain2"); + MailboxSession session = MailboxSessionUtil.create(username); + MailboxSession session2 = MailboxSessionUtil.create(username2); + + Subscription subscription = new Subscription(username, "mailbox1"); + subscriptionMapperFactory.getSubscriptionMapper(session) + .save(subscription); + + assertThat(subscriptionMapperFactory.getSubscriptionMapper(session2) + .findSubscriptionsForUser(username)) + .isEmpty(); + } +} diff --git a/mailbox/postgres/src/test/java/org/apache/james/mailbox/postgres/user/PostgresSubscriptionMapperTest.java b/mailbox/postgres/src/test/java/org/apache/james/mailbox/postgres/user/PostgresSubscriptionMapperTest.java new file mode 100644 index 00000000000..282d5cbfa42 --- /dev/null +++ b/mailbox/postgres/src/test/java/org/apache/james/mailbox/postgres/user/PostgresSubscriptionMapperTest.java @@ -0,0 +1,37 @@ +/**************************************************************** + * 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.mailbox.postgres.user; + +import org.apache.james.backends.postgres.PostgresExtension; +import org.apache.james.mailbox.store.user.SubscriptionMapper; +import org.apache.james.mailbox.store.user.SubscriptionMapperTest; +import org.junit.jupiter.api.extension.RegisterExtension; + +public class PostgresSubscriptionMapperTest extends SubscriptionMapperTest { + + @RegisterExtension + static PostgresExtension postgresExtension = PostgresExtension.withoutRowLevelSecurity(PostgresSubscriptionModule.MODULE); + + @Override + protected SubscriptionMapper createSubscriptionMapper() { + PostgresSubscriptionDAO dao = new PostgresSubscriptionDAO(postgresExtension.getDefaultPostgresExecutor()); + return new PostgresSubscriptionMapper(dao); + } +} diff --git a/mailbox/scanning-search/src/test/java/org/apache/james/mailbox/store/search/SearchThreadIdGuessingAlgorithmTest.java b/mailbox/scanning-search/src/test/java/org/apache/james/mailbox/store/search/SearchThreadIdGuessingAlgorithmTest.java index 223b6f0e1d6..75ea84cfc23 100644 --- a/mailbox/scanning-search/src/test/java/org/apache/james/mailbox/store/search/SearchThreadIdGuessingAlgorithmTest.java +++ b/mailbox/scanning-search/src/test/java/org/apache/james/mailbox/store/search/SearchThreadIdGuessingAlgorithmTest.java @@ -38,8 +38,6 @@ import org.apache.james.mailbox.store.mail.model.MimeMessageId; import org.apache.james.mailbox.store.mail.model.Subject; -import reactor.core.publisher.Flux; - public class SearchThreadIdGuessingAlgorithmTest extends ThreadIdGuessingAlgorithmContract { private InMemoryMailboxManager mailboxManager; @@ -77,7 +75,6 @@ protected MessageId initOtherBasedMessageId() { } @Override - protected Flux saveThreadData(Username username, Set mimeMessageIds, MessageId messageId, ThreadId threadId, Optional baseSubject) { - return Flux.empty(); + protected void saveThreadData(Username username, Set mimeMessageIds, MessageId messageId, ThreadId threadId, Optional baseSubject) { } } diff --git a/mailbox/store/src/main/java/org/apache/james/mailbox/store/FlagsUpdateCalculator.java b/mailbox/store/src/main/java/org/apache/james/mailbox/store/FlagsUpdateCalculator.java index 310ee47370c..4c60a12cd22 100644 --- a/mailbox/store/src/main/java/org/apache/james/mailbox/store/FlagsUpdateCalculator.java +++ b/mailbox/store/src/main/java/org/apache/james/mailbox/store/FlagsUpdateCalculator.java @@ -48,4 +48,11 @@ public Flags buildNewFlags(Flags oldFlags) { return updatedFlags; } + public Flags providedFlags() { + return providedFlags; + } + + public MessageManager.FlagsUpdateMode getMode() { + return mode; + } } diff --git a/mailbox/store/src/main/java/org/apache/james/mailbox/store/MailboxSessionMapperFactory.java b/mailbox/store/src/main/java/org/apache/james/mailbox/store/MailboxSessionMapperFactory.java index af455608bfa..3267a7ce319 100644 --- a/mailbox/store/src/main/java/org/apache/james/mailbox/store/MailboxSessionMapperFactory.java +++ b/mailbox/store/src/main/java/org/apache/james/mailbox/store/MailboxSessionMapperFactory.java @@ -124,9 +124,9 @@ public SubscriptionMapper getSubscriptionMapper(MailboxSession session) { */ public abstract SubscriptionMapper createSubscriptionMapper(MailboxSession session); - public abstract UidProvider getUidProvider(); + public abstract UidProvider getUidProvider(MailboxSession session); - public abstract ModSeqProvider getModSeqProvider(); + public abstract ModSeqProvider getModSeqProvider(MailboxSession session); /** * Call endRequest on {@link Mapper} instances diff --git a/mailbox/store/src/main/java/org/apache/james/mailbox/store/StoreMessageIdManager.java b/mailbox/store/src/main/java/org/apache/james/mailbox/store/StoreMessageIdManager.java index bf69257a9bd..49e23525a29 100644 --- a/mailbox/store/src/main/java/org/apache/james/mailbox/store/StoreMessageIdManager.java +++ b/mailbox/store/src/main/java/org/apache/james/mailbox/store/StoreMessageIdManager.java @@ -503,7 +503,7 @@ private Flux addMessageToMailboxes(MailboxMes .build()) .build()); - return save(messageIdMapper, copy, mailbox) + return save(messageIdMapper, copy, mailbox, mailboxSession) .map(metadata -> addedEvent(mailboxSession, mailbox, metadata, messageMoves)); }).sneakyThrow()); } @@ -526,10 +526,11 @@ private boolean isSingleMove(MessageMovesWithMailbox messageMoves) { return messageMoves.addedMailboxes().size() == 1 && messageMoves.removedMailboxes().size() == 1; } - private Mono save(MessageIdMapper messageIdMapper, MailboxMessage mailboxMessage, Mailbox mailbox) { + private Mono save(MessageIdMapper messageIdMapper, MailboxMessage mailboxMessage, + Mailbox mailbox, MailboxSession mailboxSession) { return Mono.zip( - mailboxSessionMapperFactory.getModSeqProvider().nextModSeqReactive(mailbox.getMailboxId()), - mailboxSessionMapperFactory.getUidProvider().nextUidReactive(mailbox.getMailboxId())) + mailboxSessionMapperFactory.getModSeqProvider(mailboxSession).nextModSeqReactive(mailbox.getMailboxId()), + mailboxSessionMapperFactory.getUidProvider(mailboxSession).nextUidReactive(mailbox.getMailboxId())) .flatMap(modSeqAndUid -> { mailboxMessage.setModSeq(modSeqAndUid.getT1()); mailboxMessage.setUid(modSeqAndUid.getT2()); diff --git a/mailbox/store/src/main/java/org/apache/james/mailbox/store/StoreMessageManager.java b/mailbox/store/src/main/java/org/apache/james/mailbox/store/StoreMessageManager.java index 9c998cf80cc..4146c1d0423 100644 --- a/mailbox/store/src/main/java/org/apache/james/mailbox/store/StoreMessageManager.java +++ b/mailbox/store/src/main/java/org/apache/james/mailbox/store/StoreMessageManager.java @@ -1055,9 +1055,4 @@ private Flux listAllMessageUids(MailboxSession session) throws Mailb return messageMapper.execute( () -> messageMapper.listAllMessageUids(mailbox)); } - - @Override - public EnumSet getSupportedMessageCapabilities() { - return messageCapabilities; - } } diff --git a/mailbox/store/src/main/java/org/apache/james/mailbox/store/quota/ListeningCurrentQuotaUpdater.java b/mailbox/store/src/main/java/org/apache/james/mailbox/store/quota/ListeningCurrentQuotaUpdater.java index 27e6acccb59..c6c7e7aea32 100644 --- a/mailbox/store/src/main/java/org/apache/james/mailbox/store/quota/ListeningCurrentQuotaUpdater.java +++ b/mailbox/store/src/main/java/org/apache/james/mailbox/store/quota/ListeningCurrentQuotaUpdater.java @@ -31,10 +31,8 @@ import org.apache.james.events.EventListener; import org.apache.james.events.Group; import org.apache.james.events.RegistrationKey; -import org.apache.james.mailbox.events.MailboxEvents; import org.apache.james.mailbox.events.MailboxEvents.Added; import org.apache.james.mailbox.events.MailboxEvents.Expunged; -import org.apache.james.mailbox.events.MailboxEvents.MailboxAdded; import org.apache.james.mailbox.events.MailboxEvents.MailboxDeletion; import org.apache.james.mailbox.events.MailboxEvents.MetaDataHoldingEvent; import org.apache.james.mailbox.events.MessageMoveEvent; @@ -79,10 +77,7 @@ public Group getDefaultGroup() { @Override public boolean isHandling(Event event) { - return event instanceof Added - || event instanceof Expunged - || event instanceof MailboxDeletion - || event instanceof MailboxAdded; + return event instanceof Added || event instanceof Expunged || event instanceof MailboxDeletion; } @Override @@ -98,9 +93,6 @@ public Publisher reactiveEvent(Event event) { } else if (event instanceof MailboxDeletion) { MailboxDeletion mailboxDeletionEvent = (MailboxDeletion) event; return handleMailboxDeletionEvent(mailboxDeletionEvent); - } else if (event instanceof MailboxAdded) { - MailboxEvents.MailboxAdded mailboxAdded = (MailboxEvents.MailboxAdded) event; - return handleMailboxAddedEvent(mailboxAdded); } return Mono.empty(); } @@ -191,16 +183,4 @@ private Mono handleMailboxDeletionEvent(MailboxDeletion mailboxDeletionEve return Mono.empty(); } - private Mono handleMailboxAddedEvent(MailboxAdded mailboxAdded) { - return provisionCurrentQuota(mailboxAdded); - } - - private Mono provisionCurrentQuota(MailboxAdded mailboxAdded) { - return Mono.from(quotaRootResolver.getQuotaRootReactive(mailboxAdded.getMailboxPath())) - .flatMap(quotaRoot -> Mono.from(currentQuotaManager.getCurrentQuotas(quotaRoot)) - .map(any -> quotaRoot) - .switchIfEmpty(Mono.defer(() -> Mono.from(currentQuotaManager.setCurrentQuotas(new QuotaOperation(quotaRoot, QuotaCountUsage.count(0), QuotaSizeUsage.ZERO))) - .thenReturn(quotaRoot)))) - .then(); - } } \ No newline at end of file diff --git a/mailbox/store/src/test/java/org/apache/james/mailbox/store/ThreadIdGuessingAlgorithmContract.java b/mailbox/store/src/test/java/org/apache/james/mailbox/store/ThreadIdGuessingAlgorithmContract.java index e415f26941f..a93e822ead1 100644 --- a/mailbox/store/src/test/java/org/apache/james/mailbox/store/ThreadIdGuessingAlgorithmContract.java +++ b/mailbox/store/src/test/java/org/apache/james/mailbox/store/ThreadIdGuessingAlgorithmContract.java @@ -75,12 +75,12 @@ public abstract class ThreadIdGuessingAlgorithmContract { protected MessageId.Factory messageIdFactory; protected ThreadIdGuessingAlgorithm testee; protected MessageId newBasedMessageId; + protected MessageId otherBasedMessageId; protected MailboxSession mailboxSession; private MailboxManager mailboxManager; private MessageManager inbox; private MessageMapper messageMapper; private CombinationManagerTestSystem testingData; - private MessageId otherBasedMessageId; private Mailbox mailbox; protected abstract CombinationManagerTestSystem createTestingData(); @@ -93,7 +93,7 @@ public abstract class ThreadIdGuessingAlgorithmContract { protected abstract MessageId initOtherBasedMessageId(); - protected abstract Flux saveThreadData(Username username, Set mimeMessageIds, MessageId messageId, ThreadId threadId, Optional baseSubject); + protected abstract void saveThreadData(Username username, Set mimeMessageIds, MessageId messageId, ThreadId threadId, Optional baseSubject); @BeforeEach void setUp() throws Exception { @@ -153,7 +153,7 @@ void givenOldMailWhenAddNewRelatedMailsThenGuessingThreadIdShouldReturnSameThrea Set mimeMessageIds = buildMimeMessageIdSet(Optional.of(new MimeMessageId("Message-ID")), Optional.of(new MimeMessageId("someInReplyTo")), Optional.of(List.of(new MimeMessageId("references1"), new MimeMessageId("references2")))); - saveThreadData(mailboxSession.getUser(), mimeMessageIds, message.getId().getMessageId(), message.getThreadId(), Optional.of(new Subject("Test"))).collectList().block(); + saveThreadData(mailboxSession.getUser(), mimeMessageIds, message.getId().getMessageId(), message.getThreadId(), Optional.of(new Subject("Test"))); // add new related mails ThreadId threadId = testee.guessThreadIdReactive(newBasedMessageId, mimeMessageId, inReplyTo, references, subject, mailboxSession).block(); @@ -186,7 +186,7 @@ void givenOldMailWhenAddNewMailsWithRelatedSubjectButHaveNonIdenticalMessageIDTh Set mimeMessageIds = buildMimeMessageIdSet(Optional.of(new MimeMessageId("Message-ID")), Optional.of(new MimeMessageId("someInReplyTo")), Optional.of(List.of(new MimeMessageId("references1"), new MimeMessageId("references2")))); - saveThreadData(mailboxSession.getUser(), mimeMessageIds, message.getId().getMessageId(), message.getThreadId(), Optional.of(new Subject("Test"))).collectList().block(); + saveThreadData(mailboxSession.getUser(), mimeMessageIds, message.getId().getMessageId(), message.getThreadId(), Optional.of(new Subject("Test"))); // add mails related to old message by subject but have non same identical Message-ID ThreadId threadId = testee.guessThreadIdReactive(newBasedMessageId, mimeMessageId, inReplyTo, references, subject, mailboxSession).block(); @@ -219,7 +219,7 @@ void givenOldMailWhenAddNewMailsWithNonRelatedSubjectButHaveSameIdenticalMessage Set mimeMessageIds = buildMimeMessageIdSet(Optional.of(new MimeMessageId("Message-ID")), Optional.of(new MimeMessageId("someInReplyTo")), Optional.of(List.of(new MimeMessageId("references1"), new MimeMessageId("references2")))); - saveThreadData(mailboxSession.getUser(), mimeMessageIds, message.getId().getMessageId(), message.getThreadId(), Optional.of(new Subject("Test"))).collectList().block(); + saveThreadData(mailboxSession.getUser(), mimeMessageIds, message.getId().getMessageId(), message.getThreadId(), Optional.of(new Subject("Test"))); // add mails related to old message by having identical Message-ID but non related subject ThreadId threadId = testee.guessThreadIdReactive(newBasedMessageId, mimeMessageId, inReplyTo, references, subject, mailboxSession).block(); @@ -252,7 +252,7 @@ void givenOldMailWhenAddNonRelatedMailsThenGuessingThreadIdShouldBasedOnGenerate Set mimeMessageIds = buildMimeMessageIdSet(Optional.of(new MimeMessageId("Message-ID")), Optional.of(new MimeMessageId("someInReplyTo")), Optional.of(List.of(new MimeMessageId("references1"), new MimeMessageId("references2")))); - saveThreadData(mailboxSession.getUser(), mimeMessageIds, message.getId().getMessageId(), message.getThreadId(), Optional.of(new Subject("Test"))).collectList().block(); + saveThreadData(mailboxSession.getUser(), mimeMessageIds, message.getId().getMessageId(), message.getThreadId(), Optional.of(new Subject("Test"))); // add mails non related to old message by both subject and identical Message-ID ThreadId threadId = testee.guessThreadIdReactive(newBasedMessageId, mimeMessageId, inReplyTo, references, subject, mailboxSession).block(); @@ -279,8 +279,6 @@ void givenThreeMailsInAThreadThenGetThreadShouldReturnAListWithThreeMessageIdsSo @Test void givenNonMailInAThreadThenGetThreadShouldThrowThreadNotFoundException() { - Flux messageIds = testee.getMessageIdsInThread(ThreadId.fromBaseMessageId(newBasedMessageId), mailboxSession); - assertThatThrownBy(() -> testee.getMessageIdsInThread(ThreadId.fromBaseMessageId(newBasedMessageId), mailboxSession).collectList().block()) .getCause() .isInstanceOf(ThreadNotFoundException.class); diff --git a/mailbox/store/src/test/java/org/apache/james/mailbox/store/mail/model/AnnotationMapperTest.java b/mailbox/store/src/test/java/org/apache/james/mailbox/store/mail/model/AnnotationMapperTest.java index c00d6b26396..974edd0fc91 100644 --- a/mailbox/store/src/test/java/org/apache/james/mailbox/store/mail/model/AnnotationMapperTest.java +++ b/mailbox/store/src/test/java/org/apache/james/mailbox/store/mail/model/AnnotationMapperTest.java @@ -204,6 +204,13 @@ void isExistedShouldReturnFalseIfAnnotationIsNotStored() { assertThat(annotationMapper.exist(mailboxId, PRIVATE_ANNOTATION)).isFalse(); } + @Test + void isExistedShouldReturnFalseIfMailboxIdExistAndAnnotationIsNotStored() { + annotationMapper.insertAnnotation(mailboxId, PRIVATE_ANNOTATION); + + assertThat(annotationMapper.exist(mailboxId, PRIVATE_USER_ANNOTATION)).isFalse(); + } + @Test void countAnnotationShouldReturnZeroIfNoMoreAnnotationBelongToMailbox() { assertThat(annotationMapper.countAnnotations(mailboxId)).isEqualTo(0); diff --git a/mailbox/store/src/test/java/org/apache/james/mailbox/store/mail/model/MailboxMapperACLTest.java b/mailbox/store/src/test/java/org/apache/james/mailbox/store/mail/model/MailboxMapperACLTest.java index 88e87684a1c..6e2f94a90f5 100644 --- a/mailbox/store/src/test/java/org/apache/james/mailbox/store/mail/model/MailboxMapperACLTest.java +++ b/mailbox/store/src/test/java/org/apache/james/mailbox/store/mail/model/MailboxMapperACLTest.java @@ -43,7 +43,7 @@ public abstract class MailboxMapperACLTest { private static final Username USER_1 = Username.of("user1"); private static final Username USER_2 = Username.of("user2"); - private Mailbox benwaInboxMailbox; + protected Mailbox benwaInboxMailbox; private MailboxMapper mailboxMapper; diff --git a/mailbox/store/src/test/java/org/apache/james/mailbox/store/mail/model/MailboxMapperTest.java b/mailbox/store/src/test/java/org/apache/james/mailbox/store/mail/model/MailboxMapperTest.java index a5a13d10367..efdb019d2a1 100644 --- a/mailbox/store/src/test/java/org/apache/james/mailbox/store/mail/model/MailboxMapperTest.java +++ b/mailbox/store/src/test/java/org/apache/james/mailbox/store/mail/model/MailboxMapperTest.java @@ -152,6 +152,21 @@ void renameShouldRemoveOldMailboxPath() { .isEmpty(); } + @Test + void renameShouldUpdateOnlyOneMailbox() { + MailboxId aliceMailboxId = mailboxMapper.create(benwaInboxPath, UidValidity.of(1L)).block().getMailboxId(); + MailboxId bobMailboxId = mailboxMapper.create(bobInboxPath, UidValidity.of(2L)).block().getMailboxId(); + + MailboxPath newMailboxPath = new MailboxPath(benwaInboxPath.getNamespace(), benwaInboxPath.getUser(), "ENBOX"); + mailboxMapper.rename(new Mailbox(newMailboxPath, UidValidity.of(1L), aliceMailboxId)).block(); + + Mailbox actualAliceMailbox = mailboxMapper.findMailboxById(aliceMailboxId).block(); + Mailbox actualBobMailbox = mailboxMapper.findMailboxById(bobMailboxId).block(); + + assertThat(actualAliceMailbox.getName()).isEqualTo("ENBOX"); + assertThat(actualBobMailbox.getName()).isEqualTo(bobInboxPath.getName()); + } + @Test void listShouldRetrieveAllMailbox() { createAll(); diff --git a/mailbox/store/src/test/java/org/apache/james/mailbox/store/mail/model/MapperProvider.java b/mailbox/store/src/test/java/org/apache/james/mailbox/store/mail/model/MapperProvider.java index b6bd054d2ef..36f5d72b0cf 100644 --- a/mailbox/store/src/test/java/org/apache/james/mailbox/store/mail/model/MapperProvider.java +++ b/mailbox/store/src/test/java/org/apache/james/mailbox/store/mail/model/MapperProvider.java @@ -59,7 +59,7 @@ enum Capabilities { MailboxId generateId(); - MessageUid generateMessageUid(); + MessageUid generateMessageUid(Mailbox mailbox); ModSeq generateModSeq(Mailbox mailbox) throws MailboxException; diff --git a/mailbox/store/src/test/java/org/apache/james/mailbox/store/mail/model/MessageIdMapperTest.java b/mailbox/store/src/test/java/org/apache/james/mailbox/store/mail/model/MessageIdMapperTest.java index c630872282f..0f26680875c 100644 --- a/mailbox/store/src/test/java/org/apache/james/mailbox/store/mail/model/MessageIdMapperTest.java +++ b/mailbox/store/src/test/java/org/apache/james/mailbox/store/mail/model/MessageIdMapperTest.java @@ -150,7 +150,7 @@ void findMailboxesShouldReturnTwoMailboxesWhenMessageExistsInTwoMailboxes() thro saveMessages(); SimpleMailboxMessage message1InOtherMailbox = SimpleMailboxMessage.copy(benwaWorkMailbox.getMailboxId(), message1); - message1InOtherMailbox.setUid(mapperProvider.generateMessageUid()); + message1InOtherMailbox.setUid(mapperProvider.generateMessageUid(benwaWorkMailbox)); message1InOtherMailbox.setModSeq(mapperProvider.generateModSeq(benwaWorkMailbox)); sut.save(message1InOtherMailbox); @@ -160,7 +160,7 @@ void findMailboxesShouldReturnTwoMailboxesWhenMessageExistsInTwoMailboxes() thro @Test void saveShouldSaveAMessage() throws Exception { - message1.setUid(mapperProvider.generateMessageUid()); + message1.setUid(mapperProvider.generateMessageUid(benwaWorkMailbox)); message1.setModSeq(mapperProvider.generateModSeq(benwaInboxMailbox)); sut.save(message1); List messages = sut.find(ImmutableList.of(message1.getMessageId()), FetchType.FULL); @@ -171,7 +171,7 @@ void saveShouldSaveAMessage() throws Exception { void saveShouldThrowWhenMailboxDoesntExist() throws Exception { Mailbox notPersistedMailbox = new Mailbox(MailboxPath.forUser(BENWA, "mybox"), UID_VALIDITY, mapperProvider.generateId()); SimpleMailboxMessage message = createMessage(notPersistedMailbox, "Subject: Test \n\nBody\n.\n", BODY_START, new PropertyBuilder()); - message.setUid(mapperProvider.generateMessageUid()); + message.setUid(mapperProvider.generateMessageUid(notPersistedMailbox)); message.setModSeq(mapperProvider.generateModSeq(notPersistedMailbox)); assertThatThrownBy(() -> sut.save(message)) @@ -180,12 +180,12 @@ void saveShouldThrowWhenMailboxDoesntExist() throws Exception { @Test void saveShouldSaveMessageInAnotherMailboxWhenMessageAlreadyInOneMailbox() throws Exception { - message1.setUid(mapperProvider.generateMessageUid()); + message1.setUid(mapperProvider.generateMessageUid(benwaInboxMailbox)); message1.setModSeq(mapperProvider.generateModSeq(benwaInboxMailbox)); sut.save(message1); SimpleMailboxMessage message1InOtherMailbox = SimpleMailboxMessage.copy(benwaWorkMailbox.getMailboxId(), message1); - message1InOtherMailbox.setUid(mapperProvider.generateMessageUid()); + message1InOtherMailbox.setUid(mapperProvider.generateMessageUid(benwaWorkMailbox)); message1InOtherMailbox.setModSeq(mapperProvider.generateModSeq(benwaWorkMailbox)); sut.save(message1InOtherMailbox); @@ -195,11 +195,11 @@ void saveShouldSaveMessageInAnotherMailboxWhenMessageAlreadyInOneMailbox() throw @Test void saveShouldWorkWhenSavingTwoTimesWithSameMessageIdAndSameMailboxId() throws Exception { - message1.setUid(mapperProvider.generateMessageUid()); + message1.setUid(mapperProvider.generateMessageUid(benwaInboxMailbox)); message1.setModSeq(mapperProvider.generateModSeq(benwaInboxMailbox)); sut.save(message1); SimpleMailboxMessage copiedMessage = SimpleMailboxMessage.copy(message1.getMailboxId(), message1); - copiedMessage.setUid(mapperProvider.generateMessageUid()); + copiedMessage.setUid(mapperProvider.generateMessageUid(benwaInboxMailbox)); copiedMessage.setModSeq(mapperProvider.generateModSeq(benwaInboxMailbox)); sut.save(copiedMessage); @@ -209,13 +209,13 @@ void saveShouldWorkWhenSavingTwoTimesWithSameMessageIdAndSameMailboxId() throws @Test void copyInMailboxShouldSaveMessageInAnotherMailbox() throws Exception { - message1.setUid(mapperProvider.generateMessageUid()); + message1.setUid(mapperProvider.generateMessageUid(benwaInboxMailbox)); message1.setModSeq(mapperProvider.generateModSeq(benwaInboxMailbox)); sut.save(message1); MailboxMessage message1InOtherMailbox = sut.find(ImmutableList.of(message1.getMessageId()), FetchType.METADATA).get(0) .copy(benwaWorkMailbox); - message1InOtherMailbox.setUid(mapperProvider.generateMessageUid()); + message1InOtherMailbox.setUid(mapperProvider.generateMessageUid(benwaWorkMailbox)); message1InOtherMailbox.setModSeq(mapperProvider.generateModSeq(benwaWorkMailbox)); sut.copyInMailbox(message1InOtherMailbox, benwaWorkMailbox); @@ -225,12 +225,12 @@ void copyInMailboxShouldSaveMessageInAnotherMailbox() throws Exception { @Test void copyInMailboxShouldWorkWhenSavingTwoTimesWithSameMessageIdAndSameMailboxId() throws Exception { - message1.setUid(mapperProvider.generateMessageUid()); + message1.setUid(mapperProvider.generateMessageUid(benwaInboxMailbox)); message1.setModSeq(mapperProvider.generateModSeq(benwaInboxMailbox)); sut.save(message1); MailboxMessage copiedMessage = sut.find(ImmutableList.of(message1.getMessageId()), FetchType.METADATA).get(0) .copy(benwaWorkMailbox); - copiedMessage.setUid(mapperProvider.generateMessageUid()); + copiedMessage.setUid(mapperProvider.generateMessageUid(benwaWorkMailbox)); copiedMessage.setModSeq(mapperProvider.generateModSeq(benwaWorkMailbox)); sut.copyInMailbox(copiedMessage, benwaWorkMailbox); @@ -250,7 +250,7 @@ void deleteShouldNotThrowWhenUnknownMessage() { @Test void deleteShouldDeleteAMessage() throws Exception { - message1.setUid(mapperProvider.generateMessageUid()); + message1.setUid(mapperProvider.generateMessageUid(benwaInboxMailbox)); message1.setModSeq(mapperProvider.generateModSeq(benwaInboxMailbox)); sut.save(message1); @@ -263,12 +263,12 @@ void deleteShouldDeleteAMessage() throws Exception { @Test void deleteShouldDeleteMessageIndicesWhenStoredInTwoMailboxes() throws Exception { - message1.setUid(mapperProvider.generateMessageUid()); + message1.setUid(mapperProvider.generateMessageUid(benwaInboxMailbox)); message1.setModSeq(mapperProvider.generateModSeq(benwaInboxMailbox)); sut.save(message1); SimpleMailboxMessage message1InOtherMailbox = SimpleMailboxMessage.copy(benwaWorkMailbox.getMailboxId(), message1); - message1InOtherMailbox.setUid(mapperProvider.generateMessageUid()); + message1InOtherMailbox.setUid(mapperProvider.generateMessageUid(benwaWorkMailbox)); message1InOtherMailbox.setModSeq(mapperProvider.generateModSeq(benwaWorkMailbox)); sut.save(message1InOtherMailbox); @@ -281,11 +281,11 @@ void deleteShouldDeleteMessageIndicesWhenStoredInTwoMailboxes() throws Exception @Test void deleteShouldDeleteMessageIndicesWhenStoredTwoTimesInTheSameMailbox() throws Exception { - message1.setUid(mapperProvider.generateMessageUid()); + message1.setUid(mapperProvider.generateMessageUid(benwaInboxMailbox)); message1.setModSeq(mapperProvider.generateModSeq(benwaInboxMailbox)); sut.save(message1); SimpleMailboxMessage copiedMessage = SimpleMailboxMessage.copy(message1.getMailboxId(), message1); - copiedMessage.setUid(mapperProvider.generateMessageUid()); + copiedMessage.setUid(mapperProvider.generateMessageUid(benwaInboxMailbox)); copiedMessage.setModSeq(mapperProvider.generateModSeq(benwaInboxMailbox)); sut.save(copiedMessage); @@ -298,12 +298,12 @@ void deleteShouldDeleteMessageIndicesWhenStoredTwoTimesInTheSameMailbox() throws @Test void deleteWithMailboxIdsShouldNotDeleteIndicesWhenMailboxIdsIsEmpty() throws Exception { - message1.setUid(mapperProvider.generateMessageUid()); + message1.setUid(mapperProvider.generateMessageUid(benwaInboxMailbox)); message1.setModSeq(mapperProvider.generateModSeq(benwaInboxMailbox)); sut.save(message1); SimpleMailboxMessage message1InOtherMailbox = SimpleMailboxMessage.copy(benwaWorkMailbox.getMailboxId(), message1); - message1InOtherMailbox.setUid(mapperProvider.generateMessageUid()); + message1InOtherMailbox.setUid(mapperProvider.generateMessageUid(benwaWorkMailbox)); message1InOtherMailbox.setModSeq(mapperProvider.generateModSeq(benwaWorkMailbox)); sut.save(message1InOtherMailbox); @@ -316,12 +316,12 @@ void deleteWithMailboxIdsShouldNotDeleteIndicesWhenMailboxIdsIsEmpty() throws Ex @Test void deleteWithMailboxIdsShouldDeleteOneIndexWhenMailboxIdsContainsOneElement() throws Exception { - message1.setUid(mapperProvider.generateMessageUid()); + message1.setUid(mapperProvider.generateMessageUid(benwaInboxMailbox)); message1.setModSeq(mapperProvider.generateModSeq(benwaInboxMailbox)); sut.save(message1); SimpleMailboxMessage message1InOtherMailbox = SimpleMailboxMessage.copy(benwaWorkMailbox.getMailboxId(), message1); - message1InOtherMailbox.setUid(mapperProvider.generateMessageUid()); + message1InOtherMailbox.setUid(mapperProvider.generateMessageUid(benwaWorkMailbox)); message1InOtherMailbox.setModSeq(mapperProvider.generateModSeq(benwaWorkMailbox)); sut.save(message1InOtherMailbox); @@ -334,12 +334,12 @@ void deleteWithMailboxIdsShouldDeleteOneIndexWhenMailboxIdsContainsOneElement() @Test void deleteWithMailboxIdsShouldDeleteIndicesWhenMailboxIdsContainsMultipleElements() throws Exception { - message1.setUid(mapperProvider.generateMessageUid()); + message1.setUid(mapperProvider.generateMessageUid(benwaInboxMailbox)); message1.setModSeq(mapperProvider.generateModSeq(benwaInboxMailbox)); sut.save(message1); SimpleMailboxMessage message1InOtherMailbox = SimpleMailboxMessage.copy(benwaWorkMailbox.getMailboxId(), message1); - message1InOtherMailbox.setUid(mapperProvider.generateMessageUid()); + message1InOtherMailbox.setUid(mapperProvider.generateMessageUid(benwaWorkMailbox)); message1InOtherMailbox.setModSeq(mapperProvider.generateModSeq(benwaWorkMailbox)); sut.save(message1InOtherMailbox); @@ -352,7 +352,7 @@ void deleteWithMailboxIdsShouldDeleteIndicesWhenMailboxIdsContainsMultipleElemen @Test void setFlagsShouldReturnUpdatedFlagsWhenMessageIsInOneMailbox() throws Exception { - message1.setUid(mapperProvider.generateMessageUid()); + message1.setUid(mapperProvider.generateMessageUid(benwaInboxMailbox)); message1.setModSeq(mapperProvider.generateModSeq(benwaInboxMailbox)); sut.save(message1); @@ -376,7 +376,7 @@ void setFlagsShouldReturnUpdatedFlagsWhenReplaceMode() throws Exception { Flags messageFlags = new FlagsBuilder().add(Flags.Flag.RECENT, Flags.Flag.FLAGGED) .build(); - message1.setUid(mapperProvider.generateMessageUid()); + message1.setUid(mapperProvider.generateMessageUid(benwaInboxMailbox)); message1.setFlags(messageFlags); message1.setModSeq(mapperProvider.generateModSeq(benwaInboxMailbox)); sut.save(message1); @@ -406,7 +406,7 @@ void setFlagsShouldReturnUpdatedFlagsWhenRemoveMode() throws Exception { Flags messageFlags = new FlagsBuilder().add(Flags.Flag.RECENT, Flags.Flag.FLAGGED) .build(); - message1.setUid(mapperProvider.generateMessageUid()); + message1.setUid(mapperProvider.generateMessageUid(benwaInboxMailbox)); message1.setFlags(messageFlags); message1.setModSeq(mapperProvider.generateModSeq(benwaInboxMailbox)); sut.save(message1); @@ -436,7 +436,7 @@ void setFlagsShouldUpdateMessageFlagsWhenRemoveMode() throws Exception { Flags messageFlags = new FlagsBuilder().add(Flags.Flag.RECENT, Flags.Flag.FLAGGED) .build(); - message1.setUid(mapperProvider.generateMessageUid()); + message1.setUid(mapperProvider.generateMessageUid(benwaInboxMailbox)); message1.setFlags(messageFlags); message1.setModSeq(mapperProvider.generateModSeq(benwaInboxMailbox)); sut.save(message1); @@ -456,7 +456,7 @@ void setFlagsShouldUpdateMessageFlagsWhenRemoveMode() throws Exception { @Test void setFlagsShouldReturnEmptyWhenMailboxIdsIsEmpty() throws Exception { - message1.setUid(mapperProvider.generateMessageUid()); + message1.setUid(mapperProvider.generateMessageUid(benwaInboxMailbox)); message1.setModSeq(mapperProvider.generateModSeq(benwaInboxMailbox)); sut.save(message1); @@ -478,7 +478,7 @@ void setFlagsShouldReturnEmptyWhenMessageIdDoesntExist() throws Exception { @Test void setFlagsShouldAddFlagsWhenAddUpdateMode() throws Exception { Flags initialFlags = new Flags(Flag.RECENT); - message1.setUid(mapperProvider.generateMessageUid()); + message1.setUid(mapperProvider.generateMessageUid(benwaInboxMailbox)); message1.setModSeq(mapperProvider.generateModSeq(benwaInboxMailbox)); message1.setFlags(initialFlags); sut.save(message1); @@ -505,12 +505,12 @@ void setFlagsShouldAddFlagsWhenAddUpdateMode() throws Exception { @Test void setFlagsShouldReturnUpdatedFlagsWhenMessageIsInTwoMailboxes() throws Exception { - message1.setUid(mapperProvider.generateMessageUid()); + message1.setUid(mapperProvider.generateMessageUid(benwaInboxMailbox)); message1.setModSeq(mapperProvider.generateModSeq(benwaInboxMailbox)); sut.save(message1); SimpleMailboxMessage message1InOtherMailbox = SimpleMailboxMessage.copy(benwaWorkMailbox.getMailboxId(), message1); - message1InOtherMailbox.setUid(mapperProvider.generateMessageUid()); + message1InOtherMailbox.setUid(mapperProvider.generateMessageUid(benwaWorkMailbox)); message1InOtherMailbox.setModSeq(mapperProvider.generateModSeq(benwaWorkMailbox)); sut.save(message1InOtherMailbox); @@ -541,7 +541,7 @@ void setFlagsShouldReturnUpdatedFlagsWhenMessageIsInTwoMailboxes() throws Except @Test void setFlagsShouldUpdateFlagsWhenMessageIsInOneMailbox() throws Exception { - message1.setUid(mapperProvider.generateMessageUid()); + message1.setUid(mapperProvider.generateMessageUid(benwaInboxMailbox)); message1.setModSeq(mapperProvider.generateModSeq(benwaInboxMailbox)); sut.save(message1); @@ -555,7 +555,7 @@ void setFlagsShouldUpdateFlagsWhenMessageIsInOneMailbox() throws Exception { @Test void setFlagsShouldNotModifyModSeqWhenMailboxIdsIsEmpty() throws Exception { - message1.setUid(mapperProvider.generateMessageUid()); + message1.setUid(mapperProvider.generateMessageUid(benwaInboxMailbox)); ModSeq modSeq = mapperProvider.generateModSeq(benwaInboxMailbox); message1.setModSeq(modSeq); sut.save(message1); @@ -571,7 +571,7 @@ void setFlagsShouldNotModifyModSeqWhenMailboxIdsIsEmpty() throws Exception { @Test void setFlagsShouldUpdateModSeqWhenMessageIsInOneMailbox() throws Exception { - message1.setUid(mapperProvider.generateMessageUid()); + message1.setUid(mapperProvider.generateMessageUid(benwaInboxMailbox)); ModSeq modSeq = mapperProvider.generateModSeq(benwaInboxMailbox); message1.setModSeq(modSeq); sut.save(message1); @@ -586,7 +586,7 @@ void setFlagsShouldUpdateModSeqWhenMessageIsInOneMailbox() throws Exception { @Test void setFlagsShouldNotModifyFlagsWhenMailboxIdsIsEmpty() throws Exception { - message1.setUid(mapperProvider.generateMessageUid()); + message1.setUid(mapperProvider.generateMessageUid(benwaInboxMailbox)); ModSeq modSeq = mapperProvider.generateModSeq(benwaInboxMailbox); message1.setModSeq(modSeq); Flags initialFlags = new Flags(Flags.Flag.DRAFT); @@ -604,12 +604,12 @@ void setFlagsShouldNotModifyFlagsWhenMailboxIdsIsEmpty() throws Exception { @Test void setFlagsShouldUpdateFlagsWhenMessageIsInTwoMailboxes() throws Exception { - message1.setUid(mapperProvider.generateMessageUid()); + message1.setUid(mapperProvider.generateMessageUid(benwaInboxMailbox)); message1.setModSeq(mapperProvider.generateModSeq(benwaInboxMailbox)); sut.save(message1); SimpleMailboxMessage message1InOtherMailbox = SimpleMailboxMessage.copy(benwaWorkMailbox.getMailboxId(), message1); - message1InOtherMailbox.setUid(mapperProvider.generateMessageUid()); + message1InOtherMailbox.setUid(mapperProvider.generateMessageUid(benwaWorkMailbox)); message1InOtherMailbox.setModSeq(mapperProvider.generateModSeq(benwaWorkMailbox)); sut.save(message1InOtherMailbox); @@ -624,16 +624,16 @@ void setFlagsShouldUpdateFlagsWhenMessageIsInTwoMailboxes() throws Exception { @Test void setFlagsShouldWorkWhenCalledOnFirstMessage() throws Exception { - message1.setUid(mapperProvider.generateMessageUid()); + message1.setUid(mapperProvider.generateMessageUid(benwaInboxMailbox)); message1.setModSeq(mapperProvider.generateModSeq(benwaInboxMailbox)); sut.save(message1); - message2.setUid(mapperProvider.generateMessageUid()); + message2.setUid(mapperProvider.generateMessageUid(benwaInboxMailbox)); message2.setModSeq(mapperProvider.generateModSeq(benwaInboxMailbox)); sut.save(message2); - message3.setUid(mapperProvider.generateMessageUid()); + message3.setUid(mapperProvider.generateMessageUid(benwaInboxMailbox)); message3.setModSeq(mapperProvider.generateModSeq(benwaInboxMailbox)); sut.save(message3); - message4.setUid(mapperProvider.generateMessageUid()); + message4.setUid(mapperProvider.generateMessageUid(benwaInboxMailbox)); message4.setModSeq(mapperProvider.generateModSeq(benwaInboxMailbox)); sut.save(message4); @@ -647,16 +647,16 @@ void setFlagsShouldWorkWhenCalledOnFirstMessage() throws Exception { @Test void setFlagsShouldWorkWhenCalledOnDuplicatedMailbox() throws Exception { - message1.setUid(mapperProvider.generateMessageUid()); + message1.setUid(mapperProvider.generateMessageUid(benwaInboxMailbox)); message1.setModSeq(mapperProvider.generateModSeq(benwaInboxMailbox)); sut.save(message1); - message2.setUid(mapperProvider.generateMessageUid()); + message2.setUid(mapperProvider.generateMessageUid(benwaInboxMailbox)); message2.setModSeq(mapperProvider.generateModSeq(benwaInboxMailbox)); sut.save(message2); - message3.setUid(mapperProvider.generateMessageUid()); + message3.setUid(mapperProvider.generateMessageUid(benwaInboxMailbox)); message3.setModSeq(mapperProvider.generateModSeq(benwaInboxMailbox)); sut.save(message3); - message4.setUid(mapperProvider.generateMessageUid()); + message4.setUid(mapperProvider.generateMessageUid(benwaInboxMailbox)); message4.setModSeq(mapperProvider.generateModSeq(benwaInboxMailbox)); sut.save(message4); @@ -671,7 +671,7 @@ void setFlagsShouldWorkWhenCalledOnDuplicatedMailbox() throws Exception { @Test public void setFlagsShouldWorkWithConcurrencyWithAdd() throws Exception { Assume.assumeTrue(mapperProvider.getSupportedCapabilities().contains(MapperProvider.Capabilities.THREAD_SAFE_FLAGS_UPDATE)); - message1.setUid(mapperProvider.generateMessageUid()); + message1.setUid(mapperProvider.generateMessageUid(benwaInboxMailbox)); message1.setModSeq(mapperProvider.generateModSeq(benwaInboxMailbox)); sut.save(message1); @@ -694,7 +694,7 @@ public void setFlagsShouldWorkWithConcurrencyWithAdd() throws Exception { @Test public void setFlagsShouldWorkWithConcurrencyWithRemove() throws Exception { Assume.assumeTrue(mapperProvider.getSupportedCapabilities().contains(MapperProvider.Capabilities.THREAD_SAFE_FLAGS_UPDATE)); - message1.setUid(mapperProvider.generateMessageUid()); + message1.setUid(mapperProvider.generateMessageUid(benwaInboxMailbox)); message1.setModSeq(mapperProvider.generateModSeq(benwaInboxMailbox)); sut.save(message1); @@ -727,7 +727,7 @@ public void setFlagsShouldWorkWithConcurrencyWithRemove() throws Exception { @Test void countMessageShouldReturnWhenCreateNewMessage() throws Exception { - message1.setUid(mapperProvider.generateMessageUid()); + message1.setUid(mapperProvider.generateMessageUid(benwaInboxMailbox)); message1.setModSeq(mapperProvider.generateModSeq(benwaInboxMailbox)); sut.save(message1); @@ -736,7 +736,7 @@ void countMessageShouldReturnWhenCreateNewMessage() throws Exception { @Test void countUnseenMessageShouldBeEmptyWhenMessageIsSeen() throws Exception { - message1.setUid(mapperProvider.generateMessageUid()); + message1.setUid(mapperProvider.generateMessageUid(benwaInboxMailbox)); message1.setModSeq(mapperProvider.generateModSeq(benwaInboxMailbox)); message1.setFlags(new Flags(Flag.SEEN)); sut.save(message1); @@ -746,7 +746,7 @@ void countUnseenMessageShouldBeEmptyWhenMessageIsSeen() throws Exception { @Test void countUnseenMessageShouldReturnWhenMessageIsNotSeen() throws Exception { - message1.setUid(mapperProvider.generateMessageUid()); + message1.setUid(mapperProvider.generateMessageUid(benwaInboxMailbox)); message1.setModSeq(mapperProvider.generateModSeq(benwaInboxMailbox)); sut.save(message1); @@ -755,7 +755,7 @@ void countUnseenMessageShouldReturnWhenMessageIsNotSeen() throws Exception { @Test void countMessageShouldBeEmptyWhenDeleteMessage() throws Exception { - message1.setUid(mapperProvider.generateMessageUid()); + message1.setUid(mapperProvider.generateMessageUid(benwaInboxMailbox)); message1.setModSeq(mapperProvider.generateModSeq(benwaInboxMailbox)); sut.save(message1); @@ -766,7 +766,7 @@ void countMessageShouldBeEmptyWhenDeleteMessage() throws Exception { @Test void countUnseenMessageShouldBeEmptyWhenDeleteMessage() throws Exception { - message1.setUid(mapperProvider.generateMessageUid()); + message1.setUid(mapperProvider.generateMessageUid(benwaInboxMailbox)); message1.setModSeq(mapperProvider.generateModSeq(benwaInboxMailbox)); sut.save(message1); @@ -777,12 +777,12 @@ void countUnseenMessageShouldBeEmptyWhenDeleteMessage() throws Exception { @Test void countUnseenMessageShouldReturnWhenDeleteMessage() throws Exception { - message1.setUid(mapperProvider.generateMessageUid()); + message1.setUid(mapperProvider.generateMessageUid(benwaInboxMailbox)); message1.setModSeq(mapperProvider.generateModSeq(benwaInboxMailbox)); message1.setFlags(new Flags(Flag.SEEN)); sut.save(message1); - message2.setUid(mapperProvider.generateMessageUid()); + message2.setUid(mapperProvider.generateMessageUid(benwaInboxMailbox)); message2.setModSeq(mapperProvider.generateModSeq(benwaInboxMailbox)); sut.save(message2); @@ -793,7 +793,7 @@ void countUnseenMessageShouldReturnWhenDeleteMessage() throws Exception { @Test void countUnseenMessageShouldTakeCareOfMessagesMarkedAsRead() throws Exception { - message1.setUid(mapperProvider.generateMessageUid()); + message1.setUid(mapperProvider.generateMessageUid(benwaInboxMailbox)); message1.setModSeq(mapperProvider.generateModSeq(benwaInboxMailbox)); sut.save(message1); @@ -804,7 +804,7 @@ void countUnseenMessageShouldTakeCareOfMessagesMarkedAsRead() throws Exception { @Test void countUnseenMessageShouldTakeCareOfMessagesMarkedAsUnread() throws Exception { - message1.setUid(mapperProvider.generateMessageUid()); + message1.setUid(mapperProvider.generateMessageUid(benwaInboxMailbox)); message1.setModSeq(mapperProvider.generateModSeq(benwaInboxMailbox)); message1.setFlags(new Flags(Flag.SEEN)); sut.save(message1); @@ -816,7 +816,7 @@ void countUnseenMessageShouldTakeCareOfMessagesMarkedAsUnread() throws Exception @Test void setFlagsShouldNotUpdateModSeqWhenNoop() throws Exception { - message1.setUid(mapperProvider.generateMessageUid()); + message1.setUid(mapperProvider.generateMessageUid(benwaInboxMailbox)); ModSeq modSeq = mapperProvider.generateModSeq(benwaInboxMailbox); message1.setModSeq(modSeq); message1.setFlags(new Flags(Flag.SEEN)); @@ -835,7 +835,7 @@ void setFlagsShouldNotUpdateModSeqWhenNoop() throws Exception { @Test void addingFlagToAMessageThatAlreadyHasThisFlagShouldResultInNoChange() throws Exception { - message1.setUid(mapperProvider.generateMessageUid()); + message1.setUid(mapperProvider.generateMessageUid(benwaInboxMailbox)); ModSeq modSeq = mapperProvider.generateModSeq(benwaInboxMailbox); message1.setModSeq(modSeq); Flags flags = new Flags(Flag.SEEN); @@ -855,7 +855,7 @@ void addingFlagToAMessageThatAlreadyHasThisFlagShouldResultInNoChange() throws E @Test void setFlagsShouldReturnUpdatedFlagsWhenNoop() throws Exception { - message1.setUid(mapperProvider.generateMessageUid()); + message1.setUid(mapperProvider.generateMessageUid(benwaInboxMailbox)); ModSeq modSeq = mapperProvider.generateModSeq(benwaInboxMailbox); message1.setModSeq(modSeq); Flags flags = new Flags(Flag.SEEN); @@ -881,7 +881,7 @@ void setFlagsShouldReturnUpdatedFlagsWhenNoop() throws Exception { @Test void countUnseenMessageShouldNotTakeCareOfOtherFlagsUpdates() throws Exception { - message1.setUid(mapperProvider.generateMessageUid()); + message1.setUid(mapperProvider.generateMessageUid(benwaInboxMailbox)); message1.setModSeq(mapperProvider.generateModSeq(benwaInboxMailbox)); message1.setFlags(new Flags(Flag.RECENT)); sut.save(message1); @@ -896,7 +896,7 @@ void deletesShouldOnlyRemoveConcernedMessages() throws Exception { saveMessages(); MailboxMessage copiedMessage = sut.find(ImmutableList.of(message1.getMessageId()), FetchType.METADATA).get(0); - copiedMessage.setUid(mapperProvider.generateMessageUid()); + copiedMessage.setUid(mapperProvider.generateMessageUid(benwaWorkMailbox)); copiedMessage.setModSeq(mapperProvider.generateModSeq(benwaWorkMailbox)); sut.copyInMailbox(copiedMessage, benwaWorkMailbox); @@ -921,7 +921,7 @@ void deletesShouldUpdateMessageCount() throws Exception { saveMessages(); MailboxMessage copiedMessage = sut.find(ImmutableList.of(message1.getMessageId()), FetchType.METADATA).get(0); - copiedMessage.setUid(mapperProvider.generateMessageUid()); + copiedMessage.setUid(mapperProvider.generateMessageUid(benwaWorkMailbox)); copiedMessage.setModSeq(mapperProvider.generateModSeq(benwaWorkMailbox)); sut.copyInMailbox(copiedMessage, benwaWorkMailbox); @@ -962,12 +962,12 @@ void setFlagsShouldReturnAllUp() throws Exception { @Test void deletesShouldUpdateUnreadCount() throws Exception { - message1.setUid(mapperProvider.generateMessageUid()); + message1.setUid(mapperProvider.generateMessageUid(benwaInboxMailbox)); message1.setModSeq(mapperProvider.generateModSeq(benwaInboxMailbox)); message1.setFlags(new Flags(Flag.SEEN)); sut.save(message1); - message2.setUid(mapperProvider.generateMessageUid()); + message2.setUid(mapperProvider.generateMessageUid(benwaInboxMailbox)); message2.setModSeq(mapperProvider.generateModSeq(benwaInboxMailbox)); sut.save(message2); @@ -993,11 +993,11 @@ void deletesShouldNotFailUponMissingMessage() { class SaveDateTests { @Test void saveMessagesShouldSetNewSaveDate() throws MailboxException { - message1.setUid(mapperProvider.generateMessageUid()); + message1.setUid(mapperProvider.generateMessageUid(benwaInboxMailbox)); message1.setModSeq(mapperProvider.generateModSeq(benwaInboxMailbox)); message1.setFlags(new Flags(Flag.SEEN)); - message2.setUid(mapperProvider.generateMessageUid()); + message2.setUid(mapperProvider.generateMessageUid(benwaInboxMailbox)); message2.setModSeq(mapperProvider.generateModSeq(benwaInboxMailbox)); sut.save(message1); @@ -1012,14 +1012,14 @@ void saveMessagesShouldSetNewSaveDate() throws MailboxException { @Test void copyInMailboxReactiveShouldSetNewSaveDate() throws MailboxException, InterruptedException { - message1.setUid(mapperProvider.generateMessageUid()); + message1.setUid(mapperProvider.generateMessageUid(benwaInboxMailbox)); message1.setModSeq(mapperProvider.generateModSeq(benwaInboxMailbox)); message1.setFlags(new Flags(Flag.SEEN)); sut.save(message1); MailboxMessage copy = sut.find(ImmutableList.of(message1.getMessageId()), FetchType.METADATA).get(0) .copy(benwaWorkMailbox); - copy.setUid(mapperProvider.generateMessageUid()); + copy.setUid(mapperProvider.generateMessageUid(benwaWorkMailbox)); copy.setModSeq(mapperProvider.generateModSeq(benwaWorkMailbox)); updatableTickingClock().setInstant(updatableTickingClock().instant().plus(8, ChronoUnit.DAYS)); diff --git a/mailbox/store/src/test/java/org/apache/james/mailbox/store/mail/model/MessageMapperTest.java b/mailbox/store/src/test/java/org/apache/james/mailbox/store/mail/model/MessageMapperTest.java index 3d16316189f..bef9c55b432 100644 --- a/mailbox/store/src/test/java/org/apache/james/mailbox/store/mail/model/MessageMapperTest.java +++ b/mailbox/store/src/test/java/org/apache/james/mailbox/store/mail/model/MessageMapperTest.java @@ -60,6 +60,7 @@ import org.apache.james.mailbox.store.mail.model.impl.PropertyBuilder; import org.apache.james.mailbox.store.mail.model.impl.SimpleMailboxMessage; import org.apache.james.util.concurrency.ConcurrentTestRunner; +import org.apache.james.util.streams.Iterators; import org.apache.james.utils.UpdatableTickingClock; import org.junit.Assume; import org.junit.jupiter.api.BeforeEach; @@ -264,11 +265,11 @@ void getHeadersBytesShouldBePresentWhenAttachmentMetadataFetchType() throws Exce void messagesCanBeRetrievedInMailboxWithRangeTypeRange() throws MailboxException, IOException { saveMessages(); Iterator retrievedMessageIterator = messageMapper - .findInMailbox(benwaInboxMailbox, MessageRange.range(message1.getUid(), message4.getUid()), MessageMapper.FetchType.FULL, LIMIT); + .findInMailbox(benwaInboxMailbox, MessageRange.range(message1.getUid(), message4.getUid()), MessageMapper.FetchType.FULL, LIMIT); assertMessages(Lists.newArrayList(retrievedMessageIterator)).containOnly(message1, message2, message3, message4); } - + @Test void messagesCanBeRetrievedInMailboxWithRangeTypeRangeContainingAHole() throws MailboxException, IOException { saveMessages(); @@ -282,7 +283,7 @@ void messagesCanBeRetrievedInMailboxWithRangeTypeRangeContainingAHole() throws M void messagesCanBeRetrievedInMailboxWithRangeTypeFrom() throws MailboxException, IOException { saveMessages(); Iterator retrievedMessageIterator = messageMapper - .findInMailbox(benwaInboxMailbox, MessageRange.from(message3.getUid()), MessageMapper.FetchType.FULL, LIMIT); + .findInMailbox(benwaInboxMailbox, MessageRange.from(message3.getUid()), MessageMapper.FetchType.FULL, LIMIT); assertMessages(Lists.newArrayList(retrievedMessageIterator)).containOnly(message3, message4, message5); } @@ -291,7 +292,7 @@ void messagesCanBeRetrievedInMailboxWithRangeTypeFromContainingAHole() throws Ma saveMessages(); messageMapper.delete(benwaInboxMailbox, message4); Iterator retrievedMessageIterator = messageMapper - .findInMailbox(benwaInboxMailbox, MessageRange.from(message3.getUid()), MessageMapper.FetchType.FULL, LIMIT); + .findInMailbox(benwaInboxMailbox, MessageRange.from(message3.getUid()), MessageMapper.FetchType.FULL, LIMIT); assertMessages(Lists.newArrayList(retrievedMessageIterator)).containOnly(message3, message5); } @@ -307,7 +308,7 @@ void messagesCanBeRetrievedInMailboxWithRangeTypeAllContainingHole() throws Mail saveMessages(); messageMapper.delete(benwaInboxMailbox, message1); Iterator retrievedMessageIterator = messageMapper - .findInMailbox(benwaInboxMailbox, MessageRange.all(), MessageMapper.FetchType.FULL, LIMIT); + .findInMailbox(benwaInboxMailbox, MessageRange.all(), MessageMapper.FetchType.FULL, LIMIT); assertMessages(Lists.newArrayList(retrievedMessageIterator)).containOnly(message2, message3, message4, message5); } @@ -679,9 +680,9 @@ void copyShouldCreateAMessageInDestination() throws MailboxException, IOExceptio assertThat(messageMapper.getLastUid(benwaInboxMailbox).get()).isGreaterThan(message6.getUid()); MailboxMessage result = messageMapper.findInMailbox(benwaInboxMailbox, - MessageRange.one(messageMapper.getLastUid(benwaInboxMailbox).get()), - MessageMapper.FetchType.FULL, - LIMIT) + MessageRange.one(messageMapper.getLastUid(benwaInboxMailbox).get()), + MessageMapper.FetchType.FULL, + LIMIT) .next(); assertThat(result).isEqualToWithoutUidAndAttachment(message7, MessageMapper.FetchType.FULL); @@ -707,11 +708,11 @@ void copiedMessageShouldBeMarkedAsRecent() throws MailboxException { MessageMetaData metaData = messageMapper.copy(benwaInboxMailbox, message); assertThat( messageMapper.findInMailbox(benwaInboxMailbox, - MessageRange.one(metaData.getUid()), - MessageMapper.FetchType.METADATA, - LIMIT - ).next() - .isRecent() + MessageRange.one(metaData.getUid()), + MessageMapper.FetchType.METADATA, + LIMIT + ).next() + .isRecent() ).isTrue(); } @@ -723,10 +724,10 @@ void copiedRecentMessageShouldBeMarkedAsRecent() throws MailboxException { MessageMetaData metaData = messageMapper.copy(benwaInboxMailbox, message); assertThat( messageMapper.findInMailbox(benwaInboxMailbox, - MessageRange.one(metaData.getUid()), - MessageMapper.FetchType.METADATA, - LIMIT - ).next() + MessageRange.one(metaData.getUid()), + MessageMapper.FetchType.METADATA, + LIMIT + ).next() .isRecent() ).isTrue(); } @@ -738,11 +739,11 @@ void copiedMessageShouldNotChangeTheFlagsOnOriginalMessage() throws MailboxExcep messageMapper.copy(benwaInboxMailbox, message); assertThat( messageMapper.findInMailbox(benwaWorkMailbox, - MessageRange.one(message6.getUid()), - MessageMapper.FetchType.METADATA, - LIMIT - ).next() - .isRecent() + MessageRange.one(message6.getUid()), + MessageMapper.FetchType.METADATA, + LIMIT + ).next() + .isRecent() ).isFalse(); } @@ -758,7 +759,7 @@ protected void flagsReplacementShouldReturnAnUpdatedFlagHighlightingTheReplaceme saveMessages(); ModSeq modSeq = messageMapper.getHighestModSeq(benwaInboxMailbox); Optional updatedFlags = messageMapper.updateFlags(benwaInboxMailbox, message1.getUid(), - new FlagsUpdateCalculator(new Flags(Flags.Flag.FLAGGED), FlagsUpdateMode.REPLACE)); + new FlagsUpdateCalculator(new Flags(Flags.Flag.FLAGGED), FlagsUpdateMode.REPLACE)); assertThat(updatedFlags) .contains(UpdatedFlags.builder() .uid(message1.getUid()) @@ -776,12 +777,12 @@ protected void flagsAdditionShouldReturnAnUpdatedFlagHighlightingTheAddition() t ModSeq modSeq = messageMapper.getHighestModSeq(benwaInboxMailbox); assertThat(messageMapper.updateFlags(benwaInboxMailbox, message1.getUid(), new FlagsUpdateCalculator(new Flags(Flags.Flag.SEEN), FlagsUpdateMode.ADD))) .contains(UpdatedFlags.builder() - .uid(message1.getUid()) - .messageId(message1.getMessageId()) - .modSeq(modSeq.next()) - .oldFlags(new Flags(Flags.Flag.FLAGGED)) - .newFlags(new FlagsBuilder().add(Flags.Flag.SEEN, Flags.Flag.FLAGGED).build()) - .build()); + .uid(message1.getUid()) + .messageId(message1.getMessageId()) + .modSeq(modSeq.next()) + .oldFlags(new Flags(Flags.Flag.FLAGGED)) + .newFlags(new FlagsBuilder().add(Flags.Flag.SEEN, Flags.Flag.FLAGGED).build()) + .build()); } @Test @@ -850,6 +851,51 @@ void updateFlagsWithRangeAllRangeShouldAffectAllMessages() throws MailboxExcepti .hasSize(5); } + @Test + public void updateFlagsOnRangeShouldReturnUpdatedFlagsWithUidOrderAsc() throws MailboxException { + saveMessages(); + + Iterator it = messageMapper.updateFlags(benwaInboxMailbox, + new FlagsUpdateCalculator(new Flags(Flags.Flag.SEEN), FlagsUpdateMode.REPLACE), + MessageRange.range(message1.getUid(), message3.getUid())); + List updatedFlagsUids = Iterators.toStream(it) + .map(UpdatedFlags::getUid) + .collect(ImmutableList.toImmutableList()); + + assertThat(updatedFlagsUids) + .containsExactly(message1.getUid(), message2.getUid(), message3.getUid()); + } + + @Test + public void updateFlagsWithRangeFromShouldReturnUpdatedFlagsWithUidOrderAsc() throws MailboxException { + saveMessages(); + + Iterator it = messageMapper.updateFlags(benwaInboxMailbox, + new FlagsUpdateCalculator(new Flags(Flags.Flag.SEEN), FlagsUpdateMode.REPLACE), + MessageRange.from(message3.getUid())); + List updatedFlagsUids = Iterators.toStream(it) + .map(UpdatedFlags::getUid) + .collect(ImmutableList.toImmutableList()); + + assertThat(updatedFlagsUids) + .containsExactly(message3.getUid(), message4.getUid(), message5.getUid()); + } + + @Test + public void updateFlagsWithRangeAllRangeShouldReturnUpdatedFlagsWithUidOrderAsc() throws MailboxException { + saveMessages(); + + Iterator it = messageMapper.updateFlags(benwaInboxMailbox, + new FlagsUpdateCalculator(new Flags(Flags.Flag.SEEN), FlagsUpdateMode.REPLACE), + MessageRange.all()); + List updatedFlagsUids = Iterators.toStream(it) + .map(UpdatedFlags::getUid) + .collect(ImmutableList.toImmutableList()); + + assertThat(updatedFlagsUids) + .containsExactly(message1.getUid(), message2.getUid(), message3.getUid(), message4.getUid(), message5.getUid()); + } + @Test void messagePropertiesShouldBeStored() throws Exception { PropertyBuilder propBuilder = new PropertyBuilder(); @@ -865,7 +911,7 @@ void messagePropertiesShouldBeStored() throws Exception { assertProperties(message.getProperties().toProperties()).containsOnly(propBuilder.toProperties()); } - + @Test void messagePropertiesShouldBeStoredWhenDuplicateEntries() throws Exception { PropertyBuilder propBuilder = new PropertyBuilder(); @@ -949,7 +995,7 @@ protected void userFlagsUpdateShouldReturnCorrectUpdatedFlagsWhenNoop() throws E saveMessages(); assertThat( - messageMapper.updateFlags(benwaInboxMailbox,message1.getUid(), + messageMapper.updateFlags(benwaInboxMailbox, message1.getUid(), new FlagsUpdateCalculator(new Flags(USER_FLAG), FlagsUpdateMode.REMOVE))) .contains( UpdatedFlags.builder() @@ -991,7 +1037,7 @@ public void setFlagsShouldWorkWithConcurrencyWithRemove() throws Exception { int updateCount = 40; ConcurrentTestRunner.builder() .operation((threadNumber, step) -> { - if (step < updateCount / 2) { + if (step < updateCount / 2) { messageMapper.updateFlags(benwaInboxMailbox, message1.getUid(), new FlagsUpdateCalculator(new Flags("custom-" + threadNumber + "-" + step), FlagsUpdateMode.ADD)); } else { @@ -1171,7 +1217,7 @@ void getApplicableFlagShouldHaveEffectWhenUnsetMessageFlagThenComputingApplicabl @Test void getApplicableFlagShouldHaveNotEffectWhenUnsetMessageFlagThenIncrementalApplicableFlags() throws Exception { - Assume.assumeTrue(mapperProvider.getSupportedCapabilities().contains(MapperProvider.Capabilities.THREAD_SAFE_FLAGS_UPDATE)); + Assume.assumeTrue(mapperProvider.getSupportedCapabilities().contains(MapperProvider.Capabilities.INCREMENTAL_APPLICABLE_FLAGS)); String customFlag1 = "custom1"; String customFlag2 = "custom2"; message1.setFlags(new Flags(customFlag1)); @@ -1265,7 +1311,8 @@ void getUidsShouldNotReturnUidsOfDeletedMessages() throws Exception { messageMapper.updateFlags(benwaInboxMailbox, new FlagsUpdateCalculator(new Flags(Flag.DELETED), FlagsUpdateMode.ADD), - MessageRange.range(message2.getUid(), message4.getUid())); + MessageRange.range(message2.getUid(), message4.getUid())).forEachRemaining(any -> { + }); List uids = messageMapper.retrieveMessagesMarkedForDeletion(benwaInboxMailbox, MessageRange.all()); messageMapper.deleteMessages(benwaInboxMailbox, uids); @@ -1397,7 +1444,7 @@ protected void saveMessages() throws MailboxException { private MailboxMessage retrieveMessageFromStorage(MailboxMessage message) throws MailboxException { return messageMapper.findInMailbox(benwaInboxMailbox, MessageRange.one(message.getUid()), MessageMapper.FetchType.METADATA, LIMIT).next(); } - + private MailboxMessage createMessage(Mailbox mailbox, MessageId messageId, String content, int bodyStart, PropertyBuilder propertyBuilder) { return new SimpleMailboxMessage(messageId, ThreadId.fromBaseMessageId(messageId), new Date(), content.length(), bodyStart, new ByteContent(content.getBytes()), new Flags(), propertyBuilder.build(), mailbox.getMailboxId()); } diff --git a/mailbox/store/src/test/java/org/apache/james/mailbox/store/quota/ListeningCurrentQuotaUpdaterTest.java b/mailbox/store/src/test/java/org/apache/james/mailbox/store/quota/ListeningCurrentQuotaUpdaterTest.java index b01935faa1e..bd0c937944f 100644 --- a/mailbox/store/src/test/java/org/apache/james/mailbox/store/quota/ListeningCurrentQuotaUpdaterTest.java +++ b/mailbox/store/src/test/java/org/apache/james/mailbox/store/quota/ListeningCurrentQuotaUpdaterTest.java @@ -42,11 +42,9 @@ import org.apache.james.events.Group; import org.apache.james.mailbox.MessageUid; import org.apache.james.mailbox.ModSeq; -import org.apache.james.mailbox.events.MailboxEvents; import org.apache.james.mailbox.events.MailboxEvents.Added; import org.apache.james.mailbox.events.MailboxEvents.Expunged; import org.apache.james.mailbox.events.MailboxEvents.MailboxDeletion; -import org.apache.james.mailbox.model.CurrentQuotas; import org.apache.james.mailbox.model.MailboxId; import org.apache.james.mailbox.model.MailboxPath; import org.apache.james.mailbox.model.MessageMetaData; @@ -196,40 +194,4 @@ void mailboxDeletionEventShouldDoNothingWhenEmptyMailbox() throws Exception { verifyNoMoreInteractions(mockedCurrentQuotaManager); } - - @Test - void mailboxAddEventShouldProvisionCurrentQuota() throws Exception { - QuotaOperation operation = new QuotaOperation(QUOTA_ROOT, QuotaCountUsage.count(0), QuotaSizeUsage.size(0)); - - MailboxEvents.MailboxAdded added; - added = mock(MailboxEvents.MailboxAdded.class); - - when(added.getMailboxId()).thenReturn(MAILBOX_ID); - when(added.getMailboxPath()).thenReturn(MAILBOX_PATH); - when(added.getUsername()).thenReturn(USERNAME_BENWA); - when(mockedQuotaRootResolver.getQuotaRootReactive(eq(MAILBOX_PATH))) - .thenReturn(Mono.just(QUOTA_ROOT)); - when(mockedCurrentQuotaManager.getCurrentQuotas(QUOTA_ROOT)).thenAnswer(any -> Mono.empty()); - when(mockedCurrentQuotaManager.setCurrentQuotas(operation)).thenAnswer(any -> Mono.empty()); - - testee.event(added); - - verify(mockedCurrentQuotaManager).setCurrentQuotas(operation); - } - - @Test - void mailboxAddEventShouldNotProvisionWhenAlreadyExist() throws Exception { - MailboxEvents.MailboxAdded added = mock(MailboxEvents.MailboxAdded.class); - when(added.getMailboxId()).thenReturn(MAILBOX_ID); - when(added.getMailboxPath()).thenReturn(MAILBOX_PATH); - when(added.getUsername()).thenReturn(USERNAME_BENWA); - when(mockedQuotaRootResolver.getQuotaRootReactive(eq(MAILBOX_PATH))) - .thenReturn(Mono.just(QUOTA_ROOT)); - when(mockedCurrentQuotaManager.getCurrentQuotas(QUOTA_ROOT)) - .thenAnswer(any -> Mono.just(CurrentQuotas.from(QUOTA))); - - testee.event(added); - - verify(mockedCurrentQuotaManager, never()).setCurrentQuotas(any()); - } } diff --git a/mpt/impl/imap-mailbox/core/src/main/resources/org/apache/james/imap/scripts/Metadata.test b/mpt/impl/imap-mailbox/core/src/main/resources/org/apache/james/imap/scripts/Metadata.test index 7e247345a59..e77ad93c049 100644 --- a/mpt/impl/imap-mailbox/core/src/main/resources/org/apache/james/imap/scripts/Metadata.test +++ b/mpt/impl/imap-mailbox/core/src/main/resources/org/apache/james/imap/scripts/Metadata.test @@ -85,7 +85,8 @@ S: \* METADATA "INBOX" \((\/private\/comment "My own comment" \/shared\/comment S: g3 OK GETMETADATA completed. C: g4 GETMETADATA "INBOX" -S: \* METADATA "INBOX" \(\/private\/comment "My own comment" \/shared\/comment "The shared comment"\) +# Regex used to be order agnostic. Annotation1 Annotation2 OR Annotation2 Annotation1 +S: \* METADATA "INBOX" \((\/private\/comment "My own comment" \/shared\/comment "The shared comment"|\/shared\/comment "The shared comment" \/private\/comment "My own comment")\) S: g4 OK GETMETADATA completed. C: g5 GETMETADATA "INBOX" /shared/comment /private/comment) @@ -102,7 +103,8 @@ S: \* METADATA "INBOX" \(\/private\/comment "My own comment"\) S: g8 OK \[METADATA LONGENTRIES 18\] GETMETADATA completed. C: g9 GETMETADATA "INBOX" (MAXSIZE 100) -S: \* METADATA "INBOX" \(\/private\/comment "My own comment" \/shared\/comment "The shared comment"\) +# Regex used to be order agnostic. Annotation1 Annotation2 OR Annotation2 Annotation1 +S: \* METADATA "INBOX" \((\/private\/comment "My own comment" \/shared\/comment "The shared comment"|\/shared\/comment "The shared comment" \/private\/comment "My own comment")\) S: g9 OK GETMETADATA completed. C: s3 SETMETADATA INBOX (/private/comment/user "My own comment for user") @@ -169,7 +171,8 @@ C: m03 SETMETADATA mailboxTest (/shared/comment "The mailboxTest shared comment" S: m03 OK SETMETADATA completed. C: m04 GETMETADATA "mailboxTest" -S: \* METADATA "mailboxTest" \(\/private\/comment "The mailboxTest private comment" \/shared\/comment "The mailboxTest shared comment"\) +# Regex used to be order agnostic. Annotation1 Annotation2 OR Annotation2 Annotation1 +S: \* METADATA "mailboxTest" \((\/private\/comment "The mailboxTest private comment" \/shared\/comment "The mailboxTest shared comment"|\/shared\/comment "The mailboxTest shared comment" \/private\/comment "The mailboxTest private comment")\) S: m04 OK GETMETADATA completed. C: m05 DELETE mailboxTest diff --git a/mpt/impl/imap-mailbox/pom.xml b/mpt/impl/imap-mailbox/pom.xml index df6453cbc99..e6b9dc7948d 100644 --- a/mpt/impl/imap-mailbox/pom.xml +++ b/mpt/impl/imap-mailbox/pom.xml @@ -41,6 +41,7 @@ jpa lucenesearch opensearch + postgres rabbitmq @@ -88,6 +89,12 @@ ${project.version} test + + ${james.groupId} + apache-james-mpt-imapmailbox-postgres + ${project.version} + test + diff --git a/mpt/impl/imap-mailbox/postgres/pom.xml b/mpt/impl/imap-mailbox/postgres/pom.xml new file mode 100644 index 00000000000..e74409d2001 --- /dev/null +++ b/mpt/impl/imap-mailbox/postgres/pom.xml @@ -0,0 +1,127 @@ + + + + 4.0.0 + + org.apache.james + apache-james-mpt-imapmailbox + 3.9.0-SNAPSHOT + + + apache-james-mpt-imapmailbox-postgres + Apache James :: MPT :: Imap Mailbox :: Postgres + + + + ${james.groupId} + apache-james-backends-postgres + test-jar + test + + + ${james.groupId} + apache-james-mailbox-api + test-jar + test + + + ${james.groupId} + apache-james-mailbox-postgres + test + + + ${james.groupId} + apache-james-mailbox-postgres + test-jar + test + + + ${james.groupId} + apache-james-mailbox-store + test + + + ${james.groupId} + apache-james-mpt-imapmailbox-core + + + ${james.groupId} + blob-memory + test + + + ${james.groupId} + blob-storage-strategy + test + + + ${james.groupId} + event-bus-api + test-jar + test + + + ${james.groupId} + event-bus-in-vm + test + + + ${james.groupId} + james-server-guice-common + test-jar + test + + + ${james.groupId} + james-server-testing + test + + + ${james.groupId} + metrics-tests + test + + + ${james.groupId} + testing-base + test + + + org.testcontainers + postgresql + test + + + + + + + org.apache.maven.plugins + maven-surefire-plugin + + -Djava.library.path= + -javaagent:"${settings.localRepository}"/org/jacoco/org.jacoco.agent/${jacoco-maven-plugin.version}/org.jacoco.agent-${jacoco-maven-plugin.version}-runtime.jar=destfile=${basedir}/target/jacoco.exec + -Xms1024m -Xmx2048m + + + + + + diff --git a/mpt/impl/imap-mailbox/postgres/src/test/java/org/apache/james/mpt/imapmailbox/postgres/PostgresAuthenticatePlainTest.java b/mpt/impl/imap-mailbox/postgres/src/test/java/org/apache/james/mpt/imapmailbox/postgres/PostgresAuthenticatePlainTest.java new file mode 100644 index 00000000000..a8d39c4fed7 --- /dev/null +++ b/mpt/impl/imap-mailbox/postgres/src/test/java/org/apache/james/mpt/imapmailbox/postgres/PostgresAuthenticatePlainTest.java @@ -0,0 +1,35 @@ +/**************************************************************** + * 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.mpt.imapmailbox.postgres; + +import org.apache.james.mpt.api.ImapHostSystem; +import org.apache.james.mpt.imapmailbox.postgres.host.PostgresHostSystemExtension; +import org.apache.james.mpt.imapmailbox.suite.AuthenticatePlain; +import org.junit.jupiter.api.extension.RegisterExtension; + +public class PostgresAuthenticatePlainTest extends AuthenticatePlain { + @RegisterExtension + public static PostgresHostSystemExtension hostSystemExtension = new PostgresHostSystemExtension(); + + @Override + protected ImapHostSystem createImapHostSystem() { + return hostSystemExtension.getHostSystem(); + } +} diff --git a/mpt/impl/imap-mailbox/postgres/src/test/java/org/apache/james/mpt/imapmailbox/postgres/PostgresAuthenticatedStateTest.java b/mpt/impl/imap-mailbox/postgres/src/test/java/org/apache/james/mpt/imapmailbox/postgres/PostgresAuthenticatedStateTest.java new file mode 100644 index 00000000000..3ac16d36dc4 --- /dev/null +++ b/mpt/impl/imap-mailbox/postgres/src/test/java/org/apache/james/mpt/imapmailbox/postgres/PostgresAuthenticatedStateTest.java @@ -0,0 +1,41 @@ +/**************************************************************** + * 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.mpt.imapmailbox.postgres; + +import org.apache.james.mpt.api.ImapHostSystem; +import org.apache.james.mpt.imapmailbox.postgres.host.PostgresHostSystemExtension; +import org.apache.james.mpt.imapmailbox.suite.AuthenticatedState; +import org.junit.jupiter.api.Disabled; +import org.junit.jupiter.api.extension.RegisterExtension; + +public class PostgresAuthenticatedStateTest extends AuthenticatedState { + @RegisterExtension + public static PostgresHostSystemExtension hostSystemExtension = new PostgresHostSystemExtension(); + + @Override + protected ImapHostSystem createImapHostSystem() { + return hostSystemExtension.getHostSystem(); + } + + @Disabled("TODO - James 3945 Should adapt to Postgres") + @Override + public void rightsCommandsShouldBeSupported() { + } +} diff --git a/mpt/impl/imap-mailbox/postgres/src/test/java/org/apache/james/mpt/imapmailbox/postgres/PostgresConcurrentSessionsTest.java b/mpt/impl/imap-mailbox/postgres/src/test/java/org/apache/james/mpt/imapmailbox/postgres/PostgresConcurrentSessionsTest.java new file mode 100644 index 00000000000..444e1d13579 --- /dev/null +++ b/mpt/impl/imap-mailbox/postgres/src/test/java/org/apache/james/mpt/imapmailbox/postgres/PostgresConcurrentSessionsTest.java @@ -0,0 +1,47 @@ +/**************************************************************** + * 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.mpt.imapmailbox.postgres; + +import org.apache.james.mpt.api.ImapHostSystem; +import org.apache.james.mpt.imapmailbox.postgres.host.PostgresHostSystemExtension; +import org.apache.james.mpt.imapmailbox.suite.ConcurrentSessions; +import org.junit.jupiter.api.extension.RegisterExtension; + +public class PostgresConcurrentSessionsTest extends ConcurrentSessions { + @RegisterExtension + public static PostgresHostSystemExtension hostSystemExtension = new PostgresHostSystemExtension(); + + @Override + protected ImapHostSystem createImapHostSystem() { + return hostSystemExtension.getHostSystem(); + } + + @Override + public void testConcurrentFetchResponseITALY() { + } + + @Override + public void testConcurrentFetchResponseKOREA() { + } + + @Override + public void testConcurrentFetchResponseUS() { + } +} diff --git a/mpt/impl/imap-mailbox/postgres/src/test/java/org/apache/james/mpt/imapmailbox/postgres/PostgresCondstoreTest.java b/mpt/impl/imap-mailbox/postgres/src/test/java/org/apache/james/mpt/imapmailbox/postgres/PostgresCondstoreTest.java new file mode 100644 index 00000000000..d8953168202 --- /dev/null +++ b/mpt/impl/imap-mailbox/postgres/src/test/java/org/apache/james/mpt/imapmailbox/postgres/PostgresCondstoreTest.java @@ -0,0 +1,35 @@ +/**************************************************************** + * 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.mpt.imapmailbox.postgres; + +import org.apache.james.mpt.host.JamesImapHostSystem; +import org.apache.james.mpt.imapmailbox.postgres.host.PostgresHostSystemExtension; +import org.apache.james.mpt.imapmailbox.suite.Condstore; +import org.junit.jupiter.api.extension.RegisterExtension; + +public class PostgresCondstoreTest extends Condstore { + @RegisterExtension + public static PostgresHostSystemExtension hostSystemExtension = new PostgresHostSystemExtension(); + + @Override + protected JamesImapHostSystem createJamesImapHostSystem() { + return hostSystemExtension.getHostSystem(); + } +} diff --git a/mpt/impl/imap-mailbox/postgres/src/test/java/org/apache/james/mpt/imapmailbox/postgres/PostgresCopyTest.java b/mpt/impl/imap-mailbox/postgres/src/test/java/org/apache/james/mpt/imapmailbox/postgres/PostgresCopyTest.java new file mode 100644 index 00000000000..e50255ad74d --- /dev/null +++ b/mpt/impl/imap-mailbox/postgres/src/test/java/org/apache/james/mpt/imapmailbox/postgres/PostgresCopyTest.java @@ -0,0 +1,39 @@ +/**************************************************************** + * 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.mpt.imapmailbox.postgres; + +import org.apache.james.mpt.api.ImapHostSystem; +import org.apache.james.mpt.imapmailbox.postgres.host.PostgresHostSystemExtension; +import org.apache.james.mpt.imapmailbox.suite.Copy; +import org.junit.jupiter.api.extension.RegisterExtension; + +public class PostgresCopyTest extends Copy { + @RegisterExtension + public static PostgresHostSystemExtension hostSystemExtension = new PostgresHostSystemExtension(); + + @Override + protected ImapHostSystem createImapHostSystem() { + return hostSystemExtension.getHostSystem(); + } + + @Override + public void copyCommandShouldRespectTheRFC() { + } +} diff --git a/mpt/impl/imap-mailbox/postgres/src/test/java/org/apache/james/mpt/imapmailbox/postgres/PostgresEventsTest.java b/mpt/impl/imap-mailbox/postgres/src/test/java/org/apache/james/mpt/imapmailbox/postgres/PostgresEventsTest.java new file mode 100644 index 00000000000..116fa312c55 --- /dev/null +++ b/mpt/impl/imap-mailbox/postgres/src/test/java/org/apache/james/mpt/imapmailbox/postgres/PostgresEventsTest.java @@ -0,0 +1,35 @@ +/**************************************************************** + * 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.mpt.imapmailbox.postgres; + +import org.apache.james.mpt.api.ImapHostSystem; +import org.apache.james.mpt.imapmailbox.postgres.host.PostgresHostSystemExtension; +import org.apache.james.mpt.imapmailbox.suite.Events; +import org.junit.jupiter.api.extension.RegisterExtension; + +public class PostgresEventsTest extends Events { + @RegisterExtension + public static PostgresHostSystemExtension hostSystemExtension = new PostgresHostSystemExtension(); + + @Override + protected ImapHostSystem createImapHostSystem() { + return hostSystemExtension.getHostSystem(); + } +} diff --git a/mpt/impl/imap-mailbox/postgres/src/test/java/org/apache/james/mpt/imapmailbox/postgres/PostgresExpungeTest.java b/mpt/impl/imap-mailbox/postgres/src/test/java/org/apache/james/mpt/imapmailbox/postgres/PostgresExpungeTest.java new file mode 100644 index 00000000000..d6cc8489002 --- /dev/null +++ b/mpt/impl/imap-mailbox/postgres/src/test/java/org/apache/james/mpt/imapmailbox/postgres/PostgresExpungeTest.java @@ -0,0 +1,35 @@ +/**************************************************************** + * 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.mpt.imapmailbox.postgres; + +import org.apache.james.mpt.api.ImapHostSystem; +import org.apache.james.mpt.imapmailbox.postgres.host.PostgresHostSystemExtension; +import org.apache.james.mpt.imapmailbox.suite.Expunge; +import org.junit.jupiter.api.extension.RegisterExtension; + +public class PostgresExpungeTest extends Expunge { + @RegisterExtension + public static PostgresHostSystemExtension hostSystemExtension = new PostgresHostSystemExtension(); + + @Override + protected ImapHostSystem createImapHostSystem() { + return hostSystemExtension.getHostSystem(); + } +} diff --git a/mpt/impl/imap-mailbox/postgres/src/test/java/org/apache/james/mpt/imapmailbox/postgres/PostgresFetchBodySectionTest.java b/mpt/impl/imap-mailbox/postgres/src/test/java/org/apache/james/mpt/imapmailbox/postgres/PostgresFetchBodySectionTest.java new file mode 100644 index 00000000000..24f06e0c30b --- /dev/null +++ b/mpt/impl/imap-mailbox/postgres/src/test/java/org/apache/james/mpt/imapmailbox/postgres/PostgresFetchBodySectionTest.java @@ -0,0 +1,35 @@ +/**************************************************************** + * 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.mpt.imapmailbox.postgres; + +import org.apache.james.mpt.api.ImapHostSystem; +import org.apache.james.mpt.imapmailbox.postgres.host.PostgresHostSystemExtension; +import org.apache.james.mpt.imapmailbox.suite.FetchBodySection; +import org.junit.jupiter.api.extension.RegisterExtension; + +public class PostgresFetchBodySectionTest extends FetchBodySection { + @RegisterExtension + public static PostgresHostSystemExtension hostSystemExtension = new PostgresHostSystemExtension(); + + @Override + protected ImapHostSystem createImapHostSystem() { + return hostSystemExtension.getHostSystem(); + } +} diff --git a/mpt/impl/imap-mailbox/postgres/src/test/java/org/apache/james/mpt/imapmailbox/postgres/PostgresFetchBodyStructureTest.java b/mpt/impl/imap-mailbox/postgres/src/test/java/org/apache/james/mpt/imapmailbox/postgres/PostgresFetchBodyStructureTest.java new file mode 100644 index 00000000000..de45b07180c --- /dev/null +++ b/mpt/impl/imap-mailbox/postgres/src/test/java/org/apache/james/mpt/imapmailbox/postgres/PostgresFetchBodyStructureTest.java @@ -0,0 +1,35 @@ +/**************************************************************** + * 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.mpt.imapmailbox.postgres; + +import org.apache.james.mpt.api.ImapHostSystem; +import org.apache.james.mpt.imapmailbox.postgres.host.PostgresHostSystemExtension; +import org.apache.james.mpt.imapmailbox.suite.FetchBodyStructure; +import org.junit.jupiter.api.extension.RegisterExtension; + +public class PostgresFetchBodyStructureTest extends FetchBodyStructure { + @RegisterExtension + public static PostgresHostSystemExtension hostSystemExtension = new PostgresHostSystemExtension(); + + @Override + protected ImapHostSystem createImapHostSystem() { + return hostSystemExtension.getHostSystem(); + } +} diff --git a/mpt/impl/imap-mailbox/postgres/src/test/java/org/apache/james/mpt/imapmailbox/postgres/PostgresFetchHeadersTest.java b/mpt/impl/imap-mailbox/postgres/src/test/java/org/apache/james/mpt/imapmailbox/postgres/PostgresFetchHeadersTest.java new file mode 100644 index 00000000000..ed908a5b89a --- /dev/null +++ b/mpt/impl/imap-mailbox/postgres/src/test/java/org/apache/james/mpt/imapmailbox/postgres/PostgresFetchHeadersTest.java @@ -0,0 +1,35 @@ +/**************************************************************** + * 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.mpt.imapmailbox.postgres; + +import org.apache.james.mpt.api.ImapHostSystem; +import org.apache.james.mpt.imapmailbox.postgres.host.PostgresHostSystemExtension; +import org.apache.james.mpt.imapmailbox.suite.FetchHeaders; +import org.junit.jupiter.api.extension.RegisterExtension; + +public class PostgresFetchHeadersTest extends FetchHeaders { + @RegisterExtension + public static PostgresHostSystemExtension hostSystemExtension = new PostgresHostSystemExtension(); + + @Override + protected ImapHostSystem createImapHostSystem() { + return hostSystemExtension.getHostSystem(); + } +} diff --git a/mpt/impl/imap-mailbox/postgres/src/test/java/org/apache/james/mpt/imapmailbox/postgres/PostgresFetchTest.java b/mpt/impl/imap-mailbox/postgres/src/test/java/org/apache/james/mpt/imapmailbox/postgres/PostgresFetchTest.java new file mode 100644 index 00000000000..358cc3180c1 --- /dev/null +++ b/mpt/impl/imap-mailbox/postgres/src/test/java/org/apache/james/mpt/imapmailbox/postgres/PostgresFetchTest.java @@ -0,0 +1,36 @@ +/**************************************************************** + * 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.mpt.imapmailbox.postgres; + +import org.apache.james.mpt.api.ImapHostSystem; +import org.apache.james.mpt.imapmailbox.postgres.host.PostgresHostSystemExtension; +import org.apache.james.mpt.imapmailbox.suite.Fetch; +import org.junit.jupiter.api.extension.RegisterExtension; + +public class PostgresFetchTest extends Fetch { + @RegisterExtension + public static PostgresHostSystemExtension hostSystemExtension = new PostgresHostSystemExtension(); + + @Override + protected ImapHostSystem createImapHostSystem() { + return hostSystemExtension.getHostSystem(); + } + +} diff --git a/mpt/impl/imap-mailbox/postgres/src/test/java/org/apache/james/mpt/imapmailbox/postgres/PostgresListingTest.java b/mpt/impl/imap-mailbox/postgres/src/test/java/org/apache/james/mpt/imapmailbox/postgres/PostgresListingTest.java new file mode 100644 index 00000000000..2069ee06784 --- /dev/null +++ b/mpt/impl/imap-mailbox/postgres/src/test/java/org/apache/james/mpt/imapmailbox/postgres/PostgresListingTest.java @@ -0,0 +1,35 @@ +/**************************************************************** + * 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.mpt.imapmailbox.postgres; + +import org.apache.james.mpt.api.ImapHostSystem; +import org.apache.james.mpt.imapmailbox.postgres.host.PostgresHostSystemExtension; +import org.apache.james.mpt.imapmailbox.suite.Listing; +import org.junit.jupiter.api.extension.RegisterExtension; + +public class PostgresListingTest extends Listing { + @RegisterExtension + public static PostgresHostSystemExtension hostSystemExtension = new PostgresHostSystemExtension(); + + @Override + protected ImapHostSystem createImapHostSystem() { + return hostSystemExtension.getHostSystem(); + } +} diff --git a/mpt/impl/imap-mailbox/postgres/src/test/java/org/apache/james/mpt/imapmailbox/postgres/PostgresMailboxAnnotationTest.java b/mpt/impl/imap-mailbox/postgres/src/test/java/org/apache/james/mpt/imapmailbox/postgres/PostgresMailboxAnnotationTest.java new file mode 100644 index 00000000000..e4c7535eb98 --- /dev/null +++ b/mpt/impl/imap-mailbox/postgres/src/test/java/org/apache/james/mpt/imapmailbox/postgres/PostgresMailboxAnnotationTest.java @@ -0,0 +1,35 @@ +/**************************************************************** + * 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.mpt.imapmailbox.postgres; + +import org.apache.james.mpt.api.ImapHostSystem; +import org.apache.james.mpt.imapmailbox.postgres.host.PostgresHostSystemExtension; +import org.apache.james.mpt.imapmailbox.suite.MailboxAnnotation; +import org.junit.jupiter.api.extension.RegisterExtension; + +public class PostgresMailboxAnnotationTest extends MailboxAnnotation { + @RegisterExtension + public static PostgresHostSystemExtension hostSystemExtension = new PostgresHostSystemExtension(); + + @Override + protected ImapHostSystem createImapHostSystem() { + return hostSystemExtension.getHostSystem(); + } +} diff --git a/mpt/impl/imap-mailbox/postgres/src/test/java/org/apache/james/mpt/imapmailbox/postgres/PostgresMailboxWithLongNameErrorTest.java b/mpt/impl/imap-mailbox/postgres/src/test/java/org/apache/james/mpt/imapmailbox/postgres/PostgresMailboxWithLongNameErrorTest.java new file mode 100644 index 00000000000..8dc66398aa6 --- /dev/null +++ b/mpt/impl/imap-mailbox/postgres/src/test/java/org/apache/james/mpt/imapmailbox/postgres/PostgresMailboxWithLongNameErrorTest.java @@ -0,0 +1,35 @@ +/**************************************************************** + * 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.mpt.imapmailbox.postgres; + +import org.apache.james.mpt.api.ImapHostSystem; +import org.apache.james.mpt.imapmailbox.postgres.host.PostgresHostSystemExtension; +import org.apache.james.mpt.imapmailbox.suite.MailboxWithLongNameError; +import org.junit.jupiter.api.extension.RegisterExtension; + +public class PostgresMailboxWithLongNameErrorTest extends MailboxWithLongNameError { + @RegisterExtension + public static PostgresHostSystemExtension hostSystemExtension = new PostgresHostSystemExtension(); + + @Override + protected ImapHostSystem createImapHostSystem() { + return hostSystemExtension.getHostSystem(); + } +} diff --git a/mpt/impl/imap-mailbox/postgres/src/test/java/org/apache/james/mpt/imapmailbox/postgres/PostgresMoveTest.java b/mpt/impl/imap-mailbox/postgres/src/test/java/org/apache/james/mpt/imapmailbox/postgres/PostgresMoveTest.java new file mode 100644 index 00000000000..8637e5d2609 --- /dev/null +++ b/mpt/impl/imap-mailbox/postgres/src/test/java/org/apache/james/mpt/imapmailbox/postgres/PostgresMoveTest.java @@ -0,0 +1,35 @@ +/**************************************************************** + * 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.mpt.imapmailbox.postgres; + +import org.apache.james.mpt.api.ImapHostSystem; +import org.apache.james.mpt.imapmailbox.postgres.host.PostgresHostSystemExtension; +import org.apache.james.mpt.imapmailbox.suite.Move; +import org.junit.jupiter.api.extension.RegisterExtension; + +public class PostgresMoveTest extends Move { + @RegisterExtension + public static PostgresHostSystemExtension hostSystemExtension = new PostgresHostSystemExtension(); + + @Override + protected ImapHostSystem createImapHostSystem() { + return hostSystemExtension.getHostSystem(); + } +} diff --git a/mpt/impl/imap-mailbox/postgres/src/test/java/org/apache/james/mpt/imapmailbox/postgres/PostgresNonAuthenticatedStateTest.java b/mpt/impl/imap-mailbox/postgres/src/test/java/org/apache/james/mpt/imapmailbox/postgres/PostgresNonAuthenticatedStateTest.java new file mode 100644 index 00000000000..5fa63f5b95b --- /dev/null +++ b/mpt/impl/imap-mailbox/postgres/src/test/java/org/apache/james/mpt/imapmailbox/postgres/PostgresNonAuthenticatedStateTest.java @@ -0,0 +1,35 @@ +/**************************************************************** + * 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.mpt.imapmailbox.postgres; + +import org.apache.james.mpt.api.ImapHostSystem; +import org.apache.james.mpt.imapmailbox.postgres.host.PostgresHostSystemExtension; +import org.apache.james.mpt.imapmailbox.suite.NonAuthenticatedState; +import org.junit.jupiter.api.extension.RegisterExtension; + +public class PostgresNonAuthenticatedStateTest extends NonAuthenticatedState { + @RegisterExtension + public static PostgresHostSystemExtension hostSystemExtension = new PostgresHostSystemExtension(); + + @Override + protected ImapHostSystem createImapHostSystem() { + return hostSystemExtension.getHostSystem(); + } +} diff --git a/mpt/impl/imap-mailbox/postgres/src/test/java/org/apache/james/mpt/imapmailbox/postgres/PostgresPartialFetchTest.java b/mpt/impl/imap-mailbox/postgres/src/test/java/org/apache/james/mpt/imapmailbox/postgres/PostgresPartialFetchTest.java new file mode 100644 index 00000000000..be90ff06e1c --- /dev/null +++ b/mpt/impl/imap-mailbox/postgres/src/test/java/org/apache/james/mpt/imapmailbox/postgres/PostgresPartialFetchTest.java @@ -0,0 +1,35 @@ +/**************************************************************** + * 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.mpt.imapmailbox.postgres; + +import org.apache.james.mpt.api.ImapHostSystem; +import org.apache.james.mpt.imapmailbox.postgres.host.PostgresHostSystemExtension; +import org.apache.james.mpt.imapmailbox.suite.PartialFetch; +import org.junit.jupiter.api.extension.RegisterExtension; + +public class PostgresPartialFetchTest extends PartialFetch { + @RegisterExtension + public static PostgresHostSystemExtension hostSystemExtension = new PostgresHostSystemExtension(); + + @Override + protected ImapHostSystem createImapHostSystem() { + return hostSystemExtension.getHostSystem(); + } +} diff --git a/mpt/impl/imap-mailbox/postgres/src/test/java/org/apache/james/mpt/imapmailbox/postgres/PostgresQuotaTest.java b/mpt/impl/imap-mailbox/postgres/src/test/java/org/apache/james/mpt/imapmailbox/postgres/PostgresQuotaTest.java new file mode 100644 index 00000000000..a19495b582c --- /dev/null +++ b/mpt/impl/imap-mailbox/postgres/src/test/java/org/apache/james/mpt/imapmailbox/postgres/PostgresQuotaTest.java @@ -0,0 +1,35 @@ +/**************************************************************** + * 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.mpt.imapmailbox.postgres; + +import org.apache.james.mpt.api.ImapHostSystem; +import org.apache.james.mpt.imapmailbox.postgres.host.PostgresHostSystemExtension; +import org.apache.james.mpt.imapmailbox.suite.QuotaTest; +import org.junit.jupiter.api.extension.RegisterExtension; + +public class PostgresQuotaTest extends QuotaTest { + @RegisterExtension + public static PostgresHostSystemExtension hostSystemExtension = new PostgresHostSystemExtension(); + + @Override + protected ImapHostSystem createImapHostSystem() { + return hostSystemExtension.getHostSystem(); + } +} diff --git a/mpt/impl/imap-mailbox/postgres/src/test/java/org/apache/james/mpt/imapmailbox/postgres/PostgresRenameTest.java b/mpt/impl/imap-mailbox/postgres/src/test/java/org/apache/james/mpt/imapmailbox/postgres/PostgresRenameTest.java new file mode 100644 index 00000000000..4ea7a04f306 --- /dev/null +++ b/mpt/impl/imap-mailbox/postgres/src/test/java/org/apache/james/mpt/imapmailbox/postgres/PostgresRenameTest.java @@ -0,0 +1,35 @@ +/**************************************************************** + * 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.mpt.imapmailbox.postgres; + +import org.apache.james.mpt.api.ImapHostSystem; +import org.apache.james.mpt.imapmailbox.postgres.host.PostgresHostSystemExtension; +import org.apache.james.mpt.imapmailbox.suite.Rename; +import org.junit.jupiter.api.extension.RegisterExtension; + +public class PostgresRenameTest extends Rename { + @RegisterExtension + public static PostgresHostSystemExtension hostSystemExtension = new PostgresHostSystemExtension(); + + @Override + protected ImapHostSystem createImapHostSystem() { + return hostSystemExtension.getHostSystem(); + } +} diff --git a/mpt/impl/imap-mailbox/postgres/src/test/java/org/apache/james/mpt/imapmailbox/postgres/PostgresSearchTest.java b/mpt/impl/imap-mailbox/postgres/src/test/java/org/apache/james/mpt/imapmailbox/postgres/PostgresSearchTest.java new file mode 100644 index 00000000000..9baf18e5f1e --- /dev/null +++ b/mpt/impl/imap-mailbox/postgres/src/test/java/org/apache/james/mpt/imapmailbox/postgres/PostgresSearchTest.java @@ -0,0 +1,35 @@ +/**************************************************************** + * 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.mpt.imapmailbox.postgres; + +import org.apache.james.mpt.api.ImapHostSystem; +import org.apache.james.mpt.imapmailbox.postgres.host.PostgresHostSystemExtension; +import org.apache.james.mpt.imapmailbox.suite.Search; +import org.junit.jupiter.api.extension.RegisterExtension; + +public class PostgresSearchTest extends Search { + @RegisterExtension + public static PostgresHostSystemExtension hostSystemExtension = new PostgresHostSystemExtension(); + + @Override + protected ImapHostSystem createImapHostSystem() { + return hostSystemExtension.getHostSystem(); + } +} diff --git a/mpt/impl/imap-mailbox/postgres/src/test/java/org/apache/james/mpt/imapmailbox/postgres/PostgresSecurityTest.java b/mpt/impl/imap-mailbox/postgres/src/test/java/org/apache/james/mpt/imapmailbox/postgres/PostgresSecurityTest.java new file mode 100644 index 00000000000..127147bd141 --- /dev/null +++ b/mpt/impl/imap-mailbox/postgres/src/test/java/org/apache/james/mpt/imapmailbox/postgres/PostgresSecurityTest.java @@ -0,0 +1,35 @@ +/**************************************************************** + * 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.mpt.imapmailbox.postgres; + +import org.apache.james.mpt.api.ImapHostSystem; +import org.apache.james.mpt.imapmailbox.postgres.host.PostgresHostSystemExtension; +import org.apache.james.mpt.imapmailbox.suite.Security; +import org.junit.jupiter.api.extension.RegisterExtension; + +public class PostgresSecurityTest extends Security { + @RegisterExtension + public static PostgresHostSystemExtension hostSystemExtension = new PostgresHostSystemExtension(); + + @Override + protected ImapHostSystem createImapHostSystem() { + return hostSystemExtension.getHostSystem(); + } +} diff --git a/mpt/impl/imap-mailbox/postgres/src/test/java/org/apache/james/mpt/imapmailbox/postgres/PostgresSelectTest.java b/mpt/impl/imap-mailbox/postgres/src/test/java/org/apache/james/mpt/imapmailbox/postgres/PostgresSelectTest.java new file mode 100644 index 00000000000..246023b1d13 --- /dev/null +++ b/mpt/impl/imap-mailbox/postgres/src/test/java/org/apache/james/mpt/imapmailbox/postgres/PostgresSelectTest.java @@ -0,0 +1,35 @@ +/**************************************************************** + * 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.mpt.imapmailbox.postgres; + +import org.apache.james.mpt.api.ImapHostSystem; +import org.apache.james.mpt.imapmailbox.postgres.host.PostgresHostSystemExtension; +import org.apache.james.mpt.imapmailbox.suite.Select; +import org.junit.jupiter.api.extension.RegisterExtension; + +public class PostgresSelectTest extends Select { + @RegisterExtension + public static PostgresHostSystemExtension hostSystemExtension = new PostgresHostSystemExtension(); + + @Override + protected ImapHostSystem createImapHostSystem() { + return hostSystemExtension.getHostSystem(); + } +} diff --git a/mpt/impl/imap-mailbox/postgres/src/test/java/org/apache/james/mpt/imapmailbox/postgres/PostgresSelectedInboxTest.java b/mpt/impl/imap-mailbox/postgres/src/test/java/org/apache/james/mpt/imapmailbox/postgres/PostgresSelectedInboxTest.java new file mode 100644 index 00000000000..9e6a273d1d4 --- /dev/null +++ b/mpt/impl/imap-mailbox/postgres/src/test/java/org/apache/james/mpt/imapmailbox/postgres/PostgresSelectedInboxTest.java @@ -0,0 +1,35 @@ +/**************************************************************** + * 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.mpt.imapmailbox.postgres; + +import org.apache.james.mpt.api.ImapHostSystem; +import org.apache.james.mpt.imapmailbox.postgres.host.PostgresHostSystemExtension; +import org.apache.james.mpt.imapmailbox.suite.SelectedInbox; +import org.junit.jupiter.api.extension.RegisterExtension; + +public class PostgresSelectedInboxTest extends SelectedInbox { + @RegisterExtension + public static PostgresHostSystemExtension hostSystemExtension = new PostgresHostSystemExtension(); + + @Override + protected ImapHostSystem createImapHostSystem() { + return hostSystemExtension.getHostSystem(); + } +} diff --git a/mpt/impl/imap-mailbox/postgres/src/test/java/org/apache/james/mpt/imapmailbox/postgres/PostgresSelectedStateTest.java b/mpt/impl/imap-mailbox/postgres/src/test/java/org/apache/james/mpt/imapmailbox/postgres/PostgresSelectedStateTest.java new file mode 100644 index 00000000000..85bc13f155e --- /dev/null +++ b/mpt/impl/imap-mailbox/postgres/src/test/java/org/apache/james/mpt/imapmailbox/postgres/PostgresSelectedStateTest.java @@ -0,0 +1,65 @@ +/**************************************************************** + * 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.mpt.imapmailbox.postgres; + +import org.apache.james.mpt.api.ImapHostSystem; +import org.apache.james.mpt.imapmailbox.postgres.host.PostgresHostSystemExtension; +import org.apache.james.mpt.imapmailbox.suite.SelectedState; +import org.junit.jupiter.api.Disabled; +import org.junit.jupiter.api.extension.RegisterExtension; + +public class PostgresSelectedStateTest extends SelectedState { + @RegisterExtension + public static PostgresHostSystemExtension hostSystemExtension = new PostgresHostSystemExtension(); + + @Override + protected ImapHostSystem createImapHostSystem() { + return hostSystemExtension.getHostSystem(); + } + + @Override + public void testCopyITALY() { + } + + @Override + public void testCopyKOREA() { + } + + @Override + public void testCopyUS() { + } + + @Override + public void testUidITALY() { + } + + @Override + public void testUidKOREA() { + } + + @Override + public void testUidUS() { + } + + @Override + @Disabled("SEARCH save date just return empty result for JPA") + public void testSearchSaveDate() { + } +} diff --git a/mpt/impl/imap-mailbox/postgres/src/test/java/org/apache/james/mpt/imapmailbox/postgres/PostgresUidSearchOnIndexTest.java b/mpt/impl/imap-mailbox/postgres/src/test/java/org/apache/james/mpt/imapmailbox/postgres/PostgresUidSearchOnIndexTest.java new file mode 100644 index 00000000000..916938eefe5 --- /dev/null +++ b/mpt/impl/imap-mailbox/postgres/src/test/java/org/apache/james/mpt/imapmailbox/postgres/PostgresUidSearchOnIndexTest.java @@ -0,0 +1,35 @@ +/**************************************************************** + * 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.mpt.imapmailbox.postgres; + +import org.apache.james.mpt.api.ImapHostSystem; +import org.apache.james.mpt.imapmailbox.postgres.host.PostgresHostSystemExtension; +import org.apache.james.mpt.imapmailbox.suite.UidSearchOnIndex; +import org.junit.jupiter.api.extension.RegisterExtension; + +public class PostgresUidSearchOnIndexTest extends UidSearchOnIndex { + @RegisterExtension + public static PostgresHostSystemExtension hostSystemExtension = new PostgresHostSystemExtension(); + + @Override + protected ImapHostSystem createImapHostSystem() { + return hostSystemExtension.getHostSystem(); + } +} diff --git a/mpt/impl/imap-mailbox/postgres/src/test/java/org/apache/james/mpt/imapmailbox/postgres/PostgresUidSearchTest.java b/mpt/impl/imap-mailbox/postgres/src/test/java/org/apache/james/mpt/imapmailbox/postgres/PostgresUidSearchTest.java new file mode 100644 index 00000000000..2f374ca2e4a --- /dev/null +++ b/mpt/impl/imap-mailbox/postgres/src/test/java/org/apache/james/mpt/imapmailbox/postgres/PostgresUidSearchTest.java @@ -0,0 +1,35 @@ +/**************************************************************** + * 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.mpt.imapmailbox.postgres; + +import org.apache.james.mpt.api.ImapHostSystem; +import org.apache.james.mpt.imapmailbox.postgres.host.PostgresHostSystemExtension; +import org.apache.james.mpt.imapmailbox.suite.UidSearch; +import org.junit.jupiter.api.extension.RegisterExtension; + +public class PostgresUidSearchTest extends UidSearch { + @RegisterExtension + public static PostgresHostSystemExtension hostSystemExtension = new PostgresHostSystemExtension(); + + @Override + protected ImapHostSystem createImapHostSystem() { + return hostSystemExtension.getHostSystem(); + } +} diff --git a/mpt/impl/imap-mailbox/postgres/src/test/java/org/apache/james/mpt/imapmailbox/postgres/PostgresUserFlagsSupportTest.java b/mpt/impl/imap-mailbox/postgres/src/test/java/org/apache/james/mpt/imapmailbox/postgres/PostgresUserFlagsSupportTest.java new file mode 100644 index 00000000000..006e41500f7 --- /dev/null +++ b/mpt/impl/imap-mailbox/postgres/src/test/java/org/apache/james/mpt/imapmailbox/postgres/PostgresUserFlagsSupportTest.java @@ -0,0 +1,35 @@ +/**************************************************************** + * 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.mpt.imapmailbox.postgres; + +import org.apache.james.mpt.api.ImapHostSystem; +import org.apache.james.mpt.imapmailbox.postgres.host.PostgresHostSystemExtension; +import org.apache.james.mpt.imapmailbox.suite.UserFlagsSupport; +import org.junit.jupiter.api.extension.RegisterExtension; + +public class PostgresUserFlagsSupportTest extends UserFlagsSupport { + @RegisterExtension + public static PostgresHostSystemExtension hostSystemExtension = new PostgresHostSystemExtension(); + + @Override + protected ImapHostSystem createImapHostSystem() { + return hostSystemExtension.getHostSystem(); + } +} diff --git a/mpt/impl/imap-mailbox/postgres/src/test/java/org/apache/james/mpt/imapmailbox/postgres/host/PostgresHostSystem.java b/mpt/impl/imap-mailbox/postgres/src/test/java/org/apache/james/mpt/imapmailbox/postgres/host/PostgresHostSystem.java new file mode 100644 index 00000000000..7e39378f8ff --- /dev/null +++ b/mpt/impl/imap-mailbox/postgres/src/test/java/org/apache/james/mpt/imapmailbox/postgres/host/PostgresHostSystem.java @@ -0,0 +1,176 @@ +/**************************************************************** + * 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.mpt.imapmailbox.postgres.host; + +import java.time.Clock; +import java.time.Instant; + +import org.apache.james.backends.postgres.PostgresConfiguration; +import org.apache.james.backends.postgres.PostgresExtension; +import org.apache.james.backends.postgres.quota.PostgresQuotaCurrentValueDAO; +import org.apache.james.backends.postgres.quota.PostgresQuotaLimitDAO; +import org.apache.james.blob.api.BlobId; +import org.apache.james.blob.api.BucketName; +import org.apache.james.blob.api.PlainBlobId; +import org.apache.james.blob.memory.MemoryBlobStoreDAO; +import org.apache.james.core.quota.QuotaCountLimit; +import org.apache.james.core.quota.QuotaSizeLimit; +import org.apache.james.events.EventBusTestFixture; +import org.apache.james.events.InVMEventBus; +import org.apache.james.events.MemoryEventDeadLetters; +import org.apache.james.events.delivery.InVmEventDelivery; +import org.apache.james.imap.api.process.ImapProcessor; +import org.apache.james.imap.encode.main.DefaultImapEncoderFactory; +import org.apache.james.imap.main.DefaultImapDecoderFactory; +import org.apache.james.imap.processor.main.DefaultImapProcessorFactory; +import org.apache.james.mailbox.AttachmentContentLoader; +import org.apache.james.mailbox.MailboxManager; +import org.apache.james.mailbox.SubscriptionManager; +import org.apache.james.mailbox.acl.MailboxACLResolver; +import org.apache.james.mailbox.acl.UnionMailboxACLResolver; +import org.apache.james.mailbox.postgres.PostgresMailboxManager; +import org.apache.james.mailbox.postgres.PostgresMailboxSessionMapperFactory; +import org.apache.james.mailbox.postgres.PostgresMessageId; +import org.apache.james.mailbox.postgres.quota.PostgresCurrentQuotaManager; +import org.apache.james.mailbox.postgres.quota.PostgresPerUserMaxQuotaManager; +import org.apache.james.mailbox.quota.CurrentQuotaManager; +import org.apache.james.mailbox.store.PreDeletionHooks; +import org.apache.james.mailbox.store.SessionProviderImpl; +import org.apache.james.mailbox.store.StoreMailboxAnnotationManager; +import org.apache.james.mailbox.store.StoreRightManager; +import org.apache.james.mailbox.store.StoreSubscriptionManager; +import org.apache.james.mailbox.store.event.MailboxAnnotationListener; +import org.apache.james.mailbox.store.extractor.DefaultTextExtractor; +import org.apache.james.mailbox.store.mail.NaiveThreadIdGuessingAlgorithm; +import org.apache.james.mailbox.store.mail.model.impl.MessageParser; +import org.apache.james.mailbox.store.quota.DefaultUserQuotaRootResolver; +import org.apache.james.mailbox.store.quota.ListeningCurrentQuotaUpdater; +import org.apache.james.mailbox.store.quota.QuotaComponents; +import org.apache.james.mailbox.store.quota.StoreQuotaManager; +import org.apache.james.mailbox.store.search.MessageSearchIndex; +import org.apache.james.mailbox.store.search.SimpleMessageSearchIndex; +import org.apache.james.metrics.logger.DefaultMetricFactory; +import org.apache.james.metrics.tests.RecordingMetricFactory; +import org.apache.james.mpt.api.ImapFeatures; +import org.apache.james.mpt.api.ImapFeatures.Feature; +import org.apache.james.mpt.host.JamesImapHostSystem; +import org.apache.james.server.blob.deduplication.DeDuplicationBlobStore; +import org.apache.james.utils.UpdatableTickingClock; + +import com.google.common.base.Preconditions; + +public class PostgresHostSystem extends JamesImapHostSystem { + + private static final ImapFeatures SUPPORTED_FEATURES = ImapFeatures.of(Feature.NAMESPACE_SUPPORT, + Feature.USER_FLAGS_SUPPORT, + Feature.ANNOTATION_SUPPORT, + Feature.QUOTA_SUPPORT, + Feature.MOVE_SUPPORT, + Feature.MOD_SEQ_SEARCH); + + + static PostgresHostSystem build(PostgresExtension postgresExtension) { + return new PostgresHostSystem(postgresExtension); + } + + private PostgresPerUserMaxQuotaManager maxQuotaManager; + private PostgresMailboxManager mailboxManager; + private final PostgresExtension postgresExtension; + + public PostgresHostSystem(PostgresExtension postgresExtension) { + this.postgresExtension = postgresExtension; + } + + public void beforeAll() { + Preconditions.checkNotNull(postgresExtension.getConnectionFactory()); + } + + @Override + public void beforeTest() throws Exception { + super.beforeTest(); + + BlobId.Factory blobIdFactory = new PlainBlobId.Factory(); + DeDuplicationBlobStore blobStore = new DeDuplicationBlobStore(new MemoryBlobStoreDAO(), BucketName.DEFAULT, blobIdFactory); + + PostgresMailboxSessionMapperFactory mapperFactory = new PostgresMailboxSessionMapperFactory(postgresExtension.getExecutorFactory(), Clock.systemUTC(), blobStore, blobIdFactory, + PostgresConfiguration.builder().username("a").password("a").build()); + + MailboxACLResolver aclResolver = new UnionMailboxACLResolver(); + MessageParser messageParser = new MessageParser(); + + + InVMEventBus eventBus = new InVMEventBus(new InVmEventDelivery(new RecordingMetricFactory()), EventBusTestFixture.RETRY_BACKOFF_CONFIGURATION, new MemoryEventDeadLetters()); + StoreRightManager storeRightManager = new StoreRightManager(mapperFactory, aclResolver, eventBus); + StoreMailboxAnnotationManager annotationManager = new StoreMailboxAnnotationManager(mapperFactory, storeRightManager); + SessionProviderImpl sessionProvider = new SessionProviderImpl(authenticator, authorizator); + DefaultUserQuotaRootResolver quotaRootResolver = new DefaultUserQuotaRootResolver(sessionProvider, mapperFactory); + CurrentQuotaManager currentQuotaManager = new PostgresCurrentQuotaManager(new PostgresQuotaCurrentValueDAO(postgresExtension.getDefaultPostgresExecutor())); + maxQuotaManager = new PostgresPerUserMaxQuotaManager(new PostgresQuotaLimitDAO(postgresExtension.getDefaultPostgresExecutor())); + StoreQuotaManager storeQuotaManager = new StoreQuotaManager(currentQuotaManager, maxQuotaManager); + ListeningCurrentQuotaUpdater quotaUpdater = new ListeningCurrentQuotaUpdater(currentQuotaManager, quotaRootResolver, eventBus, storeQuotaManager); + QuotaComponents quotaComponents = new QuotaComponents(maxQuotaManager, storeQuotaManager, quotaRootResolver); + AttachmentContentLoader attachmentContentLoader = null; + MessageSearchIndex index = new SimpleMessageSearchIndex(mapperFactory, mapperFactory, new DefaultTextExtractor(), attachmentContentLoader); + + mailboxManager = new PostgresMailboxManager(mapperFactory, sessionProvider, messageParser, + new PostgresMessageId.Factory(), + eventBus, annotationManager, storeRightManager, quotaComponents, index, new NaiveThreadIdGuessingAlgorithm(), + PreDeletionHooks.NO_PRE_DELETION_HOOK, new UpdatableTickingClock(Instant.now())); + + eventBus.register(quotaUpdater); + eventBus.register(new MailboxAnnotationListener(mapperFactory, sessionProvider)); + + SubscriptionManager subscriptionManager = new StoreSubscriptionManager(mapperFactory, mapperFactory, eventBus); + + ImapProcessor defaultImapProcessorFactory = + DefaultImapProcessorFactory.createDefaultProcessor( + mailboxManager, + eventBus, + subscriptionManager, + storeQuotaManager, + quotaRootResolver, + new DefaultMetricFactory()); + + configure(new DefaultImapDecoderFactory().buildImapDecoder(), + new DefaultImapEncoderFactory().buildImapEncoder(), + defaultImapProcessorFactory); + } + + @Override + protected MailboxManager getMailboxManager() { + return mailboxManager; + } + + @Override + public boolean supports(Feature... features) { + return SUPPORTED_FEATURES.supports(features); + } + + @Override + public void setQuotaLimits(QuotaCountLimit maxMessageQuota, QuotaSizeLimit maxStorageQuota) { + maxQuotaManager.setGlobalMaxMessage(maxMessageQuota); + maxQuotaManager.setGlobalMaxStorage(maxStorageQuota); + } + + @Override + protected void await() { + + } +} diff --git a/mpt/impl/imap-mailbox/postgres/src/test/java/org/apache/james/mpt/imapmailbox/postgres/host/PostgresHostSystemExtension.java b/mpt/impl/imap-mailbox/postgres/src/test/java/org/apache/james/mpt/imapmailbox/postgres/host/PostgresHostSystemExtension.java new file mode 100644 index 00000000000..9e53890c032 --- /dev/null +++ b/mpt/impl/imap-mailbox/postgres/src/test/java/org/apache/james/mpt/imapmailbox/postgres/host/PostgresHostSystemExtension.java @@ -0,0 +1,87 @@ +/**************************************************************** + * 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.mpt.imapmailbox.postgres.host; + +import org.apache.james.backends.postgres.PostgresExtension; +import org.apache.james.backends.postgres.PostgresModule; +import org.apache.james.backends.postgres.quota.PostgresQuotaModule; +import org.apache.james.mailbox.postgres.PostgresMailboxAggregateModule; +import org.apache.james.mpt.host.JamesImapHostSystem; +import org.junit.jupiter.api.extension.AfterAllCallback; +import org.junit.jupiter.api.extension.AfterEachCallback; +import org.junit.jupiter.api.extension.BeforeAllCallback; +import org.junit.jupiter.api.extension.BeforeEachCallback; +import org.junit.jupiter.api.extension.ExtensionContext; +import org.junit.jupiter.api.extension.ParameterContext; +import org.junit.jupiter.api.extension.ParameterResolutionException; +import org.junit.jupiter.api.extension.ParameterResolver; + +public class PostgresHostSystemExtension implements BeforeEachCallback, AfterEachCallback, BeforeAllCallback, AfterAllCallback, ParameterResolver { + private final PostgresHostSystem hostSystem; + private final PostgresExtension postgresExtension; + + public PostgresHostSystemExtension() { + this.postgresExtension = PostgresExtension.withoutRowLevelSecurity(PostgresModule.aggregateModules( + PostgresMailboxAggregateModule.MODULE, + PostgresQuotaModule.MODULE)); + try { + hostSystem = PostgresHostSystem.build(postgresExtension); + } catch (Exception e) { + throw new RuntimeException(e); + } + } + + @Override + public void afterEach(ExtensionContext extensionContext) throws Exception { + postgresExtension.afterEach(extensionContext); + hostSystem.afterTest(); + } + + @Override + public void beforeEach(ExtensionContext extensionContext) throws Exception { + postgresExtension.beforeEach(extensionContext); + hostSystem.beforeTest(); + } + + public JamesImapHostSystem getHostSystem() { + return hostSystem; + } + + @Override + public void afterAll(ExtensionContext extensionContext) throws Exception { + postgresExtension.afterAll(extensionContext); + } + + @Override + public void beforeAll(ExtensionContext extensionContext) throws Exception { + postgresExtension.beforeAll(extensionContext); + hostSystem.beforeAll(); + } + + @Override + public boolean supportsParameter(ParameterContext parameterContext, ExtensionContext extensionContext) throws ParameterResolutionException { + return false; + } + + @Override + public Object resolveParameter(ParameterContext parameterContext, ExtensionContext extensionContext) throws ParameterResolutionException { + return postgresExtension; + } +} diff --git a/mpt/impl/smtp/cassandra-rabbitmq-object-storage/pom.xml b/mpt/impl/smtp/cassandra-rabbitmq-object-storage/pom.xml index 3b9dde21e60..e35e668b22f 100644 --- a/mpt/impl/smtp/cassandra-rabbitmq-object-storage/pom.xml +++ b/mpt/impl/smtp/cassandra-rabbitmq-object-storage/pom.xml @@ -97,12 +97,26 @@ test-jar test + + ${james.groupId} + james-server-guice-opensearch + ${project.version} + test-jar + test + ${james.groupId} james-server-util test-jar test + + ${james.groupId} + queue-rabbitmq-guice + ${project.version} + test-jar + test + ${james.groupId} testing-base diff --git a/mpt/impl/smtp/cassandra/pom.xml b/mpt/impl/smtp/cassandra/pom.xml index e9ded6448d5..73eed99f44a 100644 --- a/mpt/impl/smtp/cassandra/pom.xml +++ b/mpt/impl/smtp/cassandra/pom.xml @@ -69,6 +69,13 @@ test-jar test + + ${james.groupId} + james-server-guice-opensearch + ${project.version} + test-jar + test + ${james.groupId} james-server-util diff --git a/pom.xml b/pom.xml index 86ce8689793..fafe7c72d13 100644 --- a/pom.xml +++ b/pom.xml @@ -702,6 +702,17 @@ ${project.version} test-jar + + ${james.groupId} + apache-james-backends-postgres + ${project.version} + + + ${james.groupId} + apache-james-backends-postgres + ${project.version} + test-jar + ${james.groupId} apache-james-backends-pulsar @@ -784,6 +795,11 @@ apache-james-mailbox-deleted-messages-vault-cassandra ${project.version} + + ${james.groupId} + apache-james-mailbox-deleted-messages-vault-postgres + ${project.version} + ${james.groupId} apache-james-mailbox-event-json @@ -833,6 +849,17 @@ ${project.version} test-jar + + ${james.groupId} + apache-james-mailbox-postgres + ${project.version} + + + ${james.groupId} + apache-james-mailbox-postgres + ${project.version} + test-jar + ${james.groupId} apache-james-mailbox-quota-mailing @@ -1148,6 +1175,16 @@ blob-memory-guice ${project.version} + + ${james.groupId} + blob-postgres + ${project.version} + + + ${james.groupId} + blob-postgres-guice + ${project.version} + ${james.groupId} blob-s3 @@ -1186,6 +1223,11 @@ dead-letter-cassandra ${project.version} + + ${james.groupId} + dead-letter-postgres + ${project.version} + ${james.groupId} event-bus-api @@ -1418,6 +1460,17 @@ ${project.version} test-jar + + ${james.groupId} + james-server-data-postgres + ${project.version} + + + ${james.groupId} + james-server-data-postgres + ${project.version} + test-jar + ${james.groupId} james-server-deleted-messages-vault @@ -1546,6 +1599,17 @@ ${project.version} test-jar + + ${james.groupId} + james-server-guice-sieve-postgres + ${project.version} + + + ${james.groupId} + james-server-guice-sieve-postgres + ${project.version} + test-jar + ${james.groupId} james-server-guice-smtp @@ -1681,6 +1745,17 @@ james-server-onami ${project.version} + + ${james.groupId} + james-server-postgres-app + ${project.version} + + + ${james.groupId} + james-server-postgres-app + ${project.version} + test-jar + ${james.groupId} james-server-protocols-imap4 @@ -1853,6 +1928,17 @@ ${project.version} test-jar + + ${james.groupId} + james-server-task-postgres + ${project.version} + + + ${james.groupId} + james-server-task-postgres + ${project.version} + test-jar + ${james.groupId} james-server-testing @@ -2915,6 +3001,11 @@ junit-jupiter ${testcontainers.version} + + org.testcontainers + postgresql + 1.19.8 + org.testcontainers pulsar diff --git a/server/apps/cassandra-app/pom.xml b/server/apps/cassandra-app/pom.xml index 36ddca71d4e..5d8a1f987a2 100644 --- a/server/apps/cassandra-app/pom.xml +++ b/server/apps/cassandra-app/pom.xml @@ -171,6 +171,12 @@ ${james.groupId} james-server-guice-opensearch + + ${james.groupId} + james-server-guice-opensearch + test-jar + test + ${james.groupId} james-server-guice-pop diff --git a/server/apps/cassandra-app/src/test/java/org/apache/james/JamesWithNonCompatibleElasticSearchServerTest.java b/server/apps/cassandra-app/src/test/java/org/apache/james/JamesWithNonCompatibleElasticSearchServerTest.java index 31699b491f6..543a0f941e8 100644 --- a/server/apps/cassandra-app/src/test/java/org/apache/james/JamesWithNonCompatibleElasticSearchServerTest.java +++ b/server/apps/cassandra-app/src/test/java/org/apache/james/JamesWithNonCompatibleElasticSearchServerTest.java @@ -29,6 +29,7 @@ import org.apache.james.modules.mailbox.OpenSearchStartUpCheck; import org.apache.james.util.docker.Images; import org.junit.jupiter.api.AfterAll; +import org.junit.jupiter.api.Disabled; import org.junit.jupiter.api.Test; import org.junit.jupiter.api.extension.RegisterExtension; @@ -51,6 +52,7 @@ static void afterAll() { } @Test + @Disabled("test failed, and CassandraJamesServerMain was mark as deprecated, and will be removed in the future.") void jamesShouldStopWhenStartingWithANonCompatibleElasticSearchServer(GuiceJamesServer server) throws Exception { assertThatThrownBy(server::start) .isInstanceOfSatisfying( diff --git a/server/apps/distributed-app/pom.xml b/server/apps/distributed-app/pom.xml index 7e98b1af44d..379b406d714 100644 --- a/server/apps/distributed-app/pom.xml +++ b/server/apps/distributed-app/pom.xml @@ -216,6 +216,12 @@ ${james.groupId} james-server-guice-opensearch + + ${james.groupId} + james-server-guice-opensearch + test-jar + test + ${james.groupId} james-server-guice-pop @@ -287,6 +293,12 @@ ${james.groupId} james-server-webadmin-rabbitmq + + ${james.groupId} + queue-rabbitmq-guice + test-jar + test + ${james.groupId} queue-rabbitmq-guice diff --git a/server/apps/distributed-pop3-app/pom.xml b/server/apps/distributed-pop3-app/pom.xml index 44b37d9ef08..b7546c19cfe 100644 --- a/server/apps/distributed-pop3-app/pom.xml +++ b/server/apps/distributed-pop3-app/pom.xml @@ -209,6 +209,12 @@ ${james.groupId} james-server-guice-opensearch + + ${james.groupId} + james-server-guice-opensearch + test-jar + test + ${james.groupId} james-server-guice-pop @@ -285,6 +291,12 @@ ${james.groupId} james-server-webadmin-rabbitmq + + ${james.groupId} + queue-rabbitmq-guice + test-jar + test + ${james.groupId} queue-rabbitmq-guice diff --git a/server/apps/postgres-app/README.adoc b/server/apps/postgres-app/README.adoc new file mode 100644 index 00000000000..c3c9e084d52 --- /dev/null +++ b/server/apps/postgres-app/README.adoc @@ -0,0 +1,157 @@ += Guice-Postgres Server How-to + +This server targets reactive James deployments with postgresql database. + +== Requirements + +* Java 11 SDK + +=== With Postgresql only + +Firstly, create your own user network on Docker for the James environment: + + $ docker network create --driver bridge james + +Third party compulsory dependencies: + +* Postgresql 16.0 + +[source] +---- +$ docker run -d --network james -p 5432:5432 --name=postgres --env 'POSTGRES_DB=james' --env 'POSTGRES_USER=james' --env 'POSTGRES_PASSWORD=secret1' postgres:16.0 +---- + +=== Distributed version + +Here you have the choice of using other third party softwares to handle object data storage, search indexing and event bus. + +For now, dependencies supported are: + +* OpenSearch 2.8.0 + +[source] +---- +$ docker run -d --network james -p 9200:9200 --name=opensearch --env 'discovery.type=single-node' opensearchproject/opensearch:2.8.0 +---- + +* Zenko Cloudserver or AWS S3 + +[source] +---- +$ docker run -d --network james --env 'REMOTE_MANAGEMENT_DISABLE=1' --env 'SCALITY_ACCESS_KEY_ID=accessKey1' --env 'SCALITY_SECRET_ACCESS_KEY=secretKey1' --name=s3 registry.scality.com/cloudserver/cloudserver:8.7.25 +---- + +* RabbitMQ 3.12.1 + +[source] +---- +$ docker run -d --network james -p 5672:5672 -p 15672:15672 --name=rabbitmq rabbitmq:3.12.1-management +---- + +== Running manually + +=== Running with Postgresql only + +To run James manually, you have to create a directory containing required configuration files. + +James requires the configuration to be in a subfolder of working directory that is called +**conf**. A [sample directory](https://github.com/apache/james-project/tree/master/server/apps/postgres-app/sample-configuration) +is provided with some default values you may need to replace. You will need to update its content to match your needs. + +Also you might need to add the files like in the +[sample directory](https://github.com/apache/james-project/tree/master/server/apps/postgres-app/sample-configuration-single) +to not have OpenSearch indexing enabled by default for the search. + +You also need to generate a keystore with the following command: + +[source] +---- +$ keytool -genkey -alias james -keyalg RSA -keystore conf/keystore +---- + +Once everything is set up, you just have to run the jar with: + +[source] +---- +$ java -Dworking.directory=. -Djdk.tls.ephemeralDHKeySize=2048 -Dlogback.configurationFile=conf/logback.xml -jar james-server-postgres-app.jar +---- + +In the case of quick start James without manually creating a keystore (e.g. for development), just input the command argument +`--generate-keystore` when running, James will auto-generate keystore file with the default setting that is declared in +`jmap.properties` (tls.keystoreURL, tls.secret). + +[source] +---- +$ java -Dworking.directory=. -Dlogback.configurationFile=conf/logback.xml -Djdk.tls.ephemeralDHKeySize=2048 -jar james-server-postgres-app.jar --generate-keystore +---- + +Note that binding ports below 1024 requires administrative rights. + +=== Running distributed + +If you want to use the distributed version of James Postgres app, you will need to add configuration in the **conf** folder like in the +[sample directory](https://github.com/apache/james-project/tree/master/server/apps/postgres-app/sample-configuration-distributed). + +Then you need to generate the keystore, rebuild the application jar and run it like above. + +== Docker compose + +To import the image locally: + +[source] +---- +docker image load -i target/jib-image.tar +---- + +=== With Postgresql only + +We have a docker compose for running James Postgresql app alongside Postgresql. To run it, simply type: + +.... +docker compose up -d +.... + +=== Distributed + +We also have a distributed version of the James postgresql app with: + +- OpenSearch as a search indexer +- S3 as the object storage +- RabbitMQ as the event bus + +To run it, simply type: + +.... +docker compose -f docker-compose-distributed.yml up -d +.... + +== Administration Operations +=== Clean up data + +To clean up some specific data, that will never be used again after a long time, you can execute the SQL queries `clean_up.sql`. +The never used data are: +- mailbox_change +- email_change +- vacation_notification_registry + +## Development + +### How to track the stats of the statement execution + +Using the [`pg_stat_statements` extension](https://www.postgresql.org/docs/current/pgstatstatements.html), you can track the stats of the statement execution. To install it, you can execute the following SQL query: + +```sql +create extension if not exists pg_stat_statements; +alter system set shared_preload_libraries='pg_stat_statements'; + +-- restart postgres +-- optional +alter system set pg_stat_statements.max = 100000; +alter system set pg_stat_statements.track = 'all'; +``` + +Then you can query the stats of the statement execution by executing the following SQL query: + +```sql +select query, mean_exec_time, total_exec_time, calls from pg_stat_statements order by total_exec_time desc; +``` diff --git a/server/apps/postgres-app/clean_up.sql b/server/apps/postgres-app/clean_up.sql new file mode 100644 index 00000000000..c0f8f0b8432 --- /dev/null +++ b/server/apps/postgres-app/clean_up.sql @@ -0,0 +1,26 @@ +-- This is a script to delete old rows from some tables. One of the attempts to clean up the never-used data after a long time. + +DO +$$ + DECLARE + days_to_keep INTEGER; + BEGIN + -- Set the number of days dynamically + days_to_keep := 60; + + -- Delete rows older than the specified number of days in email_change + DELETE + FROM email_change + WHERE date < current_timestamp - interval '1 day' * days_to_keep; + + -- Delete rows older than the specified number of days in mailbox_change + DELETE + FROM email_change + WHERE date < current_timestamp - interval '1 day' * days_to_keep; + + -- Delete outdated vacation notifications (older than the current UTC timestamp) + DELETE + FROM vacation_notification_registry + WHERE expiry_date < CURRENT_TIMESTAMP AT TIME ZONE 'UTC'; + END +$$; \ No newline at end of file diff --git a/server/apps/postgres-app/docker-compose-distributed.yml b/server/apps/postgres-app/docker-compose-distributed.yml new file mode 100644 index 00000000000..67d5df8c3be --- /dev/null +++ b/server/apps/postgres-app/docker-compose-distributed.yml @@ -0,0 +1,83 @@ +version: '3' + +services: + + james: + depends_on: + postgres: + condition: service_started + opensearch: + condition: service_healthy + s3: + condition: service_started + rabbitmq: + condition: service_started + image: apache/james:postgres-latest + container_name: james + hostname: james.local + command: + - --generate-keystore + ports: + - "80:80" + - "25:25" + - "110:110" + - "143:143" + - "465:465" + - "587:587" + - "993:993" + - "8000:8000" + volumes: + - ./sample-configuration-distributed/opensearch.properties:/root/conf/opensearch.properties + - ./sample-configuration-distributed/blob.properties:/root/conf/blob.properties + - ./sample-configuration-distributed/rabbitmq.properties:/root/conf/rabbitmq.properties + networks: + - james + + opensearch: + image: opensearchproject/opensearch:2.8.0 + container_name: opensearch + healthcheck: + test: curl -s http://opensearch:9200 >/dev/null || exit 1 + interval: 3s + timeout: 10s + retries: 5 + environment: + - discovery.type=single-node + - DISABLE_INSTALL_DEMO_CONFIG=true + - DISABLE_SECURITY_PLUGIN=true + networks: + - james + + postgres: + image: postgres:16.3 + container_name: postgres + ports: + - "5432:5432" + environment: + - POSTGRES_DB=james + - POSTGRES_USER=james + - POSTGRES_PASSWORD=secret1 + networks: + - james + + s3: + image: registry.scality.com/cloudserver/cloudserver:8.7.25 + container_name: s3.docker.test + environment: + - SCALITY_ACCESS_KEY_ID=accessKey1 + - SCALITY_SECRET_ACCESS_KEY=secretKey1 + - LOG_LEVEL=trace + - REMOTE_MANAGEMENT_DISABLE=1 + networks: + - james + + rabbitmq: + image: rabbitmq:3.12.1-management + ports: + - "5672:5672" + - "15672:15672" + networks: + - james + +networks: + james: \ No newline at end of file diff --git a/server/apps/postgres-app/docker-compose.yml b/server/apps/postgres-app/docker-compose.yml new file mode 100644 index 00000000000..9fcef9e03c2 --- /dev/null +++ b/server/apps/postgres-app/docker-compose.yml @@ -0,0 +1,33 @@ +version: '3' + +services: + + james: + depends_on: + - postgres + image: apache/james:postgres-latest + container_name: james + hostname: james.local + command: + - --generate-keystore + ports: + - "80:80" + - "25:25" + - "110:110" + - "143:143" + - "465:465" + - "587:587" + - "993:993" + - "8000:8000" + volumes: + - ./sample-configuration-single/search.properties:/root/conf/search.properties + - ./sample-configuration/blob.properties:/root/conf/blob.properties + + postgres: + image: postgres:16.3 + ports: + - "5432:5432" + environment: + - POSTGRES_DB=james + - POSTGRES_USER=james + - POSTGRES_PASSWORD=secret1 \ No newline at end of file diff --git a/server/apps/postgres-app/docker-configuration/webadmin.properties b/server/apps/postgres-app/docker-configuration/webadmin.properties new file mode 100644 index 00000000000..5d72d99b744 --- /dev/null +++ b/server/apps/postgres-app/docker-configuration/webadmin.properties @@ -0,0 +1,54 @@ +# 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. + +# This template file can be used as example for James Server configuration +# DO NOT USE IT AS SUCH AND ADAPT IT TO YOUR NEEDS + +# Read https://james.apache.org/server/config-webadmin.html for further details + +enabled=true +port=8000 +host=0.0.0.0 + +# Defaults to false +https.enabled=false + +# Compulsory when enabling HTTPS +#https.keystore=/path/to/keystore +#https.password=password + +# Optional when enabling HTTPS (self signed) +#https.trust.keystore +#https.trust.password + +# Defaults to false +#jwt.enabled=true +# +## If you wish to use OAuth authentication, you should provide a valid JWT public key. +## The following entry specify the link to the URL of the public key file, +## which should be a PEM format file. +## +#jwt.publickeypem.url=file://conf/jwt_publickey + +# Defaults to false +#cors.enable=true +#cors.origin + +# List of fully qualified class names that should be exposed over webadmin +# in addition to your product default routes. Routes needs to be located +# within the classpath or in the ./extensions-jars folder. +#extensions.routes= \ No newline at end of file diff --git a/server/apps/postgres-app/imap-provision-conf/provisioning.properties b/server/apps/postgres-app/imap-provision-conf/provisioning.properties new file mode 100644 index 00000000000..e2f27130e8c --- /dev/null +++ b/server/apps/postgres-app/imap-provision-conf/provisioning.properties @@ -0,0 +1,25 @@ +# IMAP (S) URL of the James server. Certificates are blindly trusted +url=imaps://localhost:993 + +# Count of mailboxes to create per user +mailbox.count=4 +# Count of messages to create per folder +message.per.folder.count=5 +# Count of messages to create in INBOX +message.inbox.count=5 + +# Count of threads of the IMAP client +thread.count=8 +# Concurrent count of users to provision simultaneously +concurrent.user.count=10 +# Connections to use per user +connection.per.user.count=2 +# Read timeout of IMAP connections. +read.timeout.ms=180000 +# Connect timeout +connect.timeout.ms=30000 + +# Count of users to offset (ignore) in the provisioning. +users.offset=0 +# Count of users to provision +# users.limit=100 \ No newline at end of file diff --git a/server/apps/postgres-app/performance-test.md b/server/apps/postgres-app/performance-test.md new file mode 100644 index 00000000000..07fea625032 --- /dev/null +++ b/server/apps/postgres-app/performance-test.md @@ -0,0 +1,11 @@ +# Performance test Postgres app + +To provision and benchmark an IMAP server backed by PostgreSQL, please have a look at following steps: +1. Build and extract the Postgres app docker image. + - `mvn clean install -DskipTests -Dmaven.skip.doc=true` + - `docker load -i ./target/jib-image.tar` +2. Run the Postgres app: `docker compose up` +3. Provision users and IMAP mailboxes + messages: `./provision.sh` +4. Performance test IMAP server using [james-gatling](https://github.com/linagora/james-gatling) + + Sample IMAP simulation: `gatling:testOnly org.apache.james.gatling.simulation.imap.PlatformValidationSimulation`. \ No newline at end of file diff --git a/server/apps/postgres-app/pom.xml b/server/apps/postgres-app/pom.xml new file mode 100644 index 00000000000..940f9d92a06 --- /dev/null +++ b/server/apps/postgres-app/pom.xml @@ -0,0 +1,519 @@ + + + + 4.0.0 + + org.apache.james + james-server + 3.9.0-SNAPSHOT + ../../pom.xml + + + james-server-postgres-app + jar + Apache James :: Server :: Postgres - Application + + + + + + ${james.groupId} + james-server-guice + ${project.version} + pom + import + + + + + + + ${james.groupId} + apache-james-backends-opensearch + test-jar + test + + + ${james.groupId} + apache-james-backends-postgres + test-jar + test + + + ${james.groupId} + apache-james-backends-rabbitmq + test-jar + test + + + ${james.groupId} + apache-james-mailbox-opensearch + test-jar + test + + + ${james.groupId} + apache-james-mailbox-postgres + test-jar + test + + + ${james.groupId} + apache-james-mailbox-quota-search-scanning + + + ${james.groupId} + apache-james-mailbox-tika + test-jar + test + + + ${james.groupId} + blob-s3 + test-jar + test + + + ${james.groupId} + blob-s3-guice + test-jar + test + + + ${james.groupId} + james-server-cli + runtime + + + ${james.groupId} + james-server-data-jmap-postgres + ${project.version} + + + ${james.groupId} + james-server-data-ldap + test-jar + test + + + ${james.groupId} + james-server-data-postgres + + + ${james.groupId} + james-server-guice-common + + + ${james.groupId} + james-server-guice-common + test-jar + test + + + ${james.groupId} + james-server-guice-data-ldap + + + ${james.groupId} + james-server-guice-data-ldap + test-jar + test + + + ${james.groupId} + james-server-guice-imap + + + ${james.groupId} + james-server-guice-jmap + test-jar + test + + + ${james.groupId} + james-server-guice-jmx + + + ${james.groupId} + james-server-guice-lmtp + + + ${james.groupId} + james-server-guice-mailbox + + + ${james.groupId} + james-server-guice-mailbox-postgres + + + ${james.groupId} + james-server-guice-managedsieve + + + ${james.groupId} + james-server-guice-memory + + + ${james.groupId} + james-server-guice-opensearch + + + ${james.groupId} + james-server-guice-opensearch + test-jar + test + + + ${james.groupId} + james-server-guice-pop + + + ${james.groupId} + james-server-guice-sieve-postgres + + + ${james.groupId} + james-server-guice-smtp + + + ${james.groupId} + james-server-guice-webadmin + + + ${james.groupId} + james-server-guice-webadmin-data + + + ${james.groupId} + james-server-guice-webadmin-jmap + + + ${james.groupId} + james-server-guice-webadmin-mailbox + + + ${james.groupId} + james-server-guice-webadmin-mailqueue + + + ${james.groupId} + james-server-guice-webadmin-mailrepository + + + ${james.groupId} + james-server-mailbox-adapter + + + ${james.groupId} + james-server-mailets + + + ${james.groupId} + james-server-postgres-common-guice + + + ${james.groupId} + james-server-postgres-common-guice + test-jar + test + + + ${james.groupId} + james-server-testing + test + + + ${james.groupId} + james-server-webadmin-core + test-jar + test + + + ${james.groupId} + queue-activemq-guice + + + ${james.groupId} + queue-rabbitmq-guice + ${project.version} + test-jar + + + ${james.groupId} + testing-base + test + + + ch.qos.logback + logback-classic + + + com.linagora + logback-elasticsearch-appender + + + io.rest-assured + rest-assured + test + + + org.awaitility + awaitility + + + org.mockito + mockito-core + test + + + org.testcontainers + postgresql + 1.19.1 + test + + + + + + + com.googlecode.maven-download-plugin + download-maven-plugin + + + install-glowroot + + wget + + package + + https://github.com/glowroot/glowroot/releases/download/v0.14.0/glowroot-0.14.0-dist.zip + true + ${project.build.directory} + 16073f10204751cd71d3b4ea93be2649 + + + + + + org.apache.maven.plugins + maven-resources-plugin + + + copy-glowroot-resources + + copy-resources + + package + + ${basedir}/target/glowroot + + + src/main/glowroot + true + + + + + + + + com.google.cloud.tools + jib-maven-plugin + + + eclipse-temurin:21-jre-jammy + + + apache/james + + postgres-latest + + + + org.apache.james.PostgresJamesServerMain + + 80 + + 143 + + 993 + + 25 + + 465 + + 587 + + 4000 + + 8000 + + + /root + + -Dlogback.configurationFile=/root/conf/logback.xml + -Dworking.directory=/root/ + + -Djdk.tls.ephemeralDHKeySize=2048 + -Dextra.props=/root/conf/jvm.properties + + USE_CURRENT_TIMESTAMP + + /logs + /root/conf + /root/extensions-jars + /root/glowroot/plugins + /root/glowroot/data + + /root/var + + + + + + sample-configuration + /root/conf + + + docker-configuration + /root/conf + + + src/main/scripts + /usr/bin + + + target/glowroot + /root + + + src/main/extensions-jars + /root/extensions-jars + + + + + /usr/bin/james-cli + 755 + + + + + + + + + buildTar + + package + + + + + org.apache.maven.plugins + maven-surefire-plugin + + false + + 1C + -Djava.library.path= + -javaagent:"${settings.localRepository}"/org/jacoco/org.jacoco.agent/${jacoco-maven-plugin.version}/org.jacoco.agent-${jacoco-maven-plugin.version}-runtime.jar=destfile=${basedir}/target/jacoco.exec + -Xms512m -Xmx1024m + + + + org.apache.maven.plugins + maven-dependency-plugin + + + copy-dependencies + + copy-dependencies + + package + + compile + runtime + ${project.build.directory}/${project.artifactId}.lib + + + + + + org.apache.maven.plugins + maven-jar-plugin + + + default-jar + + jar + + + ${project.artifactId} + + + true + ${project.artifactId}.lib/ + org.apache.james.PostgresJamesServerMain + false + + + Apache James Postgres server Application + ${project.version} + The Apache Software Foundation + Apache James Postgres server Application + ${project.version} + The Apache Software Foundation + org.apache + https://james.apache.org/server + + + + + + test-jar + + test-jar + + + + + + maven-assembly-plugin + + src/assemble/ + gnu + false + james-server-postgres-app + + + + make-assembly + + single + + package + + + + + + + diff --git a/server/apps/postgres-app/provision.sh b/server/apps/postgres-app/provision.sh new file mode 100755 index 00000000000..9a62d68dfdc --- /dev/null +++ b/server/apps/postgres-app/provision.sh @@ -0,0 +1,40 @@ +#!/bin/bash + +export WEBADMIN_BASE_URL="http://localhost:8000" +export DOMAIN_NAME="domain.org" +export USERS_COUNT=1000 + +echo "Start provisioning users." + +user_file="./imap-provision-conf/users.csv" + +# Remove old users.csv file +if [ -e "$user_file" ]; then + echo "Removing old users.csv file" + rm $user_file +fi + +# Create domain +curl -X PUT ${WEBADMIN_BASE_URL}/domains/${DOMAIN_NAME} + +for i in $(seq 1 $USERS_COUNT) +do + # Create user + echo "Creating user $i" + username=user${i}@$DOMAIN_NAME + curl -XPUT ${WEBADMIN_BASE_URL}/users/$username \ + -d '{"password":"secret"}' \ + -H "Content-Type: application/json" + + # Append user to users.csv + echo -e "$username,secret" >> $user_file +done + +echo "Finished provisioning users." + +# Provisioning IMAP mailboxes and messages. +echo "Start provisioning IMAP mailboxes and messages..." +docker run --rm -it --name james-provisioning --network host -v ./imap-provision-conf/provisioning.properties:/conf/provisioning.properties \ +-v $user_file:/conf/users.csv linagora/james-provisioning:latest +echo "Finished provisioning IMAP mailboxes and messages." + diff --git a/server/apps/postgres-app/sample-configuration-distributed/blob.properties b/server/apps/postgres-app/sample-configuration-distributed/blob.properties new file mode 100644 index 00000000000..0e761637054 --- /dev/null +++ b/server/apps/postgres-app/sample-configuration-distributed/blob.properties @@ -0,0 +1,104 @@ +# ============================================= BlobStore Implementation ================================== +# Read https://james.apache.org/server/config-blobstore.html for further details + +# Choose your BlobStore implementation +# Mandatory, allowed values are: file, s3, postgres. +implementation=s3 + +# ========================================= Deduplication ======================================== +# If you choose to enable deduplication, the mails with the same content will be stored only once. +# Warning: Once this feature is enabled, there is no turning back as turning it off will lead to the deletion of all +# the mails sharing the same content once one is deleted. +# Mandatory, Allowed values are: true, false +deduplication.enable=true + +# deduplication.family needs to be incremented every time the deduplication.generation.duration is changed +# Positive integer, defaults to 1 +# deduplication.gc.generation.family=1 + +# Duration of generation. +# Deduplication only takes place within a singe generation. +# Only items two generation old can be garbage collected. (This prevent concurrent insertions issues and +# accounts for a clock skew). +# deduplication.family needs to be incremented everytime this parameter is changed. +# Duration. Default unit: days. Defaults to 30 days. +# deduplication.gc.generation.duration=30days + +# ========================================= Encryption ======================================== +# If you choose to enable encryption, the blob content will be encrypted before storing them in the BlobStore. +# Warning: Once this feature is enabled, there is no turning back as turning it off will lead to all content being +# encrypted. This comes at a performance impact but presents you from leaking data if, for instance the third party +# offering you a S3 service is compromised. +# Optional, Allowed values are: true, false, defaults to false +encryption.aes.enable=false + +# Mandatory (if AES encryption is enabled) salt and password. Salt needs to be an hexadecimal encoded string +#encryption.aes.password=xxx +#encryption.aes.salt=73616c7479 +# Optional, defaults to PBKDF2WithHmacSHA512 +#encryption.aes.private.key.algorithm=PBKDF2WithHmacSHA512 + +# ============================================== ObjectStorage ============================================ + +# ========================================= ObjectStorage Buckets ========================================== +# bucket names prefix +# Optional, default no prefix +# objectstorage.bucketPrefix=prod- + +# Default bucket name +# Optional, default is bucketPrefix + `default` +# objectstorage.namespace=james + +# ========================================= ObjectStorage on S3 ============================================= +# Mandatory if you choose s3 storage service, S3 authentication endpoint +objectstorage.s3.endPoint=http://s3.docker.test:8000/ + +# Mandatory if you choose s3 storage service, S3 region +#objectstorage.s3.region=eu-west-1 +objectstorage.s3.region=us-east-1 + +# Mandatory if you choose aws-s3 storage service, access key id configured in S3 +objectstorage.s3.accessKeyId=accessKey1 + +# Mandatory if you choose s3 storage service, secret key configured in S3 +objectstorage.s3.secretKey=secretKey1 + +# Optional if you choose s3 storage service: The trust store file, secret, and algorithm to use +# when connecting to the storage service. If not specified falls back to Java defaults. +#objectstorage.s3.truststore.path= +#objectstorage.s3.truststore.type=JKS +#objectstorage.s3.truststore.secret= +#objectstorage.s3.truststore.algorithm=SunX509 + + +# optional: Object read in memory will be rejected if they exceed the size limit exposed here. Size, exemple `100M`. +# Supported units: K, M, G, defaults to B if no unit is specified. If unspecified, big object won't be prevented +# from being loaded in memory. This settings complements protocol limits. +# objectstorage.s3.in.read.limit=50M + +# ============================================ Blobs Exporting ============================================== +# Read https://james.apache.org/server/config-blob-export.html for further details + +# Choosing blob exporting mechanism, allowed mechanism are: localFile, linshare +# LinShare is a file sharing service, will be explained in the below section +# Optional, default is localFile +blob.export.implementation=localFile + +# ======================================= Local File Blobs Exporting ======================================== +# Optional, directory to store exported blob, directory path follows James file system format +# default is file://var/blobExporting +blob.export.localFile.directory=file://var/blobExporting + +# ======================================= LinShare File Blobs Exporting ======================================== +# LinShare is a sharing service where you can use james, connects to an existing LinShare server and shares files to +# other mail addresses as long as those addresses available in LinShare. For example you can deploy James and LinShare +# sharing the same LDAP repository +# Mandatory if you choose LinShare, url to connect to LinShare service +# blob.export.linshare.url=http://linshare:8080 + +# ======================================= LinShare Configuration BasicAuthentication =================================== +# Authentication is mandatory if you choose LinShare, TechnicalAccount is need to connect to LinShare specific service. +# For Example: It will be formalized to 'Authorization: Basic {Credential of UUID/password}' + +# blob.export.linshare.technical.account.uuid=Technical_Account_UUID +# blob.export.linshare.technical.account.password=password diff --git a/server/apps/postgres-app/sample-configuration-distributed/opensearch.properties b/server/apps/postgres-app/sample-configuration-distributed/opensearch.properties new file mode 100644 index 00000000000..df261c5dee6 --- /dev/null +++ b/server/apps/postgres-app/sample-configuration-distributed/opensearch.properties @@ -0,0 +1,101 @@ +# 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. + +# This template file can be used as example for James Server configuration +# DO NOT USE IT AS SUCH AND ADAPT IT TO YOUR NEEDS + +# Configuration file for OpenSearch +# Read https://james.apache.org/server/config-opensearch.html for further details + +opensearch.masterHost=opensearch +opensearch.port=9200 + +# Optional. Only http or https are accepted, default is http +# opensearch.hostScheme=http + +# Optional, default is `default` +# Choosing the SSL check strategy when using https scheme +# default: Use the default SSL TrustStore of the system. +# ignore: Ignore SSL Validation check (not recommended). +# override: Override the SSL Context to use a custom TrustStore containing ES server's certificate. +# opensearch.hostScheme.https.sslValidationStrategy=default + +# Optional. Required when using 'https' scheme and 'override' sslValidationStrategy +# Configure OpenSearch rest client to use this trustStore file to recognize nginx's ssl certificate. +# You need to specify both trustStorePath and trustStorePassword +# opensearch.hostScheme.https.trustStorePath=/file/to/trust/keystore.jks + +# Optional. Required when using 'https' scheme and 'override' sslValidationStrategy +# Configure OpenSearch rest client to use this trustStore file with the specified password. +# You need to specify both trustStorePath and trustStorePassword +# opensearch.hostScheme.https.trustStorePassword=myJKSPassword + +# Optional. default is `default` +# Configure OpenSearch rest client to use host name verifier during SSL handshake +# default: using the default hostname verifier provided by apache http client. +# accept_any_hostname: accept any host (not recommended). +# opensearch.hostScheme.https.hostNameVerifier=default + +# Optional. +# Basic auth username to access opensearch. +# Ignore opensearch.user and opensearch.password to not be using authentication (default behaviour). +# Otherwise, you need to specify both properties. +# opensearch.user=elasticsearch + +# Optional. +# Basic auth password to access opensearch. +# Ignore opensearch.user and opensearch.password to not be using authentication (default behaviour). +# Otherwise, you need to specify both properties. +# opensearch.password=secret + +# You can alternatively provide a list of hosts following this format : +# opensearch.hosts=host1:9200,host2:9200 +# opensearch.clusterName=cluster + +opensearch.nb.shards=5 +opensearch.nb.replica=1 +opensearch.index.waitForActiveShards=1 +opensearch.retryConnection.maxRetries=7 +opensearch.retryConnection.minDelay=3000 +# Index or not attachments (default value: true) +# Note: Attachments not implemented yet for postgresql, false for now +opensearch.indexAttachments=false + +# Search overrides allow resolution of predefined search queries against alternative sources of data +# and allow bypassing opensearch. This is useful to handle most resynchronisation queries that +# are simple enough to be resolved against Cassandra. +# +# Possible values are: +# - `org.apache.james.mailbox.postgres.search.AllSearchOverride` Some IMAP clients uses SEARCH ALL to fully list messages in +# a mailbox and detect deletions. This is typically done by clients not supporting QRESYNC and from an IMAP perspective +# is considered an optimisation as less data is transmitted compared to a FETCH command. Resolving such requests against +# Postgresql is enabled by this search override and likely desirable. +# - `org.apache.james.mailbox.postgres.search.UidSearchOverride`. Same as above but restricted by ranges. +# - `org.apache.james.mailbox.postgres.search.DeletedSearchOverride`. Find deleted messages by looking up in the relevant Postgresql +# table. +# - `org.apache.james.mailbox.postgres.search.DeletedWithRangeSearchOverride`. Same as above but limited by ranges. +# - `org.apache.james.mailbox.postgres.search.NotDeletedWithRangeSearchOverride`. List non deleted messages in a given range. +# Lists all messages and filters out deleted message thus this is based on the following heuristic: most messages are not marked as deleted. +# - `org.apache.james.mailbox.postgres.search.UnseenSearchOverride`. List unseen messages in the corresponding Postgresql index. +# +# Please note that custom overrides can be defined here. +# +# opensearch.search.overrides=org.apache.james.mailbox.postgres.search.AllSearchOverride,org.apache.james.mailbox.postgres.search.DeletedSearchOverride, org.apache.james.mailbox.postgres.search.DeletedWithRangeSearchOverride,org.apache.james.mailbox.postgres.search.NotDeletedWithRangeSearchOverride,org.apache.james.mailbox.postgres.search.UidSearchOverride,org.apache.james.mailbox.postgres.search.UnseenSearchOverride + +# Optional. Default is `false` +# When set to true, James will attempt to reindex from the indexed message when moved. If the message is not found, it will fall back to the old behavior (The message will be indexed from the blobStore source) +# opensearch.message.index.optimize.move=false \ No newline at end of file diff --git a/server/apps/postgres-app/sample-configuration-distributed/rabbitmq.properties b/server/apps/postgres-app/sample-configuration-distributed/rabbitmq.properties new file mode 100644 index 00000000000..75a562af60a --- /dev/null +++ b/server/apps/postgres-app/sample-configuration-distributed/rabbitmq.properties @@ -0,0 +1,95 @@ +# RabbitMQ configuration + +# Read https://james.apache.org/server/config-rabbitmq.html for further details + +# Mandatory +uri=amqp://rabbitmq:5672 +# If you use a vhost, specify it as well at the end of the URI +# uri=amqp://rabbitmq:5672/vhost + +# Vhost to use for creating queues and exchanges +# Optional, only use this if you have invalid URIs containing characters like '_' +# vhost=vhost1 + +# Optional, default to the host specified as part of the URI. +# Allow creating cluster aware connections. +# hosts=ip1:5672,ip2:5672 + +# RabbitMQ Administration Management +# Mandatory +management.uri=http://rabbitmq:15672 +# Mandatory +management.user=guest +# Mandatory +management.password=guest + +# Configure retries count to retrieve a connection. Exponential backoff is performed between each retries. +# Optional integer, defaults to 10 +#connection.pool.retries=10 +# Configure initial duration (in ms) between two connection retries. Exponential backoff is performed between each retries. +# Optional integer, defaults to 100 +#connection.pool.min.delay.ms=100 +# Configure retries count to retrieve a channel. Exponential backoff is performed between each retries. +# Optional integer, defaults to 3 +#channel.pool.retries=3 +# Configure timeout duration (in ms) to obtain a rabbitmq channel. Defaults to 30 seconds. +# Optional integer, defaults to 30 seconds. +#channel.pool.max.delay.ms=30000 +# Configure the size of the channel pool. +# Optional integer, defaults to 3 +#channel.pool.size=3 + +# Boolean. Whether to activate Quorum queue usage for use cases that benefits from it (work queue). +# Quorum queues enables high availability. +# False (default value) results in the usage of classic queues. +#quorum.queues.enable=true + +# Strictly positive integer. The replication factor to use when creating quorum queues. +#quorum.queues.replication.factor + +# Parameters for the Cassandra administrative view + +# Whether the Cassandra administrative view should be activated. Boolean value defaulting to true. +# Not necessarily needed for MDA deployments, mail queue management adds significant complexity. +# cassandra.view.enabled=true + +# Period of the window. Too large values will lead to wide rows while too little values might lead to many queries. +# Use the number of mail per Cassandra row, along with your expected traffic, to determine this value +# This value can only be decreased to a value dividing the current value +# Optional, default 1h +mailqueue.view.sliceWindow=1h + +# Use to distribute the emails of a given slice within your cassandra cluster +# A good value is 2*cassandraNodeCount +# This parameter can only be increased. +# Optional, default 1 +mailqueue.view.bucketCount=1 + +# Determine the probability to update the browse start pointer +# Too little value will lead to unnecessary reads. Too big value will lead to more expensive browse. +# Choose this parameter so that it get's update one time every one-two sliceWindow +# Optional, default 1000 +mailqueue.view.updateBrowseStartPace=1000 + +# Enables or disables the gauge metric on the mail queue size +# Computing the size of the mail queue is currently implemented on top of browse operation and thus have a linear complexity +# Metrics get exported periodically as configured in opensearch.properties, thus getSize is also called periodically +# Choose to disable it when the mail queue size is getting too big +# Note that this is as well a temporary workaround until we get 'getSize' method better optimized +# Optional, default false +mailqueue.size.metricsEnabled=false + +# Whether to enable task consumption on this node. Tasks are WebAdmin triggered long running jobs. +# Disable with caution (this only makes sense in a distributed setup where other nodes consume tasks). +# Defaults to true. +task.consumption.enabled=true + +# Configure task queue consumer timeout. References: https://www.rabbitmq.com/consumers.html#acknowledgement-timeout. Required at least RabbitMQ version 3.12 to have effect. +# This is used to avoid the task queue consumer (which could run very long tasks) being disconnected by RabbitMQ after the default acknowledgement timeout 30 minutes. +# Optional. Duration (support multiple time units cf `DurationParser`), defaults to 1 day. +#task.queue.consumer.timeout=1day + +# Configure queue ttl (in ms). References: https://www.rabbitmq.com/ttl.html#queue-ttl. +# This is used only on queues used to share notification patterns, are exclusive to a node. If omitted, it will not add the TTL configure when declaring queues. +# Optional integer, defaults is 3600000. +#notification.queue.ttl=3600000 diff --git a/server/apps/postgres-app/sample-configuration-single/search.properties b/server/apps/postgres-app/sample-configuration-single/search.properties new file mode 100644 index 00000000000..51833746a92 --- /dev/null +++ b/server/apps/postgres-app/sample-configuration-single/search.properties @@ -0,0 +1,2 @@ +# not for production purposes. To be replaced by PG based search. +implementation=scanning \ No newline at end of file diff --git a/server/apps/postgres-app/sample-configuration/blob.properties b/server/apps/postgres-app/sample-configuration/blob.properties new file mode 100644 index 00000000000..3a01ce1e91b --- /dev/null +++ b/server/apps/postgres-app/sample-configuration/blob.properties @@ -0,0 +1,66 @@ +# ============================================= BlobStore Implementation ================================== +# Read https://james.apache.org/server/config-blobstore.html for further details + +# Choose your BlobStore implementation +# Mandatory, allowed values are: file, s3, postgres. +implementation=postgres + +# ========================================= Deduplication ======================================== +# If you choose to enable deduplication, the mails with the same content will be stored only once. +# Warning: Once this feature is enabled, there is no turning back as turning it off will lead to the deletion of all +# the mails sharing the same content once one is deleted. +# Mandatory, Allowed values are: true, false +deduplication.enable=true + +# deduplication.family needs to be incremented every time the deduplication.generation.duration is changed +# Positive integer, defaults to 1 +# deduplication.gc.generation.family=1 + +# Duration of generation. +# Deduplication only takes place within a singe generation. +# Only items two generation old can be garbage collected. (This prevent concurrent insertions issues and +# accounts for a clock skew). +# deduplication.family needs to be incremented everytime this parameter is changed. +# Duration. Default unit: days. Defaults to 30 days. +# deduplication.gc.generation.duration=30days + +# ========================================= Encryption ======================================== +# If you choose to enable encryption, the blob content will be encrypted before storing them in the BlobStore. +# Warning: Once this feature is enabled, there is no turning back as turning it off will lead to all content being +# encrypted. This comes at a performance impact but presents you from leaking data if, for instance the third party +# offering you a S3 service is compromised. +# Optional, Allowed values are: true, false, defaults to false +encryption.aes.enable=false + +# Mandatory (if AES encryption is enabled) salt and password. Salt needs to be an hexadecimal encoded string +#encryption.aes.password=xxx +#encryption.aes.salt=73616c7479 +# Optional, defaults to PBKDF2WithHmacSHA512 +#encryption.aes.private.key.algorithm=PBKDF2WithHmacSHA512 + +# ============================================ Blobs Exporting ============================================== +# Read https://james.apache.org/server/config-blob-export.html for further details + +# Choosing blob exporting mechanism, allowed mechanism are: localFile, linshare +# LinShare is a file sharing service, will be explained in the below section +# Optional, default is localFile +blob.export.implementation=localFile + +# ======================================= Local File Blobs Exporting ======================================== +# Optional, directory to store exported blob, directory path follows James file system format +# default is file://var/blobExporting +blob.export.localFile.directory=file://var/blobExporting + +# ======================================= LinShare File Blobs Exporting ======================================== +# LinShare is a sharing service where you can use james, connects to an existing LinShare server and shares files to +# other mail addresses as long as those addresses available in LinShare. For example you can deploy James and LinShare +# sharing the same LDAP repository +# Mandatory if you choose LinShare, url to connect to LinShare service +# blob.export.linshare.url=http://linshare:8080 + +# ======================================= LinShare Configuration BasicAuthentication =================================== +# Authentication is mandatory if you choose LinShare, TechnicalAccount is need to connect to LinShare specific service. +# For Example: It will be formalized to 'Authorization: Basic {Credential of UUID/password}' + +# blob.export.linshare.technical.account.uuid=Technical_Account_UUID +# blob.export.linshare.technical.account.password=password diff --git a/server/apps/postgres-app/sample-configuration/deletedMessageVault.properties b/server/apps/postgres-app/sample-configuration/deletedMessageVault.properties new file mode 100644 index 00000000000..a6df89a2275 --- /dev/null +++ b/server/apps/postgres-app/sample-configuration/deletedMessageVault.properties @@ -0,0 +1,7 @@ +# ============================================= Deleted Messages Vault Configuration ================================== + +enabled=false + +# Retention period for your deleted messages into the vault, after which they expire and can be potentially cleaned up +# Optional, default 1y +# retentionPeriod=1y \ No newline at end of file diff --git a/server/apps/postgres-app/sample-configuration/dnsservice.xml b/server/apps/postgres-app/sample-configuration/dnsservice.xml new file mode 100644 index 00000000000..863de0e2afc --- /dev/null +++ b/server/apps/postgres-app/sample-configuration/dnsservice.xml @@ -0,0 +1,27 @@ + + + + + + + true + false + 50000 + diff --git a/server/apps/postgres-app/sample-configuration/domainlist.xml b/server/apps/postgres-app/sample-configuration/domainlist.xml new file mode 100644 index 00000000000..605439fbd0e --- /dev/null +++ b/server/apps/postgres-app/sample-configuration/domainlist.xml @@ -0,0 +1,27 @@ + + + + + + + false + false + localhost + diff --git a/server/apps/postgres-app/sample-configuration/droplists.properties b/server/apps/postgres-app/sample-configuration/droplists.properties new file mode 100644 index 00000000000..bbc27568cbc --- /dev/null +++ b/server/apps/postgres-app/sample-configuration/droplists.properties @@ -0,0 +1,3 @@ +# Configuration file for DropLists + +enabled=false \ No newline at end of file diff --git a/server/apps/postgres-app/sample-configuration/extensions.properties b/server/apps/postgres-app/sample-configuration/extensions.properties new file mode 100644 index 00000000000..2a2c23e7cb0 --- /dev/null +++ b/server/apps/postgres-app/sample-configuration/extensions.properties @@ -0,0 +1,10 @@ +# This files enables customization of users extensions injections with guice. +# A user can drop some jar-with-dependencies within the ./extensions-jars folder and +# reference classes of these jars in some of James extension mechanisms. + +# This includes mailets, matchers, mailboxListeners, preDeletionHooks, protocolHandlers, webAdmin routes + +# Upon injections, the user can reference additional guice modules, that are going to be used only upon extensions instantiation. + +#List of coma separated (',') fully qualified class names of additional guice modules to be used to instantiate extensions +#guice.extension.module= \ No newline at end of file diff --git a/server/apps/postgres-app/sample-configuration/healthcheck.properties b/server/apps/postgres-app/sample-configuration/healthcheck.properties new file mode 100644 index 00000000000..c796fee60b7 --- /dev/null +++ b/server/apps/postgres-app/sample-configuration/healthcheck.properties @@ -0,0 +1,33 @@ +# 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. + +# This template file can be used as example for James Server configuration +# DO NOT USE IT AS SUCH AND ADAPT IT TO YOUR NEEDS + +# Configuration file for Periodical Health Checks + +# Read https://james.apache.org/server/config-healthcheck.html for further details + +# Optional. Period between two PeriodicalHealthChecks. +# Units supported are (ms - millisecond, s - second, m - minute, h - hour, d - day). Default unit is millisecond. +# Default duration is 60 seconds. +# Duration must be greater or at least equals to 10 seconds. +# healthcheck.period=60s + +# List of fully qualified HealthCheck class names in addition to James' default healthchecks. +# Healthchecks need to be located within the classpath or in the ./extensions-jars folder. +# additional.healthchecks= diff --git a/server/apps/postgres-app/sample-configuration/imapserver.xml b/server/apps/postgres-app/sample-configuration/imapserver.xml new file mode 100644 index 00000000000..12991c48dc1 --- /dev/null +++ b/server/apps/postgres-app/sample-configuration/imapserver.xml @@ -0,0 +1,83 @@ + + + + + + + + + + imapserver + 0.0.0.0:143 + 200 + + + file://conf/keystore + PKCS12 + james72laBalle + org.bouncycastle.jce.provider.BouncyCastleProvider + + + + + + + + 0 + 0 + 120 + SECONDS + true + false + + true + + + + imapserver-ssl + 0.0.0.0:993 + 200 + + + file://conf/keystore + PKCS12 + james72laBalle + org.bouncycastle.jce.provider.BouncyCastleProvider + + + + + + + + 0 + 0 + 120 + SECONDS + true + + true + + + diff --git a/server/apps/postgres-app/sample-configuration/jmx.properties b/server/apps/postgres-app/sample-configuration/jmx.properties new file mode 100644 index 00000000000..e56235f9b4a --- /dev/null +++ b/server/apps/postgres-app/sample-configuration/jmx.properties @@ -0,0 +1,26 @@ +# 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. +# + +# This template file can be used as example for James Server configuration +# DO NOT USE IT AS SUCH AND ADAPT IT TO YOUR NEEDS + +# Read https://james.apache.org/server/config-system.html#jmx.properties for further details + +jmx.enabled=true +jmx.address=127.0.0.1 +jmx.port=9999 diff --git a/server/apps/postgres-app/sample-configuration/jvm.properties b/server/apps/postgres-app/sample-configuration/jvm.properties new file mode 100644 index 00000000000..186fbe4b2c3 --- /dev/null +++ b/server/apps/postgres-app/sample-configuration/jvm.properties @@ -0,0 +1,56 @@ +# ============================================= Extra JVM System Properties =========================================== +# To avoid clutter on the command line, any properties in this file will be added as system properties on server start. + +# Example: If you need an option -Dmy.property=whatever, you can instead add it here as +# my.property=whatever + +# (Optional). String (size, integer + size units, example: `12 KIB`, supported units are bytes KIB MIB GIB TIB). Defaults to 100KIB. +# This governs the threshold MimeMessageInputStreamSource relies on for storing MimeMessage content on disk. +# Below, data is stored in memory. Above data is stored on disk. +# Lower values will lead to longer processing time but will minimize heap memory usage. Modern SSD hardware +# should however support a high throughput. Higher values will lead to faster single mail processing at the cost +# of higher heap usage. +#james.message.memory.threshold=12K + +# Optional. Boolean. Defaults to false. Recommended value is false. +# Should MimeMessageWrapper use a copy of the message in memory? Or should bigger message exceeding james.message.memory.threshold +# be copied to temporary files? +#james.message.usememorycopy=false + +# Mode level of resource leak detection. It is used to detect a resource not be disposed of before it's garbage-collected. +# Example `MimeMessageInputStreamSource` +# Optional. Allowed values are: none, simple, advanced, testing +# - none: Disables resource leak detection. +# - simple: Enables output a simplistic error log if a leak is encountered and would free the resources (default). +# - advanced: Enables output an advanced error log implying the place of allocation of the underlying object and would free resources. +# - testing: Enables output an advanced error log implying the place of allocation of the underlying object and rethrow an error, that action is being taken by the development team. +#james.lifecycle.leak.detection.mode=simple + +# Should we add the host in the MDC logging context for incoming IMAP, SMTP, POP3? Doing so, a DNS resolution +# is attempted for each incoming connection, which can be costly. Remote IP is always added to the logging context. +# Optional. Boolean. Defaults to true. +#james.protocols.mdc.hostname=true + +# Manage netty leak detection level see https://netty.io/wiki/reference-counted-objects.html#leak-detection-levels +# io.netty.leakDetection.level=SIMPLE + +# Should James exit on Startup error? Boolean, defaults to true. This prevents partial startup. +# james.exit.on.startup.error=true + +# Fails explicitly on missing configuration file rather that taking implicit values. Defautls to false. +# james.fail.on.missing.configuration=true + +# JMX, when enable causes RMI to plan System.gc every hour. Set this instead to once every 1000h. +sun.rmi.dgc.server.gcInterval=3600000000 +sun.rmi.dgc.client.gcInterval=3600000000 + +# Automatically generate a JMX password upon start. CLI is able to retrieve this password. +james.jmx.credential.generation=true + +# Disable Remote Code Execution feature from JMX +# CF https://github.com/AdoptOpenJDK/openjdk-jdk11/blob/19fb8f93c59dfd791f62d41f332db9e306bc1422/src/java.management/share/classes/com/sun/jmx/remote/security/MBeanServerAccessController.java#L646 +jmx.remote.x.mlet.allow.getMBeansFromURL=false +openjpa.Multithreaded=true + +# Integer. Optional, defaults to 5000. In case of large data, this argument specifies the maximum number of rows to return in a single batch set when executing query. +#query.batch.size=5000 \ No newline at end of file diff --git a/server/apps/postgres-app/sample-configuration/jwt_publickey b/server/apps/postgres-app/sample-configuration/jwt_publickey new file mode 100644 index 00000000000..53914e0533a --- /dev/null +++ b/server/apps/postgres-app/sample-configuration/jwt_publickey @@ -0,0 +1,9 @@ +-----BEGIN PUBLIC KEY----- +MIIBIjANBgkqhkiG9w0BAQEFAAOCAQ8AMIIBCgKCAQEAtlChO/nlVP27MpdkG0Bh +16XrMRf6M4NeyGa7j5+1UKm42IKUf3lM28oe82MqIIRyvskPc11NuzSor8HmvH8H +lhDs5DyJtx2qp35AT0zCqfwlaDnlDc/QDlZv1CoRZGpQk1Inyh6SbZwYpxxwh0fi ++d/4RpE3LBVo8wgOaXPylOlHxsDizfkL8QwXItyakBfMO6jWQRrj7/9WDhGf4Hi+ +GQur1tPGZDl9mvCoRHjFrD5M/yypIPlfMGWFVEvV5jClNMLAQ9bYFuOc7H1fEWw6 +U1LZUUbJW9/CH45YXz82CYqkrfbnQxqRb2iVbVjs/sHopHd1NTiCfUtwvcYJiBVj +kwIDAQAB +-----END PUBLIC KEY----- diff --git a/server/apps/postgres-app/sample-configuration/listeners.xml b/server/apps/postgres-app/sample-configuration/listeners.xml new file mode 100644 index 00000000000..ffe9605c6d8 --- /dev/null +++ b/server/apps/postgres-app/sample-configuration/listeners.xml @@ -0,0 +1,24 @@ + + + + + + + \ No newline at end of file diff --git a/server/apps/postgres-app/sample-configuration/lmtpserver.xml b/server/apps/postgres-app/sample-configuration/lmtpserver.xml new file mode 100644 index 00000000000..723da3fb262 --- /dev/null +++ b/server/apps/postgres-app/sample-configuration/lmtpserver.xml @@ -0,0 +1,43 @@ + + + + + + + + + lmtpserver + + 127.0.0.1:24 + 200 + 1200 + + 0 + + 0 + + + 0 + + + + + + diff --git a/server/apps/postgres-app/sample-configuration/logback.xml b/server/apps/postgres-app/sample-configuration/logback.xml new file mode 100644 index 00000000000..85c261041bb --- /dev/null +++ b/server/apps/postgres-app/sample-configuration/logback.xml @@ -0,0 +1,39 @@ + + + + + true + + + + + %d{HH:mm:ss.SSS} %highlight([%-5level]) %logger{15} - %msg%n%rEx + false + + + + + /logs/james.log + + /logs/james.%i.log.tar.gz + 1 + 3 + + + + 100MB + + + + %d{HH:mm:ss.SSS} [%-5level] %logger{15} - %msg%n%rEx + false + + + + + + + + + + diff --git a/server/apps/postgres-app/sample-configuration/mailetcontainer.xml b/server/apps/postgres-app/sample-configuration/mailetcontainer.xml new file mode 100644 index 00000000000..bdc8d58c473 --- /dev/null +++ b/server/apps/postgres-app/sample-configuration/mailetcontainer.xml @@ -0,0 +1,151 @@ + + + + + + + + + + + postmaster + + + + 20 + postgres://var/mail/error/ + + + + + + + postgres://var/mail/relay-limit-exceeded/ + + + transport + + + + + + mailetContainerErrors + + + ignore + + + postgres://var/mail/error/ + propagate + + + + + + + + + + + + bcc + ignore + + + X-SMIME-Status + ignore + + + rrt-error + + + + + + local-address-error + 550 - Requested action not taken: no such user here + + + + relay + + + + + + outgoing + 5000, 100000, 500000 + 3 + 0 + 10 + true + bounces + + + + + + mailetContainerLocalAddressError + + + none + + + postgres://var/mail/address-error/ + + + + + + mailetContainerRelayDenied + + + none + + + postgres://var/mail/relay-denied/ + Warning: You are sending an e-mail to a remote server. You must be authenticated to perform such an operation + + + + + + bounces + + + false + + + + + + postgres://var/mail/rrt-error/ + true + + + + + + + + + + diff --git a/server/apps/postgres-app/sample-configuration/mailrepositorystore.xml b/server/apps/postgres-app/sample-configuration/mailrepositorystore.xml new file mode 100644 index 00000000000..445f2727f29 --- /dev/null +++ b/server/apps/postgres-app/sample-configuration/mailrepositorystore.xml @@ -0,0 +1,35 @@ + + + + + + + + postgres + + + + postgres + + + + + + diff --git a/server/apps/postgres-app/sample-configuration/managesieveserver.xml b/server/apps/postgres-app/sample-configuration/managesieveserver.xml new file mode 100644 index 00000000000..7b0b85a6eee --- /dev/null +++ b/server/apps/postgres-app/sample-configuration/managesieveserver.xml @@ -0,0 +1,65 @@ + + + + + + + + + + + + managesieveserver + + 0.0.0.0:4190 + + 200 + + + + file://conf/keystore + james72laBalle + org.bouncycastle.jce.provider.BouncyCastleProvider + + SunX509 + + + + 360 + + + 0 + + + 0 + 0 + true + + + + + + diff --git a/server/apps/postgres-app/sample-configuration/pop3server.xml b/server/apps/postgres-app/sample-configuration/pop3server.xml new file mode 100644 index 00000000000..465efe9cbfc --- /dev/null +++ b/server/apps/postgres-app/sample-configuration/pop3server.xml @@ -0,0 +1,50 @@ + + + + + + + + pop3server + 0.0.0.0:110 + 200 + + + file://conf/keystore + PKCS12 + james72laBalle + org.bouncycastle.jce.provider.BouncyCastleProvider + + + + + + + + 1200 + 0 + 0 + + + + + diff --git a/server/apps/postgres-app/sample-configuration/postgres.properties b/server/apps/postgres-app/sample-configuration/postgres.properties new file mode 100644 index 00000000000..58c7cd476c9 --- /dev/null +++ b/server/apps/postgres-app/sample-configuration/postgres.properties @@ -0,0 +1,45 @@ +# String. Optional, default to 'postgres'. Database name. +database.name=james + +# String. Optional, default to 'public'. Database schema. +database.schema=public + +# String. Optional, default to 'localhost'. Database host. +database.host=postgres + +# Integer. Optional, default to 5432. Database port. +database.port=5432 + +# String. Required. Database username. +database.username=james + +# String. Required. Database password of the user. +database.password=secret1 + +# Boolean. Optional, default to false. Whether to enable row level security. +row.level.security.enabled=false + +# String. It is required when row.level.security.enabled is true. Database username with the permission of bypassing RLS. +#database.by-pass-rls.username=bypassrlsjames + +# String. It is required when row.level.security.enabled is true. Database password of by-pass-rls user. +#database.by-pass-rls.password=secret1 + +# Integer. Optional, default to 10. Database connection pool initial size. +pool.initial.size=10 + +# Integer. Optional, default to 15. Database connection pool max size. +pool.max.size=15 + +# Integer. Optional, default to 5. rls-bypass database connection pool initial size. +by-pass-rls.pool.initial.size=5 + +# Integer. Optional, default to 10. rls-bypass database connection pool max size. +by-pass-rls.pool.max.size=10 + +# String. Optional, defaults to allow. SSLMode required to connect to the Postgresql db server. +# Check https://www.postgresql.org/docs/current/libpq-ssl.html#LIBPQ-SSL-PROTECTION for a list of supported SSLModes. +ssl.mode=allow + +## Duration. Optional, defaults to 10 second. jOOQ reactive timeout when executing Postgres query. This setting prevent jooq reactive bug from causing hanging issue. +#jooq.reactive.timeout=10second \ No newline at end of file diff --git a/server/apps/postgres-app/sample-configuration/recipientrewritetable.xml b/server/apps/postgres-app/sample-configuration/recipientrewritetable.xml new file mode 100644 index 00000000000..1a512c60351 --- /dev/null +++ b/server/apps/postgres-app/sample-configuration/recipientrewritetable.xml @@ -0,0 +1,28 @@ + + + + + + + + true + 10 + + diff --git a/server/apps/postgres-app/sample-configuration/smtpserver.xml b/server/apps/postgres-app/sample-configuration/smtpserver.xml new file mode 100644 index 00000000000..94ed2e5b6ac --- /dev/null +++ b/server/apps/postgres-app/sample-configuration/smtpserver.xml @@ -0,0 +1,159 @@ + + + + + + + + + smtpserver-global + 0.0.0.0:25 + 200 + + + file://conf/keystore + PKCS12 + james72laBalle + org.bouncycastle.jce.provider.BouncyCastleProvider + SunX509 + + + + + + + + 360 + 0 + 0 + + never + false + true + + 127.0.0.0/8 + true + 0 + true + Apache JAMES awesome SMTP Server + + + + + + + smtpserver-TLS + 0.0.0.0:465 + 200 + + + file://conf/keystore + PKCS12 + james72laBalle + org.bouncycastle.jce.provider.BouncyCastleProvider + SunX509 + + + + + + + + 360 + 0 + 0 + + forUnauthorizedAddresses + true + true + + + + 127.0.0.0/8 + true + 0 + true + Apache JAMES awesome SMTP Server + + + + + + + smtpserver-authenticated + 0.0.0.0:587 + 200 + + + file://conf/keystore + PKCS12 + james72laBalle + org.bouncycastle.jce.provider.BouncyCastleProvider + SunX509 + + + + + + + + 360 + 0 + 0 + + forUnauthorizedAddresses + true + true + + + + 127.0.0.0/8 + true + 0 + true + Apache JAMES awesome SMTP Server + + + + + + + + diff --git a/server/apps/postgres-app/sample-configuration/usersrepository.xml b/server/apps/postgres-app/sample-configuration/usersrepository.xml new file mode 100644 index 00000000000..a5390d7140d --- /dev/null +++ b/server/apps/postgres-app/sample-configuration/usersrepository.xml @@ -0,0 +1,28 @@ + + + + + + + PBKDF2-SHA512 + true + true + + diff --git a/server/apps/postgres-app/sample-configuration/webadmin.properties b/server/apps/postgres-app/sample-configuration/webadmin.properties new file mode 100644 index 00000000000..5dc74740c55 --- /dev/null +++ b/server/apps/postgres-app/sample-configuration/webadmin.properties @@ -0,0 +1,49 @@ +# 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. + +# This template file can be used as example for James Server configuration +# DO NOT USE IT AS SUCH AND ADAPT IT TO YOUR NEEDS + +# Read https://james.apache.org/server/config-webadmin.html for further details + +enabled=true +port=8000 +# Use host=0.0.0.0 to listen on all addresses +host=localhost + +# Defaults to false +https.enabled=false + +# Compulsory when enabling HTTPS +#https.keystore=/path/to/keystore +#https.password=password + +# Optional when enabling HTTPS (self signed) +#https.trust.keystore +#https.trust.password + +# Defaults to false +#jwt.enabled=true + +# Defaults to false +#cors.enable=true +#cors.origin + +# List of fully qualified class names that should be exposed over webadmin +# in addition to your product default routes. Routes needs to be located +# within the classpath or in the ./extensions-jars folder. +#extensions.routes= \ No newline at end of file diff --git a/server/apps/postgres-app/src/assemble/app.xml b/server/apps/postgres-app/src/assemble/app.xml new file mode 100644 index 00000000000..79ecba5d298 --- /dev/null +++ b/server/apps/postgres-app/src/assemble/app.xml @@ -0,0 +1,86 @@ + + + + app + + + zip + + + + + + . + 0755 + / + + README* + + + + + sample-configuration + 0755 + conf + + 0600 + + + + target/james-server-jpa-app.lib + /james-server-jpa-app.lib + 0755 + 0600 + + *.jar + + + + + + src/assemble/license-for-binary.txt + / + 0644 + LICENSE + crlf + + + README.adoc + / + 0644 + crlf + + + src/assemble/extensions-jars.txt + /extensions-jars + 0644 + crlf + README.md + + + target/james-server-postgres-app.jar + / + 0755 + james-server-postgres-app.jar + + + diff --git a/server/apps/postgres-app/src/assemble/extensions-jars.txt b/server/apps/postgres-app/src/assemble/extensions-jars.txt new file mode 100644 index 00000000000..2cea7599812 --- /dev/null +++ b/server/apps/postgres-app/src/assemble/extensions-jars.txt @@ -0,0 +1,5 @@ +# Adding Jars to JAMES + +The jar in this folder will be added to JAMES classpath when mounted under /root/extensions-jars inside the running container. + +You can use it to add you customs Mailets/Matchers. diff --git a/server/apps/postgres-app/src/assemble/license-for-binary.txt b/server/apps/postgres-app/src/assemble/license-for-binary.txt new file mode 100644 index 00000000000..682a01fab77 --- /dev/null +++ b/server/apps/postgres-app/src/assemble/license-for-binary.txt @@ -0,0 +1,1139 @@ + + + Apache License + Version 2.0, January 2004 + http://www.apache.org/licenses/ + + TERMS AND CONDITIONS FOR USE, REPRODUCTION, AND DISTRIBUTION + + 1. Definitions. + + "License" shall mean the terms and conditions for use, reproduction, + and distribution as defined by Sections 1 through 9 of this document. + + "Licensor" shall mean the copyright owner or entity authorized by + the copyright owner that is granting the License. + + "Legal Entity" shall mean the union of the acting entity and all + other entities that control, are controlled by, or are under common + control with that entity. For the purposes of this definition, + "control" means (i) the power, direct or indirect, to cause the + direction or management of such entity, whether by contract or + otherwise, or (ii) ownership of fifty percent (50%) or more of the + outstanding shares, or (iii) beneficial ownership of such entity. + + "You" (or "Your") shall mean an individual or Legal Entity + exercising permissions granted by this License. + + "Source" form shall mean the preferred form for making modifications, + including but not limited to software source code, documentation + source, and configuration files. + + "Object" form shall mean any form resulting from mechanical + transformation or translation of a Source form, including but + not limited to compiled object code, generated documentation, + and conversions to other media types. + + "Work" shall mean the work of authorship, whether in Source or + Object form, made available under the License, as indicated by a + copyright notice that is included in or attached to the work + (an example is provided in the Appendix below). + + "Derivative Works" shall mean any work, whether in Source or Object + form, that is based on (or derived from) the Work and for which the + editorial revisions, annotations, elaborations, or other modifications + represent, as a whole, an original work of authorship. For the purposes + of this License, Derivative Works shall not include works that remain + separable from, or merely link (or bind by name) to the interfaces of, + the Work and Derivative Works thereof. + + "Contribution" shall mean any work of authorship, including + the original version of the Work and any modifications or additions + to that Work or Derivative Works thereof, that is intentionally + submitted to Licensor for inclusion in the Work by the copyright owner + or by an individual or Legal Entity authorized to submit on behalf of + the copyright owner. For the purposes of this definition, "submitted" + means any form of electronic, verbal, or written communication sent + to the Licensor or its representatives, including but not limited to + communication on electronic mailing lists, source code control systems, + and issue tracking systems that are managed by, or on behalf of, the + Licensor for the purpose of discussing and improving the Work, but + excluding communication that is conspicuously marked or otherwise + designated in writing by the copyright owner as "Not a Contribution." + + "Contributor" shall mean Licensor and any individual or Legal Entity + on behalf of whom a Contribution has been received by Licensor and + subsequently incorporated within the Work. + + 2. Grant of Copyright License. Subject to the terms and conditions of + this License, each Contributor hereby grants to You a perpetual, + worldwide, non-exclusive, no-charge, royalty-free, irrevocable + copyright license to reproduce, prepare Derivative Works of, + publicly display, publicly perform, sublicense, and distribute the + Work and such Derivative Works in Source or Object form. + + 3. Grant of Patent License. Subject to the terms and conditions of + this License, each Contributor hereby grants to You a perpetual, + worldwide, non-exclusive, no-charge, royalty-free, irrevocable + (except as stated in this section) patent license to make, have made, + use, offer to sell, sell, import, and otherwise transfer the Work, + where such license applies only to those patent claims licensable + by such Contributor that are necessarily infringed by their + Contribution(s) alone or by combination of their Contribution(s) + with the Work to which such Contribution(s) was submitted. If You + institute patent litigation against any entity (including a + cross-claim or counterclaim in a lawsuit) alleging that the Work + or a Contribution incorporated within the Work constitutes direct + or contributory patent infringement, then any patent licenses + granted to You under this License for that Work shall terminate + as of the date such litigation is filed. + + 4. Redistribution. You may reproduce and distribute copies of the + Work or Derivative Works thereof in any medium, with or without + modifications, and in Source or Object form, provided that You + meet the following conditions: + + (a) You must give any other recipients of the Work or + Derivative Works a copy of this License; and + + (b) You must cause any modified files to carry prominent notices + stating that You changed the files; and + + (c) You must retain, in the Source form of any Derivative Works + that You distribute, all copyright, patent, trademark, and + attribution notices from the Source form of the Work, + excluding those notices that do not pertain to any part of + the Derivative Works; and + + (d) If the Work includes a "NOTICE" text file as part of its + distribution, then any Derivative Works that You distribute must + include a readable copy of the attribution notices contained + within such NOTICE file, excluding those notices that do not + pertain to any part of the Derivative Works, in at least one + of the following places: within a NOTICE text file distributed + as part of the Derivative Works; within the Source form or + documentation, if provided along with the Derivative Works; or, + within a display generated by the Derivative Works, if and + wherever such third-party notices normally appear. The contents + of the NOTICE file are for informational purposes only and + do not modify the License. You may add Your own attribution + notices within Derivative Works that You distribute, alongside + or as an addendum to the NOTICE text from the Work, provided + that such additional attribution notices cannot be construed + as modifying the License. + + You may add Your own copyright statement to Your modifications and + may provide additional or different license terms and conditions + for use, reproduction, or distribution of Your modifications, or + for any such Derivative Works as a whole, provided Your use, + reproduction, and distribution of the Work otherwise complies with + the conditions stated in this License. + + 5. Submission of Contributions. Unless You explicitly state otherwise, + any Contribution intentionally submitted for inclusion in the Work + by You to the Licensor shall be under the terms and conditions of + this License, without any additional terms or conditions. + Notwithstanding the above, nothing herein shall supersede or modify + the terms of any separate license agreement you may have executed + with Licensor regarding such Contributions. + + 6. Trademarks. This License does not grant permission to use the trade + names, trademarks, service marks, or product names of the Licensor, + except as required for reasonable and customary use in describing the + origin of the Work and reproducing the content of the NOTICE file. + + 7. Disclaimer of Warranty. Unless required by applicable law or + agreed to in writing, Licensor provides the Work (and each + Contributor provides its Contributions) on an "AS IS" BASIS, + WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or + implied, including, without limitation, any warranties or conditions + of TITLE, NON-INFRINGEMENT, MERCHANTABILITY, or FITNESS FOR A + PARTICULAR PURPOSE. You are solely responsible for determining the + appropriateness of using or redistributing the Work and assume any + risks associated with Your exercise of permissions under this License. + + 8. Limitation of Liability. In no event and under no legal theory, + whether in tort (including negligence), contract, or otherwise, + unless required by applicable law (such as deliberate and grossly + negligent acts) or agreed to in writing, shall any Contributor be + liable to You for damages, including any direct, indirect, special, + incidental, or consequential damages of any character arising as a + result of this License or out of the use or inability to use the + Work (including but not limited to damages for loss of goodwill, + work stoppage, computer failure or malfunction, or any and all + other commercial damages or losses), even if such Contributor + has been advised of the possibility of such damages. + + 9. Accepting Warranty or Additional Liability. While redistributing + the Work or Derivative Works thereof, You may choose to offer, + and charge a fee for, acceptance of support, warranty, indemnity, + or other liability obligations and/or rights consistent with this + License. However, in accepting such obligations, You may act only + on Your own behalf and on Your sole responsibility, not on behalf + of any other Contributor, and only if You agree to indemnify, + defend, and hold each Contributor harmless for any liability + incurred by, or claims asserted against, such Contributor by reason + of your accepting any such warranty or additional liability. + + END OF TERMS AND CONDITIONS + + +This distribution contains third party resources. +Within the bin directory + licensed under the Tanuki Software License (as follows) + + + Copyright (c) 1999, 2006 Tanuki Software, Inc. + + Permission is hereby granted, free of charge, to any person + obtaining a copy of the Java Service Wrapper and associated + documentation files (the "Software"), to deal in the Software + without restriction, including without limitation the rights + to use, copy, modify, merge, publish, distribute, sub-license, + and/or sell copies of the Software, and to permit persons to + whom the Software is furnished to do so, subject to the + following conditions: + + The above copyright notice and this permission notice shall be + included in all copies or substantial portions of the Software. + + THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, + EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES + OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND + NON-INFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT + HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, + WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING + FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR + OTHER DEALINGS IN THE SOFTWARE. + + + Portions of the Software have been derived from source code + developed by Silver Egg Technology under the following license: + + Copyright (c) 2001 Silver Egg Technology + + Permission is hereby granted, free of charge, to any person + obtaining a copy of this software and associated documentation + files (the "Software"), to deal in the Software without + restriction, including without limitation the rights to use, + copy, modify, merge, publish, distribute, sub-license, and/or + sell copies of the Software, and to permit persons to whom the + Software is furnished to do so, subject to the following + conditions: + + The above copyright notice and this permission notice shall be + included in all copies or substantial portions of the Software. + + THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, + EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES + OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND + NON-INFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT + HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, + WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING + FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR + OTHER DEALINGS IN THE SOFTWARE. + + from Tanuki Software http://www.tanukisoftware.com/ + james + james.bat + wrapper + wrapper-linux-ppc-64 + wrapper-linux-x86-32 + wrapper-linux-x86-64 + wrapper-macosx-ppc-32 + wrapper-macosx-universal-32 + wrapper-solaris-sparc-32 + wrapper-solaris-sparc-64 + wrapper-solaris-x86-32 + wrapper-windows-x86-32.exe + +Within the conf directory + licensed under the Tanuki Software License (as follows) + + + Copyright (c) 1999, 2006 Tanuki Software, Inc. + + Permission is hereby granted, free of charge, to any person + obtaining a copy of the Java Service Wrapper and associated + documentation files (the "Software"), to deal in the Software + without restriction, including without limitation the rights + to use, copy, modify, merge, publish, distribute, sub-license, + and/or sell copies of the Software, and to permit persons to + whom the Software is furnished to do so, subject to the + following conditions: + + The above copyright notice and this permission notice shall be + included in all copies or substantial portions of the Software. + + THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, + EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES + OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND + NON-INFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT + HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, + WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING + FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR + OTHER DEALINGS IN THE SOFTWARE. + + + Portions of the Software have been derived from source code + developed by Silver Egg Technology under the following license: + + Copyright (c) 2001 Silver Egg Technology + + Permission is hereby granted, free of charge, to any person + obtaining a copy of this software and associated documentation + files (the "Software"), to deal in the Software without + restriction, including without limitation the rights to use, + copy, modify, merge, publish, distribute, sub-license, and/or + sell copies of the Software, and to permit persons to whom the + Software is furnished to do so, subject to the following + conditions: + + The above copyright notice and this permission notice shall be + included in all copies or substantial portions of the Software. + + THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, + EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES + OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND + NON-INFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT + HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, + WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING + FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR + OTHER DEALINGS IN THE SOFTWARE. + + from Tanuki Software http://www.tanukisoftware.com/ + wrapper.conf + +Within the lib directory + placed in the public domain + by Doug Lea + concurrent-1.3.4.jar + by Drew Noakes + metadata-extractor-2.4.0-beta-1.jar + by The AOP Alliance http://aopalliance.sourceforge.net/ + aopalliance-1.0.jar + + licensed under the Apache License, Version 2 http://www.apache.org/licenses/LICENSE-2.0.txt (as above) + from Boilerpipe http://code.google.com/p/boilerpipe/ + boilerpipe-1.1.0.jar + from FuseSource http://www.fusesource.org + commons-management-1.0.jar + from JBoss, a division of Red Hat, Inc. http://www.jboss.org + netty-3.2.4.Final.jar + from John Cowan http://home.ccil.org/~cowan/XML/tagsoup/ + tagsoup-1.2.jar + from Oracle http://www.oracle.com + rome-0.9.jar + from The JASYPT team http://www.jasypt.org + jasypt-1.6.jar + from The Spring Framework Project http://www.springframework.org + spring-aop-3.1.RELEASE.jar + spring-asm-3.1.RELEASE.jar + spring-beans-3.1.RELEASE.jar + spring-context-3.1.RELEASE.jar + spring-core-3.1.RELEASE.jar + spring-expression-3.1.RELEASE.jar + spring-jdbc-3.1.RELEASE.jar + spring-jms-3.1.RELEASE.jar + spring-orm-3.1.RELEASE.jar + spring-tx-3.1.RELEASE.jar + spring-web-3.1.RELEASE.jar + + licensed under the BSD (3-clause) http://www.opensource.org/licenses/BSD-3-Clause (as follows) + + ASM: a very small and fast Java bytecode manipulation framework + Copyright (c) 2000-2007 INRIA, France Telecom + All rights reserved. + + Redistribution and use in source and binary forms, with or without + modification, are permitted provided that the following conditions + are met: + 1. Redistributions of source code must retain the above copyright + notice, this list of conditions and the following disclaimer. + 2. Redistributions in binary form must reproduce the above copyright + notice, this list of conditions and the following disclaimer in the + documentation and/or other materials provided with the distribution. + 3. Neither the name of the copyright holders nor the names of its + contributors may be used to endorse or promote products derived from + this software without specific prior written permission. + + THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" + AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE + IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE + ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS BE + LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR + CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF + SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS + INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN + CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) + ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF + THE POSSIBILITY OF SUCH DAMAGE. + + from OW2 http://www.ow2.org/ + asm-3.1.jar + + licensed under the BSD (2-clause) http://www.opensource.org/licenses/BSD-2-Clause (as follows) + + dnsjava is placed under the BSD license. Several files are also under + additional licenses; see the individual files for details. + + Copyright (c) 1998-2011, Brian Wellington. + All rights reserved. + + Redistribution and use in source and binary forms, with or without + modification, are permitted provided that the following conditions are met: + + * Redistributions of source code must retain the above copyright notice, + this list of conditions and the following disclaimer. + + * Redistributions in binary form must reproduce the above copyright notice, + this list of conditions and the following disclaimer in the documentation + and/or other materials provided with the distribution. + + THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" + AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE + IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE + ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE + LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR + CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF + SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS + INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN + CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) + ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE + POSSIBILITY OF SUCH DAMAGE. + + from Brian Wellington + dnsjava-2.1.8.jar + + licensed under the BSD (3-clause) http://www.opensource.org/licenses/BSD-3-Clause (as follows) + + Copyright (c) 2002-2007, A. Abram White + All rights reserved. + + Redistribution and use in source and binary forms, with or without + modification, are permitted provided that the following conditions are met: + + * Redistributions of source code must retain the above copyright notice, + this list of conditions and the following disclaimer. + * Redistributions in binary form must reproduce the above copyright notice, + this list of conditions and the following disclaimer in the documentation + and/or other materials provided with the distribution. + * Neither the name of 'serp' nor the names of its contributors + may be used to endorse or promote products derived from this software + without specific prior written permission. + + THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND + ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED + WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE + DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS BE LIABLE FOR + ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES + (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; + LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON + ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT + (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS + SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. + + from The Serp Project http://serp.sourceforge.net/ + serp-1.13.1.jar + + licensed under the Bouncy Castle Licence http://www.bouncycastle.org/licence.html (as follows) + + Copyright (c) 2000 - 2011 The Legion Of The Bouncy Castle (http://www.bouncycastle.org) + + Permission is hereby granted, free of charge, to any person obtaining a copy of this software + and associated documentation files (the "Software"), to deal in the Software without restriction, + including without limitation the rights to use, copy, modify, merge, publish, distribute, sublicense, + and/or sell copies of the Software, and to permit persons to whom the Software is furnished to + do so, subject to the following conditions: + + The above copyright notice and this permission notice shall be included in all copies or substantial + portions of the Software. + + THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR + IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS + FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS + OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER + IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION + WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. + + from The Legion of the Bouncy Castle http://www.bouncycastle.org/ + bcmail-jdk15-1.45.jar + bcprov-jdk15-1.45.jar + + licensed under the COMMON DEVELOPMENT AND DISTRIBUTION LICENSE (CDDL) Version 1.0 http://www.opensource.org/licenses/CDDL-1.0 (as follows) + + + COMMON DEVELOPMENT AND DISTRIBUTION LICENSE (CDDL) Version 1.0 + + 1. Definitions. + + 1.1. "Contributor" means each individual or entity that + creates or contributes to the creation of Modifications. + + 1.2. "Contributor Version" means the combination of the + Original Software, prior Modifications used by a + Contributor (if any), and the Modifications made by that + particular Contributor. + + 1.3. "Covered Software" means (a) the Original Software, or + (b) Modifications, or (c) the combination of files + containing Original Software with files containing + Modifications, in each case including portions thereof. + + 1.4. "Executable" means the Covered Software in any form + other than Source Code. + + 1.5. "Initial Developer" means the individual or entity + that first makes Original Software available under this + License. + + 1.6. "Larger Work" means a work which combines Covered + Software or portions thereof with code not governed by the + terms of this License. + + 1.7. "License" means this document. + + 1.8. "Licensable" means having the right to grant, to the + maximum extent possible, whether at the time of the initial + grant or subsequently acquired, any and all of the rights + conveyed herein. + + 1.9. "Modifications" means the Source Code and Executable + form of any of the following: + + A. Any file that results from an addition to, + deletion from or modification of the contents of a + file containing Original Software or previous + Modifications; + + B. Any new file that contains any part of the + Original Software or previous Modification; or + + C. Any new file that is contributed or otherwise made + available under the terms of this License. + + 1.10. "Original Software" means the Source Code and + Executable form of computer software code that is + originally released under this License. + + 1.11. "Patent Claims" means any patent claim(s), now owned + or hereafter acquired, including without limitation, + method, process, and apparatus claims, in any patent + Licensable by grantor. + + 1.12. "Source Code" means (a) the common form of computer + software code in which modifications are made and (b) + associated documentation included in or with such code. + + 1.13. "You" (or "Your") means an individual or a legal + entity exercising rights under, and complying with all of + the terms of, this License. For legal entities, "You" + includes any entity which controls, is controlled by, or is + under common control with You. For purposes of this + definition, "control" means (a) the power, direct or + indirect, to cause the direction or management of such + entity, whether by contract or otherwise, or (b) ownership + of more than fifty percent (50%) of the outstanding shares + or beneficial ownership of such entity. + + 2. License Grants. + + 2.1. The Initial Developer Grant. + + Conditioned upon Your compliance with Section 3.1 below and + subject to third party intellectual property claims, the + Initial Developer hereby grants You a world-wide, + royalty-free, non-exclusive license: + + (a) under intellectual property rights (other than + patent or trademark) Licensable by Initial Developer, + to use, reproduce, modify, display, perform, + sublicense and distribute the Original Software (or + portions thereof), with or without Modifications, + and/or as part of a Larger Work; and + + (b) under Patent Claims infringed by the making, + using or selling of Original Software, to make, have + made, use, practice, sell, and offer for sale, and/or + otherwise dispose of the Original Software (or + portions thereof). + + (c) The licenses granted in Sections 2.1(a) and (b) + are effective on the date Initial Developer first + distributes or otherwise makes the Original Software + available to a third party under the terms of this + License. + + (d) Notwithstanding Section 2.1(b) above, no patent + license is granted: (1) for code that You delete from + the Original Software, or (2) for infringements + caused by: (i) the modification of the Original + Software, or (ii) the combination of the Original + Software with other software or devices. + + 2.2. Contributor Grant. + + Conditioned upon Your compliance with Section 3.1 below and + subject to third party intellectual property claims, each + Contributor hereby grants You a world-wide, royalty-free, + non-exclusive license: + + (a) under intellectual property rights (other than + patent or trademark) Licensable by Contributor to + use, reproduce, modify, display, perform, sublicense + and distribute the Modifications created by such + Contributor (or portions thereof), either on an + unmodified basis, with other Modifications, as + Covered Software and/or as part of a Larger Work; and + + (b) under Patent Claims infringed by the making, + using, or selling of Modifications made by that + Contributor either alone and/or in combination with + its Contributor Version (or portions of such + combination), to make, use, sell, offer for sale, + have made, and/or otherwise dispose of: (1) + Modifications made by that Contributor (or portions + thereof); and (2) the combination of Modifications + made by that Contributor with its Contributor Version + (or portions of such combination). + + (c) The licenses granted in Sections 2.2(a) and + 2.2(b) are effective on the date Contributor first + distributes or otherwise makes the Modifications + available to a third party. + + (d) Notwithstanding Section 2.2(b) above, no patent + license is granted: (1) for any code that Contributor + has deleted from the Contributor Version; (2) for + infringements caused by: (i) third party + modifications of Contributor Version, or (ii) the + combination of Modifications made by that Contributor + with other software (except as part of the + Contributor Version) or other devices; or (3) under + Patent Claims infringed by Covered Software in the + absence of Modifications made by that Contributor. + + 3. Distribution Obligations. + + 3.1. Availability of Source Code. + + Any Covered Software that You distribute or otherwise make + available in Executable form must also be made available in + Source Code form and that Source Code form must be + distributed only under the terms of this License. You must + include a copy of this License with every copy of the + Source Code form of the Covered Software You distribute or + otherwise make available. You must inform recipients of any + such Covered Software in Executable form as to how they can + obtain such Covered Software in Source Code form in a + reasonable manner on or through a medium customarily used + for software exchange. + + 3.2. Modifications. + + The Modifications that You create or to which You + contribute are governed by the terms of this License. You + represent that You believe Your Modifications are Your + original creation(s) and/or You have sufficient rights to + grant the rights conveyed by this License. + + 3.3. Required Notices. + + You must include a notice in each of Your Modifications + that identifies You as the Contributor of the Modification. + You may not remove or alter any copyright, patent or + trademark notices contained within the Covered Software, or + any notices of licensing or any descriptive text giving + attribution to any Contributor or the Initial Developer. + + 3.4. Application of Additional Terms. + + You may not offer or impose any terms on any Covered + Software in Source Code form that alters or restricts the + applicable version of this License or the recipients' + rights hereunder. You may choose to offer, and to charge a + fee for, warranty, support, indemnity or liability + obligations to one or more recipients of Covered Software. + However, you may do so only on Your own behalf, and not on + behalf of the Initial Developer or any Contributor. You + must make it absolutely clear that any such warranty, + support, indemnity or liability obligation is offered by + You alone, and You hereby agree to indemnify the Initial + Developer and every Contributor for any liability incurred + by the Initial Developer or such Contributor as a result of + warranty, support, indemnity or liability terms You offer. + + 3.5. Distribution of Executable Versions. + + You may distribute the Executable form of the Covered + Software under the terms of this License or under the terms + of a license of Your choice, which may contain terms + different from this License, provided that You are in + compliance with the terms of this License and that the + license for the Executable form does not attempt to limit + or alter the recipient's rights in the Source Code form + from the rights set forth in this License. If You + distribute the Covered Software in Executable form under a + different license, You must make it absolutely clear that + any terms which differ from this License are offered by You + alone, not by the Initial Developer or Contributor. You + hereby agree to indemnify the Initial Developer and every + Contributor for any liability incurred by the Initial + Developer or such Contributor as a result of any such terms + You offer. + + 3.6. Larger Works. + + You may create a Larger Work by combining Covered Software + with other code not governed by the terms of this License + and distribute the Larger Work as a single product. In such + a case, You must make sure the requirements of this License + are fulfilled for the Covered Software. + + 4. Versions of the License. + + 4.1. New Versions. + + Sun Microsystems, Inc. is the initial license steward and + may publish revised and/or new versions of this License + from time to time. Each version will be given a + distinguishing version number. Except as provided in + Section 4.3, no one other than the license steward has the + right to modify this License. + + 4.2. Effect of New Versions. + + You may always continue to use, distribute or otherwise + make the Covered Software available under the terms of the + version of the License under which You originally received + the Covered Software. If the Initial Developer includes a + notice in the Original Software prohibiting it from being + distributed or otherwise made available under any + subsequent version of the License, You must distribute and + make the Covered Software available under the terms of the + version of the License under which You originally received + the Covered Software. Otherwise, You may also choose to + use, distribute or otherwise make the Covered Software + available under the terms of any subsequent version of the + License published by the license steward. + + 4.3. Modified Versions. + + When You are an Initial Developer and You want to create a + new license for Your Original Software, You may create and + use a modified version of this License if You: (a) rename + the license and remove any references to the name of the + license steward (except to note that the license differs + from this License); and (b) otherwise make it clear that + the license contains terms which differ from this License. + + 5. DISCLAIMER OF WARRANTY. + + COVERED SOFTWARE IS PROVIDED UNDER THIS LICENSE ON AN "AS IS" + BASIS, WITHOUT WARRANTY OF ANY KIND, EITHER EXPRESSED OR IMPLIED, + INCLUDING, WITHOUT LIMITATION, WARRANTIES THAT THE COVERED + SOFTWARE IS FREE OF DEFECTS, MERCHANTABLE, FIT FOR A PARTICULAR + PURPOSE OR NON-INFRINGING. THE ENTIRE RISK AS TO THE QUALITY AND + PERFORMANCE OF THE COVERED SOFTWARE IS WITH YOU. SHOULD ANY + COVERED SOFTWARE PROVE DEFECTIVE IN ANY RESPECT, YOU (NOT THE + INITIAL DEVELOPER OR ANY OTHER CONTRIBUTOR) ASSUME THE COST OF + ANY NECESSARY SERVICING, REPAIR OR CORRECTION. THIS DISCLAIMER OF + WARRANTY CONSTITUTES AN ESSENTIAL PART OF THIS LICENSE. NO USE OF + ANY COVERED SOFTWARE IS AUTHORIZED HEREUNDER EXCEPT UNDER THIS + DISCLAIMER. + + 6. TERMINATION. + + 6.1. This License and the rights granted hereunder will + terminate automatically if You fail to comply with terms + herein and fail to cure such breach within 30 days of + becoming aware of the breach. Provisions which, by their + nature, must remain in effect beyond the termination of + this License shall survive. + + 6.2. If You assert a patent infringement claim (excluding + declaratory judgment actions) against Initial Developer or + a Contributor (the Initial Developer or Contributor against + whom You assert such claim is referred to as "Participant") + alleging that the Participant Software (meaning the + Contributor Version where the Participant is a Contributor + or the Original Software where the Participant is the + Initial Developer) directly or indirectly infringes any + patent, then any and all rights granted directly or + indirectly to You by such Participant, the Initial + Developer (if the Initial Developer is not the Participant) + and all Contributors under Sections 2.1 and/or 2.2 of this + License shall, upon 60 days notice from Participant + terminate prospectively and automatically at the expiration + of such 60 day notice period, unless if within such 60 day + period You withdraw Your claim with respect to the + Participant Software against such Participant either + unilaterally or pursuant to a written agreement with + Participant. + + 6.3. In the event of termination under Sections 6.1 or 6.2 + above, all end user licenses that have been validly granted + by You or any distributor hereunder prior to termination + (excluding licenses granted to You by any distributor) + shall survive termination. + + 7. LIMITATION OF LIABILITY. + + UNDER NO CIRCUMSTANCES AND UNDER NO LEGAL THEORY, WHETHER TORT + (INCLUDING NEGLIGENCE), CONTRACT, OR OTHERWISE, SHALL YOU, THE + INITIAL DEVELOPER, ANY OTHER CONTRIBUTOR, OR ANY DISTRIBUTOR OF + COVERED SOFTWARE, OR ANY SUPPLIER OF ANY OF SUCH PARTIES, BE + LIABLE TO ANY PERSON FOR ANY INDIRECT, SPECIAL, INCIDENTAL, OR + CONSEQUENTIAL DAMAGES OF ANY CHARACTER INCLUDING, WITHOUT + LIMITATION, DAMAGES FOR LOST PROFITS, LOSS OF GOODWILL, WORK + STOPPAGE, COMPUTER FAILURE OR MALFUNCTION, OR ANY AND ALL OTHER + COMMERCIAL DAMAGES OR LOSSES, EVEN IF SUCH PARTY SHALL HAVE BEEN + INFORMED OF THE POSSIBILITY OF SUCH DAMAGES. THIS LIMITATION OF + LIABILITY SHALL NOT APPLY TO LIABILITY FOR DEATH OR PERSONAL + INJURY RESULTING FROM SUCH PARTY'S NEGLIGENCE TO THE EXTENT + APPLICABLE LAW PROHIBITS SUCH LIMITATION. SOME JURISDICTIONS DO + NOT ALLOW THE EXCLUSION OR LIMITATION OF INCIDENTAL OR + CONSEQUENTIAL DAMAGES, SO THIS EXCLUSION AND LIMITATION MAY NOT + APPLY TO YOU. + + 8. U.S. GOVERNMENT END USERS. + + The Covered Software is a "commercial item," as that term is + defined in 48 C.F.R. 2.101 (Oct. 1995), consisting of "commercial + computer software" (as that term is defined at 48 C.F.R. ? + 252.227-7014(a)(1)) and "commercial computer software + documentation" as such terms are used in 48 C.F.R. 12.212 (Sept. + 1995). Consistent with 48 C.F.R. 12.212 and 48 C.F.R. 227.7202-1 + through 227.7202-4 (June 1995), all U.S. Government End Users + acquire Covered Software with only those rights set forth herein. + This U.S. Government Rights clause is in lieu of, and supersedes, + any other FAR, DFAR, or other clause or provision that addresses + Government rights in computer software under this License. + + 9. MISCELLANEOUS. + + This License represents the complete agreement concerning subject + matter hereof. If any provision of this License is held to be + unenforceable, such provision shall be reformed only to the + extent necessary to make it enforceable. This License shall be + governed by the law of the jurisdiction specified in a notice + contained within the Original Software (except to the extent + applicable law, if any, provides otherwise), excluding such + jurisdiction's conflict-of-law provisions. Any litigation + relating to this License shall be subject to the jurisdiction of + the courts located in the jurisdiction and venue specified in a + notice contained within the Original Software, with the losing + party responsible for costs, including, without limitation, court + costs and reasonable attorneys' fees and expenses. The + application of the United Nations Convention on Contracts for the + International Sale of Goods is expressly excluded. Any law or + regulation which provides that the language of a contract shall + be construed against the drafter shall not apply to this License. + You agree that You alone are responsible for compliance with the + United States export administration regulations (and the export + control laws and regulation of any other countries) when You use, + distribute or otherwise make available any Covered Software. + + 10. RESPONSIBILITY FOR CLAIMS. + + As between Initial Developer and the Contributors, each party is + responsible for claims and damages arising, directly or + indirectly, out of its utilization of rights under this License + and You agree to work with Initial Developer and Contributors to + distribute such responsibility on an equitable basis. Nothing + herein is intended or shall be deemed to constitute any admission + of liability. + + from Oracle http://www.oracle.com + mail-1.4.4.jar + + licensed under the Day Specification License with Addendum http://www.day.com/content/dam/day/downloads/jsr283/LICENSE.txt (as follows) + + + Day Management AG ("Licensor") is willing to license this specification to you ONLY UPON + THE CONDITION THAT YOU ACCEPT ALL OF THE TERMS CONTAINED IN THIS LICENSE AGREEMENT + ("Agreement"). Please read the terms and conditions of this Agreement carefully. + + Content Repository for JavaTM Technology API Specification ("Specification") + Version: 2.0 + Status: FCS + Release: 10 August 2009 + + Copyright 2009 Day Management AG + Barf?sserplatz 6, 4001 Basel, Switzerland. + All rights reserved. + + NOTICE; LIMITED LICENSE GRANTS + + 1. License for Purposes of Evaluation and Developing Applications. Licensor hereby grants + you a fully-paid, non-exclusive, non-transferable, worldwide, limited license (without the + right to sublicense), under Licensor's applicable intellectual property rights to view, + download, use and reproduce the Specification only for the purpose of internal evaluation. + This includes developing applications intended to run on an implementation of the + Specification provided that such applications do not themselves implement any portion(s) + of the Specification. + + 2. License for the Distribution of Compliant Implementations. Licensor also grants you a + perpetual, non-exclusive, non-transferable, worldwide, fully paid-up, royalty free, limited + license (without the right to sublicense) under any applicable copyrights or, subject to + the provisions of subsection 4 below, patent rights it may have covering the Specification + to create and/or distribute an Independent Implementation of the Specification that: + + (a) fully implements the Specification including all its required interfaces and + functionality; + (b) does not modify, subset, superset or otherwise extend the Licensor Name Space, + or include any public or protected packages, classes, Java interfaces, fields + or methods within the Licensor Name Space other than those required/authorized + by the Specification or Specifications being implemented; and + (c) passes the Technology Compatibility Kit (including satisfying the requirements + of the applicable TCK Users Guide) for such Specification ("Compliant Implementation"). + In addition, the foregoing license is expressly conditioned on your not acting + outside its scope. No license is granted hereunder for any other purpose (including, + for example, modifying the Specification, other than to the extent of your fair use + rights, or distributing the Specification to third parties). + + 3. Pass-through Conditions. You need not include limitations (a)-(c) from the previous paragraph + or any other particular "pass through" requirements in any license You grant concerning the + use of your Independent Implementation or products derived from it. However, except with + respect to Independent Implementations (and products derived from them) that satisfy + limitations (a)-(c) from the previous paragraph, You may neither: + + (a) grant or otherwise pass through to your licensees any licenses under Licensor's + applicable intellectual property rights; nor + (b) authorize your licensees to make any claims concerning their implementation's + compliance with the Specification. + + 4. Reciprocity Concerning Patent Licenses. With respect to any patent claims covered by the + license granted under subparagraph 2 above that would be infringed by all technically + feasible implementations of the Specification, such license is conditioned upon your + offering on fair, reasonable and non-discriminatory terms, to any party seeking it from + You, a perpetual, non-exclusive, non-transferable, worldwide license under Your patent + rights that are or would be infringed by all technically feasible implementations of the + Specification to develop, distribute and use a Compliant Implementation. + + 5. Definitions. For the purposes of this Agreement: "Independent Implementation" shall mean an + implementation of the Specification that neither derives from any of Licensor's source code + or binary code materials nor, except with an appropriate and separate license from Licensor, + includes any of Licensor's source code or binary code materials; "Licensor Name Space" shall + mean the public class or interface declarations whose names begin with "java", "javax", + "javax.jcr" or their equivalents in any subsequent naming convention adopted by Licensor + through the Java Community Process, or any recognized successors or replacements thereof; + and "Technology Compatibility Kit" or "TCK" shall mean the test suite and accompanying TCK + User's Guide provided by Licensor which corresponds to the particular version of the + Specification being tested. + + 6. Termination. This Agreement will terminate immediately without notice from Licensor if + you fail to comply with any material provision of or act outside the scope of the licenses + granted above. + + 7. Trademarks. No right, title, or interest in or to any trademarks, service marks, or trade + names of Licensor is granted hereunder. Java is a registered trademark of Sun Microsystems, + Inc. in the United States and other countries. + + 8. Disclaimer of Warranties. The Specification is provided "AS IS". LICENSOR MAKES NO + REPRESENTATIONS OR WARRANTIES, EITHER EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO, + WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE, NON-INFRINGEMENT + (INCLUDING AS A CONSEQUENCE OF ANY PRACTICE OR IMPLEMENTATION OF THE SPECIFICATION), + OR THAT THE CONTENTS OF THE SPECIFICATION ARE SUITABLE FOR ANY PURPOSE. This document + does not represent any commitment to release or implement any portion of the Specification + in any product. + + The Specification could include technical inaccuracies or typographical errors. Changes are + periodically added to the information therein; these changes will be incorporated into new + versions of the Specification, if any. Licensor may make improvements and/or changes to the + product(s) and/or the program(s) described in the Specification at any time. Any use of such + changes in the Specification will be governed by the then-current license for the applicable + version of the Specification. + + 9. Limitation of Liability. TO THE EXTENT NOT PROHIBITED BY LAW, IN NO EVENT WILL LICENSOR + BE LIABLE FOR ANY DAMAGES, INCLUDING WITHOUT LIMITATION, LOST REVENUE, PROFITS OR DATA, OR + FOR SPECIAL, INDIRECT, CONSEQUENTIAL, INCIDENTAL OR PUNITIVE DAMAGES, HOWEVER CAUSED AND + REGARDLESS OF THE THEORY OF LIABILITY, ARISING OUT OF OR RELATED TO ANY FURNISHING, + PRACTICING, MODIFYING OR ANY USE OF THE SPECIFICATION, EVEN IF LICENSOR HAS BEEN ADVISED + OF THE POSSIBILITY OF SUCH DAMAGES. + + 10. Report. If you provide Licensor with any comments or suggestions in connection with your + use of the Specification ("Feedback"), you hereby: (i) agree that such Feedback is provided + on a non-proprietary and non-confidential basis, and (ii) grant Licensor a perpetual, + non-exclusive, worldwide, fully paid-up, irrevocable license, with the right to sublicense + through multiple levels of sublicensees, to incorporate, disclose, and use without + limitation the Feedback for any purpose related to the Specification and future versions, + implementations, and test suites thereof. + + Day Specification License Addendum + + In addition to the permissions granted under the Specification + License, Day Management AG hereby grants to You a perpetual, + worldwide, non-exclusive, no-charge, royalty-free, irrevocable + license to reproduce, publicly display, publicly perform, + sublicense, and distribute unmodified copies of the Content + Repository for Java Technology API (JCR 2.0) Java Archive (JAR) + file ("jcr-2.0.jar") and to make, have made, use, offer to sell, + sell, import, and otherwise transfer said file on its own or + as part of a larger work that makes use of the JCR API. + + With respect to any patent claims covered by this license + that would be infringed by all technically feasible implementations + of the Specification, such license is conditioned upon your + offering on fair, reasonable and non-discriminatory terms, + to any party seeking it from You, a perpetual, non-exclusive, + non-transferable, worldwide license under Your patent rights + that are or would be infringed by all technically feasible + implementations of the Specification to develop, distribute + and use a Compliant Implementation. + + + from Day Software http://www.day.com + jcr-2.0.jar + + licensed under the MIT License http://www.opensource.org/licenses/mit-license.php (as follows) + + Copyright (c) 2004-2008 QOS.ch + All rights reserved. + + Permission is hereby granted, free of charge, to any person obtaining + a copy of this software and associated documentation files (the + "Software"), to deal in the Software without restriction, including + without limitation the rights to use, copy, modify, merge, publish, + distribute, sublicense, and/or sell copies of the Software, and to + permit persons to whom the Software is furnished to do so, subject to + the following conditions: + + The above copyright notice and this permission notice shall be + included in all copies or substantial portions of the Software. + + THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, + EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF + MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND + NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE + LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION + OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION + WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. + + from QOS.ch http://www.qos.ch + jcl-over-slf4j-1.6.1.jar + slf4j-api-1.6.1.jar + slf4j-log4j12-1.6.1.jar + + licensed under the Tanuki Software License (as follows) + + + Copyright (c) 1999, 2006 Tanuki Software, Inc. + + Permission is hereby granted, free of charge, to any person + obtaining a copy of the Java Service Wrapper and associated + documentation files (the "Software"), to deal in the Software + without restriction, including without limitation the rights + to use, copy, modify, merge, publish, distribute, sub-license, + and/or sell copies of the Software, and to permit persons to + whom the Software is furnished to do so, subject to the + following conditions: + + The above copyright notice and this permission notice shall be + included in all copies or substantial portions of the Software. + + THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, + EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES + OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND + NON-INFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT + HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, + WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING + FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR + OTHER DEALINGS IN THE SOFTWARE. + + + Portions of the Software have been derived from source code + developed by Silver Egg Technology under the following license: + + Copyright (c) 2001 Silver Egg Technology + + Permission is hereby granted, free of charge, to any person + obtaining a copy of this software and associated documentation + files (the "Software"), to deal in the Software without + restriction, including without limitation the rights to use, + copy, modify, merge, publish, distribute, sub-license, and/or + sell copies of the Software, and to permit persons to whom the + Software is furnished to do so, subject to the following + conditions: + + The above copyright notice and this permission notice shall be + included in all copies or substantial portions of the Software. + + THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, + EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES + OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND + NON-INFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT + HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, + WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING + FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR + OTHER DEALINGS IN THE SOFTWARE. + + from Tanuki Software http://www.tanukisoftware.com/ + libwrapper-linux-ppc-64.so + libwrapper-linux-x86-32.so + libwrapper-linux-x86-64.so + libwrapper-macosx-ppc-32.jnilib + libwrapper-macosx-universal-32.jnilib + libwrapper-solaris-sparc-32.so + libwrapper-solaris-sparc-64.so + libwrapper-solaris-x86-32.so + wrapper-windows-x86-32.dll + wrapper.jar + + + licensed under the Day Specification License http://www.day.com/content/dam/day/downloads/jsr283/LICENSE.txt (as follows) + + In addition to the permissions granted under the Specification + License, Day Management AG hereby grants to You a perpetual, + worldwide, non-exclusive, no-charge, royalty-free, irrevocable + license to reproduce, publicly display, publicly perform, + sublicense, and distribute unmodified copies of the Content + Repository for Java Technology API (JCR 2.0) Java Archive (JAR) + file ("jcr-2.0.jar") and to make, have made, use, offer to sell, + sell, import, and otherwise transfer said file on its own or + as part of a larger work that makes use of the JCR API. + + With respect to any patent claims covered by this license + that would be infringed by all technically feasible implementations + of the Specification, such license is conditioned upon your + offering on fair, reasonable and non-discriminatory terms, + to any party seeking it from You, a perpetual, non-exclusive, + non-transferable, worldwide license under Your patent rights + that are or would be infringed by all technically feasible + implementations of the Specification to develop, distribute + and use a Compliant Implementation. + + + licensed under the BSD (3-clause style) http://jetm.void.fm/license.html (as follows) + + Copyright (c) 2004, 2005, 2006, 2007 void.fm + All rights reserved. + + Redistribution and use in source and binary forms, with or without modification, + are permitted provided that the following conditions are met: + + * Redistributions of source code must retain the above copyright notice, this list + of conditions and the following disclaimer. + + * Redistributions in binary form must reproduce the above copyright notice, this + list of conditions and the following disclaimer in the documentation and/or + other materials provided with the distribution. + + * Neither the name void.fm nor the names of its contributors may be + used to endorse or promote products derived from this software without specific + prior written permission. + + THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND + ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED + WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. + IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, + INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, + BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, + DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF + LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE + OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED + OF THE POSSIBILITY OF SUCH DAMAGE. + + from JETM http://jetm.void.fm + jetm-1.2.3.jar + jetm-optional-1.2.3.jar diff --git a/server/apps/postgres-app/src/main/extensions-jars/README.md b/server/apps/postgres-app/src/main/extensions-jars/README.md new file mode 100644 index 00000000000..dab5c40e60d --- /dev/null +++ b/server/apps/postgres-app/src/main/extensions-jars/README.md @@ -0,0 +1,5 @@ +# Adding Jars to JAMES + +The jar in this folder will be added to JAMES classpath when mounted under /root/extensions-jars inside the running container. + +You can use it to add your custom Mailets/Matchers. diff --git a/server/apps/postgres-app/src/main/glowroot/admin.json b/server/apps/postgres-app/src/main/glowroot/admin.json new file mode 100644 index 00000000000..c75c59d555a --- /dev/null +++ b/server/apps/postgres-app/src/main/glowroot/admin.json @@ -0,0 +1,5 @@ +{ + "web": { + "bindAddress": "0.0.0.0" + } +} \ No newline at end of file diff --git a/server/apps/postgres-app/src/main/glowroot/plugins/imap.json b/server/apps/postgres-app/src/main/glowroot/plugins/imap.json new file mode 100644 index 00000000000..d27904feb5e --- /dev/null +++ b/server/apps/postgres-app/src/main/glowroot/plugins/imap.json @@ -0,0 +1,19 @@ +{ + "name": "IMAP Plugin", + "id": "imap", + "instrumentation": [ + { + "className": "org.apache.james.imap.processor.base.AbstractChainedProcessor", + "methodName": "doProcess", + "methodParameterTypes": [ + ".." + ], + "captureKind": "transaction", + "transactionType": "IMAP", + "transactionNameTemplate": "IMAP processor : {{this.class.name}}", + "alreadyInTransactionBehavior": "capture-trace-entry", + "traceEntryMessageTemplate": "{{this.class.name}}.{{methodName}}", + "timerName": "imapProcessor" + } + ] +} \ No newline at end of file diff --git a/server/apps/postgres-app/src/main/glowroot/plugins/jmap.json b/server/apps/postgres-app/src/main/glowroot/plugins/jmap.json new file mode 100644 index 00000000000..9afce4bf94c --- /dev/null +++ b/server/apps/postgres-app/src/main/glowroot/plugins/jmap.json @@ -0,0 +1,19 @@ +{ + "name": "JMAP Plugin", + "id": "jmap", + "instrumentation": [ + { + "className": "org.apache.james.jmap.draft.methods.Method", + "methodName": "processToStream", + "methodParameterTypes": [ + ".." + ], + "captureKind": "transaction", + "transactionType": "JMAP", + "transactionNameTemplate": "JMAP method : {{this.class.name}}", + "alreadyInTransactionBehavior": "capture-trace-entry", + "traceEntryMessageTemplate": "{{this.class.name}}.{{methodName}}", + "timerName": "jmapMethod" + } + ] +} \ No newline at end of file diff --git a/server/apps/postgres-app/src/main/glowroot/plugins/mailboxListener.json b/server/apps/postgres-app/src/main/glowroot/plugins/mailboxListener.json new file mode 100644 index 00000000000..54a55ac1e4c --- /dev/null +++ b/server/apps/postgres-app/src/main/glowroot/plugins/mailboxListener.json @@ -0,0 +1,19 @@ +{ + "name": "MailboxListener Plugin", + "id": "mailboxListener", + "instrumentation": [ + { + "className": "org.apache.james.mailbox.events.MailboxListener", + "methodName": "event", + "methodParameterTypes": [ + ".." + ], + "captureKind": "transaction", + "transactionType": "MailboxListener", + "transactionNameTemplate": "MailboxListener : {{this.class.name}}", + "alreadyInTransactionBehavior": "capture-trace-entry", + "traceEntryMessageTemplate": "{{this.class.name}}.{{methodName}}", + "timerName": "mailboxListener" + } + ] +} \ No newline at end of file diff --git a/server/apps/postgres-app/src/main/glowroot/plugins/pop3.json b/server/apps/postgres-app/src/main/glowroot/plugins/pop3.json new file mode 100644 index 00000000000..a5bcdccce1f --- /dev/null +++ b/server/apps/postgres-app/src/main/glowroot/plugins/pop3.json @@ -0,0 +1,19 @@ +{ + "name": "POP3 Plugin", + "id": "pop3", + "instrumentation": [ + { + "className": "org.apache.james.protocols.pop3.core.AbstractPOP3CommandHandler", + "methodName": "onCommand", + "methodParameterTypes": [ + ".." + ], + "captureKind": "transaction", + "transactionType": "POP3", + "transactionNameTemplate": "POP3 Command: {{this.class.name}}", + "alreadyInTransactionBehavior": "capture-trace-entry", + "traceEntryMessageTemplate": "{{this.class.name}}.{{methodName}}", + "timerName": "pop3Timer" + } + ] +} diff --git a/server/apps/postgres-app/src/main/glowroot/plugins/smtp.json b/server/apps/postgres-app/src/main/glowroot/plugins/smtp.json new file mode 100644 index 00000000000..393bac9d9c3 --- /dev/null +++ b/server/apps/postgres-app/src/main/glowroot/plugins/smtp.json @@ -0,0 +1,19 @@ +{ + "name": "SMTP Plugin", + "id": "smtp", + "instrumentation": [ + { + "className": "org.apache.james.protocols.smtp.core.AbstractHookableCmdHandler", + "methodName": "onCommand", + "methodParameterTypes": [ + ".." + ], + "captureKind": "transaction", + "transactionType": "SMTP", + "transactionNameTemplate": "SMTP command : {{this.class.name}}", + "alreadyInTransactionBehavior": "capture-trace-entry", + "traceEntryMessageTemplate": "{{this.class.name}}.{{methodName}}", + "timerName": "smtpProcessor" + } + ] +} \ No newline at end of file diff --git a/server/apps/postgres-app/src/main/glowroot/plugins/spooler.json b/server/apps/postgres-app/src/main/glowroot/plugins/spooler.json new file mode 100644 index 00000000000..fd7732de8b2 --- /dev/null +++ b/server/apps/postgres-app/src/main/glowroot/plugins/spooler.json @@ -0,0 +1,45 @@ +{ + "name": "Spooler Plugin", + "id": "spooler", + "instrumentation": [ + { + "className": "org.apache.james.mailetcontainer.api.MailProcessor", + "methodName": "service", + "methodParameterTypes": [ + ".." + ], + "captureKind": "transaction", + "transactionType": "Spooler", + "transactionNameTemplate": "Mailet processor : {{this.class.name}}", + "alreadyInTransactionBehavior": "capture-trace-entry", + "traceEntryMessageTemplate": "{{this.class.name}}.{{methodName}}", + "timerName": "mailetProcessor" + }, + { + "className": "org.apache.mailet.Mailet", + "methodName": "service", + "methodParameterTypes": [ + ".." + ], + "captureKind": "transaction", + "transactionType": "Mailet", + "transactionNameTemplate": "Mailet : {{this.class.name}}", + "alreadyInTransactionBehavior": "capture-trace-entry", + "traceEntryMessageTemplate": "{{this.class.name}}.{{methodName}}", + "timerName": "mailet" + }, + { + "className": "org.apache.mailet.Matcher", + "methodName": "match", + "methodParameterTypes": [ + ".." + ], + "captureKind": "transaction", + "transactionType": "Matcher", + "transactionNameTemplate": "Mailet processor : {{this.class.name}}", + "alreadyInTransactionBehavior": "capture-trace-entry", + "traceEntryMessageTemplate": "{{this.class.name}}.{{methodName}}", + "timerName": "matcher" + } + ] +} \ No newline at end of file diff --git a/server/apps/postgres-app/src/main/glowroot/plugins/task.json b/server/apps/postgres-app/src/main/glowroot/plugins/task.json new file mode 100644 index 00000000000..8f04c69e741 --- /dev/null +++ b/server/apps/postgres-app/src/main/glowroot/plugins/task.json @@ -0,0 +1,19 @@ +{ + "name": "Task Plugin", + "id": "task", + "instrumentation": [ + { + "className": "org.apache.james.task.Task", + "methodName": "run", + "methodParameterTypes": [ + ".." + ], + "captureKind": "transaction", + "transactionType": "TASK", + "transactionNameTemplate": "TASK : {{this.class.name}}", + "alreadyInTransactionBehavior": "capture-trace-entry", + "traceEntryMessageTemplate": "{{this.class.name}}.{{methodName}}", + "timerName": "task" + } + ] +} \ No newline at end of file diff --git a/server/apps/postgres-app/src/main/java/org/apache/james/PostgresJamesConfiguration.java b/server/apps/postgres-app/src/main/java/org/apache/james/PostgresJamesConfiguration.java new file mode 100644 index 00000000000..21b9c633c79 --- /dev/null +++ b/server/apps/postgres-app/src/main/java/org/apache/james/PostgresJamesConfiguration.java @@ -0,0 +1,320 @@ +/**************************************************************** + * 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; + +import java.io.File; +import java.io.FileNotFoundException; +import java.util.Optional; + +import org.apache.commons.configuration2.ex.ConfigurationException; +import org.apache.james.backends.postgres.PostgresConfiguration; +import org.apache.james.data.UsersRepositoryModuleChooser; +import org.apache.james.filesystem.api.FileSystem; +import org.apache.james.filesystem.api.JamesDirectoriesProvider; +import org.apache.james.jmap.JMAPModule; +import org.apache.james.modules.blobstore.BlobStoreConfiguration; +import org.apache.james.server.core.JamesServerResourceLoader; +import org.apache.james.server.core.MissingArgumentException; +import org.apache.james.server.core.configuration.Configuration; +import org.apache.james.server.core.configuration.FileConfigurationProvider; +import org.apache.james.server.core.filesystem.FileSystemImpl; +import org.apache.james.utils.PropertiesProvider; +import org.apache.james.vault.VaultConfiguration; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + +import com.github.fge.lambdas.Throwing; +import com.google.common.base.Preconditions; + +public class PostgresJamesConfiguration implements Configuration { + + private static final Logger LOGGER = LoggerFactory.getLogger("org.apache.james.CONFIGURATION"); + + private static final BlobStoreConfiguration.BlobStoreImplName DEFAULT_BLOB_STORE = BlobStoreConfiguration.BlobStoreImplName.POSTGRES; + + public enum EventBusImpl { + IN_MEMORY, RABBITMQ; + + public static EventBusImpl from(PropertiesProvider configurationProvider) { + try { + configurationProvider.getConfiguration("rabbitmq"); + return EventBusImpl.RABBITMQ; + } catch (FileNotFoundException e) { + LOGGER.info("RabbitMQ configuration was not found, defaulting to in memory event bus"); + return EventBusImpl.IN_MEMORY; + } catch (ConfigurationException e) { + LOGGER.warn("Error reading rabbitmq.xml, defaulting to in memory event bus", e); + return EventBusImpl.IN_MEMORY; + } + } + } + + public static class Builder { + private Optional rootDirectory; + private Optional configurationPath; + private Optional usersRepositoryImplementation; + private Optional searchConfiguration; + private Optional blobStoreConfiguration; + private Optional eventBusImpl; + private Optional deletedMessageVaultConfiguration; + private Optional jmapEnabled; + private Optional dropListsEnabled; + private Optional rlsEnabled; + + private Builder() { + searchConfiguration = Optional.empty(); + rootDirectory = Optional.empty(); + configurationPath = Optional.empty(); + usersRepositoryImplementation = Optional.empty(); + blobStoreConfiguration = Optional.empty(); + eventBusImpl = Optional.empty(); + deletedMessageVaultConfiguration = Optional.empty(); + jmapEnabled = Optional.empty(); + dropListsEnabled = Optional.empty(); + rlsEnabled = Optional.empty(); + } + + public Builder workingDirectory(String path) { + rootDirectory = Optional.of(path); + return this; + } + + public Builder workingDirectory(File file) { + rootDirectory = Optional.of(file.getAbsolutePath()); + return this; + } + + public Builder useWorkingDirectoryEnvProperty() { + rootDirectory = Optional.ofNullable(System.getProperty(WORKING_DIRECTORY)); + if (!rootDirectory.isPresent()) { + throw new MissingArgumentException("Server needs a working.directory env entry"); + } + return this; + } + + public Builder configurationPath(ConfigurationPath path) { + configurationPath = Optional.of(path); + return this; + } + + public Builder configurationFromClasspath() { + configurationPath = Optional.of(new ConfigurationPath(FileSystem.CLASSPATH_PROTOCOL)); + return this; + } + + public Builder usersRepository(UsersRepositoryModuleChooser.Implementation implementation) { + this.usersRepositoryImplementation = Optional.of(implementation); + return this; + } + + public Builder searchConfiguration(SearchConfiguration searchConfiguration) { + this.searchConfiguration = Optional.of(searchConfiguration); + return this; + } + + public Builder blobStore(BlobStoreConfiguration blobStoreConfiguration) { + this.blobStoreConfiguration = Optional.of(blobStoreConfiguration); + return this; + } + + public Builder eventBusImpl(EventBusImpl eventBusImpl) { + this.eventBusImpl = Optional.of(eventBusImpl); + return this; + } + + public Builder deletedMessageVaultConfiguration(VaultConfiguration vaultConfiguration) { + this.deletedMessageVaultConfiguration = Optional.of(vaultConfiguration); + return this; + } + + public Builder jmapEnabled(Optional jmapEnabled) { + this.jmapEnabled = jmapEnabled; + return this; + } + + public Builder enableDropLists() { + this.dropListsEnabled = Optional.of(true); + return this; + } + + public Builder rlsEnabled(Optional rlsEnabled) { + this.rlsEnabled = rlsEnabled; + return this; + } + + public PostgresJamesConfiguration build() { + ConfigurationPath configurationPath = this.configurationPath.orElse(new ConfigurationPath(FileSystem.FILE_PROTOCOL_AND_CONF)); + JamesServerResourceLoader directories = new JamesServerResourceLoader(rootDirectory + .orElseThrow(() -> new MissingArgumentException("Server needs a working.directory env entry"))); + + FileSystemImpl fileSystem = new FileSystemImpl(directories); + PropertiesProvider propertiesProvider = new PropertiesProvider(fileSystem, configurationPath); + + SearchConfiguration searchConfiguration = this.searchConfiguration.orElseGet(Throwing.supplier( + () -> SearchConfiguration.parse(propertiesProvider))); + + BlobStoreConfiguration blobStoreConfiguration = this.blobStoreConfiguration.orElseGet(Throwing.supplier( + () -> BlobStoreConfiguration.parse(propertiesProvider, DEFAULT_BLOB_STORE))); + Preconditions.checkState(!blobStoreConfiguration.getImplementation().equals(BlobStoreConfiguration.BlobStoreImplName.CASSANDRA), "Cassandra BlobStore is not supported by postgres-app."); + Preconditions.checkState(!blobStoreConfiguration.cacheEnabled(), "BlobStore caching is not supported by postgres-app."); + + FileConfigurationProvider configurationProvider = new FileConfigurationProvider(fileSystem, Basic.builder() + .configurationPath(configurationPath) + .workingDirectory(directories.getRootDirectory()) + .build()); + UsersRepositoryModuleChooser.Implementation usersRepositoryChoice = usersRepositoryImplementation.orElseGet( + () -> UsersRepositoryModuleChooser.Implementation.parse(configurationProvider)); + + EventBusImpl eventBusImpl = this.eventBusImpl.orElseGet(() -> EventBusImpl.from(propertiesProvider)); + + VaultConfiguration deletedMessageVaultConfiguration = this.deletedMessageVaultConfiguration.orElseGet(() -> { + try { + return VaultConfiguration.from(propertiesProvider.getConfiguration("deletedMessageVault")); + } catch (FileNotFoundException e) { + return VaultConfiguration.DEFAULT; + } catch (ConfigurationException e) { + throw new RuntimeException(e); + } + }); + + boolean rlsEnabled = this.rlsEnabled.orElse(readRLSEnabledFromFile(propertiesProvider)); + + boolean jmapEnabled = this.jmapEnabled.orElseGet(() -> { + try { + return JMAPModule.parseConfiguration(propertiesProvider).isEnabled(); + } catch (FileNotFoundException e) { + return false; + } catch (ConfigurationException e) { + throw new RuntimeException(e); + } + }); + + boolean dropListsEnabled = this.dropListsEnabled.orElseGet(() -> { + try { + return configurationProvider.getConfiguration("droplists").getBoolean("enabled", false); + } catch (ConfigurationException e) { + return false; + } + }); + + LOGGER.info("BlobStore configuration {}", blobStoreConfiguration); + return new PostgresJamesConfiguration( + configurationPath, + directories, + searchConfiguration, + usersRepositoryChoice, + blobStoreConfiguration, + eventBusImpl, + deletedMessageVaultConfiguration, + jmapEnabled, + dropListsEnabled, + rlsEnabled); + } + + private boolean readRLSEnabledFromFile(PropertiesProvider propertiesProvider) { + try { + return PostgresConfiguration.from(propertiesProvider.getConfiguration(PostgresConfiguration.POSTGRES_CONFIGURATION_NAME)) + .getRowLevelSecurity() + .isRowLevelSecurityEnabled(); + } catch (FileNotFoundException | ConfigurationException e) { + return false; + } + } + } + + public static Builder builder() { + return new Builder(); + } + + private final ConfigurationPath configurationPath; + private final JamesDirectoriesProvider directories; + private final SearchConfiguration searchConfiguration; + private final UsersRepositoryModuleChooser.Implementation usersRepositoryImplementation; + private final BlobStoreConfiguration blobStoreConfiguration; + private final EventBusImpl eventBusImpl; + private final VaultConfiguration deletedMessageVaultConfiguration; + private final boolean jmapEnabled; + private final boolean dropListsEnabled; + private final boolean rlsEnabled; + + private PostgresJamesConfiguration(ConfigurationPath configurationPath, + JamesDirectoriesProvider directories, + SearchConfiguration searchConfiguration, + UsersRepositoryModuleChooser.Implementation usersRepositoryImplementation, + BlobStoreConfiguration blobStoreConfiguration, + EventBusImpl eventBusImpl, + VaultConfiguration deletedMessageVaultConfiguration, + boolean jmapEnabled, + boolean dropListsEnabled, + boolean rlsEnabled) { + this.configurationPath = configurationPath; + this.directories = directories; + this.searchConfiguration = searchConfiguration; + this.usersRepositoryImplementation = usersRepositoryImplementation; + this.blobStoreConfiguration = blobStoreConfiguration; + this.eventBusImpl = eventBusImpl; + this.deletedMessageVaultConfiguration = deletedMessageVaultConfiguration; + this.jmapEnabled = jmapEnabled; + this.dropListsEnabled = dropListsEnabled; + this.rlsEnabled = rlsEnabled; + } + + @Override + public ConfigurationPath configurationPath() { + return configurationPath; + } + + @Override + public JamesDirectoriesProvider directories() { + return directories; + } + + public SearchConfiguration searchConfiguration() { + return searchConfiguration; + } + + public UsersRepositoryModuleChooser.Implementation getUsersRepositoryImplementation() { + return usersRepositoryImplementation; + } + + public BlobStoreConfiguration blobStoreConfiguration() { + return blobStoreConfiguration; + } + + public EventBusImpl eventBusImpl() { + return eventBusImpl; + } + + public VaultConfiguration getDeletedMessageVaultConfiguration() { + return deletedMessageVaultConfiguration; + } + + public boolean isJmapEnabled() { + return jmapEnabled; + } + + public boolean isDropListsEnabled() { + return dropListsEnabled; + } + + public boolean isRlsEnabled() { + return rlsEnabled; + } +} diff --git a/server/apps/postgres-app/src/main/java/org/apache/james/PostgresJamesServerMain.java b/server/apps/postgres-app/src/main/java/org/apache/james/PostgresJamesServerMain.java new file mode 100644 index 00000000000..75bf39d9461 --- /dev/null +++ b/server/apps/postgres-app/src/main/java/org/apache/james/PostgresJamesServerMain.java @@ -0,0 +1,272 @@ +/**************************************************************** + * 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; + +import java.util.List; +import java.util.Set; +import java.util.function.Function; + +import org.apache.james.data.UsersRepositoryModuleChooser; +import org.apache.james.eventsourcing.eventstore.EventNestedTypes; +import org.apache.james.jmap.JMAPListenerModule; +import org.apache.james.json.DTO; +import org.apache.james.json.DTOModule; +import org.apache.james.modules.BlobExportMechanismModule; +import org.apache.james.modules.DistributedTaskSerializationModule; +import org.apache.james.modules.MailboxModule; +import org.apache.james.modules.MailetProcessingModule; +import org.apache.james.modules.RunArgumentsModule; +import org.apache.james.modules.TasksCleanupTaskSerializationModule; +import org.apache.james.modules.blobstore.BlobStoreCacheModulesChooser; +import org.apache.james.modules.blobstore.BlobStoreModulesChooser; +import org.apache.james.modules.data.PostgresDLPConfigurationStoreModule; +import org.apache.james.modules.data.PostgresDataJmapModule; +import org.apache.james.modules.data.PostgresDataModule; +import org.apache.james.modules.data.PostgresDelegationStoreModule; +import org.apache.james.modules.data.PostgresDropListsModule; +import org.apache.james.modules.data.PostgresEventStoreModule; +import org.apache.james.modules.data.PostgresUsersRepositoryModule; +import org.apache.james.modules.data.PostgresVacationModule; +import org.apache.james.modules.data.SievePostgresRepositoryModules; +import org.apache.james.modules.event.JMAPEventBusModule; +import org.apache.james.modules.event.RabbitMQEventBusModule; +import org.apache.james.modules.events.PostgresDeadLetterModule; +import org.apache.james.modules.mailbox.DefaultEventModule; +import org.apache.james.modules.mailbox.PostgresDeletedMessageVaultModule; +import org.apache.james.modules.mailbox.PostgresMailboxModule; +import org.apache.james.modules.mailbox.RLSSupportPostgresMailboxModule; +import org.apache.james.modules.mailbox.TikaMailboxModule; +import org.apache.james.modules.plugins.QuotaMailingModule; +import org.apache.james.modules.protocols.IMAPServerModule; +import org.apache.james.modules.protocols.JMAPServerModule; +import org.apache.james.modules.protocols.JmapEventBusModule; +import org.apache.james.modules.protocols.LMTPServerModule; +import org.apache.james.modules.protocols.ManageSieveServerModule; +import org.apache.james.modules.protocols.POP3ServerModule; +import org.apache.james.modules.protocols.ProtocolHandlerModule; +import org.apache.james.modules.protocols.SMTPServerModule; +import org.apache.james.modules.queue.activemq.ActiveMQQueueModule; +import org.apache.james.modules.queue.rabbitmq.FakeMailQueueViewModule; +import org.apache.james.modules.queue.rabbitmq.RabbitMQMailQueueModule; +import org.apache.james.modules.queue.rabbitmq.RabbitMQModule; +import org.apache.james.modules.server.DKIMMailetModule; +import org.apache.james.modules.server.DLPRoutesModule; +import org.apache.james.modules.server.DataRoutesModules; +import org.apache.james.modules.server.DropListsRoutesModule; +import org.apache.james.modules.server.InconsistencyQuotasSolvingRoutesModule; +import org.apache.james.modules.server.JMXServerModule; +import org.apache.james.modules.server.JmapTasksModule; +import org.apache.james.modules.server.JmapUploadCleanupModule; +import org.apache.james.modules.server.MailQueueRoutesModule; +import org.apache.james.modules.server.MailRepositoriesRoutesModule; +import org.apache.james.modules.server.MailboxRoutesModule; +import org.apache.james.modules.server.MailboxesExportRoutesModule; +import org.apache.james.modules.server.RabbitMailQueueRoutesModule; +import org.apache.james.modules.server.ReIndexingModule; +import org.apache.james.modules.server.SieveRoutesModule; +import org.apache.james.modules.server.TaskManagerModule; +import org.apache.james.modules.server.UserIdentityModule; +import org.apache.james.modules.server.WebAdminReIndexingTaskSerializationModule; +import org.apache.james.modules.server.WebAdminServerModule; +import org.apache.james.modules.task.DistributedTaskManagerModule; +import org.apache.james.modules.task.PostgresTaskExecutionDetailsProjectionGuiceModule; +import org.apache.james.modules.vault.DeletedMessageVaultRoutesModule; +import org.apache.james.modules.webadmin.TasksCleanupRoutesModule; +import org.apache.james.vault.VaultConfiguration; + +import com.google.common.collect.ImmutableList; +import com.google.common.collect.ImmutableSet; +import com.google.inject.Module; +import com.google.inject.TypeLiteral; +import com.google.inject.name.Names; +import com.google.inject.util.Modules; + +public class PostgresJamesServerMain implements JamesServerMain { + + private static final Module EVENT_STORE_JSON_SERIALIZATION_DEFAULT_MODULE = binder -> + binder.bind(new TypeLiteral>>() { + }).annotatedWith(Names.named(EventNestedTypes.EVENT_NESTED_TYPES_INJECTION_NAME)) + .toInstance(ImmutableSet.of()); + + private static final Module WEBADMIN = Modules.combine( + new WebAdminServerModule(), + new DataRoutesModules(), + new InconsistencyQuotasSolvingRoutesModule(), + new MailboxRoutesModule(), + new MailQueueRoutesModule(), + new MailRepositoriesRoutesModule(), + new ReIndexingModule(), + new SieveRoutesModule(), + new WebAdminReIndexingTaskSerializationModule(), + new MailboxesExportRoutesModule(), + new UserIdentityModule(), + new DLPRoutesModule(), + new JmapUploadCleanupModule(), + new JmapTasksModule(), + new TasksCleanupRoutesModule(), + new TasksCleanupTaskSerializationModule()); + + private static final Module PROTOCOLS = Modules.combine( + new IMAPServerModule(), + new LMTPServerModule(), + new ManageSieveServerModule(), + new POP3ServerModule(), + new ProtocolHandlerModule(), + new SMTPServerModule(), + WEBADMIN); + + private static final Module POSTGRES_SERVER_MODULE = Modules.combine( + new BlobExportMechanismModule(), + new PostgresDelegationStoreModule(), + new PostgresMailboxModule(), + new PostgresDeadLetterModule(), + new PostgresDataModule(), + new MailboxModule(), + new SievePostgresRepositoryModules(), + new PostgresEventStoreModule(), + new TikaMailboxModule(), + new PostgresDLPConfigurationStoreModule(), + new PostgresVacationModule(), + EVENT_STORE_JSON_SERIALIZATION_DEFAULT_MODULE); + + public static final Module JMAP = Modules.combine( + new PostgresJmapModule(), + new PostgresDataJmapModule(), + new JmapEventBusModule(), + new JMAPServerModule()); + + public static final Module PLUGINS = new QuotaMailingModule(); + + private static final Function POSTGRES_MODULE_AGGREGATE = configuration -> + Modules.override(Modules.combine( + new MailetProcessingModule(), + new DKIMMailetModule(), + POSTGRES_SERVER_MODULE, + JMAP, + PROTOCOLS, + PLUGINS)) + .with(chooseEventBusModules(configuration)); + + public static void main(String[] args) throws Exception { + ExtraProperties.initialize(); + + PostgresJamesConfiguration configuration = PostgresJamesConfiguration.builder() + .useWorkingDirectoryEnvProperty() + .build(); + + LOGGER.info("Loading configuration {}", configuration.toString()); + GuiceJamesServer server = createServer(configuration) + .combineWith(new JMXServerModule()) + .overrideWith(new RunArgumentsModule(args)); + + JamesServerMain.main(server); + } + + public static GuiceJamesServer createServer(PostgresJamesConfiguration configuration) { + SearchConfiguration searchConfiguration = configuration.searchConfiguration(); + + return GuiceJamesServer.forConfiguration(configuration) + .combineWith(POSTGRES_MODULE_AGGREGATE.apply(configuration)) + .combineWith(SearchModuleChooser.chooseModules(searchConfiguration)) + .combineWith(chooseUsersRepositoryModule(configuration)) + .combineWith(chooseBlobStoreModules(configuration)) + .combineWith(chooseDeletedMessageVaultModules(configuration.getDeletedMessageVaultConfiguration())) + .combineWith(chooseRLSSupportPostgresMailboxModule(configuration)) + .overrideWith(chooseJmapModules(configuration)) + .overrideWith(chooseTaskManagerModules(configuration)) + .overrideWith(chooseDropListsModule(configuration)); + } + + private static List chooseUsersRepositoryModule(PostgresJamesConfiguration configuration) { + return List.of(PostgresUsersRepositoryModule.USER_CONFIGURATION_MODULE, + Modules.combine(new UsersRepositoryModuleChooser(new PostgresUsersRepositoryModule()) + .chooseModules(configuration.getUsersRepositoryImplementation()))); + } + + private static List chooseBlobStoreModules(PostgresJamesConfiguration configuration) { + ImmutableList.Builder builder = ImmutableList.builder() + .addAll(BlobStoreModulesChooser.chooseModules(configuration.blobStoreConfiguration())) + .add(new BlobStoreCacheModulesChooser.CacheDisabledModule()); + + return builder.build(); + } + + public static List chooseTaskManagerModules(PostgresJamesConfiguration configuration) { + switch (configuration.eventBusImpl()) { + case IN_MEMORY: + return List.of(new TaskManagerModule(), new PostgresTaskExecutionDetailsProjectionGuiceModule()); + case RABBITMQ: + return List.of(new DistributedTaskManagerModule()); + default: + throw new RuntimeException("Unsupported event-bus implementation " + configuration.eventBusImpl().name()); + } + } + + public static List chooseEventBusModules(PostgresJamesConfiguration configuration) { + switch (configuration.eventBusImpl()) { + case IN_MEMORY: + return List.of( + new DefaultEventModule(), + new ActiveMQQueueModule()); + case RABBITMQ: + return List.of( + Modules.override(new DefaultEventModule()).with(new RabbitMQEventBusModule()), + new RabbitMQModule(), + new RabbitMQMailQueueModule(), + new FakeMailQueueViewModule(), + new RabbitMailQueueRoutesModule(), + new DistributedTaskSerializationModule()); + default: + throw new RuntimeException("Unsupported event-bus implementation " + configuration.eventBusImpl().name()); + } + } + + private static Module chooseDeletedMessageVaultModules(VaultConfiguration vaultConfiguration) { + if (vaultConfiguration.isEnabled()) { + return Modules.combine(new PostgresDeletedMessageVaultModule(), new DeletedMessageVaultRoutesModule()); + } + + return Modules.EMPTY_MODULE; + } + + private static Module chooseJmapModules(PostgresJamesConfiguration configuration) { + if (configuration.isJmapEnabled()) { + return Modules.combine(new JMAPEventBusModule(), new JMAPListenerModule()); + } + return binder -> { + }; + } + + private static Module chooseDropListsModule(PostgresJamesConfiguration configuration) { + if (configuration.isDropListsEnabled()) { + return Modules.combine(new PostgresDropListsModule(), new DropListsRoutesModule()); + } + return binder -> { + + }; + } + + private static Module chooseRLSSupportPostgresMailboxModule(PostgresJamesConfiguration configuration) { + if (configuration.isRlsEnabled()) { + return new RLSSupportPostgresMailboxModule(); + } + return Modules.EMPTY_MODULE; + } +} diff --git a/server/apps/postgres-app/src/main/java/org/apache/james/PostgresJmapModule.java b/server/apps/postgres-app/src/main/java/org/apache/james/PostgresJmapModule.java new file mode 100644 index 00000000000..2d85fd2b8e5 --- /dev/null +++ b/server/apps/postgres-app/src/main/java/org/apache/james/PostgresJmapModule.java @@ -0,0 +1,84 @@ +/**************************************************************** + * 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; + +import org.apache.james.backends.postgres.PostgresModule; +import org.apache.james.eventsourcing.Event; +import org.apache.james.eventsourcing.eventstore.dto.EventDTO; +import org.apache.james.eventsourcing.eventstore.dto.EventDTOModule; +import org.apache.james.jmap.api.change.EmailChangeRepository; +import org.apache.james.jmap.api.change.Limit; +import org.apache.james.jmap.api.change.MailboxChangeRepository; +import org.apache.james.jmap.api.change.State; +import org.apache.james.jmap.api.filtering.FilteringRuleSetDefineDTOModules; +import org.apache.james.jmap.api.pushsubscription.PushSubscriptionRepository; +import org.apache.james.jmap.api.upload.UploadUsageRepository; +import org.apache.james.jmap.postgres.PostgresDataJMapAggregateModule; +import org.apache.james.jmap.postgres.change.PostgresEmailChangeRepository; +import org.apache.james.jmap.postgres.change.PostgresMailboxChangeRepository; +import org.apache.james.jmap.postgres.change.PostgresStateFactory; +import org.apache.james.jmap.postgres.pushsubscription.PostgresPushSubscriptionRepository; +import org.apache.james.jmap.postgres.upload.PostgresUploadUsageRepository; +import org.apache.james.mailbox.AttachmentManager; +import org.apache.james.mailbox.MessageIdManager; +import org.apache.james.mailbox.RightManager; +import org.apache.james.mailbox.store.StoreAttachmentManager; +import org.apache.james.mailbox.store.StoreMessageIdManager; +import org.apache.james.mailbox.store.StoreRightManager; + +import com.google.inject.AbstractModule; +import com.google.inject.Scopes; +import com.google.inject.TypeLiteral; +import com.google.inject.multibindings.Multibinder; +import com.google.inject.name.Names; + +public class PostgresJmapModule extends AbstractModule { + + @Override + protected void configure() { + Multibinder.newSetBinder(binder(), PostgresModule.class).addBinding().toInstance(PostgresDataJMapAggregateModule.MODULE); + + bind(EmailChangeRepository.class).to(PostgresEmailChangeRepository.class); + bind(PostgresEmailChangeRepository.class).in(Scopes.SINGLETON); + + bind(MailboxChangeRepository.class).to(PostgresMailboxChangeRepository.class); + bind(PostgresMailboxChangeRepository.class).in(Scopes.SINGLETON); + + bind(Limit.class).annotatedWith(Names.named(PostgresEmailChangeRepository.LIMIT_NAME)).toInstance(Limit.of(256)); + bind(Limit.class).annotatedWith(Names.named(PostgresMailboxChangeRepository.LIMIT_NAME)).toInstance(Limit.of(256)); + + bind(UploadUsageRepository.class).to(PostgresUploadUsageRepository.class); + + bind(MessageIdManager.class).to(StoreMessageIdManager.class); + bind(AttachmentManager.class).to(StoreAttachmentManager.class); + bind(StoreMessageIdManager.class).in(Scopes.SINGLETON); + bind(StoreAttachmentManager.class).in(Scopes.SINGLETON); + bind(RightManager.class).to(StoreRightManager.class); + bind(StoreRightManager.class).in(Scopes.SINGLETON); + + bind(State.Factory.class).to(PostgresStateFactory.class); + + bind(PushSubscriptionRepository.class).to(PostgresPushSubscriptionRepository.class); + + Multibinder> eventDTOModuleBinder = Multibinder.newSetBinder(binder(), new TypeLiteral<>() {}); + eventDTOModuleBinder.addBinding().toInstance(FilteringRuleSetDefineDTOModules.FILTERING_RULE_SET_DEFINED); + eventDTOModuleBinder.addBinding().toInstance(FilteringRuleSetDefineDTOModules.FILTERING_INCREMENT); + } +} diff --git a/server/apps/postgres-app/src/main/resources/defaultMailetContainer.xml b/server/apps/postgres-app/src/main/resources/defaultMailetContainer.xml new file mode 100644 index 00000000000..3822f0c210f --- /dev/null +++ b/server/apps/postgres-app/src/main/resources/defaultMailetContainer.xml @@ -0,0 +1,87 @@ + + + + + + + + transport + + + + + + + + + + + + X-UserIsAuth + true + + + bcc + + + + + local-address-error + 550 - Requested action not taken: no such user here + + + outgoing + 5000, 100000, 500000 + 3 + 0 + 10 + true + bounces + + + relay-denied + + + + + + + + + + none + + + + + + + none + + + + + + + false + + + + \ No newline at end of file diff --git a/server/apps/postgres-app/src/main/scripts/james-cli b/server/apps/postgres-app/src/main/scripts/james-cli new file mode 100755 index 00000000000..652b623405b --- /dev/null +++ b/server/apps/postgres-app/src/main/scripts/james-cli @@ -0,0 +1,4 @@ +#!/bin/bash + +unset JAVA_TOOL_OPTIONS +java -cp /root/resources:/root/classes:/root/libs/* org.apache.james.cli.ServerCmd "$@" \ No newline at end of file diff --git a/server/apps/postgres-app/src/test/java/org/apache/james/BodyDeduplicationIntegrationTest.java b/server/apps/postgres-app/src/test/java/org/apache/james/BodyDeduplicationIntegrationTest.java new file mode 100644 index 00000000000..05263b0ef60 --- /dev/null +++ b/server/apps/postgres-app/src/test/java/org/apache/james/BodyDeduplicationIntegrationTest.java @@ -0,0 +1,134 @@ +/**************************************************************** + * 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; + +import static org.apache.james.data.UsersRepositoryModuleChooser.Implementation.DEFAULT; +import static org.assertj.core.api.Assertions.assertThat; + +import java.nio.charset.StandardCharsets; + +import org.apache.james.backends.postgres.PostgresExtension; +import org.apache.james.mailbox.DefaultMailboxes; +import org.apache.james.mailbox.postgres.mail.PostgresMessageModule; +import org.apache.james.modules.MailboxProbeImpl; +import org.apache.james.modules.blobstore.BlobStoreConfiguration; +import org.apache.james.modules.protocols.ImapGuiceProbe; +import org.apache.james.modules.protocols.SmtpGuiceProbe; +import org.apache.james.util.Port; +import org.apache.james.utils.DataProbeImpl; +import org.apache.james.utils.SMTPMessageSender; +import org.apache.james.utils.SpoolerProbe; +import org.apache.james.utils.TestIMAPClient; +import org.jooq.impl.DSL; +import org.junit.jupiter.api.BeforeEach; +import org.junit.jupiter.api.Test; +import org.junit.jupiter.api.extension.RegisterExtension; + +import com.google.common.collect.ImmutableList; +import com.google.common.io.Resources; + +import reactor.core.publisher.Mono; + +class BodyDeduplicationIntegrationTest implements MailsShouldBeWellReceived { + static PostgresExtension postgresExtension = PostgresExtension.empty(); + + @RegisterExtension + static JamesServerExtension jamesServerExtension = new JamesServerBuilder(tmpDir -> + PostgresJamesConfiguration.builder() + .workingDirectory(tmpDir) + .configurationFromClasspath() + .searchConfiguration(SearchConfiguration.scanning()) + .blobStore(BlobStoreConfiguration.builder() + .postgres() + .disableCache() + .deduplication() + .noCryptoConfig()) + .usersRepository(DEFAULT) + .build()) + .server(PostgresJamesServerMain::createServer) + .extension(postgresExtension) + .build(); + + private static final String PASSWORD = "123456"; + private static final String YET_ANOTHER_USER = "yet-another-user@" + DOMAIN; + + private TestIMAPClient testIMAPClient; + private SMTPMessageSender smtpMessageSender; + + @BeforeEach + void setUp() { + this.testIMAPClient = new TestIMAPClient(); + this.smtpMessageSender = new SMTPMessageSender(DOMAIN); + } + + @Override + public int imapPort(GuiceJamesServer server) { + return server.getProbe(ImapGuiceProbe.class).getImapPort(); + } + + @Override + public int smtpPort(GuiceJamesServer server) { + return server.getProbe(SmtpGuiceProbe.class).getSmtpPort().getValue(); + } + + @Test + void bodyBlobsShouldBeDeDeduplicated(GuiceJamesServer server) throws Exception { + server.getProbe(DataProbeImpl.class).fluent() + .addDomain(DOMAIN) + .addUser(JAMES_USER, PASSWORD) + .addUser(OTHER_USER, PASSWORD_OTHER) + .addUser(YET_ANOTHER_USER, PASSWORD); + + MailboxProbeImpl mailboxProbe = server.getProbe(MailboxProbeImpl.class); + mailboxProbe.createMailbox("#private", JAMES_USER, DefaultMailboxes.INBOX); + mailboxProbe.createMailbox("#private", OTHER_USER, DefaultMailboxes.INBOX); + mailboxProbe.createMailbox("#private", YET_ANOTHER_USER, DefaultMailboxes.INBOX); + + Port smtpPort = server.getProbe(SmtpGuiceProbe.class).getSmtpPort(); + String message = Resources.toString(Resources.getResource("eml/htmlMail.eml"), StandardCharsets.UTF_8); + + // Given a mail sent to 3 recipients + smtpMessageSender.connect(JAMES_SERVER_HOST, smtpPort); + sendUniqueMessageToUsers(smtpMessageSender, message, ImmutableList.of(JAMES_USER, OTHER_USER, YET_ANOTHER_USER)); + CALMLY_AWAIT.untilAsserted(() -> assertThat(server.getProbe(SpoolerProbe.class).processingFinished()).isTrue()); + + // When 3 mails are received + testIMAPClient.connect(JAMES_SERVER_HOST, server.getProbe(ImapGuiceProbe.class).getImapPort()) + .login(JAMES_USER, PASSWORD) + .select(TestIMAPClient.INBOX) + .awaitMessageCount(CALMLY_AWAIT, 1); + testIMAPClient.connect(JAMES_SERVER_HOST, server.getProbe(ImapGuiceProbe.class).getImapPort()) + .login(OTHER_USER, PASSWORD_OTHER) + .select(TestIMAPClient.INBOX) + .awaitMessageCount(CALMLY_AWAIT, 1); + testIMAPClient.connect(JAMES_SERVER_HOST, server.getProbe(ImapGuiceProbe.class).getImapPort()) + .login(YET_ANOTHER_USER, PASSWORD) + .select(TestIMAPClient.INBOX) + .awaitMessageCount(CALMLY_AWAIT, 1); + + // Then the body blobs are deduplicated + int distinctBlobCount = postgresExtension.getDefaultPostgresExecutor() + .executeCount(dslContext -> Mono.from(dslContext.select(DSL.countDistinct(PostgresMessageModule.MessageTable.BODY_BLOB_ID)) + .from(PostgresMessageModule.MessageTable.TABLE_NAME))) + .block(); + + assertThat(distinctBlobCount).isEqualTo(1); + } +} diff --git a/server/apps/postgres-app/src/test/java/org/apache/james/DistributedPostgresJamesServerTest.java b/server/apps/postgres-app/src/test/java/org/apache/james/DistributedPostgresJamesServerTest.java new file mode 100644 index 00000000000..e34aaed6e20 --- /dev/null +++ b/server/apps/postgres-app/src/test/java/org/apache/james/DistributedPostgresJamesServerTest.java @@ -0,0 +1,135 @@ +/**************************************************************** + * 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; + +import static org.apache.james.data.UsersRepositoryModuleChooser.Implementation.DEFAULT; +import static org.assertj.core.api.Assertions.assertThat; +import static org.awaitility.Durations.FIVE_HUNDRED_MILLISECONDS; +import static org.awaitility.Durations.ONE_MINUTE; +import static org.hamcrest.Matchers.equalTo; + +import org.apache.james.PostgresJamesConfiguration.EventBusImpl; +import org.apache.james.backends.postgres.PostgresExtension; +import org.apache.james.core.healthcheck.ResultStatus; +import org.apache.james.core.quota.QuotaSizeLimit; +import org.apache.james.modules.AwsS3BlobStoreExtension; +import org.apache.james.modules.QuotaProbesImpl; +import org.apache.james.modules.RabbitMQExtension; +import org.apache.james.modules.blobstore.BlobStoreConfiguration; +import org.apache.james.modules.protocols.ImapGuiceProbe; +import org.apache.james.modules.protocols.SmtpGuiceProbe; +import org.apache.james.utils.DataProbeImpl; +import org.apache.james.utils.SMTPMessageSender; +import org.apache.james.utils.TestIMAPClient; +import org.apache.james.utils.WebAdminGuiceProbe; +import org.apache.james.webadmin.WebAdminUtils; +import org.awaitility.Awaitility; +import org.awaitility.core.ConditionFactory; +import org.eclipse.jetty.http.HttpStatus; +import org.junit.jupiter.api.BeforeEach; +import org.junit.jupiter.api.Test; +import org.junit.jupiter.api.extension.RegisterExtension; + +import com.google.common.base.Strings; + +import io.restassured.specification.RequestSpecification; + +class DistributedPostgresJamesServerTest implements JamesServerConcreteContract { + static PostgresExtension postgresExtension = PostgresExtension.empty(); + + @RegisterExtension + static RabbitMQExtension rabbitMQExtension = new RabbitMQExtension(); + + @RegisterExtension + static JamesServerExtension jamesServerExtension = new JamesServerBuilder(tmpDir -> + PostgresJamesConfiguration.builder() + .workingDirectory(tmpDir) + .configurationFromClasspath() + .usersRepository(DEFAULT) + .blobStore(BlobStoreConfiguration.builder() + .s3() + .disableCache() + .deduplication() + .noCryptoConfig()) + .searchConfiguration(SearchConfiguration.openSearch()) + .eventBusImpl(EventBusImpl.RABBITMQ) + .build()) + .server(PostgresJamesServerMain::createServer) + .extension(postgresExtension) + .extension(new AwsS3BlobStoreExtension()) + .extension(new DockerOpenSearchExtension()) + .extension(rabbitMQExtension) + .lifeCycle(JamesServerExtension.Lifecycle.PER_CLASS) + .build(); + + private static final ConditionFactory AWAIT = Awaitility.await() + .atMost(ONE_MINUTE) + .with() + .pollInterval(FIVE_HUNDRED_MILLISECONDS); + static final String DOMAIN = "james.local"; + private static final String USER = "toto@" + DOMAIN; + private static final String PASSWORD = "123456"; + + private TestIMAPClient testIMAPClient; + private SMTPMessageSender smtpMessageSender; + private RequestSpecification webAdminApi; + + @BeforeEach + void setUp(GuiceJamesServer guiceJamesServer) { + this.testIMAPClient = new TestIMAPClient(); + this.smtpMessageSender = new SMTPMessageSender(DOMAIN); + this.webAdminApi = WebAdminUtils.spec(guiceJamesServer.getProbe(WebAdminGuiceProbe.class).getWebAdminPort()); + } + + @Test + void guiceServerShouldUpdateQuota(GuiceJamesServer jamesServer) throws Exception { + jamesServer.getProbe(DataProbeImpl.class) + .fluent() + .addDomain(DOMAIN) + .addUser(USER, PASSWORD); + jamesServer.getProbe(QuotaProbesImpl.class).setGlobalMaxStorage(QuotaSizeLimit.size(50 * 1024)); + + // ~ 12 KB email + int imapPort = jamesServer.getProbe(ImapGuiceProbe.class).getImapPort(); + smtpMessageSender.connect(JAMES_SERVER_HOST, jamesServer.getProbe(SmtpGuiceProbe.class).getSmtpPort()) + .authenticate(USER, PASSWORD) + .sendMessageWithHeaders(USER, USER, "header: toto\\r\\n\\r\\n" + Strings.repeat("0123456789\n", 1024)); + AWAIT.until(() -> testIMAPClient.connect(JAMES_SERVER_HOST, imapPort) + .login(USER, PASSWORD) + .select(TestIMAPClient.INBOX) + .hasAMessage()); + + AWAIT.untilAsserted(() -> assertThat(testIMAPClient.connect(JAMES_SERVER_HOST, imapPort) + .login(USER, PASSWORD) + .getQuotaRoot(TestIMAPClient.INBOX)) + .startsWith("* QUOTAROOT \"INBOX\" #private&toto@james.local\r\n" + + "* QUOTA #private&toto@james.local (STORAGE 12 50)\r\n") + .endsWith("OK GETQUOTAROOT completed.\r\n")); + } + + @Test + void healthCheckShouldBeHealthy() { + webAdminApi.when() + .get("/healthcheck") + .then() + .statusCode(HttpStatus.OK_200) + .body("status", equalTo(ResultStatus.HEALTHY.getValue())); + } +} diff --git a/server/apps/postgres-app/src/test/java/org/apache/james/JamesCapabilitiesServerTest.java b/server/apps/postgres-app/src/test/java/org/apache/james/JamesCapabilitiesServerTest.java new file mode 100644 index 00000000000..347f4f1a8a1 --- /dev/null +++ b/server/apps/postgres-app/src/test/java/org/apache/james/JamesCapabilitiesServerTest.java @@ -0,0 +1,64 @@ +/**************************************************************** + * 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; + +import static org.apache.james.data.UsersRepositoryModuleChooser.Implementation.DEFAULT; +import static org.mockito.Mockito.mock; +import static org.mockito.Mockito.when; + +import java.util.EnumSet; + +import org.apache.james.PostgresJamesConfiguration.EventBusImpl; +import org.apache.james.backends.postgres.PostgresExtension; +import org.apache.james.mailbox.MailboxManager; +import org.junit.jupiter.api.Test; +import org.junit.jupiter.api.extension.RegisterExtension; + +class JamesCapabilitiesServerTest { + private static MailboxManager mailboxManager() { + MailboxManager mailboxManager = mock(MailboxManager.class); + when(mailboxManager.getSupportedMailboxCapabilities()) + .thenReturn(EnumSet.noneOf(MailboxManager.MailboxCapabilities.class)); + when(mailboxManager.getSupportedMessageCapabilities()) + .thenReturn(EnumSet.noneOf(MailboxManager.MessageCapabilities.class)); + when(mailboxManager.getSupportedSearchCapabilities()) + .thenReturn(EnumSet.noneOf(MailboxManager.SearchCapabilities.class)); + return mailboxManager; + } + + static PostgresExtension postgresExtension = PostgresExtension.empty(); + + @RegisterExtension + static JamesServerExtension jamesServerExtension = new JamesServerBuilder(tmpDir -> + PostgresJamesConfiguration.builder() + .workingDirectory(tmpDir) + .configurationFromClasspath() + .searchConfiguration(SearchConfiguration.scanning()) + .usersRepository(DEFAULT) + .eventBusImpl(EventBusImpl.IN_MEMORY) + .build()) + .server(configuration -> PostgresJamesServerMain.createServer(configuration)) + .extension(postgresExtension) + .build(); + + @Test + void startShouldSucceedWhenRequiredCapabilities(GuiceJamesServer server) { + + } +} diff --git a/server/apps/postgres-app/src/test/java/org/apache/james/JamesServerConcreteContract.java b/server/apps/postgres-app/src/test/java/org/apache/james/JamesServerConcreteContract.java new file mode 100644 index 00000000000..3ac19242eeb --- /dev/null +++ b/server/apps/postgres-app/src/test/java/org/apache/james/JamesServerConcreteContract.java @@ -0,0 +1,52 @@ +/**************************************************************** + * 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; + +import org.apache.james.modules.protocols.ImapGuiceProbe; +import org.apache.james.modules.protocols.LmtpGuiceProbe; +import org.apache.james.modules.protocols.Pop3GuiceProbe; +import org.apache.james.modules.protocols.SmtpGuiceProbe; + +public interface JamesServerConcreteContract extends JamesServerContract { + @Override + default int imapPort(GuiceJamesServer server) { + return server.getProbe(ImapGuiceProbe.class).getImapPort(); + } + + @Override + default int imapsPort(GuiceJamesServer server) { + return server.getProbe(ImapGuiceProbe.class).getImapStartTLSPort(); + } + + @Override + default int smtpPort(GuiceJamesServer server) { + return server.getProbe(SmtpGuiceProbe.class).getSmtpPort().getValue(); + } + + @Override + default int lmtpPort(GuiceJamesServer server) { + return server.getProbe(LmtpGuiceProbe.class).getLmtpPort(); + } + + @Override + default int pop3Port(GuiceJamesServer server) { + return server.getProbe(Pop3GuiceProbe.class).getPop3Port(); + } +} diff --git a/server/apps/postgres-app/src/test/java/org/apache/james/PostgresBlobStoreIntegrationTest.java b/server/apps/postgres-app/src/test/java/org/apache/james/PostgresBlobStoreIntegrationTest.java new file mode 100644 index 00000000000..72d8bab7475 --- /dev/null +++ b/server/apps/postgres-app/src/test/java/org/apache/james/PostgresBlobStoreIntegrationTest.java @@ -0,0 +1,59 @@ +/**************************************************************** + * 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; + +import static org.apache.james.data.UsersRepositoryModuleChooser.Implementation.DEFAULT; + +import org.apache.james.backends.postgres.PostgresExtension; +import org.apache.james.modules.blobstore.BlobStoreConfiguration; +import org.apache.james.modules.protocols.ImapGuiceProbe; +import org.apache.james.modules.protocols.SmtpGuiceProbe; +import org.junit.jupiter.api.extension.RegisterExtension; + +public class PostgresBlobStoreIntegrationTest implements MailsShouldBeWellReceived { + + @RegisterExtension + static JamesServerExtension jamesServerExtension = new JamesServerBuilder(tmpDir -> + PostgresJamesConfiguration.builder() + .workingDirectory(tmpDir) + .configurationFromClasspath() + .searchConfiguration(SearchConfiguration.scanning()) + .blobStore(BlobStoreConfiguration.builder() + .postgres() + .disableCache() + .deduplication() + .noCryptoConfig()) + .usersRepository(DEFAULT) + .build()) + .server(PostgresJamesServerMain::createServer) + .extension(PostgresExtension.empty()) + .build(); + + @Override + public int imapPort(GuiceJamesServer server) { + return server.getProbe(ImapGuiceProbe.class).getImapPort(); + } + + @Override + public int smtpPort(GuiceJamesServer server) { + return server.getProbe(SmtpGuiceProbe.class).getSmtpPort().getValue(); + } + +} \ No newline at end of file diff --git a/server/apps/postgres-app/src/test/java/org/apache/james/PostgresJamesServerTest.java b/server/apps/postgres-app/src/test/java/org/apache/james/PostgresJamesServerTest.java new file mode 100644 index 00000000000..6d7ba64109a --- /dev/null +++ b/server/apps/postgres-app/src/test/java/org/apache/james/PostgresJamesServerTest.java @@ -0,0 +1,106 @@ +/**************************************************************** + * 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; + +import static org.apache.james.data.UsersRepositoryModuleChooser.Implementation.DEFAULT; +import static org.assertj.core.api.Assertions.assertThat; +import static org.awaitility.Durations.FIVE_HUNDRED_MILLISECONDS; +import static org.awaitility.Durations.ONE_MINUTE; + +import org.apache.james.PostgresJamesConfiguration.EventBusImpl; +import org.apache.james.backends.postgres.PostgresExtension; +import org.apache.james.core.quota.QuotaSizeLimit; +import org.apache.james.modules.QuotaProbesImpl; +import org.apache.james.modules.protocols.ImapGuiceProbe; +import org.apache.james.modules.protocols.SmtpGuiceProbe; +import org.apache.james.utils.DataProbeImpl; +import org.apache.james.utils.SMTPMessageSender; +import org.apache.james.utils.TestIMAPClient; +import org.apache.james.vault.VaultConfiguration; +import org.awaitility.Awaitility; +import org.awaitility.core.ConditionFactory; +import org.junit.jupiter.api.BeforeEach; +import org.junit.jupiter.api.Test; +import org.junit.jupiter.api.extension.RegisterExtension; + +import com.google.common.base.Strings; + +class PostgresJamesServerTest implements JamesServerConcreteContract { + static PostgresExtension postgresExtension = PostgresExtension.empty(); + + @RegisterExtension + static JamesServerExtension jamesServerExtension = new JamesServerBuilder(tmpDir -> + PostgresJamesConfiguration.builder() + .workingDirectory(tmpDir) + .configurationFromClasspath() + .searchConfiguration(SearchConfiguration.scanning()) + .usersRepository(DEFAULT) + .eventBusImpl(EventBusImpl.IN_MEMORY) + .deletedMessageVaultConfiguration(VaultConfiguration.ENABLED_DEFAULT) + .build()) + .server(PostgresJamesServerMain::createServer) + .extension(postgresExtension) + .lifeCycle(JamesServerExtension.Lifecycle.PER_CLASS) + .build(); + + private static final ConditionFactory AWAIT = Awaitility.await() + .atMost(ONE_MINUTE) + .with() + .pollInterval(FIVE_HUNDRED_MILLISECONDS); + static final String DOMAIN = "james.local"; + private static final String USER = "toto@" + DOMAIN; + private static final String PASSWORD = "123456"; + + private TestIMAPClient testIMAPClient; + private SMTPMessageSender smtpMessageSender; + + @BeforeEach + void setUp() { + this.testIMAPClient = new TestIMAPClient(); + this.smtpMessageSender = new SMTPMessageSender(DOMAIN); + } + + @Test + void guiceServerShouldUpdateQuota(GuiceJamesServer jamesServer) throws Exception { + jamesServer.getProbe(DataProbeImpl.class) + .fluent() + .addDomain(DOMAIN) + .addUser(USER, PASSWORD); + jamesServer.getProbe(QuotaProbesImpl.class).setGlobalMaxStorage(QuotaSizeLimit.size(50 * 1024)); + + // ~ 12 KB email + int imapPort = jamesServer.getProbe(ImapGuiceProbe.class).getImapPort(); + smtpMessageSender.connect(JAMES_SERVER_HOST, jamesServer.getProbe(SmtpGuiceProbe.class).getSmtpPort()) + .authenticate(USER, PASSWORD) + .sendMessageWithHeaders(USER, USER, "header: toto\\r\\n\\r\\n" + Strings.repeat("0123456789\n", 1024)); + AWAIT.until(() -> testIMAPClient.connect(JAMES_SERVER_HOST, imapPort) + .login(USER, PASSWORD) + .select(TestIMAPClient.INBOX) + .hasAMessage()); + + assertThat( + testIMAPClient.connect(JAMES_SERVER_HOST, imapPort) + .login(USER, PASSWORD) + .getQuotaRoot(TestIMAPClient.INBOX)) + .startsWith("* QUOTAROOT \"INBOX\" #private&toto@james.local\r\n" + + "* QUOTA #private&toto@james.local (STORAGE 12 50)\r\n") + .endsWith("OK GETQUOTAROOT completed.\r\n"); + } +} diff --git a/server/apps/postgres-app/src/test/java/org/apache/james/PostgresJmapJamesServerTest.java b/server/apps/postgres-app/src/test/java/org/apache/james/PostgresJmapJamesServerTest.java new file mode 100644 index 00000000000..e0ba197e70b --- /dev/null +++ b/server/apps/postgres-app/src/test/java/org/apache/james/PostgresJmapJamesServerTest.java @@ -0,0 +1,48 @@ +/**************************************************************** + * 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; + +import static org.apache.james.data.UsersRepositoryModuleChooser.Implementation.DEFAULT; + +import org.apache.james.backends.postgres.PostgresExtension; +import org.apache.james.jmap.JmapJamesServerContract; +import org.apache.james.modules.TestJMAPServerModule; +import org.apache.james.vault.VaultConfiguration; +import org.junit.jupiter.api.extension.RegisterExtension; + +class PostgresJmapJamesServerTest implements JmapJamesServerContract { + static PostgresExtension postgresExtension = PostgresExtension.empty(); + + @RegisterExtension + static JamesServerExtension jamesServerExtension = new JamesServerBuilder(tmpDir -> + PostgresJamesConfiguration.builder() + .workingDirectory(tmpDir) + .configurationFromClasspath() + .searchConfiguration(SearchConfiguration.scanning()) + .usersRepository(DEFAULT) + .eventBusImpl(PostgresJamesConfiguration.EventBusImpl.IN_MEMORY) + .deletedMessageVaultConfiguration(VaultConfiguration.ENABLED_DEFAULT) + .build()) + .server(configuration -> PostgresJamesServerMain.createServer(configuration) + .overrideWith(new TestJMAPServerModule())) + .extension(postgresExtension) + .lifeCycle(JamesServerExtension.Lifecycle.PER_CLASS) + .build(); +} diff --git a/server/apps/postgres-app/src/test/java/org/apache/james/PostgresWithLDAPJamesServerTest.java b/server/apps/postgres-app/src/test/java/org/apache/james/PostgresWithLDAPJamesServerTest.java new file mode 100644 index 00000000000..a0be3e81710 --- /dev/null +++ b/server/apps/postgres-app/src/test/java/org/apache/james/PostgresWithLDAPJamesServerTest.java @@ -0,0 +1,64 @@ +/**************************************************************** + * 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; + +import static org.apache.james.MailsShouldBeWellReceived.JAMES_SERVER_HOST; +import static org.apache.james.data.UsersRepositoryModuleChooser.Implementation.LDAP; +import static org.assertj.core.api.Assertions.assertThat; + +import java.io.IOException; + +import org.apache.commons.net.imap.IMAPClient; +import org.apache.james.PostgresJamesConfiguration.EventBusImpl; +import org.apache.james.backends.postgres.PostgresExtension; +import org.apache.james.data.LdapTestExtension; +import org.apache.james.modules.protocols.ImapGuiceProbe; +import org.apache.james.user.ldap.DockerLdapSingleton; +import org.junit.jupiter.api.Test; +import org.junit.jupiter.api.extension.RegisterExtension; + +class PostgresWithLDAPJamesServerTest { + static PostgresExtension postgresExtension = PostgresExtension.empty(); + + @RegisterExtension + static JamesServerExtension jamesServerExtension = new JamesServerBuilder(tmpDir -> + PostgresJamesConfiguration.builder() + .workingDirectory(tmpDir) + .configurationFromClasspath() + .searchConfiguration(SearchConfiguration.openSearch()) + .usersRepository(LDAP) + .eventBusImpl(EventBusImpl.IN_MEMORY) + .build()) + .server(PostgresJamesServerMain::createServer) + .lifeCycle(JamesServerExtension.Lifecycle.PER_CLASS) + .extension(new LdapTestExtension()) + .extension(new DockerOpenSearchExtension()) + .extension(postgresExtension) + .build(); + + + @Test + void userFromLdapShouldLoginViaImapProtocol(GuiceJamesServer server) throws IOException { + IMAPClient imapClient = new IMAPClient(); + imapClient.connect(JAMES_SERVER_HOST, server.getProbe(ImapGuiceProbe.class).getImapPort()); + + assertThat(imapClient.login(DockerLdapSingleton.JAMES_USER.asString(), DockerLdapSingleton.PASSWORD)).isTrue(); + } +} diff --git a/server/apps/postgres-app/src/test/java/org/apache/james/PostgresWithOpenSearchDisabledTest.java b/server/apps/postgres-app/src/test/java/org/apache/james/PostgresWithOpenSearchDisabledTest.java new file mode 100644 index 00000000000..49555b377b0 --- /dev/null +++ b/server/apps/postgres-app/src/test/java/org/apache/james/PostgresWithOpenSearchDisabledTest.java @@ -0,0 +1,142 @@ +/**************************************************************** + * 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; + +import static org.apache.james.data.UsersRepositoryModuleChooser.Implementation.DEFAULT; +import static org.assertj.core.api.Assertions.assertThat; + +import java.nio.charset.StandardCharsets; + +import org.apache.james.PostgresJamesConfiguration.EventBusImpl; +import org.apache.james.backends.opensearch.OpenSearchConfiguration; +import org.apache.james.backends.postgres.PostgresExtension; +import org.apache.james.core.Domain; +import org.apache.james.mailbox.DefaultMailboxes; +import org.apache.james.mailbox.opensearch.events.OpenSearchListeningMessageSearchIndex; +import org.apache.james.modules.EventDeadLettersProbe; +import org.apache.james.modules.MailboxProbeImpl; +import org.apache.james.modules.protocols.ImapGuiceProbe; +import org.apache.james.modules.protocols.SmtpGuiceProbe; +import org.apache.james.util.Host; +import org.apache.james.util.Port; +import org.apache.james.utils.DataProbeImpl; +import org.apache.james.utils.SMTPMessageSender; +import org.apache.james.utils.TestIMAPClient; +import org.junit.jupiter.api.Disabled; +import org.junit.jupiter.api.Test; +import org.junit.jupiter.api.extension.RegisterExtension; + +import com.google.common.io.Resources; + +public class PostgresWithOpenSearchDisabledTest implements MailsShouldBeWellReceived { + static PostgresExtension postgresExtension = PostgresExtension.empty(); + + @RegisterExtension + static JamesServerExtension jamesServerExtension = new JamesServerBuilder(tmpDir -> + PostgresJamesConfiguration.builder() + .workingDirectory(tmpDir) + .configurationFromClasspath() + .searchConfiguration(SearchConfiguration.openSearchDisabled()) + .usersRepository(DEFAULT) + .eventBusImpl(EventBusImpl.IN_MEMORY) + .build()) + .server(configuration -> PostgresJamesServerMain.createServer(configuration) + .overrideWith(binder -> binder.bind(OpenSearchConfiguration.class) + .toInstance(OpenSearchConfiguration.builder() + .addHost(Host.from("127.0.0.1", 9042)) + .build()))) + .extension(postgresExtension) + .build(); + + @Override + public int imapPort(GuiceJamesServer server) { + return server.getProbe(ImapGuiceProbe.class).getImapPort(); + } + + @Override + public int smtpPort(GuiceJamesServer server) { + return server.getProbe(SmtpGuiceProbe.class).getSmtpPort().getValue(); + } + + @Test + void mailsShouldBeKeptInDeadLetterForLaterIndexing(GuiceJamesServer server) throws Exception { + server.getProbe(DataProbeImpl.class).fluent() + .addDomain(DOMAIN) + .addUser(JAMES_USER, PASSWORD) + .addUser(SENDER, PASSWORD); + + MailboxProbeImpl mailboxProbe = server.getProbe(MailboxProbeImpl.class); + mailboxProbe.createMailbox("#private", JAMES_USER, DefaultMailboxes.INBOX); + + Port smtpPort = Port.of(smtpPort(server)); + String message = Resources.toString(Resources.getResource("eml/htmlMail.eml"), StandardCharsets.UTF_8); + + try (SMTPMessageSender sender = new SMTPMessageSender(Domain.LOCALHOST.asString())) { + sender.connect(JAMES_SERVER_HOST, smtpPort).authenticate(SENDER, PASSWORD); + sendUniqueMessage(sender, message); + } + + CALMLY_AWAIT.until(() -> server.getProbe(EventDeadLettersProbe.class).getEventDeadLetters() + .groupsWithFailedEvents().collectList().block().contains(new OpenSearchListeningMessageSearchIndex.OpenSearchListeningMessageSearchIndexGroup())); + } + + @Test + void searchShouldFail(GuiceJamesServer server) throws Exception { + server.getProbe(DataProbeImpl.class).fluent() + .addDomain(DOMAIN) + .addUser(JAMES_USER, PASSWORD) + .addUser(SENDER, PASSWORD); + + MailboxProbeImpl mailboxProbe = server.getProbe(MailboxProbeImpl.class); + mailboxProbe.createMailbox("#private", JAMES_USER, DefaultMailboxes.INBOX); + + try (TestIMAPClient reader = new TestIMAPClient()) { + int imapPort = imapPort(server); + reader.connect(JAMES_SERVER_HOST, imapPort) + .login(JAMES_USER, PASSWORD) + .select(TestIMAPClient.INBOX); + + assertThat(reader.sendCommand("SEARCH SUBJECT thy")) + .contains("NO SEARCH processing failed"); + } + } + + @Test + @Disabled("Overrides not implemented yet for Postgresql") + void searchShouldSucceedOnSearchOverrides(GuiceJamesServer server) throws Exception { + server.getProbe(DataProbeImpl.class).fluent() + .addDomain(DOMAIN) + .addUser(JAMES_USER, PASSWORD) + .addUser(SENDER, PASSWORD); + + MailboxProbeImpl mailboxProbe = server.getProbe(MailboxProbeImpl.class); + mailboxProbe.createMailbox("#private", JAMES_USER, DefaultMailboxes.INBOX); + + try (TestIMAPClient reader = new TestIMAPClient()) { + int imapPort = imapPort(server); + reader.connect(JAMES_SERVER_HOST, imapPort) + .login(JAMES_USER, PASSWORD) + .select(TestIMAPClient.INBOX); + + assertThat(reader.sendCommand("SEARCH UNSEEN")) + .contains("OK SEARCH"); + } + } +} diff --git a/server/apps/postgres-app/src/test/java/org/apache/james/PostgresWithTikaTest.java b/server/apps/postgres-app/src/test/java/org/apache/james/PostgresWithTikaTest.java new file mode 100644 index 00000000000..48bea4ee511 --- /dev/null +++ b/server/apps/postgres-app/src/test/java/org/apache/james/PostgresWithTikaTest.java @@ -0,0 +1,46 @@ +/**************************************************************** + * 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; + +import static org.apache.james.data.UsersRepositoryModuleChooser.Implementation.DEFAULT; + +import org.apache.james.PostgresJamesConfiguration.EventBusImpl; +import org.apache.james.backends.postgres.PostgresExtension; +import org.junit.jupiter.api.extension.RegisterExtension; + +public class PostgresWithTikaTest implements JamesServerConcreteContract { + static PostgresExtension postgresExtension = PostgresExtension.empty(); + + @RegisterExtension + static JamesServerExtension jamesServerExtension = new JamesServerBuilder(tmpDir -> + PostgresJamesConfiguration.builder() + .workingDirectory(tmpDir) + .configurationFromClasspath() + .searchConfiguration(SearchConfiguration.openSearch()) + .usersRepository(DEFAULT) + .eventBusImpl(EventBusImpl.IN_MEMORY) + .build()) + .server(PostgresJamesServerMain::createServer) + .extension(new DockerOpenSearchExtension()) + .extension(new TikaExtension()) + .extension(postgresExtension) + .lifeCycle(JamesServerExtension.Lifecycle.PER_CLASS) + .build(); +} diff --git a/server/apps/postgres-app/src/test/java/org/apache/james/WithScanningSearchImmutableTest.java b/server/apps/postgres-app/src/test/java/org/apache/james/WithScanningSearchImmutableTest.java new file mode 100644 index 00000000000..a5e653af6d3 --- /dev/null +++ b/server/apps/postgres-app/src/test/java/org/apache/james/WithScanningSearchImmutableTest.java @@ -0,0 +1,44 @@ +/**************************************************************** + * 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; + +import static org.apache.james.data.UsersRepositoryModuleChooser.Implementation.DEFAULT; + +import org.apache.james.PostgresJamesConfiguration.EventBusImpl; +import org.apache.james.backends.postgres.PostgresExtension; +import org.junit.jupiter.api.extension.RegisterExtension; + +public class WithScanningSearchImmutableTest implements JamesServerConcreteContract { + static PostgresExtension postgresExtension = PostgresExtension.empty(); + + @RegisterExtension + static JamesServerExtension jamesServerExtension = new JamesServerBuilder(tmpDir -> + PostgresJamesConfiguration.builder() + .workingDirectory(tmpDir) + .configurationFromClasspath() + .searchConfiguration(SearchConfiguration.scanning()) + .usersRepository(DEFAULT) + .eventBusImpl(EventBusImpl.IN_MEMORY) + .build()) + .server(PostgresJamesServerMain::createServer) + .extension(postgresExtension) + .lifeCycle(JamesServerExtension.Lifecycle.PER_CLASS) + .build(); +} diff --git a/server/apps/postgres-app/src/test/java/org/apache/james/WithScanningSearchMutableTest.java b/server/apps/postgres-app/src/test/java/org/apache/james/WithScanningSearchMutableTest.java new file mode 100644 index 00000000000..d9c8ef34779 --- /dev/null +++ b/server/apps/postgres-app/src/test/java/org/apache/james/WithScanningSearchMutableTest.java @@ -0,0 +1,56 @@ +/**************************************************************** + * 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; + +import static org.apache.james.data.UsersRepositoryModuleChooser.Implementation.DEFAULT; + +import org.apache.james.PostgresJamesConfiguration.EventBusImpl; +import org.apache.james.backends.postgres.PostgresExtension; +import org.apache.james.modules.protocols.ImapGuiceProbe; +import org.apache.james.modules.protocols.SmtpGuiceProbe; +import org.junit.jupiter.api.extension.RegisterExtension; + +public class WithScanningSearchMutableTest implements MailsShouldBeWellReceived { + static PostgresExtension postgresExtension = PostgresExtension.empty(); + + @RegisterExtension + static JamesServerExtension jamesServerExtension = new JamesServerBuilder(tmpDir -> + PostgresJamesConfiguration.builder() + .workingDirectory(tmpDir) + .configurationFromClasspath() + .searchConfiguration(SearchConfiguration.scanning()) + .usersRepository(DEFAULT) + .eventBusImpl(EventBusImpl.IN_MEMORY) + .build()) + .server(PostgresJamesServerMain::createServer) + .extension(postgresExtension) + .lifeCycle(JamesServerExtension.Lifecycle.PER_TEST) + .build(); + + @Override + public int imapPort(GuiceJamesServer server) { + return server.getProbe(ImapGuiceProbe.class).getImapPort(); + } + + @Override + public int smtpPort(GuiceJamesServer server) { + return server.getProbe(SmtpGuiceProbe.class).getSmtpPort().getValue(); + } +} diff --git a/server/apps/postgres-app/src/test/resources/dnsservice.xml b/server/apps/postgres-app/src/test/resources/dnsservice.xml new file mode 100644 index 00000000000..6e4fbd2efb5 --- /dev/null +++ b/server/apps/postgres-app/src/test/resources/dnsservice.xml @@ -0,0 +1,25 @@ + + + + + true + false + 50000 + diff --git a/server/apps/postgres-app/src/test/resources/domainlist.xml b/server/apps/postgres-app/src/test/resources/domainlist.xml new file mode 100644 index 00000000000..fe17431a1ea --- /dev/null +++ b/server/apps/postgres-app/src/test/resources/domainlist.xml @@ -0,0 +1,24 @@ + + + + + false + false + diff --git a/server/apps/postgres-app/src/test/resources/eml/htmlMail.eml b/server/apps/postgres-app/src/test/resources/eml/htmlMail.eml new file mode 100644 index 00000000000..c8213f4d7ea --- /dev/null +++ b/server/apps/postgres-app/src/test/resources/eml/htmlMail.eml @@ -0,0 +1,81 @@ +Delivered-To: mister@james.org +Received: by 10.28.170.202 with SMTP id t193csp327634wme; + Thu, 4 Jun 2015 00:36:15 -0700 (PDT) +X-Received: by 10.180.77.195 with SMTP id u3mr5042880wiw.30.1433403375307; + Thu, 04 Jun 2015 00:36:15 -0700 (PDT) +Return-Path: +Received: from o7.email.airbnb.com (o7.email.airbnb.com. [167.89.32.249]) + by mx.google.com with ESMTPS id i2si5691730wjz.123.2015.06.04.00.36.13 + for + (version=TLSv1.2 cipher=ECDHE-RSA-AES128-GCM-SHA256 bits=128/128); + Thu, 04 Jun 2015 00:36:15 -0700 (PDT) +Received-SPF: pass (google.com: domain of bounces+1453977-062b-mister=james.org@email.airbnb.com designates 167.89.32.249 as permitted sender) client-ip=167.89.32.249; +Authentication-Results: mx.google.com; + spf=pass (google.com: domain of bounces+1453977-062b-mister=james.org@email.airbnb.com designates 167.89.32.249 as permitted sender) smtp.mail=bounces+1453977-062b-mister=james.org@email.airbnb.com; + dkim=pass header.i=@email.airbnb.com; + dmarc=pass (p=REJECT dis=NONE) header.from=airbnb.com +DKIM-Signature: v=1; a=rsa-sha1; c=relaxed; d=email.airbnb.com; + h=from:to:subject:mime-version:content-type:content-transfer-encoding; + s=s20150428; bh=2mhWUwzjtQTC0KljgpaEsuvrqok=; b=EhC2QHKb5+63egDD + qDCAepUELCeUZXCkw8+31kGT+O1va3iAKvQSAvzXJ3qJlIL9FgdeFk8sR78Vszn/ + A73vp6NGjAW60M4gUZjxEOIPzGKIS95KfmHxg10fXUOFW0uePojNEg4ZPCcuitrZ + HuWvzHK5I2siGnqupiivwxDgs5c= +DKIM-Signature: v=1; a=rsa-sha1; c=relaxed; d=sendgrid.info; + h=from:to:subject:mime-version:content-type:content-transfer-encoding:x-feedback-id; + s=smtpapi; bh=2mhWUwzjtQTC0KljgpaEsuvrqok=; b=FPiYMmNJLCrL2e8v/0 + DQC4voofe8nuuE7rhXZ25oqNAhAQja4oKIysJ1qAME2aEaqh+N5aJlCEuHrSG/7+ + NAQ0OY8KaJ2zlnxAbmgJETOjnf4oGdAa+nU/nVVEPfN2NRcBCNLGQZ80U4T5k8Xi + PakIuZvKDTRq7PiosSCSHT/FQ= +Received: by filter0490p1mdw1.sendgrid.net with SMTP id filter0490p1mdw1.13271.556FFFE7B + 2015-06-04 07:36:09.249601779 +0000 UTC +Received: from i-dee0850e.inst.aws.airbnb.com (ec2-54-90-154-187.compute-1.amazonaws.com [54.90.154.187]) + by ismtpd-017 (SG) with ESMTP id 14dbd7fa6b4.779a.254b43 + for ; Thu, 04 Jun 2015 07:36:09 +0000 (UTC) +Received: by i-dee0850e.inst.aws.airbnb.com (Postfix, from userid 1041) + id 19CBA24C60; Thu, 4 Jun 2015 07:36:09 +0000 (UTC) +Date: Thu, 04 Jun 2015 07:36:08 +0000 +From: Airbnb +To: mister@james.org +Message-ID: <556fffe8cac78_7ed0e0fe204457be@i-dee0850e.mail> +Subject: Text and Html not similar +Mime-Version: 1.0 +Content-Type: multipart/alternative; + boundary="--==_mimepart_556fffe8c7e84_7ed0e0fe20445637"; + charset=UTF-8 +Content-Transfer-Encoding: 7bit +X-User-ID: 32692788 +X-Locale: fr +X-Category: engagement +X-Template: low_intent_top_destinations +recipients: +sent-on: +X-SG-EID: mgVKhb3i1xMIKbRk82EYOUTMOPmiNk6g5BaWGQQKDTQchtClhw7VcIxig2BMwy1QMCr7h56hNVush8 + 4aRV0ieMn+WZ1XVnpY0OcmMYNZnuuvlOoNkBaiuiqeWuKVZO9c9S5OyxPy7WQeI0mccenq35NpLqjI + nQt7IAl2sIUksUD4lM8Ai0u2C88YW13cL+Lo +X-SG-ID: pQ7zy0fBcyQB3Gm22dZtqT6AR3zbAquH5ABZFkQfSKaxWRhz0YhtD36Li5uybRUjnPsuB21NpreKvG + t8iQBUn2ygs6hx6sMcgyI7L7bAY28p14Qj47KqA3JXbtMa0Xa3wdZaUUjZpemCg078XxMM5VaSHdDO + ChUhSV+z9RAJ38wAdUfXkpbO+m97vpU+mtWzVBoOrSiWCVYoNxPhvE4yIQ== +X-Feedback-ID: 1453977:N5+DXWRfRBXSDDbqVYXPNg8MjWYWwZibliGo1i/oSqY=:Ibl/atjs+SOcHCINmWWv/YvIGzDUihUrks9jdHsNF1+pafkc987UhcxmuyggxNgdCkEmMZDb9gJcndcUJy5KOw==:SG + +----==_mimepart_556fffe8c7e84_7ed0e0fe20445637 +Content-Type: text/plain; + charset=UTF-8 +Content-Transfer-Encoding: quoted-printable + +The text/plain part is not matching the html one. + +----==_mimepart_556fffe8c7e84_7ed0e0fe20445637 +Content-Type: text/html; charset=utf-8 +Content-Transfer-Encoding: 7bit + + + + + + + + This is a mail with beautifull html content which contains a banana.
+ + + +----==_mimepart_556fffe8c7e84_7ed0e0fe20445637-- diff --git a/server/apps/postgres-app/src/test/resources/fakemailrepositorystore.xml b/server/apps/postgres-app/src/test/resources/fakemailrepositorystore.xml new file mode 100644 index 00000000000..2d19a802da9 --- /dev/null +++ b/server/apps/postgres-app/src/test/resources/fakemailrepositorystore.xml @@ -0,0 +1,31 @@ + + + + + + + + + file + + + + + diff --git a/server/apps/postgres-app/src/test/resources/imapserver.xml b/server/apps/postgres-app/src/test/resources/imapserver.xml new file mode 100644 index 00000000000..3434dbce390 --- /dev/null +++ b/server/apps/postgres-app/src/test/resources/imapserver.xml @@ -0,0 +1,57 @@ + + + + + + + + imapserver + 0.0.0.0:0 + 200 + + + classpath://keystore + james72laBalle + org.bouncycastle.jce.provider.BouncyCastleProvider + + 0 + 0 + false + false + + + imapserver-ssl + 0.0.0.0:0 + 200 + + + classpath://keystore + james72laBalle + org.bouncycastle.jce.provider.BouncyCastleProvider + + 0 + 0 + false + + diff --git a/server/apps/postgres-app/src/test/resources/keystore b/server/apps/postgres-app/src/test/resources/keystore new file mode 100644 index 00000000000..536a6c792b0 Binary files /dev/null and b/server/apps/postgres-app/src/test/resources/keystore differ diff --git a/server/apps/postgres-app/src/test/resources/lmtpserver.xml b/server/apps/postgres-app/src/test/resources/lmtpserver.xml new file mode 100644 index 00000000000..30caa8c185f --- /dev/null +++ b/server/apps/postgres-app/src/test/resources/lmtpserver.xml @@ -0,0 +1,42 @@ + + + + + + + lmtpserver + + 127.0.0.1:0 + 200 + 1200 + + 0 + + 0 + + + 0 + + + + false + + + diff --git a/server/apps/postgres-app/src/test/resources/mailetcontainer.xml b/server/apps/postgres-app/src/test/resources/mailetcontainer.xml new file mode 100644 index 00000000000..b9b7a7eba44 --- /dev/null +++ b/server/apps/postgres-app/src/test/resources/mailetcontainer.xml @@ -0,0 +1,132 @@ + + + + + + + + postmaster + + + + 20 + postgres://var/mail/error/ + + + + + + + + transport + + + + + + ignore + + + postgres://var/mail/error/ + propagate + + + + + + + + + + + + bcc + + + rrt-error + + + + + ignore + + + ignore + + + + + local-address-error + 550 - Requested action not taken: no such user here + + + + outgoing + 5000, 100000, 500000 + 3 + 0 + 10 + true + bounces + + + relay-denied + + + + + + none + + + postgres://var/mail/address-error/ + + + + + + none + + + postgres://var/mail/relay-denied/ + Warning: You are sending an e-mail to a remote server. You must be authentified to perform such an operation + + + + + + false + + + + + + postgres://var/mail/rrt-error/ + true + + + + + + + + + + diff --git a/server/apps/postgres-app/src/test/resources/mailrepositorystore.xml b/server/apps/postgres-app/src/test/resources/mailrepositorystore.xml new file mode 100644 index 00000000000..689745af60f --- /dev/null +++ b/server/apps/postgres-app/src/test/resources/mailrepositorystore.xml @@ -0,0 +1,31 @@ + + + + + + + + + postgres + + + + + diff --git a/server/apps/postgres-app/src/test/resources/managesieveserver.xml b/server/apps/postgres-app/src/test/resources/managesieveserver.xml new file mode 100644 index 00000000000..b644fa43177 --- /dev/null +++ b/server/apps/postgres-app/src/test/resources/managesieveserver.xml @@ -0,0 +1,66 @@ + + + + + + + + + + + + managesieveserver + + 0.0.0.0:0 + + 200 + + + + file://conf/keystore + james72laBalle + org.bouncycastle.jce.provider.BouncyCastleProvider + + SunX509 + + + + 360 + + + 0 + + + 0 + 0 + true + false + + + + + + diff --git a/server/apps/postgres-app/src/test/resources/pop3server.xml b/server/apps/postgres-app/src/test/resources/pop3server.xml new file mode 100644 index 00000000000..6e4473aae2b --- /dev/null +++ b/server/apps/postgres-app/src/test/resources/pop3server.xml @@ -0,0 +1,43 @@ + + + + + + + pop3server + 0.0.0.0:0 + 200 + + + file://conf/keystore + james72laBalle + org.bouncycastle.jce.provider.BouncyCastleProvider + + 1200 + 0 + 0 + + + + false + + diff --git a/server/apps/postgres-app/src/test/resources/smtpserver.xml b/server/apps/postgres-app/src/test/resources/smtpserver.xml new file mode 100644 index 00000000000..36ac142375e --- /dev/null +++ b/server/apps/postgres-app/src/test/resources/smtpserver.xml @@ -0,0 +1,111 @@ + + + + + + + smtpserver-global + 0.0.0.0:0 + 200 + + file://conf/keystore + james72laBalle + org.bouncycastle.jce.provider.BouncyCastleProvider + SunX509 + + 360 + 0 + 0 + + never + false + true + + false + 0 + true + Apache JAMES awesome SMTP Server + + + + + false + + + smtpserver-TLS + 0.0.0.0:0 + 200 + + file://conf/keystore + james72laBalle + org.bouncycastle.jce.provider.BouncyCastleProvider + SunX509 + + 360 + 0 + 0 + + forUnauthorizedAddresses + false + true + + + false + 0 + true + Apache JAMES awesome SMTP Server + + + + + false + + + smtpserver-authenticated + 0.0.0.0:0 + 200 + + file://conf/keystore + james72laBalle + org.bouncycastle.jce.provider.BouncyCastleProvider + SunX509 + + 360 + 0 + 0 + + forUnauthorizedAddresses + false + true + + + false + 0 + true + Apache JAMES awesome SMTP Server + + + + + false + + + + diff --git a/server/apps/postgres-app/src/test/resources/usersrepository.xml b/server/apps/postgres-app/src/test/resources/usersrepository.xml new file mode 100644 index 00000000000..a5390d7140d --- /dev/null +++ b/server/apps/postgres-app/src/test/resources/usersrepository.xml @@ -0,0 +1,28 @@ + + + + + + + PBKDF2-SHA512 + true + true + + diff --git a/server/apps/postgres-app/src/test/resources/webadmin.properties b/server/apps/postgres-app/src/test/resources/webadmin.properties new file mode 100644 index 00000000000..3386a14238a --- /dev/null +++ b/server/apps/postgres-app/src/test/resources/webadmin.properties @@ -0,0 +1,25 @@ +# 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. + +# This template file can be used as example for James Server configuration +# DO NOT USE IT AS SUCH AND ADAPT IT TO YOUR NEEDS + +# Read https://james.apache.org/server/config-webadmin.html for further details + +enabled=true +port=0 +host=127.0.0.1 \ No newline at end of file diff --git a/server/blob/blob-postgres/pom.xml b/server/blob/blob-postgres/pom.xml new file mode 100644 index 00000000000..d5bb4bfd06f --- /dev/null +++ b/server/blob/blob-postgres/pom.xml @@ -0,0 +1,141 @@ + + + + 4.0.0 + + + org.apache.james + james-server-blob + 3.9.0-SNAPSHOT + ../pom.xml + + + blob-postgres + + Apache James :: Server :: Blob :: Postgres + + + + ${james.groupId} + apache-james-backends-postgres + + + ${james.groupId} + apache-james-backends-postgres + test-jar + test + + + ${james.groupId} + blob-api + + + ${james.groupId} + blob-api + test-jar + test + + + ${james.groupId} + blob-storage-strategy + + + ${james.groupId} + blob-storage-strategy + test-jar + test + + + ${james.groupId} + james-server-guice-common + test-jar + test + + + ${james.groupId} + james-server-testing + test + + + ${james.groupId} + james-server-util + test + + + ${james.groupId} + metrics-tests + test + + + ${james.groupId} + testing-base + test + + + commons-io + commons-io + + + io.projectreactor + reactor-core + + + org.awaitility + awaitility + test + + + org.mockito + mockito-core + test + + + org.testcontainers + junit-jupiter + test + + + org.testcontainers + postgresql + test + + + org.testcontainers + testcontainers + test + + + + + + org.apache.maven.plugins + maven-surefire-plugin + + -Djava.library.path= + -javaagent:"${settings.localRepository}"/org/jacoco/org.jacoco.agent/${jacoco-maven-plugin.version}/org.jacoco.agent-${jacoco-maven-plugin.version}-runtime.jar=destfile=${basedir}/target/jacoco.exec + -Xms1024m -Xmx2048m + true + 1800 + + + + + + diff --git a/server/blob/blob-postgres/src/main/java/org/apache/james/blob/postgres/PostgresBlobStorageModule.java b/server/blob/blob-postgres/src/main/java/org/apache/james/blob/postgres/PostgresBlobStorageModule.java new file mode 100644 index 00000000000..d5eab5e4eb5 --- /dev/null +++ b/server/blob/blob-postgres/src/main/java/org/apache/james/blob/postgres/PostgresBlobStorageModule.java @@ -0,0 +1,62 @@ +/**************************************************************** + * 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.blob.postgres; + +import static org.apache.james.blob.postgres.PostgresBlobStorageModule.PostgresBlobStorageTable.BUCKET_NAME_INDEX; +import static org.jooq.impl.SQLDataType.BLOB; + +import org.apache.james.backends.postgres.PostgresIndex; +import org.apache.james.backends.postgres.PostgresModule; +import org.apache.james.backends.postgres.PostgresTable; +import org.jooq.Field; +import org.jooq.Record; +import org.jooq.Table; +import org.jooq.impl.DSL; +import org.jooq.impl.SQLDataType; + +public interface PostgresBlobStorageModule { + interface PostgresBlobStorageTable { + Table TABLE_NAME = DSL.table("blob_storage"); + + Field BUCKET_NAME = DSL.field("bucket_name", SQLDataType.VARCHAR(200).notNull()); + Field BLOB_ID = DSL.field("blob_id", SQLDataType.VARCHAR(200).notNull()); + Field DATA = DSL.field("data", BLOB.notNull()); + Field SIZE = DSL.field("size", SQLDataType.INTEGER.notNull()); + + PostgresTable TABLE = PostgresTable.name(TABLE_NAME.getName()) + .createTableStep(((dsl, tableName) -> dsl.createTableIfNotExists(tableName) + .column(BUCKET_NAME) + .column(BLOB_ID) + .column(DATA) + .column(SIZE) + .constraint(DSL.primaryKey(BUCKET_NAME, BLOB_ID)))) + .disableRowLevelSecurity() + .build(); + + PostgresIndex BUCKET_NAME_INDEX = PostgresIndex.name("blob_storage_bucket_name_index") + .createIndexStep((dsl, indexName) -> dsl.createIndexIfNotExists(indexName) + .on(TABLE_NAME, BUCKET_NAME)); + } + + PostgresModule MODULE = PostgresModule.builder() + .addTable(PostgresBlobStorageTable.TABLE) + .addIndex(BUCKET_NAME_INDEX) + .build(); +} diff --git a/server/blob/blob-postgres/src/main/java/org/apache/james/blob/postgres/PostgresBlobStoreDAO.java b/server/blob/blob-postgres/src/main/java/org/apache/james/blob/postgres/PostgresBlobStoreDAO.java new file mode 100644 index 00000000000..5003f61179a --- /dev/null +++ b/server/blob/blob-postgres/src/main/java/org/apache/james/blob/postgres/PostgresBlobStoreDAO.java @@ -0,0 +1,180 @@ +/**************************************************************** + * 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.blob.postgres; + +import static org.apache.james.blob.postgres.PostgresBlobStorageModule.PostgresBlobStorageTable.BLOB_ID; +import static org.apache.james.blob.postgres.PostgresBlobStorageModule.PostgresBlobStorageTable.BUCKET_NAME; +import static org.apache.james.blob.postgres.PostgresBlobStorageModule.PostgresBlobStorageTable.DATA; +import static org.apache.james.blob.postgres.PostgresBlobStorageModule.PostgresBlobStorageTable.SIZE; +import static org.apache.james.blob.postgres.PostgresBlobStorageModule.PostgresBlobStorageTable.TABLE_NAME; + +import java.io.ByteArrayInputStream; +import java.io.IOException; +import java.io.InputStream; +import java.util.Collection; +import java.util.List; +import java.util.Optional; +import java.util.function.Function; + +import jakarta.inject.Inject; + +import org.apache.commons.io.IOUtils; +import org.apache.james.backends.postgres.utils.PostgresExecutor; +import org.apache.james.backends.postgres.utils.PostgresUtils; +import org.apache.james.blob.api.BlobId; +import org.apache.james.blob.api.BlobStoreDAO; +import org.apache.james.blob.api.BucketName; +import org.apache.james.blob.api.ObjectNotFoundException; +import org.apache.james.blob.api.ObjectStoreIOException; +import org.jooq.impl.DSL; + +import com.google.common.base.Preconditions; +import com.google.common.collect.ImmutableList; +import com.google.common.io.ByteSource; + +import reactor.core.publisher.Flux; +import reactor.core.publisher.Mono; + +public class PostgresBlobStoreDAO implements BlobStoreDAO { + private final PostgresExecutor postgresExecutor; + private final BlobId.Factory blobIdFactory; + + @Inject + public PostgresBlobStoreDAO(PostgresExecutor postgresExecutor, BlobId.Factory blobIdFactory) { + this.postgresExecutor = postgresExecutor; + this.blobIdFactory = blobIdFactory; + } + + @Override + public InputStream read(BucketName bucketName, BlobId blobId) throws ObjectStoreIOException, ObjectNotFoundException { + return Mono.from(readReactive(bucketName, blobId)) + .block(); + } + + @Override + public Mono readReactive(BucketName bucketName, BlobId blobId) { + return Mono.from(readBytes(bucketName, blobId)) + .map(ByteArrayInputStream::new); + } + + @Override + public Mono readBytes(BucketName bucketName, BlobId blobId) { + return postgresExecutor.executeRow(dsl -> Mono.from(dsl.select(DATA) + .from(TABLE_NAME) + .where(BUCKET_NAME.eq(bucketName.asString())) + .and(BLOB_ID.eq(blobId.asString())))) + .map(record -> record.get(DATA)) + .switchIfEmpty(Mono.error(() -> new ObjectNotFoundException("Blob " + blobId + " does not exist in bucket " + bucketName))); + } + + @Override + public Mono save(BucketName bucketName, BlobId blobId, byte[] data) { + Preconditions.checkNotNull(data); + + return postgresExecutor.executeVoid(dslContext -> + Mono.from(dslContext.insertInto(TABLE_NAME, BUCKET_NAME, BLOB_ID, DATA, SIZE) + .values(bucketName.asString(), + blobId.asString(), + data, + data.length) + .onConflict(BUCKET_NAME, BLOB_ID) + .doUpdate() + .set(DATA, data) + .set(SIZE, data.length))); + } + + @Override + public Mono save(BucketName bucketName, BlobId blobId, InputStream inputStream) { + Preconditions.checkNotNull(inputStream); + + return Mono.fromCallable(() -> { + try { + return IOUtils.toByteArray(inputStream); + } catch (IOException e) { + throw new ObjectStoreIOException("IOException occurred", e); + } + }).flatMap(bytes -> save(bucketName, blobId, bytes)); + } + + @Override + public Mono save(BucketName bucketName, BlobId blobId, ByteSource content) { + return Mono.fromCallable(() -> { + try { + return content.read(); + } catch (IOException e) { + throw new ObjectStoreIOException("IOException occurred", e); + } + }).flatMap(bytes -> save(bucketName, blobId, bytes)); + } + + @Override + public Mono delete(BucketName bucketName, BlobId blobId) { + return postgresExecutor.executeVoid(dsl -> Mono.from(dsl.deleteFrom(TABLE_NAME) + .where(BUCKET_NAME.eq(bucketName.asString())) + .and(BLOB_ID.eq(blobId.asString())))); + } + + @Override + public Mono delete(BucketName bucketName, Collection blobIds) { + if (blobIds.isEmpty()) { + return Mono.empty(); + } + return postgresExecutor.executeVoid(dsl -> Mono.from(dsl.deleteFrom(TABLE_NAME) + .where(BUCKET_NAME.eq(bucketName.asString())) + .and(BLOB_ID.in(blobIds.stream().map(BlobId::asString).collect(ImmutableList.toImmutableList()))))); + } + + @Override + public Mono deleteBucket(BucketName bucketName) { + return postgresExecutor.executeVoid(dsl -> Mono.from(dsl.deleteFrom(TABLE_NAME) + .where(BUCKET_NAME.eq(bucketName.asString())))); + } + + @Override + public Flux listBuckets() { + return postgresExecutor.executeRows(dsl -> Flux.from(dsl.selectDistinct(BUCKET_NAME) + .from(TABLE_NAME))) + .map(record -> BucketName.of(record.get(BUCKET_NAME))); + } + + @Override + public Flux listBlobs(BucketName bucketName) { + return Flux.defer(() -> listBlobsBatch(bucketName, Optional.empty(), PostgresUtils.QUERY_BATCH_SIZE)) + .expand(blobIds -> { + if (blobIds.isEmpty() || blobIds.size() < PostgresUtils.QUERY_BATCH_SIZE) { + return Mono.empty(); + } + return listBlobsBatch(bucketName, Optional.of(blobIds.getLast()), PostgresUtils.QUERY_BATCH_SIZE); + }) + .flatMapIterable(Function.identity()); + } + + private Mono> listBlobsBatch(BucketName bucketName, Optional blobIdFrom, int batchSize) { + return postgresExecutor.executeRows(dsl -> Flux.from(dsl.select(BLOB_ID) + .from(TABLE_NAME) + .where(BUCKET_NAME.eq(bucketName.asString())) + .and(blobIdFrom.map(blobId -> BLOB_ID.greaterThan(blobId.asString())).orElseGet(DSL::noCondition)) + .orderBy(BLOB_ID.asc()) + .limit(batchSize))) + .map(record -> blobIdFactory.parse(record.get(BLOB_ID))) + .collectList() + .switchIfEmpty(Mono.just(ImmutableList.of())); + } +} diff --git a/server/blob/blob-postgres/src/test/java/org/apache/james/blob/postgres/PostgresBlobStoreDAOTest.java b/server/blob/blob-postgres/src/test/java/org/apache/james/blob/postgres/PostgresBlobStoreDAOTest.java new file mode 100644 index 00000000000..388cf3906c3 --- /dev/null +++ b/server/blob/blob-postgres/src/test/java/org/apache/james/blob/postgres/PostgresBlobStoreDAOTest.java @@ -0,0 +1,96 @@ +/**************************************************************** + * 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.blob.postgres; + +import static org.apache.james.blob.api.BlobStoreDAOFixture.TEST_BLOB_ID; +import static org.apache.james.blob.api.BlobStoreDAOFixture.TEST_BUCKET_NAME; + +import java.io.ByteArrayInputStream; +import java.time.Duration; +import java.util.concurrent.ExecutionException; + +import org.apache.james.backends.postgres.PostgresExtension; +import org.apache.james.blob.api.BlobStoreDAO; +import org.apache.james.blob.api.BlobStoreDAOContract; +import org.apache.james.blob.api.PlainBlobId; +import org.apache.james.util.concurrency.ConcurrentTestRunner; +import org.junit.jupiter.api.BeforeEach; +import org.junit.jupiter.api.Disabled; +import org.junit.jupiter.api.extension.RegisterExtension; +import org.junit.jupiter.params.ParameterizedTest; +import org.junit.jupiter.params.provider.MethodSource; + +import com.google.common.io.ByteSource; + +import reactor.core.publisher.Mono; + +class PostgresBlobStoreDAOTest implements BlobStoreDAOContract { + static Duration CONCURRENT_TEST_DURATION = Duration.ofMinutes(5); + + @RegisterExtension + static PostgresExtension postgresExtension = PostgresExtension.withoutRowLevelSecurity(PostgresBlobStorageModule.MODULE, PostgresExtension.PoolSize.LARGE); + + private PostgresBlobStoreDAO blobStore; + + @BeforeEach + void setUp() { + blobStore = new PostgresBlobStoreDAO(postgresExtension.getDefaultPostgresExecutor(), new PlainBlobId.Factory()); + } + + @Override + public BlobStoreDAO testee() { + return blobStore; + } + + @Override + @Disabled("Not supported") + public void listBucketsShouldReturnBucketsWithNoBlob() { + } + + @Override + @ParameterizedTest(name = "[{index}] {0}") + @MethodSource("blobs") + public void concurrentSaveByteSourceShouldReturnConsistentValues(String description, byte[] bytes) throws ExecutionException, InterruptedException { + Mono.from(testee().save(TEST_BUCKET_NAME, TEST_BLOB_ID, bytes)).block(); + ConcurrentTestRunner.builder() + .randomlyDistributedReactorOperations( + (threadNumber, step) -> testee().save(TEST_BUCKET_NAME, TEST_BLOB_ID, ByteSource.wrap(bytes)), + (threadNumber, step) -> checkConcurrentSaveOperation(bytes) + ) + .threadCount(10) + .operationCount(20) + .runSuccessfullyWithin(CONCURRENT_TEST_DURATION); + } + + @Override + @ParameterizedTest(name = "[{index}] {0}") + @MethodSource("blobs") + public void concurrentSaveInputStreamShouldReturnConsistentValues(String description, byte[] bytes) throws ExecutionException, InterruptedException { + Mono.from(testee().save(TEST_BUCKET_NAME, TEST_BLOB_ID, bytes)).block(); + ConcurrentTestRunner.builder() + .randomlyDistributedReactorOperations( + (threadNumber, step) -> testee().save(TEST_BUCKET_NAME, TEST_BLOB_ID, new ByteArrayInputStream(bytes)), + (threadNumber, step) -> checkConcurrentSaveOperation(bytes) + ) + .threadCount(10) + .operationCount(20) + .runSuccessfullyWithin(CONCURRENT_TEST_DURATION); + } +} \ No newline at end of file diff --git a/server/blob/blob-postgres/src/test/java/org/apache/james/blob/postgres/PostgresBlobStoreGCAlgorithmTest.java b/server/blob/blob-postgres/src/test/java/org/apache/james/blob/postgres/PostgresBlobStoreGCAlgorithmTest.java new file mode 100644 index 00000000000..212966bf1de --- /dev/null +++ b/server/blob/blob-postgres/src/test/java/org/apache/james/blob/postgres/PostgresBlobStoreGCAlgorithmTest.java @@ -0,0 +1,103 @@ +/**************************************************************** + * 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.blob.postgres; + +import static org.assertj.core.api.AssertionsForClassTypes.assertThat; +import static org.mockito.Mockito.when; + +import java.util.List; +import java.util.UUID; +import java.util.stream.IntStream; + +import org.apache.james.backends.postgres.PostgresExtension; +import org.apache.james.blob.api.BlobId; +import org.apache.james.blob.api.BlobStore; +import org.apache.james.blob.api.BlobStoreDAO; +import org.apache.james.blob.api.PlainBlobId; +import org.apache.james.server.blob.deduplication.BloomFilterGCAlgorithm; +import org.apache.james.server.blob.deduplication.BloomFilterGCAlgorithmContract; +import org.apache.james.task.Task; +import org.junit.jupiter.api.AfterAll; +import org.junit.jupiter.api.BeforeAll; +import org.junit.jupiter.api.BeforeEach; +import org.junit.jupiter.api.Test; +import org.junit.jupiter.api.extension.RegisterExtension; + +import reactor.core.publisher.Flux; +import reactor.core.publisher.Mono; + +public class PostgresBlobStoreGCAlgorithmTest implements BloomFilterGCAlgorithmContract { + + @RegisterExtension + static PostgresExtension postgresExtension = PostgresExtension.withoutRowLevelSecurity(PostgresBlobStorageModule.MODULE, PostgresExtension.PoolSize.LARGE); + private PostgresBlobStoreDAO blobStore; + + @BeforeAll + static void setUpClass() { + // We set the batch size to 10 to test the batching + System.setProperty("james.postgresql.query.batch.size", "10"); + } + + @AfterAll + static void tearDownClass() { + System.clearProperty("james.postgresql.query.batch.size"); + } + + @BeforeEach + void beforeEach() { + blobStore = new PostgresBlobStoreDAO(postgresExtension.getDefaultPostgresExecutor(), new PlainBlobId.Factory()); + } + + @Override + public BlobStoreDAO blobStoreDAO() { + return blobStore; + } + + @Test + void gcShouldSuccessWhenBatchSizeIsSmallerThanAllBlobEntries() { + BlobStore blobStore = blobStore(); + int orphanBlobCount = 200; + List referencedBlobIds = IntStream.range(0, 100) + .mapToObj(index -> Mono.from(blobStore.save(DEFAULT_BUCKET, UUID.randomUUID().toString(), BlobStore.StoragePolicy.HIGH_PERFORMANCE)).block()) + .toList(); + List orphanBlobIds = IntStream.range(0, orphanBlobCount) + .mapToObj(index -> Mono.from(blobStore.save(DEFAULT_BUCKET, UUID.randomUUID().toString(), BlobStore.StoragePolicy.HIGH_PERFORMANCE)).block()) + .toList(); + + when(BLOB_REFERENCE_SOURCE.listReferencedBlobs()).thenReturn(Flux.fromIterable(referencedBlobIds)); + CLOCK.setInstant(NOW.plusMonths(2).toInstant()); + + BloomFilterGCAlgorithm.Context context = new BloomFilterGCAlgorithm.Context(EXPECTED_BLOB_COUNT, ASSOCIATED_PROBABILITY); + BloomFilterGCAlgorithm bloomFilterGCAlgorithm = bloomFilterGCAlgorithm(); + Task.Result result = Mono.from(bloomFilterGCAlgorithm.gc(EXPECTED_BLOB_COUNT, DELETION_WINDOW_SIZE, ASSOCIATED_PROBABILITY, DEFAULT_BUCKET, context)).block(); + + assertThat(result).isEqualTo(Task.Result.COMPLETED); + BloomFilterGCAlgorithm.Context.Snapshot snapshot = context.snapshot(); + + assertThat(snapshot.getReferenceSourceCount()) + .isEqualTo(referencedBlobIds.size()); + assertThat(snapshot.getBlobCount()) + .isEqualTo(referencedBlobIds.size() + orphanBlobIds.size()); + + assertThat(snapshot.getGcedBlobCount()) + .isLessThanOrEqualTo(orphanBlobIds.size()) + .isGreaterThan(0); + } +} \ No newline at end of file diff --git a/server/blob/pom.xml b/server/blob/pom.xml index d429b1ad4fa..bd2aaa9f6ba 100644 --- a/server/blob/pom.xml +++ b/server/blob/pom.xml @@ -41,6 +41,7 @@ blob-export-file blob-file blob-memory + blob-postgres blob-s3 blob-storage-strategy diff --git a/server/container/guice/blob/postgres/pom.xml b/server/container/guice/blob/postgres/pom.xml new file mode 100644 index 00000000000..f42dcd8ea60 --- /dev/null +++ b/server/container/guice/blob/postgres/pom.xml @@ -0,0 +1,53 @@ + + + + + 4.0.0 + + org.apache.james + james-server-guice + 3.9.0-SNAPSHOT + ../../pom.xml + + + blob-postgres-guice + jar + + Apache James :: Server :: Blob Postgres - guice injection + Blob modules on Postgres storage + + + + ${james.groupId} + blob-api + + + ${james.groupId} + blob-postgres + + + com.google.inject + guice + + + + \ No newline at end of file diff --git a/server/container/guice/blob/postgres/src/main/java/modules/BlobPostgresModule.java b/server/container/guice/blob/postgres/src/main/java/modules/BlobPostgresModule.java new file mode 100644 index 00000000000..162e1176a78 --- /dev/null +++ b/server/container/guice/blob/postgres/src/main/java/modules/BlobPostgresModule.java @@ -0,0 +1,35 @@ +/**************************************************************** + * 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 modules; + +import org.apache.james.backends.postgres.PostgresModule; +import org.apache.james.blob.postgres.PostgresBlobStorageModule; + +import com.google.inject.AbstractModule; +import com.google.inject.multibindings.Multibinder; + +public class BlobPostgresModule extends AbstractModule { + + @Override + protected void configure() { + Multibinder postgresDataDefinitions = Multibinder.newSetBinder(binder(), PostgresModule.class); + postgresDataDefinitions.addBinding().toInstance(PostgresBlobStorageModule.MODULE); + } +} diff --git a/server/apps/distributed-app/src/test/java/org/apache/james/modules/AwsS3BlobStoreExtension.java b/server/container/guice/blob/s3/src/test/java/org/apache/james/modules/AwsS3BlobStoreExtension.java similarity index 100% rename from server/apps/distributed-app/src/test/java/org/apache/james/modules/AwsS3BlobStoreExtension.java rename to server/container/guice/blob/s3/src/test/java/org/apache/james/modules/AwsS3BlobStoreExtension.java diff --git a/server/container/guice/cassandra/src/main/java/org/apache/james/modules/data/CassandraJmapModule.java b/server/container/guice/cassandra/src/main/java/org/apache/james/modules/data/CassandraJmapModule.java index cebbbb50e2f..fce2ca29ea8 100644 --- a/server/container/guice/cassandra/src/main/java/org/apache/james/modules/data/CassandraJmapModule.java +++ b/server/container/guice/cassandra/src/main/java/org/apache/james/modules/data/CassandraJmapModule.java @@ -35,7 +35,9 @@ import org.apache.james.jmap.api.filtering.impl.FilterUsernameChangeTaskStep; import org.apache.james.jmap.api.identity.CustomIdentityDAO; import org.apache.james.jmap.api.identity.IdentityUserDeletionTaskStep; +import org.apache.james.jmap.api.projections.DefaultEmailQueryViewManager; import org.apache.james.jmap.api.projections.EmailQueryView; +import org.apache.james.jmap.api.projections.EmailQueryViewManager; import org.apache.james.jmap.api.projections.MessageFastViewProjection; import org.apache.james.jmap.api.projections.MessageFastViewProjectionHealthCheck; import org.apache.james.jmap.api.pushsubscription.PushDeleteUserDataTaskStep; @@ -95,6 +97,8 @@ protected void configure() { bind(CassandraEmailQueryView.class).in(Scopes.SINGLETON); bind(EmailQueryView.class).to(CassandraEmailQueryView.class); + bind(DefaultEmailQueryViewManager.class).in(Scopes.SINGLETON); + bind(EmailQueryViewManager.class).to(DefaultEmailQueryViewManager.class); Multibinder cassandraDataDefinitions = Multibinder.newSetBinder(binder(), CassandraModule.class); cassandraDataDefinitions.addBinding().toInstance(CassandraMessageFastViewProjectionModule.MODULE); diff --git a/server/container/guice/cassandra/src/main/java/org/apache/james/modules/mailbox/CassandraQuotaMailingModule.java b/server/container/guice/cassandra/src/main/java/org/apache/james/modules/mailbox/CassandraQuotaMailingModule.java index fd9145772f8..98d15767083 100644 --- a/server/container/guice/cassandra/src/main/java/org/apache/james/modules/mailbox/CassandraQuotaMailingModule.java +++ b/server/container/guice/cassandra/src/main/java/org/apache/james/modules/mailbox/CassandraQuotaMailingModule.java @@ -22,7 +22,7 @@ import org.apache.james.eventsourcing.Event; import org.apache.james.eventsourcing.eventstore.dto.EventDTO; import org.apache.james.eventsourcing.eventstore.dto.EventDTOModule; -import org.apache.james.mailbox.quota.cassandra.dto.QuotaEventDTOModules; +import org.apache.james.mailbox.quota.mailing.events.QuotaEventDTOModules; import com.google.inject.AbstractModule; import com.google.inject.TypeLiteral; diff --git a/server/container/guice/common/src/main/java/org/apache/james/GuiceJamesServer.java b/server/container/guice/common/src/main/java/org/apache/james/GuiceJamesServer.java index 98fcb294230..ab32d103292 100644 --- a/server/container/guice/common/src/main/java/org/apache/james/GuiceJamesServer.java +++ b/server/container/guice/common/src/main/java/org/apache/james/GuiceJamesServer.java @@ -92,8 +92,8 @@ public void start() throws Exception { preDestroy = injector.getInstance(Key.get(new TypeLiteral>() { })); injector.getInstance(ConfigurationSanitizingPerformer.class).sanitize(); - injector.getInstance(StartUpChecksPerformer.class).performCheck(); injector.getInstance(InitializationOperations.class).initModules(); + injector.getInstance(StartUpChecksPerformer.class).performCheck(); isStartedProbe.notifyStarted(); LOGGER.info("JAMES server started in: {}ms", stopwatch.elapsed(TimeUnit.MILLISECONDS)); } catch (Throwable e) { diff --git a/server/container/guice/common/src/main/java/org/apache/james/utils/InitializationOperations.java b/server/container/guice/common/src/main/java/org/apache/james/utils/InitializationOperations.java index 61adb875d35..a8cf0f45fb0 100644 --- a/server/container/guice/common/src/main/java/org/apache/james/utils/InitializationOperations.java +++ b/server/container/guice/common/src/main/java/org/apache/james/utils/InitializationOperations.java @@ -47,6 +47,7 @@ private Set processStartables() { return startables.get().stream() .flatMap(this::configurationPerformerFor) .distinct() + .sorted((a, b) -> Integer.compare(b.priority(), a.priority())) .peek(Throwing.consumer(InitializationOperation::initModule).sneakyThrow()) .collect(Collectors.toSet()); } diff --git a/server/container/guice/common/src/test/java/org/apache/james/JamesServerExtension.java b/server/container/guice/common/src/test/java/org/apache/james/JamesServerExtension.java index 85ff5ae53bf..b96ff32bd8b 100644 --- a/server/container/guice/common/src/test/java/org/apache/james/JamesServerExtension.java +++ b/server/container/guice/common/src/test/java/org/apache/james/JamesServerExtension.java @@ -214,4 +214,4 @@ private File createTmpDir() { public void await() { awaitCondition.await(); } -} +} \ No newline at end of file diff --git a/server/container/guice/configuration/src/main/java/org/apache/james/utils/InitializationOperation.java b/server/container/guice/configuration/src/main/java/org/apache/james/utils/InitializationOperation.java index 57417555909..7ddd75daada 100644 --- a/server/container/guice/configuration/src/main/java/org/apache/james/utils/InitializationOperation.java +++ b/server/container/guice/configuration/src/main/java/org/apache/james/utils/InitializationOperation.java @@ -27,6 +27,8 @@ public interface InitializationOperation { + int DEFAULT_PRIORITY = 0; + void initModule() throws Exception; /** @@ -41,4 +43,9 @@ public interface InitializationOperation { default List> requires() { return ImmutableList.of(); } + + default int priority() { + return DEFAULT_PRIORITY; + } + } diff --git a/server/container/guice/configuration/src/main/java/org/apache/james/utils/InitilizationOperationBuilder.java b/server/container/guice/configuration/src/main/java/org/apache/james/utils/InitilizationOperationBuilder.java index 84df2dad646..2896237d2bf 100644 --- a/server/container/guice/configuration/src/main/java/org/apache/james/utils/InitilizationOperationBuilder.java +++ b/server/container/guice/configuration/src/main/java/org/apache/james/utils/InitilizationOperationBuilder.java @@ -19,6 +19,8 @@ package org.apache.james.utils; +import static org.apache.james.utils.InitializationOperation.DEFAULT_PRIORITY; + import java.util.Arrays; import java.util.List; @@ -41,7 +43,11 @@ public interface RequireInit { } public static RequireInit forClass(Class type) { - return init -> new PrivateImpl(init, type); + return init -> new PrivateImpl(init, type, DEFAULT_PRIORITY); + } + + public static RequireInit forClass(Class type, int priority) { + return init -> new PrivateImpl(init, type, priority); } public static class PrivateImpl implements InitializationOperation { @@ -49,9 +55,12 @@ public static class PrivateImpl implements InitializationOperation { private final Class type; private List> requires; - private PrivateImpl(Init init, Class type) { + private final int priority; + + private PrivateImpl(Init init, Class type, int priority) { this.init = init; this.type = type; + this.priority = priority; /* Class requirements are by default infered from the parameters of the first @Inject annotated constructor. @@ -85,5 +94,10 @@ public PrivateImpl requires(List> requires) { public List> requires() { return requires; } + + @Override + public int priority() { + return priority; + } } } diff --git a/server/container/guice/distributed/pom.xml b/server/container/guice/distributed/pom.xml index f8c2c965dee..2df8a3efc16 100644 --- a/server/container/guice/distributed/pom.xml +++ b/server/container/guice/distributed/pom.xml @@ -63,6 +63,10 @@ ${james.groupId} blob-file
+ + ${james.groupId} + blob-postgres-guice + ${james.groupId} blob-s3-guice diff --git a/server/container/guice/distributed/src/main/java/org/apache/james/modules/blobstore/BlobStoreCacheModulesChooser.java b/server/container/guice/distributed/src/main/java/org/apache/james/modules/blobstore/BlobStoreCacheModulesChooser.java index 8789aac7ee8..5d6d9736695 100644 --- a/server/container/guice/distributed/src/main/java/org/apache/james/modules/blobstore/BlobStoreCacheModulesChooser.java +++ b/server/container/guice/distributed/src/main/java/org/apache/james/modules/blobstore/BlobStoreCacheModulesChooser.java @@ -53,7 +53,7 @@ public class BlobStoreCacheModulesChooser { private static final Logger LOGGER = LoggerFactory.getLogger(BlobStoreCacheModulesChooser.class); - static class CacheDisabledModule extends AbstractModule { + public static class CacheDisabledModule extends AbstractModule { @Provides @Named(MetricableBlobStore.BLOB_STORE_IMPLEMENTATION) @Singleton diff --git a/server/container/guice/distributed/src/main/java/org/apache/james/modules/blobstore/BlobStoreConfiguration.java b/server/container/guice/distributed/src/main/java/org/apache/james/modules/blobstore/BlobStoreConfiguration.java index 5b344994e6b..84d39ca0d5a 100644 --- a/server/container/guice/distributed/src/main/java/org/apache/james/modules/blobstore/BlobStoreConfiguration.java +++ b/server/container/guice/distributed/src/main/java/org/apache/james/modules/blobstore/BlobStoreConfiguration.java @@ -59,6 +59,10 @@ default RequireCache file() { default RequireCache s3() { return implementation(BlobStoreImplName.S3); } + + default RequireCache postgres() { + return implementation(BlobStoreImplName.POSTGRES); + } } @FunctionalInterface @@ -108,7 +112,8 @@ public static RequireImplementation builder() { public enum BlobStoreImplName { CASSANDRA("cassandra"), FILE("file"), - S3("s3"); + S3("s3"), + POSTGRES("postgres"); static String supportedImplNames() { return Stream.of(BlobStoreImplName.values()) @@ -151,13 +156,17 @@ public static BlobStoreConfiguration parse(org.apache.james.server.core.configur } public static BlobStoreConfiguration parse(PropertiesProvider propertiesProvider) throws ConfigurationException { + return parse(propertiesProvider, BlobStoreImplName.CASSANDRA); + } + + public static BlobStoreConfiguration parse(PropertiesProvider propertiesProvider, BlobStoreImplName defaultBlobStore) throws ConfigurationException { try { Configuration configuration = propertiesProvider.getConfigurations(ConfigurationComponent.NAMES); return BlobStoreConfiguration.from(configuration); } catch (FileNotFoundException e) { - LOGGER.warn("Could not find " + ConfigurationComponent.NAME + " configuration file, using cassandra blobstore as the default"); + LOGGER.warn("Could not find " + ConfigurationComponent.NAME + " configuration file, using " + defaultBlobStore.getName() + " blobstore as the default"); return BlobStoreConfiguration.builder() - .cassandra() + .implementation(defaultBlobStore) .disableCache() .passthrough() .noCryptoConfig(); @@ -238,7 +247,7 @@ public StorageStrategy storageStrategy() { return storageStrategy; } - BlobStoreImplName getImplementation() { + public BlobStoreImplName getImplementation() { return implementation; } diff --git a/server/container/guice/distributed/src/main/java/org/apache/james/modules/blobstore/BlobStoreModulesChooser.java b/server/container/guice/distributed/src/main/java/org/apache/james/modules/blobstore/BlobStoreModulesChooser.java index 7b5c461be09..41f1388a26f 100644 --- a/server/container/guice/distributed/src/main/java/org/apache/james/modules/blobstore/BlobStoreModulesChooser.java +++ b/server/container/guice/distributed/src/main/java/org/apache/james/modules/blobstore/BlobStoreModulesChooser.java @@ -38,6 +38,7 @@ import org.apache.james.blob.objectstorage.aws.sse.S3SSECConfiguration; import org.apache.james.blob.objectstorage.aws.sse.S3SSECustomerKeyFactory; import org.apache.james.blob.objectstorage.aws.sse.S3SSECustomerKeyFactory.SingleCustomerKeyFactory; +import org.apache.james.blob.postgres.PostgresBlobStoreDAO; import org.apache.james.core.healthcheck.HealthCheck; import org.apache.james.modules.blobstore.validation.BlobStoreConfigurationValidationStartUpCheck.StorageStrategySupplier; import org.apache.james.modules.blobstore.validation.StoragePolicyConfigurationSanityEnforcementModule; @@ -61,6 +62,8 @@ import com.google.inject.name.Named; import com.google.inject.name.Names; +import modules.BlobPostgresModule; + public class BlobStoreModulesChooser { private static final String UNENCRYPTED = "unencrypted"; @@ -108,6 +111,17 @@ protected void configure() { } } + static class PostgresBlobStoreDAODeclarationModule extends AbstractModule { + @Override + protected void configure() { + install(new BlobPostgresModule()); + + install(new DefaultBucketModule()); + + bind(BlobStoreDAO.class).annotatedWith(Names.named(UNENCRYPTED)).to(PostgresBlobStoreDAO.class); + } + } + static class NoEncryptionModule extends AbstractModule { @Provides @Singleton @@ -154,6 +168,8 @@ public static Module chooseBlobStoreDAOModule(BlobStoreConfiguration.BlobStoreIm return new ObjectStorageBlobStoreDAODeclarationModule(); case FILE: return new FileBlobStoreDAODeclarationModule(); + case POSTGRES: + return new PostgresBlobStoreDAODeclarationModule(); default: throw new RuntimeException("Unsupported blobStore implementation " + implementation); } diff --git a/server/container/guice/distributed/src/main/java/org/apache/james/modules/plugins/QuotaMailingModule.java b/server/container/guice/distributed/src/main/java/org/apache/james/modules/plugins/QuotaMailingModule.java new file mode 100644 index 00000000000..d5de19c2912 --- /dev/null +++ b/server/container/guice/distributed/src/main/java/org/apache/james/modules/plugins/QuotaMailingModule.java @@ -0,0 +1,40 @@ +/**************************************************************** + * 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.modules.plugins; + +import static org.apache.james.mailbox.quota.mailing.events.QuotaEventDTOModules.QUOTA_THRESHOLD_CHANGE; + +import org.apache.james.eventsourcing.Event; +import org.apache.james.eventsourcing.eventstore.dto.EventDTO; +import org.apache.james.eventsourcing.eventstore.dto.EventDTOModule; + +import com.google.inject.AbstractModule; +import com.google.inject.TypeLiteral; +import com.google.inject.multibindings.Multibinder; + +public class QuotaMailingModule extends AbstractModule { + @Override + protected void configure() { + Multibinder> eventDTOModuleBinder = Multibinder.newSetBinder(binder(), new TypeLiteral>() {}); + + eventDTOModuleBinder.addBinding() + .toInstance(QUOTA_THRESHOLD_CHANGE); + } +} \ No newline at end of file diff --git a/server/container/guice/distributed/src/test/java/org/apache/james/modules/blobstore/BlobStoreConfigurationTest.java b/server/container/guice/distributed/src/test/java/org/apache/james/modules/blobstore/BlobStoreConfigurationTest.java index 4bd6ea3ece6..65aff8ff8ba 100644 --- a/server/container/guice/distributed/src/test/java/org/apache/james/modules/blobstore/BlobStoreConfigurationTest.java +++ b/server/container/guice/distributed/src/test/java/org/apache/james/modules/blobstore/BlobStoreConfigurationTest.java @@ -234,13 +234,30 @@ void provideChoosingConfigurationShouldReturnFileFactoryWhenConfigurationImplIsF .noCryptoConfig()); } + @Test + void provideChoosingConfigurationShouldReturnPostgresFactoryWhenConfigurationImplIsPostgres() throws Exception { + PropertiesConfiguration configuration = new PropertiesConfiguration(); + configuration.addProperty("implementation", BlobStoreConfiguration.BlobStoreImplName.POSTGRES.getName()); + configuration.addProperty("deduplication.enable", "false"); + FakePropertiesProvider propertyProvider = FakePropertiesProvider.builder() + .register(ConfigurationComponent.NAME, configuration) + .build(); + + assertThat(parse(propertyProvider)) + .isEqualTo(BlobStoreConfiguration.builder() + .postgres() + .disableCache() + .passthrough() + .noCryptoConfig()); + } + @Test void fromShouldThrowWhenBlobStoreImplIsMissing() { PropertiesConfiguration configuration = new PropertiesConfiguration(); assertThatThrownBy(() -> BlobStoreConfiguration.from(configuration)) .isInstanceOf(IllegalStateException.class) - .hasMessage("implementation property is missing please use one of supported values in: cassandra, file, s3"); + .hasMessage("implementation property is missing please use one of supported values in: " + supportedBlobStores()); } @Test @@ -250,7 +267,7 @@ void fromShouldThrowWhenBlobStoreImplIsNull() { assertThatThrownBy(() -> BlobStoreConfiguration.from(configuration)) .isInstanceOf(IllegalStateException.class) - .hasMessage("implementation property is missing please use one of supported values in: cassandra, file, s3"); + .hasMessage("implementation property is missing please use one of supported values in: " + supportedBlobStores()); } @Test @@ -260,7 +277,7 @@ void fromShouldThrowWhenBlobStoreImplIsEmpty() { assertThatThrownBy(() -> BlobStoreConfiguration.from(configuration)) .isInstanceOf(IllegalStateException.class) - .hasMessage("implementation property is missing please use one of supported values in: cassandra, file, s3"); + .hasMessage("implementation property is missing please use one of supported values in: " + supportedBlobStores()); } @Test @@ -270,7 +287,11 @@ void fromShouldThrowWhenBlobStoreImplIsNotInSupportedList() { assertThatThrownBy(() -> BlobStoreConfiguration.from(configuration)) .isInstanceOf(IllegalArgumentException.class) - .hasMessage("un_supported is not a valid name of BlobStores, please use one of supported values in: cassandra, file, s3"); + .hasMessage("un_supported is not a valid name of BlobStores, please use one of supported values in: " + supportedBlobStores()); + } + + private String supportedBlobStores() { + return "cassandra, file, s3, postgres"; } @Test diff --git a/server/container/guice/mailbox-postgres/pom.xml b/server/container/guice/mailbox-postgres/pom.xml new file mode 100644 index 00000000000..28da17432dc --- /dev/null +++ b/server/container/guice/mailbox-postgres/pom.xml @@ -0,0 +1,83 @@ + + + + + 4.0.0 + + + org.apache.james + james-server-guice + 3.9.0-SNAPSHOT + ../pom.xml + + + james-server-guice-mailbox-postgres + jar + Apache James :: Server :: Postgres - Guice injection + + + + ${james.groupId} + apache-james-mailbox-deleted-messages-vault-postgres + + + ${james.groupId} + apache-james-mailbox-postgres + + + ${james.groupId} + apache-james-mailbox-quota-search-scanning + + + ${james.groupId} + blob-memory-guice + + + ${james.groupId} + james-server-data-postgres + + + ${james.groupId} + james-server-guice-mailbox + + + ${james.groupId} + james-server-guice-webadmin-data + + + ${james.groupId} + james-server-mailbox-adapter + + + ${james.groupId} + james-server-postgres-common-guice + + + ${james.groupId} + testing-base + test + + + com.google.inject + guice + + + + diff --git a/server/container/guice/mailbox-postgres/src/main/java/org/apache/james/modules/mailbox/PostgresDeletedMessageVaultModule.java b/server/container/guice/mailbox-postgres/src/main/java/org/apache/james/modules/mailbox/PostgresDeletedMessageVaultModule.java new file mode 100644 index 00000000000..d444bc4f1f8 --- /dev/null +++ b/server/container/guice/mailbox-postgres/src/main/java/org/apache/james/modules/mailbox/PostgresDeletedMessageVaultModule.java @@ -0,0 +1,50 @@ +/**************************************************************** + * 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.modules.mailbox; + +import org.apache.james.backends.postgres.PostgresModule; +import org.apache.james.mailbox.postgres.DeleteMessageListener; +import org.apache.james.modules.vault.DeletedMessageVaultModule; +import org.apache.james.vault.metadata.DeletedMessageMetadataVault; +import org.apache.james.vault.metadata.PostgresDeletedMessageMetadataModule; +import org.apache.james.vault.metadata.PostgresDeletedMessageMetadataVault; +import org.apache.james.vault.metadata.PostgresDeletedMessageVaultDeletionCallback; + +import com.google.inject.AbstractModule; +import com.google.inject.Scopes; +import com.google.inject.multibindings.Multibinder; + +public class PostgresDeletedMessageVaultModule extends AbstractModule { + @Override + protected void configure() { + install(new DeletedMessageVaultModule()); + + Multibinder postgresDataDefinitions = Multibinder.newSetBinder(binder(), PostgresModule.class); + postgresDataDefinitions.addBinding().toInstance(PostgresDeletedMessageMetadataModule.MODULE); + + bind(PostgresDeletedMessageMetadataVault.class).in(Scopes.SINGLETON); + bind(DeletedMessageMetadataVault.class) + .to(PostgresDeletedMessageMetadataVault.class); + + Multibinder.newSetBinder(binder(), DeleteMessageListener.DeletionCallback.class) + .addBinding() + .to(PostgresDeletedMessageVaultDeletionCallback.class); + } +} diff --git a/server/container/guice/mailbox-postgres/src/main/java/org/apache/james/modules/mailbox/PostgresMailboxModule.java b/server/container/guice/mailbox-postgres/src/main/java/org/apache/james/modules/mailbox/PostgresMailboxModule.java new file mode 100644 index 00000000000..de42727b537 --- /dev/null +++ b/server/container/guice/mailbox-postgres/src/main/java/org/apache/james/modules/mailbox/PostgresMailboxModule.java @@ -0,0 +1,192 @@ +/**************************************************************** + * 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.modules.mailbox; + +import static org.apache.james.modules.Names.MAILBOXMANAGER_NAME; + +import jakarta.inject.Singleton; + +import org.apache.james.adapter.mailbox.ACLUsernameChangeTaskStep; +import org.apache.james.adapter.mailbox.DelegationStoreAuthorizator; +import org.apache.james.adapter.mailbox.MailboxUserDeletionTaskStep; +import org.apache.james.adapter.mailbox.MailboxUsernameChangeTaskStep; +import org.apache.james.adapter.mailbox.QuotaUsernameChangeTaskStep; +import org.apache.james.adapter.mailbox.UserRepositoryAuthenticator; +import org.apache.james.adapter.mailbox.UserRepositoryAuthorizator; +import org.apache.james.backends.postgres.PostgresModule; +import org.apache.james.blob.api.BlobReferenceSource; +import org.apache.james.events.EventListener; +import org.apache.james.eventsourcing.Event; +import org.apache.james.eventsourcing.eventstore.dto.EventDTO; +import org.apache.james.eventsourcing.eventstore.dto.EventDTOModule; +import org.apache.james.mailbox.AttachmentContentLoader; +import org.apache.james.mailbox.AttachmentIdFactory; +import org.apache.james.mailbox.AttachmentManager; +import org.apache.james.mailbox.Authenticator; +import org.apache.james.mailbox.Authorizator; +import org.apache.james.mailbox.MailboxCounterCorrector; +import org.apache.james.mailbox.MailboxManager; +import org.apache.james.mailbox.MailboxPathLocker; +import org.apache.james.mailbox.MessageIdManager; +import org.apache.james.mailbox.RightManager; +import org.apache.james.mailbox.SessionProvider; +import org.apache.james.mailbox.SubscriptionManager; +import org.apache.james.mailbox.UuidBackedAttachmentIdFactory; +import org.apache.james.mailbox.acl.MailboxACLResolver; +import org.apache.james.mailbox.acl.UnionMailboxACLResolver; +import org.apache.james.mailbox.indexer.MessageIdReIndexer; +import org.apache.james.mailbox.indexer.ReIndexer; +import org.apache.james.mailbox.model.MailboxId; +import org.apache.james.mailbox.model.MessageId; +import org.apache.james.mailbox.postgres.DeleteMessageListener; +import org.apache.james.mailbox.postgres.PostgresMailboxAggregateModule; +import org.apache.james.mailbox.postgres.PostgresMailboxId; +import org.apache.james.mailbox.postgres.PostgresMailboxManager; +import org.apache.james.mailbox.postgres.PostgresMailboxSessionMapperFactory; +import org.apache.james.mailbox.postgres.PostgresMessageId; +import org.apache.james.mailbox.postgres.PostgresThreadIdGuessingAlgorithm; +import org.apache.james.mailbox.postgres.mail.PostgresAttachmentBlobReferenceSource; +import org.apache.james.mailbox.postgres.mail.PostgresMessageBlobReferenceSource; +import org.apache.james.mailbox.postgres.mail.dao.PostgresMessageDAO; +import org.apache.james.mailbox.postgres.mail.eventsourcing.acl.ACLModule; +import org.apache.james.mailbox.store.MailboxManagerConfiguration; +import org.apache.james.mailbox.store.MailboxSessionMapperFactory; +import org.apache.james.mailbox.store.NoMailboxPathLocker; +import org.apache.james.mailbox.store.SessionProviderImpl; +import org.apache.james.mailbox.store.StoreAttachmentManager; +import org.apache.james.mailbox.store.StoreMailboxManager; +import org.apache.james.mailbox.store.StoreMessageIdManager; +import org.apache.james.mailbox.store.StoreRightManager; +import org.apache.james.mailbox.store.StoreSubscriptionManager; +import org.apache.james.mailbox.store.event.MailboxAnnotationListener; +import org.apache.james.mailbox.store.event.MailboxSubscriptionListener; +import org.apache.james.mailbox.store.mail.AttachmentMapperFactory; +import org.apache.james.mailbox.store.mail.MailboxMapperFactory; +import org.apache.james.mailbox.store.mail.MessageMapperFactory; +import org.apache.james.mailbox.store.mail.ThreadIdGuessingAlgorithm; +import org.apache.james.mailbox.store.user.SubscriptionMapperFactory; +import org.apache.james.modules.data.PostgresCommonModule; +import org.apache.james.user.api.DeleteUserDataTaskStep; +import org.apache.james.user.api.UsernameChangeTaskStep; +import org.apache.james.utils.MailboxManagerDefinition; +import org.apache.mailbox.tools.indexer.MessageIdReIndexerImpl; +import org.apache.mailbox.tools.indexer.ReIndexerImpl; + +import com.google.inject.AbstractModule; +import com.google.inject.Inject; +import com.google.inject.Scopes; +import com.google.inject.TypeLiteral; +import com.google.inject.multibindings.Multibinder; +import com.google.inject.name.Names; + +public class PostgresMailboxModule extends AbstractModule { + + @Override + protected void configure() { + install(new PostgresCommonModule()); + + Multibinder postgresDataDefinitions = Multibinder.newSetBinder(binder(), PostgresModule.class); + postgresDataDefinitions.addBinding().toInstance(PostgresMailboxAggregateModule.MODULE); + + install(new PostgresQuotaModule()); + + bind(PostgresMailboxSessionMapperFactory.class).in(Scopes.SINGLETON); + bind(PostgresMailboxManager.class).in(Scopes.SINGLETON); + bind(NoMailboxPathLocker.class).in(Scopes.SINGLETON); + bind(StoreSubscriptionManager.class).in(Scopes.SINGLETON); + bind(UserRepositoryAuthenticator.class).in(Scopes.SINGLETON); + bind(UserRepositoryAuthorizator.class).in(Scopes.SINGLETON); + bind(UnionMailboxACLResolver.class).in(Scopes.SINGLETON); + bind(PostgresMessageId.Factory.class).in(Scopes.SINGLETON); + bind(PostgresThreadIdGuessingAlgorithm.class).in(Scopes.SINGLETON); + bind(ReIndexerImpl.class).in(Scopes.SINGLETON); + bind(SessionProviderImpl.class).in(Scopes.SINGLETON); + bind(StoreMessageIdManager.class).in(Scopes.SINGLETON); + bind(StoreRightManager.class).in(Scopes.SINGLETON); + + bind(SubscriptionMapperFactory.class).to(PostgresMailboxSessionMapperFactory.class); + bind(MessageMapperFactory.class).to(PostgresMailboxSessionMapperFactory.class); + bind(MailboxMapperFactory.class).to(PostgresMailboxSessionMapperFactory.class); + bind(MailboxSessionMapperFactory.class).to(PostgresMailboxSessionMapperFactory.class); + bind(MessageId.Factory.class).to(PostgresMessageId.Factory.class); + bind(ThreadIdGuessingAlgorithm.class).to(PostgresThreadIdGuessingAlgorithm.class); + + bind(SubscriptionManager.class).to(StoreSubscriptionManager.class); + bind(MailboxPathLocker.class).to(NoMailboxPathLocker.class); + bind(Authenticator.class).to(UserRepositoryAuthenticator.class); + bind(MailboxManager.class).to(PostgresMailboxManager.class); + bind(StoreMailboxManager.class).to(PostgresMailboxManager.class); + bind(SessionProvider.class).to(SessionProviderImpl.class); + bind(Authorizator.class).to(DelegationStoreAuthorizator.class); + bind(MailboxId.Factory.class).to(PostgresMailboxId.Factory.class); + bind(MailboxACLResolver.class).to(UnionMailboxACLResolver.class); + bind(MessageIdManager.class).to(StoreMessageIdManager.class); + bind(RightManager.class).to(StoreRightManager.class); + bind(AttachmentIdFactory.class).to(UuidBackedAttachmentIdFactory.class); + bind(AttachmentManager.class).to(StoreAttachmentManager.class); + bind(AttachmentContentLoader.class).to(AttachmentManager.class); + bind(AttachmentMapperFactory.class).to(PostgresMailboxSessionMapperFactory.class); + bind(MailboxCounterCorrector.class).toInstance(MailboxCounterCorrector.DEFAULT); + + bind(ReIndexer.class).to(ReIndexerImpl.class); + bind(MessageIdReIndexer.class).to(MessageIdReIndexerImpl.class); + + bind(PostgresMessageDAO.class).in(Scopes.SINGLETON); + + Multibinder.newSetBinder(binder(), MailboxManagerDefinition.class).addBinding().to(PostgresMailboxManagerDefinition.class); + + Multibinder.newSetBinder(binder(), EventListener.GroupEventListener.class) + .addBinding() + .to(MailboxAnnotationListener.class); + + Multibinder.newSetBinder(binder(), EventListener.GroupEventListener.class) + .addBinding() + .to(MailboxSubscriptionListener.class); + + Multibinder.newSetBinder(binder(), EventListener.ReactiveGroupEventListener.class) + .addBinding().to(DeleteMessageListener.class); + Multibinder.newSetBinder(binder(), DeleteMessageListener.DeletionCallback.class); + + bind(MailboxManager.class).annotatedWith(Names.named(MAILBOXMANAGER_NAME)).to(MailboxManager.class); + bind(MailboxManagerConfiguration.class).toInstance(MailboxManagerConfiguration.DEFAULT); + + Multibinder usernameChangeTaskStepMultibinder = Multibinder.newSetBinder(binder(), UsernameChangeTaskStep.class); + usernameChangeTaskStepMultibinder.addBinding().to(MailboxUsernameChangeTaskStep.class); + usernameChangeTaskStepMultibinder.addBinding().to(ACLUsernameChangeTaskStep.class); + usernameChangeTaskStepMultibinder.addBinding().to(QuotaUsernameChangeTaskStep.class); + + Multibinder deleteUserDataTaskStepMultibinder = Multibinder.newSetBinder(binder(), DeleteUserDataTaskStep.class); + deleteUserDataTaskStepMultibinder.addBinding().to(MailboxUserDeletionTaskStep.class); + + Multibinder blobReferenceSourceMultibinder = Multibinder.newSetBinder(binder(), BlobReferenceSource.class); + blobReferenceSourceMultibinder.addBinding().to(PostgresMessageBlobReferenceSource.class); + blobReferenceSourceMultibinder.addBinding().to(PostgresAttachmentBlobReferenceSource.class); + + Multibinder.newSetBinder(binder(), new TypeLiteral>() {}) + .addBinding().toInstance(ACLModule.ACL_UPDATE); + } + + @Singleton + private static class PostgresMailboxManagerDefinition extends MailboxManagerDefinition { + @Inject + private PostgresMailboxManagerDefinition(PostgresMailboxManager manager) { + super("postgres-mailboxmanager", manager); + } + } +} \ No newline at end of file diff --git a/server/container/guice/mailbox-postgres/src/main/java/org/apache/james/modules/mailbox/PostgresQuotaModule.java b/server/container/guice/mailbox-postgres/src/main/java/org/apache/james/modules/mailbox/PostgresQuotaModule.java new file mode 100644 index 00000000000..8e7ea84e288 --- /dev/null +++ b/server/container/guice/mailbox-postgres/src/main/java/org/apache/james/modules/mailbox/PostgresQuotaModule.java @@ -0,0 +1,68 @@ +/**************************************************************** + * 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.modules.mailbox; + +import org.apache.james.backends.postgres.PostgresModule; +import org.apache.james.backends.postgres.quota.PostgresQuotaCurrentValueDAO; +import org.apache.james.events.EventListener; +import org.apache.james.mailbox.postgres.quota.PostgresCurrentQuotaManager; +import org.apache.james.mailbox.postgres.quota.PostgresPerUserMaxQuotaManager; +import org.apache.james.mailbox.quota.CurrentQuotaManager; +import org.apache.james.mailbox.quota.MaxQuotaManager; +import org.apache.james.mailbox.quota.QuotaManager; +import org.apache.james.mailbox.quota.QuotaRootDeserializer; +import org.apache.james.mailbox.quota.QuotaRootResolver; +import org.apache.james.mailbox.quota.UserQuotaRootResolver; +import org.apache.james.mailbox.store.quota.DefaultUserQuotaRootResolver; +import org.apache.james.mailbox.store.quota.ListeningCurrentQuotaUpdater; +import org.apache.james.mailbox.store.quota.QuotaUpdater; +import org.apache.james.mailbox.store.quota.StoreQuotaManager; + +import com.google.inject.AbstractModule; +import com.google.inject.Scopes; +import com.google.inject.multibindings.Multibinder; + +public class PostgresQuotaModule extends AbstractModule { + + @Override + protected void configure() { + Multibinder postgresDataDefinitions = Multibinder.newSetBinder(binder(), PostgresModule.class); + postgresDataDefinitions.addBinding().toInstance(org.apache.james.backends.postgres.quota.PostgresQuotaModule.MODULE); + + bind(DefaultUserQuotaRootResolver.class).in(Scopes.SINGLETON); + bind(PostgresPerUserMaxQuotaManager.class).in(Scopes.SINGLETON); + bind(StoreQuotaManager.class).in(Scopes.SINGLETON); + bind(PostgresQuotaCurrentValueDAO.class).in(Scopes.SINGLETON); + bind(PostgresCurrentQuotaManager.class).in(Scopes.SINGLETON); + + bind(UserQuotaRootResolver.class).to(DefaultUserQuotaRootResolver.class); + bind(QuotaRootResolver.class).to(DefaultUserQuotaRootResolver.class); + bind(QuotaRootDeserializer.class).to(DefaultUserQuotaRootResolver.class); + bind(MaxQuotaManager.class).to(PostgresPerUserMaxQuotaManager.class); + bind(QuotaManager.class).to(StoreQuotaManager.class); + bind(CurrentQuotaManager.class).to(PostgresCurrentQuotaManager.class); + + bind(ListeningCurrentQuotaUpdater.class).in(Scopes.SINGLETON); + bind(QuotaUpdater.class).to(ListeningCurrentQuotaUpdater.class); + Multibinder.newSetBinder(binder(), EventListener.ReactiveGroupEventListener.class) + .addBinding() + .to(ListeningCurrentQuotaUpdater.class); + } +} diff --git a/server/container/guice/mailbox-postgres/src/main/java/org/apache/james/modules/mailbox/RLSSupportPostgresMailboxModule.java b/server/container/guice/mailbox-postgres/src/main/java/org/apache/james/modules/mailbox/RLSSupportPostgresMailboxModule.java new file mode 100644 index 00000000000..0217fa0d107 --- /dev/null +++ b/server/container/guice/mailbox-postgres/src/main/java/org/apache/james/modules/mailbox/RLSSupportPostgresMailboxModule.java @@ -0,0 +1,34 @@ +/**************************************************************** + * 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.modules.mailbox; + +import org.apache.james.backends.postgres.PostgresModule; +import org.apache.james.mailbox.postgres.mail.PostgresMailboxMemberModule; + +import com.google.inject.AbstractModule; +import com.google.inject.multibindings.Multibinder; + +public class RLSSupportPostgresMailboxModule extends AbstractModule { + @Override + protected void configure() { + Multibinder postgresDataDefinitions = Multibinder.newSetBinder(binder(), PostgresModule.class); + postgresDataDefinitions.addBinding().toInstance(PostgresMailboxMemberModule.MODULE); + } +} diff --git a/server/container/guice/memory/src/main/java/org/apache/james/modules/data/MemoryDataJmapModule.java b/server/container/guice/memory/src/main/java/org/apache/james/modules/data/MemoryDataJmapModule.java index cea7c2d0d6e..4c92db5f0af 100644 --- a/server/container/guice/memory/src/main/java/org/apache/james/modules/data/MemoryDataJmapModule.java +++ b/server/container/guice/memory/src/main/java/org/apache/james/modules/data/MemoryDataJmapModule.java @@ -26,7 +26,9 @@ import org.apache.james.jmap.api.filtering.impl.FilterUsernameChangeTaskStep; import org.apache.james.jmap.api.identity.CustomIdentityDAO; import org.apache.james.jmap.api.identity.IdentityUserDeletionTaskStep; +import org.apache.james.jmap.api.projections.DefaultEmailQueryViewManager; import org.apache.james.jmap.api.projections.EmailQueryView; +import org.apache.james.jmap.api.projections.EmailQueryViewManager; import org.apache.james.jmap.api.projections.MessageFastViewProjection; import org.apache.james.jmap.api.projections.MessageFastViewProjectionHealthCheck; import org.apache.james.jmap.api.pushsubscription.PushDeleteUserDataTaskStep; @@ -67,6 +69,8 @@ protected void configure() { bind(MemoryEmailQueryView.class).in(Scopes.SINGLETON); bind(EmailQueryView.class).to(MemoryEmailQueryView.class); + bind(DefaultEmailQueryViewManager.class).in(Scopes.SINGLETON); + bind(EmailQueryViewManager.class).to(DefaultEmailQueryViewManager.class); bind(MessageFastViewProjectionHealthCheck.class).in(Scopes.SINGLETON); Multibinder.newSetBinder(binder(), HealthCheck.class) diff --git a/server/container/guice/opensearch/pom.xml b/server/container/guice/opensearch/pom.xml index 2f12db9c703..f927cbad3bc 100644 --- a/server/container/guice/opensearch/pom.xml +++ b/server/container/guice/opensearch/pom.xml @@ -39,6 +39,12 @@ + + ${james.groupId} + apache-james-backends-opensearch + test-jar + test + ${james.groupId} apache-james-mailbox-opensearch @@ -55,6 +61,12 @@ ${james.groupId} apache-james-mailbox-tika + + ${james.groupId} + apache-james-mailbox-tika + test-jar + test + ${james.groupId} james-server-filesystem-api @@ -65,6 +77,12 @@ ${james.groupId} james-server-guice-common + + ${james.groupId} + james-server-guice-common + test-jar + test + ${james.groupId} james-server-guice-webadmin-mailbox diff --git a/server/apps/cassandra-app/src/test/java/org/apache/james/DockerOpenSearchExtension.java b/server/container/guice/opensearch/src/test/java/org/apache/james/DockerOpenSearchExtension.java similarity index 100% rename from server/apps/cassandra-app/src/test/java/org/apache/james/DockerOpenSearchExtension.java rename to server/container/guice/opensearch/src/test/java/org/apache/james/DockerOpenSearchExtension.java diff --git a/server/apps/cassandra-app/src/test/java/org/apache/james/DockerOpenSearchRule.java b/server/container/guice/opensearch/src/test/java/org/apache/james/DockerOpenSearchRule.java similarity index 96% rename from server/apps/cassandra-app/src/test/java/org/apache/james/DockerOpenSearchRule.java rename to server/container/guice/opensearch/src/test/java/org/apache/james/DockerOpenSearchRule.java index d8abc842012..84c18b4b4d0 100644 --- a/server/apps/cassandra-app/src/test/java/org/apache/james/DockerOpenSearchRule.java +++ b/server/container/guice/opensearch/src/test/java/org/apache/james/DockerOpenSearchRule.java @@ -21,7 +21,7 @@ import org.apache.james.backends.opensearch.DockerOpenSearch; import org.apache.james.backends.opensearch.DockerOpenSearchSingleton; -import org.apache.james.modules.TestDockerOpenSearchModule; +import org.apache.james.modules.mailbox.TestDockerOpenSearchModule; import org.junit.runner.Description; import org.junit.runners.model.Statement; diff --git a/server/apps/cassandra-app/src/test/java/org/apache/james/TikaExtension.java b/server/container/guice/opensearch/src/test/java/org/apache/james/TikaExtension.java similarity index 100% rename from server/apps/cassandra-app/src/test/java/org/apache/james/TikaExtension.java rename to server/container/guice/opensearch/src/test/java/org/apache/james/TikaExtension.java diff --git a/server/apps/cassandra-app/src/test/java/org/apache/james/modules/TestTikaModule.java b/server/container/guice/opensearch/src/test/java/org/apache/james/modules/TestTikaModule.java similarity index 100% rename from server/apps/cassandra-app/src/test/java/org/apache/james/modules/TestTikaModule.java rename to server/container/guice/opensearch/src/test/java/org/apache/james/modules/TestTikaModule.java diff --git a/server/apps/cassandra-app/src/test/java/org/apache/james/modules/TestDockerOpenSearchModule.java b/server/container/guice/opensearch/src/test/java/org/apache/james/modules/mailbox/TestDockerOpenSearchModule.java similarity index 98% rename from server/apps/cassandra-app/src/test/java/org/apache/james/modules/TestDockerOpenSearchModule.java rename to server/container/guice/opensearch/src/test/java/org/apache/james/modules/mailbox/TestDockerOpenSearchModule.java index 466647f6e84..850ccd7c5df 100644 --- a/server/apps/cassandra-app/src/test/java/org/apache/james/modules/TestDockerOpenSearchModule.java +++ b/server/container/guice/opensearch/src/test/java/org/apache/james/modules/mailbox/TestDockerOpenSearchModule.java @@ -17,7 +17,7 @@ * under the License. * ****************************************************************/ -package org.apache.james.modules; +package org.apache.james.modules.mailbox; import org.apache.james.CleanupTasksPerformer; import org.apache.james.backends.opensearch.DockerOpenSearch; diff --git a/server/container/guice/pom.xml b/server/container/guice/pom.xml index 9cca7cd195f..f6a817372af 100644 --- a/server/container/guice/pom.xml +++ b/server/container/guice/pom.xml @@ -37,6 +37,7 @@ blob/deduplication-gc blob/export blob/memory + blob/postgres blob/s3 cassandra common @@ -50,6 +51,7 @@ mailbox mailbox-jpa mailbox-plugin-deleted-messages-vault + mailbox-postgres mailet mailrepository-blob mailrepository-cassandra @@ -57,6 +59,7 @@ memory onami opensearch + postgres-common protocols/imap protocols/jmap protocols/lmtp @@ -78,6 +81,7 @@ queue/rabbitmq sieve-file sieve-jpa + sieve-postgres testing utils @@ -156,6 +160,11 @@ james-server-guice-mailbox-jpa ${project.version} + + ${james.groupId} + james-server-guice-mailbox-postgres + ${project.version} + ${james.groupId} james-server-guice-mailet @@ -176,6 +185,12 @@ james-server-guice-opensearch ${project.version} + + ${james.groupId} + james-server-guice-opensearch + ${project.version} + test-jar + ${james.groupId} james-server-guice-pop @@ -191,6 +206,11 @@ james-server-guice-sieve-jpa ${project.version} + + ${james.groupId} + james-server-guice-sieve-postgres + ${project.version} + ${james.groupId} james-server-guice-smtp @@ -252,6 +272,17 @@ ${project.version} test-jar + + ${james.groupId} + james-server-postgres-common-guice + ${project.version} + + + ${james.groupId} + james-server-postgres-common-guice + ${project.version} + test-jar + ${james.groupId} mailrepository-blob @@ -287,6 +318,12 @@ queue-rabbitmq-guice ${project.version} + + ${james.groupId} + queue-rabbitmq-guice + ${project.version} + test-jar + com.linagora logback-elasticsearch-appender diff --git a/server/container/guice/postgres-common/pom.xml b/server/container/guice/postgres-common/pom.xml new file mode 100644 index 00000000000..c0d95997d3e --- /dev/null +++ b/server/container/guice/postgres-common/pom.xml @@ -0,0 +1,103 @@ + + + + + 4.0.0 + + + org.apache.james + james-server-guice + 3.9.0-SNAPSHOT + ../pom.xml + + + james-server-postgres-common-guice + jar + + Apache James :: Server :: Postgres - guice common + + + empty + + + + + ${james.groupId} + apache-james-backends-postgres + test-jar + test + + + ${james.groupId} + dead-letter-postgres + + + ${james.groupId} + event-sourcing-event-store-postgres + ${project.version} + + + ${james.groupId} + james-server-data-file + + + ${james.groupId} + james-server-data-jmap-postgres + ${project.version} + + + ${james.groupId} + james-server-data-postgres + + + ${james.groupId} + james-server-guice-common + + + ${james.groupId} + james-server-guice-common + test-jar + test + + + ${james.groupId} + james-server-guice-distributed + + + ${james.groupId} + james-server-mailbox-adapter + + + ${james.groupId} + james-server-task-postgres + + + ${james.groupId} + testing-base + test + + + org.postgresql + postgresql + 42.7.0 + test + + + diff --git a/server/container/guice/postgres-common/src/main/java/org/apache/james/modules/data/PostgresCommonModule.java b/server/container/guice/postgres-common/src/main/java/org/apache/james/modules/data/PostgresCommonModule.java new file mode 100644 index 00000000000..c9c51a7ae62 --- /dev/null +++ b/server/container/guice/postgres-common/src/main/java/org/apache/james/modules/data/PostgresCommonModule.java @@ -0,0 +1,180 @@ +/**************************************************************** + * 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.modules.data; + +import static org.apache.james.backends.postgres.PostgresTableManager.INITIALIZATION_PRIORITY; +import static org.apache.james.backends.postgres.utils.PostgresExecutor.DEFAULT_INJECT; + +import java.io.FileNotFoundException; +import java.util.Set; + +import org.apache.commons.configuration2.ex.ConfigurationException; +import org.apache.james.backends.postgres.PostgresConfiguration; +import org.apache.james.backends.postgres.PostgresModule; +import org.apache.james.backends.postgres.PostgresTableManager; +import org.apache.james.backends.postgres.RowLevelSecurity; +import org.apache.james.backends.postgres.utils.JamesPostgresConnectionFactory; +import org.apache.james.backends.postgres.utils.PoolBackedPostgresConnectionFactory; +import org.apache.james.backends.postgres.utils.PostgresConnectionClosure; +import org.apache.james.backends.postgres.utils.PostgresExecutor; +import org.apache.james.backends.postgres.utils.PostgresHealthCheck; +import org.apache.james.core.healthcheck.HealthCheck; +import org.apache.james.metrics.api.MetricFactory; +import org.apache.james.utils.InitializationOperation; +import org.apache.james.utils.InitilizationOperationBuilder; +import org.apache.james.utils.PropertiesProvider; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + +import com.google.inject.AbstractModule; +import com.google.inject.Provides; +import com.google.inject.Scopes; +import com.google.inject.Singleton; +import com.google.inject.multibindings.Multibinder; +import com.google.inject.multibindings.ProvidesIntoSet; +import com.google.inject.name.Named; + +import io.r2dbc.postgresql.PostgresqlConnectionConfiguration; +import io.r2dbc.postgresql.PostgresqlConnectionFactory; +import io.r2dbc.spi.ConnectionFactory; + +public class PostgresCommonModule extends AbstractModule { + private static final Logger LOGGER = LoggerFactory.getLogger("POSTGRES"); + + @Override + public void configure() { + Multibinder.newSetBinder(binder(), PostgresModule.class); + + bind(PostgresExecutor.Factory.class).in(Scopes.SINGLETON); + bind(PostgresConnectionClosure.class).asEagerSingleton(); + + Multibinder.newSetBinder(binder(), HealthCheck.class) + .addBinding().to(PostgresHealthCheck.class); + } + + @Provides + @Singleton + PostgresConfiguration provideConfiguration(PropertiesProvider propertiesProvider) throws FileNotFoundException, ConfigurationException { + return PostgresConfiguration.from(propertiesProvider.getConfiguration(PostgresConfiguration.POSTGRES_CONFIGURATION_NAME)); + } + + @Provides + @Singleton + JamesPostgresConnectionFactory provideJamesPostgresConnectionFactory(PostgresConfiguration postgresConfiguration, + ConnectionFactory connectionFactory) { + return new PoolBackedPostgresConnectionFactory(postgresConfiguration.getRowLevelSecurity(), + postgresConfiguration.poolInitialSize(), + postgresConfiguration.poolMaxSize(), + connectionFactory); + } + + @Provides + @Named(JamesPostgresConnectionFactory.BY_PASS_RLS_INJECT) + @Singleton + JamesPostgresConnectionFactory provideJamesPostgresConnectionFactoryWithRLSBypass(PostgresConfiguration postgresConfiguration, + JamesPostgresConnectionFactory jamesPostgresConnectionFactory, + @Named(JamesPostgresConnectionFactory.BY_PASS_RLS_INJECT) ConnectionFactory connectionFactory) { + if (!postgresConfiguration.getRowLevelSecurity().isRowLevelSecurityEnabled()) { + return jamesPostgresConnectionFactory; + } + return new PoolBackedPostgresConnectionFactory(RowLevelSecurity.DISABLED, + postgresConfiguration.byPassRLSPoolInitialSize(), + postgresConfiguration.byPassRLSPoolMaxSize(), + connectionFactory); + } + + @Provides + @Singleton + ConnectionFactory postgresqlConnectionFactory(PostgresConfiguration postgresConfiguration) { + return new PostgresqlConnectionFactory(PostgresqlConnectionConfiguration.builder() + .host(postgresConfiguration.getHost()) + .port(postgresConfiguration.getPort()) + .username(postgresConfiguration.getDefaultCredential().getUsername()) + .password(postgresConfiguration.getDefaultCredential().getPassword()) + .database(postgresConfiguration.getDatabaseName()) + .schema(postgresConfiguration.getDatabaseSchema()) + .sslMode(postgresConfiguration.getSslMode()) + .build()); + } + + @Provides + @Named(JamesPostgresConnectionFactory.BY_PASS_RLS_INJECT) + @Singleton + ConnectionFactory postgresqlConnectionFactoryRLSBypass(PostgresConfiguration postgresConfiguration) { + return new PostgresqlConnectionFactory(PostgresqlConnectionConfiguration.builder() + .host(postgresConfiguration.getHost()) + .port(postgresConfiguration.getPort()) + .username(postgresConfiguration.getByPassRLSCredential().getUsername()) + .password(postgresConfiguration.getByPassRLSCredential().getPassword()) + .database(postgresConfiguration.getDatabaseName()) + .schema(postgresConfiguration.getDatabaseSchema()) + .sslMode(postgresConfiguration.getSslMode()) + .build()); + } + + @Provides + @Singleton + PostgresModule composePostgresDataDefinitions(Set modules) { + return PostgresModule.aggregateModules(modules); + } + + @Provides + @Singleton + PostgresTableManager postgresTableManager(PostgresExecutor postgresExecutor, + PostgresModule postgresModule, + PostgresConfiguration postgresConfiguration) { + return new PostgresTableManager(postgresExecutor, postgresModule, postgresConfiguration); + } + + @Provides + @Named(PostgresExecutor.BY_PASS_RLS_INJECT) + @Singleton + PostgresExecutor.Factory postgresExecutorFactoryWithRLSBypass(@Named(PostgresExecutor.BY_PASS_RLS_INJECT) JamesPostgresConnectionFactory singlePostgresConnectionFactory, + PostgresConfiguration postgresConfiguration, + MetricFactory metricFactory) { + return new PostgresExecutor.Factory(singlePostgresConnectionFactory, postgresConfiguration, metricFactory); + } + + @Provides + @Named(DEFAULT_INJECT) + @Singleton + PostgresExecutor defaultPostgresExecutor(PostgresExecutor.Factory factory) { + return factory.create(); + } + + @Provides + @Named(PostgresExecutor.BY_PASS_RLS_INJECT) + @Singleton + PostgresExecutor postgresExecutorWithRLSBypass(@Named(PostgresExecutor.BY_PASS_RLS_INJECT) PostgresExecutor.Factory factory) { + return factory.create(); + } + + @Provides + @Singleton + PostgresExecutor postgresExecutor(@Named(DEFAULT_INJECT) PostgresExecutor postgresExecutor) { + return postgresExecutor; + } + + @ProvidesIntoSet + InitializationOperation provisionPostgresTablesAndIndexes(PostgresTableManager postgresTableManager) { + return InitilizationOperationBuilder + .forClass(PostgresTableManager.class, INITIALIZATION_PRIORITY) + .init(postgresTableManager::initPostgres); + } +} diff --git a/server/container/guice/postgres-common/src/main/java/org/apache/james/modules/data/PostgresDLPConfigurationStoreModule.java b/server/container/guice/postgres-common/src/main/java/org/apache/james/modules/data/PostgresDLPConfigurationStoreModule.java new file mode 100644 index 00000000000..f5a765b41f7 --- /dev/null +++ b/server/container/guice/postgres-common/src/main/java/org/apache/james/modules/data/PostgresDLPConfigurationStoreModule.java @@ -0,0 +1,44 @@ +/**************************************************************** + * 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.modules.data; + +import org.apache.james.dlp.api.DLPConfigurationStore; +import org.apache.james.dlp.eventsourcing.EventSourcingDLPConfigurationStore; +import org.apache.james.dlp.eventsourcing.cassandra.DLPConfigurationModules; +import org.apache.james.eventsourcing.Event; +import org.apache.james.eventsourcing.eventstore.dto.EventDTO; +import org.apache.james.eventsourcing.eventstore.dto.EventDTOModule; + +import com.google.inject.AbstractModule; +import com.google.inject.Scopes; +import com.google.inject.TypeLiteral; +import com.google.inject.multibindings.Multibinder; + +public class PostgresDLPConfigurationStoreModule extends AbstractModule { + + @Override + protected void configure() { + bind(EventSourcingDLPConfigurationStore.class).in(Scopes.SINGLETON); + bind(DLPConfigurationStore.class).to(EventSourcingDLPConfigurationStore.class); + Multibinder> eventDTOModuleBinder = Multibinder.newSetBinder(binder(), new TypeLiteral<>() {}); + eventDTOModuleBinder.addBinding().toInstance(DLPConfigurationModules.DLP_CONFIGURATION_STORE); + eventDTOModuleBinder.addBinding().toInstance(DLPConfigurationModules.DLP_CONFIGURATION_CLEAR); + } +} diff --git a/server/container/guice/postgres-common/src/main/java/org/apache/james/modules/data/PostgresDataJmapModule.java b/server/container/guice/postgres-common/src/main/java/org/apache/james/modules/data/PostgresDataJmapModule.java new file mode 100644 index 00000000000..b9a34ab1941 --- /dev/null +++ b/server/container/guice/postgres-common/src/main/java/org/apache/james/modules/data/PostgresDataJmapModule.java @@ -0,0 +1,86 @@ +/**************************************************************** + * 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.modules.data; + +import org.apache.james.core.healthcheck.HealthCheck; +import org.apache.james.jmap.api.filtering.FilteringManagement; +import org.apache.james.jmap.api.filtering.FiltersDeleteUserDataTaskStep; +import org.apache.james.jmap.api.filtering.impl.EventSourcingFilteringManagement; +import org.apache.james.jmap.api.filtering.impl.FilterUsernameChangeTaskStep; +import org.apache.james.jmap.api.identity.CustomIdentityDAO; +import org.apache.james.jmap.api.identity.IdentityUserDeletionTaskStep; +import org.apache.james.jmap.api.projections.EmailQueryView; +import org.apache.james.jmap.api.projections.EmailQueryViewManager; +import org.apache.james.jmap.api.projections.MessageFastViewProjection; +import org.apache.james.jmap.api.projections.MessageFastViewProjectionHealthCheck; +import org.apache.james.jmap.api.pushsubscription.PushDeleteUserDataTaskStep; +import org.apache.james.jmap.api.upload.UploadRepository; +import org.apache.james.jmap.postgres.filtering.PostgresFilteringProjection; +import org.apache.james.jmap.postgres.identity.PostgresCustomIdentityDAO; +import org.apache.james.jmap.postgres.projections.PostgresEmailQueryView; +import org.apache.james.jmap.postgres.projections.PostgresEmailQueryViewManager; +import org.apache.james.jmap.postgres.projections.PostgresMessageFastViewProjection; +import org.apache.james.jmap.postgres.upload.PostgresUploadRepository; +import org.apache.james.mailbox.store.extractor.DefaultTextExtractor; +import org.apache.james.user.api.DeleteUserDataTaskStep; +import org.apache.james.user.api.UsernameChangeTaskStep; + +import com.google.inject.AbstractModule; +import com.google.inject.Scopes; +import com.google.inject.multibindings.Multibinder; + +public class PostgresDataJmapModule extends AbstractModule { + + @Override + protected void configure() { + bind(UploadRepository.class).to(PostgresUploadRepository.class); + + bind(PostgresCustomIdentityDAO.class).in(Scopes.SINGLETON); + bind(CustomIdentityDAO.class).to(PostgresCustomIdentityDAO.class); + + bind(EventSourcingFilteringManagement.class).in(Scopes.SINGLETON); + bind(FilteringManagement.class).to(EventSourcingFilteringManagement.class).asEagerSingleton(); + bind(PostgresFilteringProjection.class).in(Scopes.SINGLETON); + bind(EventSourcingFilteringManagement.ReadProjection.class).to(PostgresFilteringProjection.class); + + bind(DefaultTextExtractor.class).in(Scopes.SINGLETON); + + bind(PostgresMessageFastViewProjection.class).in(Scopes.SINGLETON); + bind(MessageFastViewProjection.class).to(PostgresMessageFastViewProjection.class); + + bind(PostgresEmailQueryView.class).in(Scopes.SINGLETON); + bind(EmailQueryView.class).to(PostgresEmailQueryView.class); + bind(PostgresEmailQueryView.class).in(Scopes.SINGLETON); + bind(EmailQueryViewManager.class).to(PostgresEmailQueryViewManager.class); + + bind(MessageFastViewProjectionHealthCheck.class).in(Scopes.SINGLETON); + Multibinder.newSetBinder(binder(), HealthCheck.class) + .addBinding() + .to(MessageFastViewProjectionHealthCheck.class); + Multibinder.newSetBinder(binder(), UsernameChangeTaskStep.class) + .addBinding() + .to(FilterUsernameChangeTaskStep.class); + + Multibinder deleteUserDataTaskSteps = Multibinder.newSetBinder(binder(), DeleteUserDataTaskStep.class); + deleteUserDataTaskSteps.addBinding().to(FiltersDeleteUserDataTaskStep.class); + deleteUserDataTaskSteps.addBinding().to(IdentityUserDeletionTaskStep.class); + deleteUserDataTaskSteps.addBinding().to(PushDeleteUserDataTaskStep.class); + } +} diff --git a/server/container/guice/postgres-common/src/main/java/org/apache/james/modules/data/PostgresDataModule.java b/server/container/guice/postgres-common/src/main/java/org/apache/james/modules/data/PostgresDataModule.java new file mode 100644 index 00000000000..e6860792b62 --- /dev/null +++ b/server/container/guice/postgres-common/src/main/java/org/apache/james/modules/data/PostgresDataModule.java @@ -0,0 +1,34 @@ +/**************************************************************** + * 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.modules.data; + +import org.apache.james.CoreDataModule; + +import com.google.inject.AbstractModule; + +public class PostgresDataModule extends AbstractModule { + @Override + protected void configure() { + install(new CoreDataModule()); + install(new PostgresDomainListModule()); + install(new PostgresRecipientRewriteTableModule()); + install(new PostgresMailRepositoryModule()); + } +} diff --git a/server/container/guice/postgres-common/src/main/java/org/apache/james/modules/data/PostgresDelegationStoreModule.java b/server/container/guice/postgres-common/src/main/java/org/apache/james/modules/data/PostgresDelegationStoreModule.java new file mode 100644 index 00000000000..886b21c7386 --- /dev/null +++ b/server/container/guice/postgres-common/src/main/java/org/apache/james/modules/data/PostgresDelegationStoreModule.java @@ -0,0 +1,39 @@ +/**************************************************************** + * 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.modules.data; + +import org.apache.james.user.api.DelegationStore; +import org.apache.james.user.api.DelegationUsernameChangeTaskStep; +import org.apache.james.user.api.UsernameChangeTaskStep; +import org.apache.james.user.postgres.PostgresDelegationStore; + +import com.google.inject.AbstractModule; +import com.google.inject.multibindings.Multibinder; + +public class PostgresDelegationStoreModule extends AbstractModule { + @Override + public void configure() { + bind(DelegationStore.class).to(PostgresDelegationStore.class); + bind(PostgresDelegationStore.UserExistencePredicate.class).to(PostgresDelegationStore.UserExistencePredicateImplementation.class); + + Multibinder.newSetBinder(binder(), UsernameChangeTaskStep.class) + .addBinding().to(DelegationUsernameChangeTaskStep.class); + } +} diff --git a/server/container/guice/postgres-common/src/main/java/org/apache/james/modules/data/PostgresDomainListModule.java b/server/container/guice/postgres-common/src/main/java/org/apache/james/modules/data/PostgresDomainListModule.java new file mode 100644 index 00000000000..728c1ad0513 --- /dev/null +++ b/server/container/guice/postgres-common/src/main/java/org/apache/james/modules/data/PostgresDomainListModule.java @@ -0,0 +1,49 @@ +/**************************************************************** + * 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.modules.data; + +import org.apache.james.backends.postgres.PostgresModule; +import org.apache.james.domainlist.api.DomainList; +import org.apache.james.domainlist.lib.DomainListConfiguration; +import org.apache.james.domainlist.postgres.PostgresDomainList; +import org.apache.james.domainlist.postgres.PostgresDomainModule; +import org.apache.james.utils.InitializationOperation; +import org.apache.james.utils.InitilizationOperationBuilder; + +import com.google.inject.AbstractModule; +import com.google.inject.Scopes; +import com.google.inject.multibindings.Multibinder; +import com.google.inject.multibindings.ProvidesIntoSet; + +public class PostgresDomainListModule extends AbstractModule { + @Override + public void configure() { + bind(PostgresDomainList.class).in(Scopes.SINGLETON); + bind(DomainList.class).to(PostgresDomainList.class); + Multibinder.newSetBinder(binder(), PostgresModule.class).addBinding().toInstance(PostgresDomainModule.MODULE); + } + + @ProvidesIntoSet + InitializationOperation configureDomainList(DomainListConfiguration configuration, PostgresDomainList postgresDomainList) { + return InitilizationOperationBuilder + .forClass(PostgresDomainList.class) + .init(() -> postgresDomainList.configure(configuration)); + } +} diff --git a/server/container/guice/postgres-common/src/main/java/org/apache/james/modules/data/PostgresDropListsModule.java b/server/container/guice/postgres-common/src/main/java/org/apache/james/modules/data/PostgresDropListsModule.java new file mode 100644 index 00000000000..d2f4397295b --- /dev/null +++ b/server/container/guice/postgres-common/src/main/java/org/apache/james/modules/data/PostgresDropListsModule.java @@ -0,0 +1,33 @@ +/**************************************************************** + * 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.modules.data; + +import org.apache.james.droplists.api.DropList; +import org.apache.james.droplists.postgres.PostgresDropList; + +import com.google.inject.AbstractModule; +import com.google.inject.Scopes; + +public class PostgresDropListsModule extends AbstractModule { + @Override + protected void configure() { + bind(DropList.class).to(PostgresDropList.class).in(Scopes.SINGLETON); + } +} diff --git a/server/container/guice/postgres-common/src/main/java/org/apache/james/modules/data/PostgresEventStoreModule.java b/server/container/guice/postgres-common/src/main/java/org/apache/james/modules/data/PostgresEventStoreModule.java new file mode 100644 index 00000000000..843ea4031ea --- /dev/null +++ b/server/container/guice/postgres-common/src/main/java/org/apache/james/modules/data/PostgresEventStoreModule.java @@ -0,0 +1,45 @@ +/**************************************************************** + * 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.modules.data; + +import org.apache.james.backends.postgres.PostgresModule; +import org.apache.james.eventsourcing.Event; +import org.apache.james.eventsourcing.eventstore.EventStore; +import org.apache.james.eventsourcing.eventstore.dto.EventDTO; +import org.apache.james.eventsourcing.eventstore.dto.EventDTOModule; +import org.apache.james.eventsourcing.eventstore.postgres.PostgresEventStore; + +import com.google.inject.AbstractModule; +import com.google.inject.Scopes; +import com.google.inject.TypeLiteral; +import com.google.inject.multibindings.Multibinder; + +public class PostgresEventStoreModule extends AbstractModule { + @Override + protected void configure() { + bind(PostgresEventStore.class).in(Scopes.SINGLETON); + bind(EventStore.class).to(PostgresEventStore.class); + + Multibinder postgresDataDefinitions = Multibinder.newSetBinder(binder(), PostgresModule.class); + postgresDataDefinitions.addBinding().toInstance(org.apache.james.eventsourcing.eventstore.postgres.PostgresEventStoreModule.MODULE); + + Multibinder.newSetBinder(binder(), new TypeLiteral>() {}); + } +} diff --git a/server/container/guice/postgres-common/src/main/java/org/apache/james/modules/data/PostgresMailRepositoryModule.java b/server/container/guice/postgres-common/src/main/java/org/apache/james/modules/data/PostgresMailRepositoryModule.java new file mode 100644 index 00000000000..550fb7c8cfc --- /dev/null +++ b/server/container/guice/postgres-common/src/main/java/org/apache/james/modules/data/PostgresMailRepositoryModule.java @@ -0,0 +1,62 @@ +/**************************************************************** + * 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.modules.data; + +import org.apache.commons.configuration2.BaseHierarchicalConfiguration; +import org.apache.james.backends.postgres.PostgresModule; +import org.apache.james.blob.api.BlobReferenceSource; +import org.apache.james.mailrepository.api.MailRepositoryFactory; +import org.apache.james.mailrepository.api.MailRepositoryUrlStore; +import org.apache.james.mailrepository.api.Protocol; +import org.apache.james.mailrepository.memory.MailRepositoryStoreConfiguration; +import org.apache.james.mailrepository.postgres.PostgresMailRepository; +import org.apache.james.mailrepository.postgres.PostgresMailRepositoryBlobReferenceSource; +import org.apache.james.mailrepository.postgres.PostgresMailRepositoryContentDAO; +import org.apache.james.mailrepository.postgres.PostgresMailRepositoryFactory; +import org.apache.james.mailrepository.postgres.PostgresMailRepositoryUrlStore; + +import com.google.common.collect.ImmutableList; +import com.google.inject.AbstractModule; +import com.google.inject.Scopes; +import com.google.inject.multibindings.Multibinder; + +public class PostgresMailRepositoryModule extends AbstractModule { + @Override + protected void configure() { + bind(PostgresMailRepositoryContentDAO.class).in(Scopes.SINGLETON); + bind(PostgresMailRepositoryUrlStore.class).in(Scopes.SINGLETON); + + bind(MailRepositoryUrlStore.class).to(PostgresMailRepositoryUrlStore.class); + + bind(MailRepositoryStoreConfiguration.Item.class) + .toProvider(() -> new MailRepositoryStoreConfiguration.Item( + ImmutableList.of(new Protocol("postgres")), + PostgresMailRepository.class.getName(), + new BaseHierarchicalConfiguration())); + + Multibinder.newSetBinder(binder(), MailRepositoryFactory.class) + .addBinding().to(PostgresMailRepositoryFactory.class); + Multibinder.newSetBinder(binder(), PostgresModule.class) + .addBinding().toInstance(org.apache.james.mailrepository.postgres.PostgresMailRepositoryModule.MODULE); + + Multibinder blobReferenceSourceMultibinder = Multibinder.newSetBinder(binder(), BlobReferenceSource.class); + blobReferenceSourceMultibinder.addBinding().to(PostgresMailRepositoryBlobReferenceSource.class); + } +} diff --git a/server/container/guice/postgres-common/src/main/java/org/apache/james/modules/data/PostgresRecipientRewriteTableModule.java b/server/container/guice/postgres-common/src/main/java/org/apache/james/modules/data/PostgresRecipientRewriteTableModule.java new file mode 100644 index 00000000000..363c9879b8b --- /dev/null +++ b/server/container/guice/postgres-common/src/main/java/org/apache/james/modules/data/PostgresRecipientRewriteTableModule.java @@ -0,0 +1,59 @@ +/**************************************************************** + * 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.modules.data; + +import org.apache.james.backends.postgres.PostgresModule; +import org.apache.james.rrt.api.AliasReverseResolver; +import org.apache.james.rrt.api.CanSendFrom; +import org.apache.james.rrt.api.RecipientRewriteTable; +import org.apache.james.rrt.lib.AliasReverseResolverImpl; +import org.apache.james.rrt.lib.CanSendFromImpl; +import org.apache.james.rrt.postgres.PostgresRecipientRewriteTable; +import org.apache.james.rrt.postgres.PostgresRecipientRewriteTableDAO; +import org.apache.james.server.core.configuration.ConfigurationProvider; +import org.apache.james.utils.InitializationOperation; +import org.apache.james.utils.InitilizationOperationBuilder; + +import com.google.inject.AbstractModule; +import com.google.inject.Scopes; +import com.google.inject.multibindings.Multibinder; +import com.google.inject.multibindings.ProvidesIntoSet; + +public class PostgresRecipientRewriteTableModule extends AbstractModule { + @Override + public void configure() { + bind(PostgresRecipientRewriteTable.class).in(Scopes.SINGLETON); + bind(PostgresRecipientRewriteTableDAO.class).in(Scopes.SINGLETON); + bind(RecipientRewriteTable.class).to(PostgresRecipientRewriteTable.class); + bind(AliasReverseResolverImpl.class).in(Scopes.SINGLETON); + bind(AliasReverseResolver.class).to(AliasReverseResolverImpl.class); + bind(CanSendFromImpl.class).in(Scopes.SINGLETON); + bind(CanSendFrom.class).to(CanSendFromImpl.class); + + Multibinder.newSetBinder(binder(), PostgresModule.class).addBinding().toInstance(org.apache.james.rrt.postgres.PostgresRecipientRewriteTableModule.MODULE); + } + + @ProvidesIntoSet + InitializationOperation configureRecipientRewriteTable(ConfigurationProvider configurationProvider, PostgresRecipientRewriteTable recipientRewriteTable) { + return InitilizationOperationBuilder + .forClass(PostgresRecipientRewriteTable.class) + .init(() -> recipientRewriteTable.configure(configurationProvider.getConfiguration("recipientrewritetable"))); + } +} diff --git a/server/container/guice/postgres-common/src/main/java/org/apache/james/modules/data/PostgresUsersRepositoryModule.java b/server/container/guice/postgres-common/src/main/java/org/apache/james/modules/data/PostgresUsersRepositoryModule.java new file mode 100644 index 00000000000..506258c5344 --- /dev/null +++ b/server/container/guice/postgres-common/src/main/java/org/apache/james/modules/data/PostgresUsersRepositoryModule.java @@ -0,0 +1,71 @@ +/**************************************************************** + * 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.modules.data; + +import org.apache.commons.configuration2.ex.ConfigurationException; +import org.apache.james.backends.postgres.PostgresModule; +import org.apache.james.server.core.configuration.ConfigurationProvider; +import org.apache.james.user.api.UsersRepository; +import org.apache.james.user.lib.UsersDAO; +import org.apache.james.user.postgres.PostgresUserModule; +import org.apache.james.user.postgres.PostgresUsersDAO; +import org.apache.james.user.postgres.PostgresUsersRepository; +import org.apache.james.user.postgres.PostgresUsersRepositoryConfiguration; +import org.apache.james.utils.InitializationOperation; +import org.apache.james.utils.InitilizationOperationBuilder; + +import com.google.inject.AbstractModule; +import com.google.inject.Provides; +import com.google.inject.Scopes; +import com.google.inject.Singleton; +import com.google.inject.multibindings.Multibinder; +import com.google.inject.multibindings.ProvidesIntoSet; + +public class PostgresUsersRepositoryModule extends AbstractModule { + + public static AbstractModule USER_CONFIGURATION_MODULE = new AbstractModule() { + @Provides + @Singleton + public PostgresUsersRepositoryConfiguration provideConfiguration(ConfigurationProvider configurationProvider) throws ConfigurationException { + return PostgresUsersRepositoryConfiguration.from( + configurationProvider.getConfiguration("usersrepository")); + } + }; + + @Override + public void configure() { + bind(PostgresUsersRepository.class).in(Scopes.SINGLETON); + bind(UsersRepository.class).to(PostgresUsersRepository.class); + + bind(PostgresUsersDAO.class).in(Scopes.SINGLETON); + bind(UsersDAO.class).to(PostgresUsersDAO.class); + + Multibinder postgresDataDefinitions = Multibinder.newSetBinder(binder(), PostgresModule.class); + postgresDataDefinitions.addBinding().toInstance(PostgresUserModule.MODULE); + } + + @ProvidesIntoSet + InitializationOperation configureInitialization(ConfigurationProvider configurationProvider, PostgresUsersRepository usersRepository) { + return InitilizationOperationBuilder + .forClass(PostgresUsersRepository.class) + .init(() -> usersRepository.configure(configurationProvider.getConfiguration("usersrepository"))); + } + +} diff --git a/server/container/guice/postgres-common/src/main/java/org/apache/james/modules/data/PostgresVacationModule.java b/server/container/guice/postgres-common/src/main/java/org/apache/james/modules/data/PostgresVacationModule.java new file mode 100644 index 00000000000..c7dddf4fd4a --- /dev/null +++ b/server/container/guice/postgres-common/src/main/java/org/apache/james/modules/data/PostgresVacationModule.java @@ -0,0 +1,56 @@ +/**************************************************************** + * 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.modules.data; + +import org.apache.james.DefaultVacationService; +import org.apache.james.backends.postgres.PostgresModule; +import org.apache.james.user.api.DeleteUserDataTaskStep; +import org.apache.james.vacation.api.NotificationRegistry; +import org.apache.james.vacation.api.VacationDeleteUserTaskStep; +import org.apache.james.vacation.api.VacationRepository; +import org.apache.james.vacation.api.VacationService; +import org.apache.james.vacation.postgres.PostgresNotificationRegistry; +import org.apache.james.vacation.postgres.PostgresVacationRepository; + +import com.google.inject.AbstractModule; +import com.google.inject.Scopes; +import com.google.inject.multibindings.Multibinder; + +public class PostgresVacationModule extends AbstractModule { + + @Override + public void configure() { + bind(DefaultVacationService.class).in(Scopes.SINGLETON); + bind(VacationService.class).to(DefaultVacationService.class); + + bind(PostgresVacationRepository.class).in(Scopes.SINGLETON); + bind(VacationRepository.class).to(PostgresVacationRepository.class); + + bind(PostgresNotificationRegistry.class).in(Scopes.SINGLETON); + bind(NotificationRegistry.class).to(PostgresNotificationRegistry.class); + + Multibinder postgresVacationModules = Multibinder.newSetBinder(binder(), PostgresModule.class); + postgresVacationModules.addBinding().toInstance(org.apache.james.vacation.postgres.PostgresVacationModule.MODULE); + + Multibinder deleteUserDataTaskSteps = Multibinder.newSetBinder(binder(), DeleteUserDataTaskStep.class); + deleteUserDataTaskSteps.addBinding().to(VacationDeleteUserTaskStep.class); + } + +} diff --git a/server/container/guice/postgres-common/src/main/java/org/apache/james/modules/events/PostgresDeadLetterModule.java b/server/container/guice/postgres-common/src/main/java/org/apache/james/modules/events/PostgresDeadLetterModule.java new file mode 100644 index 00000000000..9745ea79c1a --- /dev/null +++ b/server/container/guice/postgres-common/src/main/java/org/apache/james/modules/events/PostgresDeadLetterModule.java @@ -0,0 +1,47 @@ +/**************************************************************** + * 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.modules.events; + +import org.apache.james.backends.postgres.PostgresModule; +import org.apache.james.core.healthcheck.HealthCheck; +import org.apache.james.events.EventDeadLetters; +import org.apache.james.events.EventDeadLettersHealthCheck; +import org.apache.james.events.PostgresEventDeadLetters; +import org.apache.james.events.PostgresEventDeadLettersModule; + +import com.google.inject.AbstractModule; +import com.google.inject.Scopes; +import com.google.inject.multibindings.Multibinder; + +public class PostgresDeadLetterModule extends AbstractModule { + @Override + protected void configure() { + Multibinder.newSetBinder(binder(), PostgresModule.class) + .addBinding().toInstance(PostgresEventDeadLettersModule.MODULE); + + bind(PostgresEventDeadLetters.class).in(Scopes.SINGLETON); + bind(EventDeadLetters.class).to(PostgresEventDeadLetters.class); + + bind(EventDeadLettersHealthCheck.class).in(Scopes.SINGLETON); + Multibinder.newSetBinder(binder(), HealthCheck.class) + .addBinding() + .to(EventDeadLettersHealthCheck.class); + } +} diff --git a/server/container/guice/postgres-common/src/main/java/org/apache/james/modules/task/DistributedTaskManagerModule.java b/server/container/guice/postgres-common/src/main/java/org/apache/james/modules/task/DistributedTaskManagerModule.java new file mode 100644 index 00000000000..694158b409e --- /dev/null +++ b/server/container/guice/postgres-common/src/main/java/org/apache/james/modules/task/DistributedTaskManagerModule.java @@ -0,0 +1,109 @@ +/**************************************************************** + * 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.modules.task; + +import static org.apache.james.modules.queue.rabbitmq.RabbitMQModule.RABBITMQ_CONFIGURATION_NAME; + +import java.io.FileNotFoundException; + +import jakarta.inject.Singleton; + +import org.apache.commons.configuration2.Configuration; +import org.apache.commons.configuration2.ex.ConfigurationException; +import org.apache.james.backends.rabbitmq.SimpleConnectionPool; +import org.apache.james.core.healthcheck.HealthCheck; +import org.apache.james.modules.server.HostnameModule; +import org.apache.james.modules.server.TaskSerializationModule; +import org.apache.james.task.TaskManager; +import org.apache.james.task.eventsourcing.EventSourcingTaskManager; +import org.apache.james.task.eventsourcing.TerminationSubscriber; +import org.apache.james.task.eventsourcing.WorkQueueSupplier; +import org.apache.james.task.eventsourcing.distributed.CancelRequestQueueName; +import org.apache.james.task.eventsourcing.distributed.DistributedTaskManagerHealthCheck; +import org.apache.james.task.eventsourcing.distributed.RabbitMQTerminationSubscriber; +import org.apache.james.task.eventsourcing.distributed.RabbitMQWorkQueue; +import org.apache.james.task.eventsourcing.distributed.RabbitMQWorkQueueConfiguration; +import org.apache.james.task.eventsourcing.distributed.RabbitMQWorkQueueConfiguration$; +import org.apache.james.task.eventsourcing.distributed.RabbitMQWorkQueueReconnectionHandler; +import org.apache.james.task.eventsourcing.distributed.RabbitMQWorkQueueSupplier; +import org.apache.james.task.eventsourcing.distributed.TerminationQueueName; +import org.apache.james.task.eventsourcing.distributed.TerminationReconnectionHandler; +import org.apache.james.utils.InitializationOperation; +import org.apache.james.utils.InitilizationOperationBuilder; +import org.apache.james.utils.PropertiesProvider; + +import com.google.inject.AbstractModule; +import com.google.inject.Provides; +import com.google.inject.Scopes; +import com.google.inject.multibindings.Multibinder; +import com.google.inject.multibindings.ProvidesIntoSet; + +public class DistributedTaskManagerModule extends AbstractModule { + + @Override + protected void configure() { + install(new HostnameModule()); + install(new TaskSerializationModule()); + install(new PostgresTaskExecutionDetailsProjectionGuiceModule()); + + bind(EventSourcingTaskManager.class).in(Scopes.SINGLETON); + bind(RabbitMQWorkQueueSupplier.class).in(Scopes.SINGLETON); + bind(RabbitMQTerminationSubscriber.class).in(Scopes.SINGLETON); + bind(TerminationSubscriber.class).to(RabbitMQTerminationSubscriber.class); + bind(TaskManager.class).to(EventSourcingTaskManager.class); + bind(WorkQueueSupplier.class).to(RabbitMQWorkQueueSupplier.class); + bind(CancelRequestQueueName.class).toInstance(CancelRequestQueueName.generate()); + bind(TerminationQueueName.class).toInstance(TerminationQueueName.generate()); + + Multibinder reconnectionHandlerMultibinder = Multibinder.newSetBinder(binder(), SimpleConnectionPool.ReconnectionHandler.class); + reconnectionHandlerMultibinder.addBinding().to(RabbitMQWorkQueueReconnectionHandler.class); + reconnectionHandlerMultibinder.addBinding().to(TerminationReconnectionHandler.class); + + Multibinder.newSetBinder(binder(), HealthCheck.class) + .addBinding() + .to(DistributedTaskManagerHealthCheck.class); + } + + @Provides + @Singleton + private RabbitMQWorkQueueConfiguration getWorkQueueConfiguration(PropertiesProvider propertiesProvider) throws ConfigurationException { + try { + Configuration configuration = propertiesProvider.getConfiguration(RABBITMQ_CONFIGURATION_NAME); + return RabbitMQWorkQueueConfiguration$.MODULE$.from(configuration); + } catch (FileNotFoundException e) { + return RabbitMQWorkQueueConfiguration$.MODULE$.enabled(); + } + } + + @ProvidesIntoSet + InitializationOperation terminationSubscriber(RabbitMQTerminationSubscriber instance) { + return InitilizationOperationBuilder + .forClass(RabbitMQTerminationSubscriber.class) + .init(instance::start); + } + + @ProvidesIntoSet + InitializationOperation workQueue(EventSourcingTaskManager instance) { + return InitilizationOperationBuilder + .forClass(RabbitMQWorkQueue.class) + .init(instance::start); + } + +} diff --git a/server/container/guice/postgres-common/src/main/java/org/apache/james/modules/task/PostgresTaskExecutionDetailsProjectionGuiceModule.java b/server/container/guice/postgres-common/src/main/java/org/apache/james/modules/task/PostgresTaskExecutionDetailsProjectionGuiceModule.java new file mode 100644 index 00000000000..9f7bb0694a5 --- /dev/null +++ b/server/container/guice/postgres-common/src/main/java/org/apache/james/modules/task/PostgresTaskExecutionDetailsProjectionGuiceModule.java @@ -0,0 +1,40 @@ +/**************************************************************** + * 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.modules.task; + +import org.apache.james.backends.postgres.PostgresModule; +import org.apache.james.task.eventsourcing.TaskExecutionDetailsProjection; +import org.apache.james.task.eventsourcing.postgres.PostgresTaskExecutionDetailsProjection; +import org.apache.james.task.eventsourcing.postgres.PostgresTaskExecutionDetailsProjectionModule; + +import com.google.inject.AbstractModule; +import com.google.inject.Scopes; +import com.google.inject.multibindings.Multibinder; + +public class PostgresTaskExecutionDetailsProjectionGuiceModule extends AbstractModule { + @Override + protected void configure() { + bind(TaskExecutionDetailsProjection.class).to(PostgresTaskExecutionDetailsProjection.class) + .in(Scopes.SINGLETON); + + Multibinder postgresDataDefinitions = Multibinder.newSetBinder(binder(), PostgresModule.class); + postgresDataDefinitions.addBinding().toInstance(PostgresTaskExecutionDetailsProjectionModule.MODULE()); + } +} diff --git a/server/container/guice/protocols/webadmin-jmap/src/main/java/org/apache/james/modules/server/JmapUploadCleanupModule.java b/server/container/guice/protocols/webadmin-jmap/src/main/java/org/apache/james/modules/server/JmapUploadCleanupModule.java index d746556705d..632e7af2734 100644 --- a/server/container/guice/protocols/webadmin-jmap/src/main/java/org/apache/james/modules/server/JmapUploadCleanupModule.java +++ b/server/container/guice/protocols/webadmin-jmap/src/main/java/org/apache/james/modules/server/JmapUploadCleanupModule.java @@ -19,7 +19,7 @@ package org.apache.james.modules.server; -import org.apache.james.jmap.cassandra.upload.CassandraUploadRepository; +import org.apache.james.jmap.api.upload.UploadRepository; import org.apache.james.server.task.json.dto.AdditionalInformationDTO; import org.apache.james.server.task.json.dto.AdditionalInformationDTOModule; import org.apache.james.server.task.json.dto.TaskDTO; @@ -46,8 +46,8 @@ protected void configure() { } @ProvidesIntoSet - public TaskDTOModule uploadRepositoryCleanupTask(CassandraUploadRepository cassandraUploadRepository) { - return UploadCleanupTaskDTO.module(cassandraUploadRepository); + public TaskDTOModule uploadRepositoryCleanupTask(UploadRepository uploadRepository) { + return UploadCleanupTaskDTO.module(uploadRepository); } @ProvidesIntoSet diff --git a/server/container/guice/queue/rabbitmq/pom.xml b/server/container/guice/queue/rabbitmq/pom.xml index 0fc3035c5a9..d2e6e480cf3 100644 --- a/server/container/guice/queue/rabbitmq/pom.xml +++ b/server/container/guice/queue/rabbitmq/pom.xml @@ -36,6 +36,23 @@ ${james.groupId} apache-james-backends-rabbitmq + + ${james.groupId} + apache-james-backends-rabbitmq + test-jar + test + + + ${james.groupId} + james-server-guice-common + test + + + ${james.groupId} + james-server-guice-common + test-jar + test + ${james.groupId} james-server-guice-configuration diff --git a/server/apps/distributed-app/src/test/java/org/apache/james/modules/DockerRabbitMQRule.java b/server/container/guice/queue/rabbitmq/src/test/java/org/apache/james/modules/DockerRabbitMQRule.java similarity index 100% rename from server/apps/distributed-app/src/test/java/org/apache/james/modules/DockerRabbitMQRule.java rename to server/container/guice/queue/rabbitmq/src/test/java/org/apache/james/modules/DockerRabbitMQRule.java diff --git a/server/apps/distributed-app/src/test/java/org/apache/james/modules/RabbitMQExtension.java b/server/container/guice/queue/rabbitmq/src/test/java/org/apache/james/modules/RabbitMQExtension.java similarity index 99% rename from server/apps/distributed-app/src/test/java/org/apache/james/modules/RabbitMQExtension.java rename to server/container/guice/queue/rabbitmq/src/test/java/org/apache/james/modules/RabbitMQExtension.java index 2da7a4105c1..743371f4b77 100644 --- a/server/apps/distributed-app/src/test/java/org/apache/james/modules/RabbitMQExtension.java +++ b/server/container/guice/queue/rabbitmq/src/test/java/org/apache/james/modules/RabbitMQExtension.java @@ -59,4 +59,4 @@ public boolean supportsParameter(ParameterContext parameterContext, ExtensionCon public Object resolveParameter(ParameterContext parameterContext, ExtensionContext extensionContext) throws ParameterResolutionException { return dockerRabbitMQ(); } -} +} \ No newline at end of file diff --git a/server/apps/distributed-app/src/test/java/org/apache/james/modules/TestRabbitMQModule.java b/server/container/guice/queue/rabbitmq/src/test/java/org/apache/james/modules/TestRabbitMQModule.java similarity index 98% rename from server/apps/distributed-app/src/test/java/org/apache/james/modules/TestRabbitMQModule.java rename to server/container/guice/queue/rabbitmq/src/test/java/org/apache/james/modules/TestRabbitMQModule.java index 60835d933b7..e068482b5ba 100644 --- a/server/apps/distributed-app/src/test/java/org/apache/james/modules/TestRabbitMQModule.java +++ b/server/container/guice/queue/rabbitmq/src/test/java/org/apache/james/modules/TestRabbitMQModule.java @@ -33,6 +33,7 @@ import org.apache.james.queue.rabbitmq.RabbitMQMailQueueManagement; import org.apache.james.queue.rabbitmq.view.RabbitMQMailQueueConfiguration; import org.apache.james.queue.rabbitmq.view.cassandra.configuration.CassandraMailQueueViewConfiguration; +import org.apache.james.task.Task; import com.google.inject.AbstractModule; import com.google.inject.Provides; @@ -102,7 +103,7 @@ public QueueCleanUp(RabbitMQMailQueueManagement api) { public Result run() { api.deleteAllQueues(); - return Result.COMPLETED; + return Task.Result.COMPLETED; } } } diff --git a/server/container/guice/sieve-postgres/pom.xml b/server/container/guice/sieve-postgres/pom.xml new file mode 100644 index 00000000000..512875ef11f --- /dev/null +++ b/server/container/guice/sieve-postgres/pom.xml @@ -0,0 +1,53 @@ + + + + + 4.0.0 + + + org.apache.james + james-server-guice + 3.9.0-SNAPSHOT + + + james-server-guice-sieve-postgres + jar + + Apache James :: Server :: Guice :: Sieve :: Postgres + Sieve Postgres modules for Guice implementation of James server + + + + ${james.groupId} + james-server-data-postgres + + + + ${james.groupId} + james-server-testing + test + + + com.google.inject + guice + + + + diff --git a/server/container/guice/sieve-postgres/src/main/java/org/apache/james/modules/data/SievePostgresRepositoryModules.java b/server/container/guice/sieve-postgres/src/main/java/org/apache/james/modules/data/SievePostgresRepositoryModules.java new file mode 100644 index 00000000000..a3191352624 --- /dev/null +++ b/server/container/guice/sieve-postgres/src/main/java/org/apache/james/modules/data/SievePostgresRepositoryModules.java @@ -0,0 +1,46 @@ +/**************************************************************** + * 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.modules.data; + +import org.apache.james.backends.postgres.PostgresModule; +import org.apache.james.sieve.postgres.PostgresSieveModule; +import org.apache.james.sieve.postgres.PostgresSieveQuotaDAO; +import org.apache.james.sieve.postgres.PostgresSieveRepository; +import org.apache.james.sieve.postgres.PostgresSieveScriptDAO; +import org.apache.james.sieverepository.api.SieveQuotaRepository; +import org.apache.james.sieverepository.api.SieveRepository; + +import com.google.inject.AbstractModule; +import com.google.inject.Scopes; +import com.google.inject.multibindings.Multibinder; + +public class SievePostgresRepositoryModules extends AbstractModule { + @Override + protected void configure() { + Multibinder.newSetBinder(binder(), PostgresModule.class).addBinding().toInstance(PostgresSieveModule.MODULE); + + bind(PostgresSieveQuotaDAO.class).in(Scopes.SINGLETON); + bind(PostgresSieveScriptDAO.class).in(Scopes.SINGLETON); + + bind(PostgresSieveRepository.class).in(Scopes.SINGLETON); + bind(SieveRepository.class).to(PostgresSieveRepository.class); + bind(SieveQuotaRepository.class).to(PostgresSieveRepository.class); + } +} diff --git a/server/data/data-api/src/main/java/org/apache/james/vacation/api/Vacation.java b/server/data/data-api/src/main/java/org/apache/james/vacation/api/Vacation.java index 31804a516e5..c378d63700c 100644 --- a/server/data/data-api/src/main/java/org/apache/james/vacation/api/Vacation.java +++ b/server/data/data-api/src/main/java/org/apache/james/vacation/api/Vacation.java @@ -173,13 +173,20 @@ public boolean equals(Object o) { Vacation vacation = (Vacation) o; return Objects.equals(this.isEnabled, vacation.isEnabled) && - Objects.equals(this.fromDate, vacation.fromDate) && - Objects.equals(this.toDate, vacation.toDate) && + compareZonedDateTimeAcrossTimeZone(this.fromDate, vacation.fromDate) && + compareZonedDateTimeAcrossTimeZone(this.toDate, vacation.toDate) && Objects.equals(this.textBody, vacation.textBody) && Objects.equals(this.subject, vacation.subject) && Objects.equals(this.htmlBody, vacation.htmlBody); } + private boolean compareZonedDateTimeAcrossTimeZone(Optional thisZonedDateTimeOptional, Optional thatZonedDateTimeOptional) { + return thisZonedDateTimeOptional.map(thisZonedDateTime -> thatZonedDateTimeOptional + .map(thisZonedDateTime::isEqual) + .orElse(false)) + .orElseGet(thatZonedDateTimeOptional::isEmpty); + } + @Override public int hashCode() { return Objects.hash(isEnabled, fromDate, toDate, textBody, subject, htmlBody); diff --git a/server/data/data-api/src/test/java/org/apache/james/vacation/api/NotificationRegistryContract.java b/server/data/data-api/src/test/java/org/apache/james/vacation/api/NotificationRegistryContract.java index b5196b8167c..5f13091c4b7 100644 --- a/server/data/data-api/src/test/java/org/apache/james/vacation/api/NotificationRegistryContract.java +++ b/server/data/data-api/src/test/java/org/apache/james/vacation/api/NotificationRegistryContract.java @@ -115,7 +115,7 @@ default void registerShouldNotPersistWhenExpiryDateIsPresent() { notificationRegistry().register(ACCOUNT_ID, recipientId(), Optional.of(ZONED_DATE_TIME)).block(); - assertThat(notificationRegistry().isRegistered(ACCOUNT_ID, recipientId()).block()).isTrue(); + assertThat(notificationRegistry().isRegistered(ACCOUNT_ID, recipientId()).block()).isFalse(); } @Test diff --git a/server/data/data-cassandra/src/main/java/org/apache/james/sieve/cassandra/CassandraSieveQuotaDAOV2.java b/server/data/data-cassandra/src/main/java/org/apache/james/sieve/cassandra/CassandraSieveQuotaDAOV2.java index 668ffd71646..798b23d13bc 100644 --- a/server/data/data-cassandra/src/main/java/org/apache/james/sieve/cassandra/CassandraSieveQuotaDAOV2.java +++ b/server/data/data-cassandra/src/main/java/org/apache/james/sieve/cassandra/CassandraSieveQuotaDAOV2.java @@ -51,14 +51,14 @@ public CassandraSieveQuotaDAOV2(CassandraQuotaCurrentValueDao currentValueDao, C @Override public Mono spaceUsedBy(Username username) { - CassandraQuotaCurrentValueDao.QuotaKey quotaKey = asQuotaKey(username); + QuotaCurrentValue.Key quotaKey = asQuotaKey(username); return currentValueDao.getQuotaCurrentValue(quotaKey).map(QuotaCurrentValue::getCurrentValue) .switchIfEmpty(Mono.just(0L)); } - private CassandraQuotaCurrentValueDao.QuotaKey asQuotaKey(Username username) { - return CassandraQuotaCurrentValueDao.QuotaKey.of( + private QuotaCurrentValue.Key asQuotaKey(Username username) { + return QuotaCurrentValue.Key.of( QUOTA_COMPONENT, username.asString(), QuotaType.SIZE); @@ -66,7 +66,7 @@ private CassandraQuotaCurrentValueDao.QuotaKey asQuotaKey(Username username) { @Override public Mono updateSpaceUsed(Username username, long spaceUsed) { - CassandraQuotaCurrentValueDao.QuotaKey quotaKey = asQuotaKey(username); + QuotaCurrentValue.Key quotaKey = asQuotaKey(username); return currentValueDao.deleteQuotaCurrentValue(quotaKey) .then(currentValueDao.increase(quotaKey, spaceUsed)); @@ -93,7 +93,7 @@ public Mono setQuota(QuotaSizeLimit quota) { @Override public Mono removeQuota() { - return limitDao.deleteQuotaLimit(CassandraQuotaLimitDao.QuotaLimitKey.of(QUOTA_COMPONENT, QuotaScope.GLOBAL, GLOBAL, QuotaType.SIZE)); + return limitDao.deleteQuotaLimit(QuotaLimit.QuotaLimitKey.of(QUOTA_COMPONENT, QuotaScope.GLOBAL, GLOBAL, QuotaType.SIZE)); } @Override @@ -117,7 +117,7 @@ public Mono setQuota(Username username, QuotaSizeLimit quota) { @Override public Mono removeQuota(Username username) { - return limitDao.deleteQuotaLimit(CassandraQuotaLimitDao.QuotaLimitKey.of( + return limitDao.deleteQuotaLimit(QuotaLimit.QuotaLimitKey.of( QUOTA_COMPONENT, QuotaScope.USER, username.asString(), QuotaType.SIZE)); } diff --git a/server/data/data-cassandra/src/main/java/org/apache/james/vacation/cassandra/CassandraNotificationRegistry.java b/server/data/data-cassandra/src/main/java/org/apache/james/vacation/cassandra/CassandraNotificationRegistry.java index 76033531252..43091bd04b9 100644 --- a/server/data/data-cassandra/src/main/java/org/apache/james/vacation/cassandra/CassandraNotificationRegistry.java +++ b/server/data/data-cassandra/src/main/java/org/apache/james/vacation/cassandra/CassandraNotificationRegistry.java @@ -81,6 +81,6 @@ public Mono flush(AccountId accountId) { } private boolean isValid(Optional waitDelay) { - return waitDelay.isEmpty() || waitDelay.get() >= 0; + return waitDelay.isEmpty() || waitDelay.get() > 0; } } diff --git a/server/data/data-jmap-cassandra/src/main/java/org/apache/james/jmap/cassandra/upload/CassandraUploadRepository.java b/server/data/data-jmap-cassandra/src/main/java/org/apache/james/jmap/cassandra/upload/CassandraUploadRepository.java index b66ed6ca15a..9de6f27c023 100644 --- a/server/data/data-jmap-cassandra/src/main/java/org/apache/james/jmap/cassandra/upload/CassandraUploadRepository.java +++ b/server/data/data-jmap-cassandra/src/main/java/org/apache/james/jmap/cassandra/upload/CassandraUploadRepository.java @@ -46,9 +46,7 @@ import reactor.core.publisher.Mono; public class CassandraUploadRepository implements UploadRepository { - public static final BucketName UPLOAD_BUCKET = BucketName.of("jmap-uploads"); - public static final Duration EXPIRE_DURATION = Duration.ofDays(7); private final UploadDAO uploadDAO; private final BlobStore blobStore; private final Clock clock; @@ -91,10 +89,11 @@ public Flux listUploads(Username user) { .map(UploadDAO.UploadRepresentation::toUploadMetaData); } - public Mono purge() { - Instant sevenDaysAgo = clock.instant().minus(EXPIRE_DURATION); + @Override + public Mono deleteByUploadDateBefore(Duration expireDuration) { + Instant expirationTime = clock.instant().minus(expireDuration); return Flux.from(uploadDAO.all()) - .filter(upload -> upload.getUploadDate().isBefore(sevenDaysAgo)) + .filter(upload -> upload.getUploadDate().isBefore(expirationTime)) .flatMap(upload -> Mono.from(blobStore.delete(UPLOAD_BUCKET, upload.getBlobId())) .then(uploadDAO.delete(upload.getUser(), upload.getId())), DEFAULT_CONCURRENCY) .then(); diff --git a/server/data/data-jmap-cassandra/src/main/java/org/apache/james/jmap/cassandra/upload/CassandraUploadUsageRepository.java b/server/data/data-jmap-cassandra/src/main/java/org/apache/james/jmap/cassandra/upload/CassandraUploadUsageRepository.java index 6978cfc8971..513100b9cfc 100644 --- a/server/data/data-jmap-cassandra/src/main/java/org/apache/james/jmap/cassandra/upload/CassandraUploadUsageRepository.java +++ b/server/data/data-jmap-cassandra/src/main/java/org/apache/james/jmap/cassandra/upload/CassandraUploadUsageRepository.java @@ -24,6 +24,7 @@ import org.apache.james.backends.cassandra.components.CassandraQuotaCurrentValueDao; import org.apache.james.core.Username; import org.apache.james.core.quota.QuotaComponent; +import org.apache.james.core.quota.QuotaCurrentValue; import org.apache.james.core.quota.QuotaSizeUsage; import org.apache.james.core.quota.QuotaType; import org.apache.james.jmap.api.upload.UploadUsageRepository; @@ -43,19 +44,19 @@ public CassandraUploadUsageRepository(CassandraQuotaCurrentValueDao cassandraQuo @Override public Mono increaseSpace(Username username, QuotaSizeUsage usage) { - return cassandraQuotaCurrentValueDao.increase(CassandraQuotaCurrentValueDao.QuotaKey.of(QuotaComponent.JMAP_UPLOADS, username.asString(), QuotaType.SIZE), + return cassandraQuotaCurrentValueDao.increase(QuotaCurrentValue.Key.of(QuotaComponent.JMAP_UPLOADS, username.asString(), QuotaType.SIZE), usage.asLong()); } @Override public Mono decreaseSpace(Username username, QuotaSizeUsage usage) { - return cassandraQuotaCurrentValueDao.decrease(CassandraQuotaCurrentValueDao.QuotaKey.of(QuotaComponent.JMAP_UPLOADS, username.asString(), QuotaType.SIZE), + return cassandraQuotaCurrentValueDao.decrease(QuotaCurrentValue.Key.of(QuotaComponent.JMAP_UPLOADS, username.asString(), QuotaType.SIZE), usage.asLong()); } @Override public Mono getSpaceUsage(Username username) { - return cassandraQuotaCurrentValueDao.getQuotaCurrentValue(CassandraQuotaCurrentValueDao.QuotaKey.of(QuotaComponent.JMAP_UPLOADS, username.asString(), QuotaType.SIZE)) + return cassandraQuotaCurrentValueDao.getQuotaCurrentValue(QuotaCurrentValue.Key.of(QuotaComponent.JMAP_UPLOADS, username.asString(), QuotaType.SIZE)) .map(quotaCurrentValue -> QuotaSizeUsage.size(quotaCurrentValue.getCurrentValue())).defaultIfEmpty(DEFAULT_QUOTA_SIZE_USAGE); } diff --git a/server/data/data-jmap-cassandra/src/test/java/org/apache/james/jmap/cassandra/upload/CassandraUploadRepositoryTest.java b/server/data/data-jmap-cassandra/src/test/java/org/apache/james/jmap/cassandra/upload/CassandraUploadRepositoryTest.java index 22c47139711..d598677c3e6 100644 --- a/server/data/data-jmap-cassandra/src/test/java/org/apache/james/jmap/cassandra/upload/CassandraUploadRepositoryTest.java +++ b/server/data/data-jmap-cassandra/src/test/java/org/apache/james/jmap/cassandra/upload/CassandraUploadRepositoryTest.java @@ -29,6 +29,7 @@ import org.apache.james.jmap.api.upload.UploadRepository; import org.apache.james.jmap.api.upload.UploadRepositoryContract; import org.apache.james.server.blob.deduplication.DeDuplicationBlobStore; +import org.apache.james.utils.UpdatableTickingClock; import org.junit.jupiter.api.BeforeEach; import org.junit.jupiter.api.Disabled; import org.junit.jupiter.api.extension.RegisterExtension; @@ -39,10 +40,11 @@ class CassandraUploadRepositoryTest implements UploadRepositoryContract { @RegisterExtension static CassandraClusterExtension cassandra = new CassandraClusterExtension(UploadModule.MODULE); private CassandraUploadRepository testee; + private UpdatableTickingClock clock; @BeforeEach void setUp() { - Clock clock = Clock.systemUTC(); + clock = new UpdatableTickingClock(Clock.systemUTC().instant()); testee = new CassandraUploadRepository(new UploadDAO(cassandra.getCassandraCluster().getConf(), new PlainBlobId.Factory()), new DeDuplicationBlobStore(new MemoryBlobStoreDAO(), BucketName.of("default"), new PlainBlobId.Factory()), clock); @@ -70,4 +72,9 @@ public void deleteShouldReturnTrueWhenRowExists() { public void deleteShouldReturnFalseWhenRowDoesNotExist() { UploadRepositoryContract.super.deleteShouldReturnFalseWhenRowDoesNotExist(); } + + @Override + public UpdatableTickingClock clock() { + return clock; + } } \ No newline at end of file diff --git a/server/data/data-jmap-postgres/pom.xml b/server/data/data-jmap-postgres/pom.xml new file mode 100644 index 00000000000..ffb09f7ff0a --- /dev/null +++ b/server/data/data-jmap-postgres/pom.xml @@ -0,0 +1,156 @@ + + + + + 4.0.0 + + + org.apache.james + james-server + 3.9.0-SNAPSHOT + ../../pom.xml + + + james-server-data-jmap-postgres + jar + + Apache James :: Server :: Data :: JMAP :: PostgreSQL persistence + + + 5.3.7 + + + + + ${james.groupId} + apache-james-backends-postgres + + + ${james.groupId} + apache-james-backends-postgres + test-jar + test + + + ${james.groupId} + apache-james-mailbox-api + test-jar + test + + + ${james.groupId} + apache-james-mailbox-postgres + + + ${james.groupId} + blob-memory + test + + + ${james.groupId} + blob-storage-strategy + test + + + ${james.groupId} + event-sourcing-event-store-postgres + ${project.version} + + + ${james.groupId} + james-json + test-jar + test + + + ${james.groupId} + james-server-data-jmap + + + ${james.groupId} + james-server-data-jmap + test-jar + test + + + ${james.groupId} + james-server-guice-common + test-jar + test + + + ${james.groupId} + james-server-testing + test + + + ${james.groupId} + metrics-tests + test + + + ${james.groupId} + testing-base + test + + + com.github.f4b6a3 + uuid-creator + ${uuid-creator.version} + + + com.google.guava + guava + + + net.javacrumbs.json-unit + json-unit-assertj + test + + + org.awaitility + awaitility + test + + + org.testcontainers + postgresql + test + + + + + + + net.alchim31.maven + scala-maven-plugin + + + org.apache.maven.plugins + maven-surefire-plugin + + true + 2 + + + + + + diff --git a/server/data/data-jmap-postgres/src/main/java/org/apache/james/jmap/postgres/PostgresDataJMapAggregateModule.java b/server/data/data-jmap-postgres/src/main/java/org/apache/james/jmap/postgres/PostgresDataJMapAggregateModule.java new file mode 100644 index 00000000000..6943fbd9f9a --- /dev/null +++ b/server/data/data-jmap-postgres/src/main/java/org/apache/james/jmap/postgres/PostgresDataJMapAggregateModule.java @@ -0,0 +1,42 @@ +/**************************************************************** + * 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.jmap.postgres; + +import org.apache.james.backends.postgres.PostgresModule; +import org.apache.james.jmap.postgres.change.PostgresEmailChangeModule; +import org.apache.james.jmap.postgres.change.PostgresMailboxChangeModule; +import org.apache.james.jmap.postgres.filtering.PostgresFilteringProjectionModule; +import org.apache.james.jmap.postgres.identity.PostgresCustomIdentityModule; +import org.apache.james.jmap.postgres.projections.PostgresEmailQueryViewModule; +import org.apache.james.jmap.postgres.projections.PostgresMessageFastViewProjectionModule; +import org.apache.james.jmap.postgres.pushsubscription.PostgresPushSubscriptionModule; +import org.apache.james.jmap.postgres.upload.PostgresUploadModule; + +public interface PostgresDataJMapAggregateModule { + PostgresModule MODULE = PostgresModule.aggregateModules( + PostgresUploadModule.MODULE, + PostgresMessageFastViewProjectionModule.MODULE, + PostgresEmailChangeModule.MODULE, + PostgresMailboxChangeModule.MODULE, + PostgresPushSubscriptionModule.MODULE, + PostgresFilteringProjectionModule.MODULE, + PostgresCustomIdentityModule.MODULE, + PostgresEmailQueryViewModule.MODULE); +} diff --git a/server/data/data-jmap-postgres/src/main/java/org/apache/james/jmap/postgres/change/PostgresEmailChangeDAO.java b/server/data/data-jmap-postgres/src/main/java/org/apache/james/jmap/postgres/change/PostgresEmailChangeDAO.java new file mode 100644 index 00000000000..7e651562f99 --- /dev/null +++ b/server/data/data-jmap-postgres/src/main/java/org/apache/james/jmap/postgres/change/PostgresEmailChangeDAO.java @@ -0,0 +1,118 @@ +/**************************************************************** + * 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.jmap.postgres.change; + +import static org.apache.james.jmap.postgres.change.PostgresEmailChangeModule.PostgresEmailChangeTable.ACCOUNT_ID; +import static org.apache.james.jmap.postgres.change.PostgresEmailChangeModule.PostgresEmailChangeTable.CREATED; +import static org.apache.james.jmap.postgres.change.PostgresEmailChangeModule.PostgresEmailChangeTable.DATE; +import static org.apache.james.jmap.postgres.change.PostgresEmailChangeModule.PostgresEmailChangeTable.DESTROYED; +import static org.apache.james.jmap.postgres.change.PostgresEmailChangeModule.PostgresEmailChangeTable.IS_SHARED; +import static org.apache.james.jmap.postgres.change.PostgresEmailChangeModule.PostgresEmailChangeTable.STATE; +import static org.apache.james.jmap.postgres.change.PostgresEmailChangeModule.PostgresEmailChangeTable.TABLE_NAME; +import static org.apache.james.jmap.postgres.change.PostgresEmailChangeModule.PostgresEmailChangeTable.UPDATED; + +import java.util.Arrays; +import java.util.List; +import java.util.UUID; + +import org.apache.james.backends.postgres.utils.PostgresExecutor; +import org.apache.james.jmap.api.change.EmailChange; +import org.apache.james.jmap.api.change.State; +import org.apache.james.jmap.api.model.AccountId; +import org.apache.james.mailbox.model.MessageId; +import org.apache.james.mailbox.postgres.PostgresMessageId; +import org.jooq.Record; + +import com.google.common.collect.ImmutableList; + +import reactor.core.publisher.Flux; +import reactor.core.publisher.Mono; + +public class PostgresEmailChangeDAO { + private final PostgresExecutor postgresExecutor; + + public PostgresEmailChangeDAO(PostgresExecutor postgresExecutor) { + this.postgresExecutor = postgresExecutor; + } + + public Mono insert(EmailChange change) { + return postgresExecutor.executeVoid(dslContext -> Mono.from(dslContext.insertInto(TABLE_NAME) + .set(ACCOUNT_ID, change.getAccountId().getIdentifier()) + .set(STATE, change.getState().getValue()) + .set(IS_SHARED, change.isShared()) + .set(CREATED, convertToUUIDArray(change.getCreated())) + .set(UPDATED, convertToUUIDArray(change.getUpdated())) + .set(DESTROYED, convertToUUIDArray(change.getDestroyed())) + .set(DATE, change.getDate().toOffsetDateTime()))); + } + + public Flux getAllChanges(AccountId accountId) { + return postgresExecutor.executeRows(dslContext -> Flux.from(dslContext.selectFrom(TABLE_NAME) + .where(ACCOUNT_ID.eq(accountId.getIdentifier())))) + .map(record -> readRecord(record, accountId)); + } + + public Flux getChangesSince(AccountId accountId, State state) { + return postgresExecutor.executeRows(dslContext -> Flux.from(dslContext.selectFrom(TABLE_NAME) + .where(ACCOUNT_ID.eq(accountId.getIdentifier())) + .and(STATE.greaterOrEqual(state.getValue())) + .orderBy(STATE))) + .map(record -> readRecord(record, accountId)); + } + + public Mono latestState(AccountId accountId) { + return postgresExecutor.executeRow(dslContext -> Mono.from(dslContext.select(STATE) + .from(TABLE_NAME) + .where(ACCOUNT_ID.eq(accountId.getIdentifier())) + .orderBy(STATE.desc()) + .limit(1))) + .map(record -> State.of(record.get(STATE))); + } + + public Mono latestStateNotDelegated(AccountId accountId) { + return postgresExecutor.executeRow(dslContext -> Mono.from(dslContext.select(STATE) + .from(TABLE_NAME) + .where(ACCOUNT_ID.eq(accountId.getIdentifier())) + .and(IS_SHARED.eq(false)) + .orderBy(STATE.desc()) + .limit(1))) + .map(record -> State.of(record.get(STATE))); + } + + private UUID[] convertToUUIDArray(List messageIds) { + return messageIds.stream().map(PostgresMessageId.class::cast).map(PostgresMessageId::asUuid).toArray(UUID[]::new); + } + + private EmailChange readRecord(Record record, AccountId accountId) { + return EmailChange.builder() + .accountId(accountId) + .state(State.of(record.get(STATE))) + .date(record.get(DATE).toZonedDateTime()) + .isShared(record.get(IS_SHARED)) + .created(convertToMessageIdList(record.get(CREATED))) + .updated(convertToMessageIdList(record.get(UPDATED))) + .destroyed(convertToMessageIdList(record.get(DESTROYED))) + .build(); + } + + private List convertToMessageIdList(UUID[] uuids) { + return Arrays.stream(uuids).map(PostgresMessageId.Factory::of).collect(ImmutableList.toImmutableList()); + } +} diff --git a/server/data/data-jmap-postgres/src/main/java/org/apache/james/jmap/postgres/change/PostgresEmailChangeModule.java b/server/data/data-jmap-postgres/src/main/java/org/apache/james/jmap/postgres/change/PostgresEmailChangeModule.java new file mode 100644 index 00000000000..442078212ac --- /dev/null +++ b/server/data/data-jmap-postgres/src/main/java/org/apache/james/jmap/postgres/change/PostgresEmailChangeModule.java @@ -0,0 +1,71 @@ +/**************************************************************** + * 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.jmap.postgres.change; + +import static org.apache.james.jmap.postgres.change.PostgresEmailChangeModule.PostgresEmailChangeTable.INDEX; +import static org.apache.james.jmap.postgres.change.PostgresEmailChangeModule.PostgresEmailChangeTable.TABLE; + +import java.time.OffsetDateTime; +import java.util.UUID; + +import org.apache.james.backends.postgres.PostgresIndex; +import org.apache.james.backends.postgres.PostgresModule; +import org.apache.james.backends.postgres.PostgresTable; +import org.jooq.Field; +import org.jooq.Record; +import org.jooq.Table; +import org.jooq.impl.DSL; +import org.jooq.impl.SQLDataType; + +public interface PostgresEmailChangeModule { + interface PostgresEmailChangeTable { + Table TABLE_NAME = DSL.table("email_change"); + + Field ACCOUNT_ID = DSL.field("account_id", SQLDataType.VARCHAR.notNull()); + Field STATE = DSL.field("state", SQLDataType.UUID.notNull()); + Field DATE = DSL.field("date", SQLDataType.TIMESTAMPWITHTIMEZONE.notNull()); + Field IS_SHARED = DSL.field("is_shared", SQLDataType.BOOLEAN.notNull()); + Field CREATED = DSL.field("created", SQLDataType.UUID.getArrayDataType().notNull()); + Field UPDATED = DSL.field("updated", SQLDataType.UUID.getArrayDataType().notNull()); + Field DESTROYED = DSL.field("destroyed", SQLDataType.UUID.getArrayDataType().notNull()); + + PostgresTable TABLE = PostgresTable.name(TABLE_NAME.getName()) + .createTableStep(((dsl, tableName) -> dsl.createTableIfNotExists(tableName) + .column(ACCOUNT_ID) + .column(STATE) + .column(DATE) + .column(IS_SHARED) + .column(CREATED) + .column(UPDATED) + .column(DESTROYED) + .constraint(DSL.primaryKey(ACCOUNT_ID, STATE, IS_SHARED)))) + .supportsRowLevelSecurity() + .build(); + + PostgresIndex INDEX = PostgresIndex.name("idx_email_change_date") + .createIndexStep((dslContext, indexName) -> dslContext.createIndexIfNotExists(indexName) + .on(TABLE_NAME, DATE)); + } + + PostgresModule MODULE = PostgresModule.builder() + .addTable(TABLE) + .addIndex(INDEX) + .build(); +} diff --git a/server/data/data-jmap-postgres/src/main/java/org/apache/james/jmap/postgres/change/PostgresEmailChangeRepository.java b/server/data/data-jmap-postgres/src/main/java/org/apache/james/jmap/postgres/change/PostgresEmailChangeRepository.java new file mode 100644 index 00000000000..94afb643b30 --- /dev/null +++ b/server/data/data-jmap-postgres/src/main/java/org/apache/james/jmap/postgres/change/PostgresEmailChangeRepository.java @@ -0,0 +1,115 @@ +/**************************************************************** + * 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.jmap.postgres.change; + +import java.util.Optional; + +import jakarta.inject.Inject; +import jakarta.inject.Named; + +import org.apache.james.backends.postgres.utils.PostgresExecutor; +import org.apache.james.core.Username; +import org.apache.james.jmap.api.change.EmailChange; +import org.apache.james.jmap.api.change.EmailChangeRepository; +import org.apache.james.jmap.api.change.EmailChanges; +import org.apache.james.jmap.api.change.Limit; +import org.apache.james.jmap.api.change.State; +import org.apache.james.jmap.api.exception.ChangeNotFoundException; +import org.apache.james.jmap.api.model.AccountId; + +import com.google.common.base.Preconditions; + +import reactor.core.publisher.Flux; +import reactor.core.publisher.Mono; + +public class PostgresEmailChangeRepository implements EmailChangeRepository { + public static final String LIMIT_NAME = "emailChangeDefaultLimit"; + + private final PostgresExecutor.Factory executorFactory; + private final Limit defaultLimit; + + @Inject + public PostgresEmailChangeRepository(PostgresExecutor.Factory executorFactory, @Named(LIMIT_NAME) Limit defaultLimit) { + this.executorFactory = executorFactory; + this.defaultLimit = defaultLimit; + } + + @Override + public Mono save(EmailChange change) { + PostgresEmailChangeDAO emailChangeDAO = createPostgresEmailChangeDAO(change.getAccountId()); + return emailChangeDAO.insert(change); + } + + @Override + public Mono getSinceState(AccountId accountId, State state, Optional maxChanges) { + Preconditions.checkNotNull(accountId); + Preconditions.checkNotNull(state); + maxChanges.ifPresent(limit -> Preconditions.checkArgument(limit.getValue() > 0, "maxChanges must be a positive integer")); + + PostgresEmailChangeDAO emailChangeDAO = createPostgresEmailChangeDAO(accountId); + if (state.equals(State.INITIAL)) { + return emailChangeDAO.getAllChanges(accountId) + .filter(change -> !change.isShared()) + .collect(new EmailChanges.Builder.EmailChangeCollector(state, maxChanges.orElse(defaultLimit))); + } + + return emailChangeDAO.getChangesSince(accountId, state) + .switchIfEmpty(Flux.error(() -> new ChangeNotFoundException(state, String.format("State '%s' could not be found", state.getValue())))) + .filter(change -> !change.isShared()) + .filter(change -> !change.getState().equals(state)) + .collect(new EmailChanges.Builder.EmailChangeCollector(state, maxChanges.orElse(defaultLimit))); + } + + @Override + public Mono getSinceStateWithDelegation(AccountId accountId, State state, Optional maxChanges) { + Preconditions.checkNotNull(accountId); + Preconditions.checkNotNull(state); + maxChanges.ifPresent(limit -> Preconditions.checkArgument(limit.getValue() > 0, "maxChanges must be a positive integer")); + + PostgresEmailChangeDAO emailChangeDAO = createPostgresEmailChangeDAO(accountId); + if (state.equals(State.INITIAL)) { + return emailChangeDAO.getAllChanges(accountId) + .collect(new EmailChanges.Builder.EmailChangeCollector(state, maxChanges.orElse(defaultLimit))); + } + + return emailChangeDAO.getChangesSince(accountId, state) + .switchIfEmpty(Flux.error(() -> new ChangeNotFoundException(state, String.format("State '%s' could not be found", state.getValue())))) + .filter(change -> !change.getState().equals(state)) + .collect(new EmailChanges.Builder.EmailChangeCollector(state, maxChanges.orElse(defaultLimit))); + } + + @Override + public Mono getLatestState(AccountId accountId) { + PostgresEmailChangeDAO emailChangeDAO = createPostgresEmailChangeDAO(accountId); + return emailChangeDAO.latestStateNotDelegated(accountId) + .switchIfEmpty(Mono.just(State.INITIAL)); + } + + @Override + public Mono getLatestStateWithDelegation(AccountId accountId) { + PostgresEmailChangeDAO emailChangeDAO = createPostgresEmailChangeDAO(accountId); + return emailChangeDAO.latestState(accountId) + .switchIfEmpty(Mono.just(State.INITIAL)); + } + + private PostgresEmailChangeDAO createPostgresEmailChangeDAO(AccountId accountId) { + return new PostgresEmailChangeDAO(executorFactory.create(Username.of(accountId.getIdentifier()).getDomainPart())); + } +} diff --git a/server/data/data-jmap-postgres/src/main/java/org/apache/james/jmap/postgres/change/PostgresMailboxChangeDAO.java b/server/data/data-jmap-postgres/src/main/java/org/apache/james/jmap/postgres/change/PostgresMailboxChangeDAO.java new file mode 100644 index 00000000000..5a183fd2ba6 --- /dev/null +++ b/server/data/data-jmap-postgres/src/main/java/org/apache/james/jmap/postgres/change/PostgresMailboxChangeDAO.java @@ -0,0 +1,126 @@ +/**************************************************************** + * 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.jmap.postgres.change; + +import static org.apache.james.jmap.postgres.change.PostgresMailboxChangeModule.PostgresMailboxChangeTable.ACCOUNT_ID; +import static org.apache.james.jmap.postgres.change.PostgresMailboxChangeModule.PostgresMailboxChangeTable.CREATED; +import static org.apache.james.jmap.postgres.change.PostgresMailboxChangeModule.PostgresMailboxChangeTable.DATE; +import static org.apache.james.jmap.postgres.change.PostgresMailboxChangeModule.PostgresMailboxChangeTable.DESTROYED; +import static org.apache.james.jmap.postgres.change.PostgresMailboxChangeModule.PostgresMailboxChangeTable.IS_COUNT_CHANGE; +import static org.apache.james.jmap.postgres.change.PostgresMailboxChangeModule.PostgresMailboxChangeTable.IS_SHARED; +import static org.apache.james.jmap.postgres.change.PostgresMailboxChangeModule.PostgresMailboxChangeTable.STATE; +import static org.apache.james.jmap.postgres.change.PostgresMailboxChangeModule.PostgresMailboxChangeTable.TABLE_NAME; +import static org.apache.james.jmap.postgres.change.PostgresMailboxChangeModule.PostgresMailboxChangeTable.UPDATED; + +import java.util.Arrays; +import java.util.List; +import java.util.UUID; + +import org.apache.james.backends.postgres.utils.PostgresExecutor; +import org.apache.james.jmap.api.change.MailboxChange; +import org.apache.james.jmap.api.change.State; +import org.apache.james.jmap.api.model.AccountId; +import org.apache.james.mailbox.model.MailboxId; +import org.apache.james.mailbox.postgres.PostgresMailboxId; +import org.jooq.Record; + +import com.google.common.collect.ImmutableList; + +import reactor.core.publisher.Flux; +import reactor.core.publisher.Mono; + +public class PostgresMailboxChangeDAO { + private final PostgresExecutor postgresExecutor; + + public PostgresMailboxChangeDAO(PostgresExecutor postgresExecutor) { + this.postgresExecutor = postgresExecutor; + } + + public Mono insert(MailboxChange change) { + return postgresExecutor.executeVoid(dslContext -> Mono.from(dslContext.insertInto(TABLE_NAME) + .set(ACCOUNT_ID, change.getAccountId().getIdentifier()) + .set(STATE, change.getState().getValue()) + .set(IS_SHARED, change.isShared()) + .set(IS_COUNT_CHANGE, change.isCountChange()) + .set(CREATED, toUUIDArray(change.getCreated())) + .set(UPDATED, toUUIDArray(change.getUpdated())) + .set(DESTROYED, toUUIDArray(change.getDestroyed())) + .set(DATE, change.getDate().toOffsetDateTime()))); + } + + public Flux getAllChanges(AccountId accountId) { + return postgresExecutor.executeRows(dslContext -> Flux.from(dslContext.selectFrom(TABLE_NAME) + .where(ACCOUNT_ID.eq(accountId.getIdentifier())))) + .map(record -> readRecord(record, accountId)); + } + + public Flux getChangesSince(AccountId accountId, State state) { + return postgresExecutor.executeRows(dslContext -> Flux.from(dslContext.selectFrom(TABLE_NAME) + .where(ACCOUNT_ID.eq(accountId.getIdentifier())) + .and(STATE.greaterOrEqual(state.getValue())) + .orderBy(STATE))) + .map(record -> readRecord(record, accountId)); + } + + public Mono latestState(AccountId accountId) { + return postgresExecutor.executeRow(dslContext -> Mono.from(dslContext.select(STATE) + .from(TABLE_NAME) + .where(ACCOUNT_ID.eq(accountId.getIdentifier())) + .orderBy(STATE.desc()) + .limit(1))) + .map(record -> State.of(record.get(STATE))); + } + + public Mono latestStateNotDelegated(AccountId accountId) { + return postgresExecutor.executeRow(dslContext -> Mono.from(dslContext.select(STATE) + .from(TABLE_NAME) + .where(ACCOUNT_ID.eq(accountId.getIdentifier())) + .and(IS_SHARED.eq(false)) + .orderBy(STATE.desc()) + .limit(1))) + .map(record -> State.of(record.get(STATE))); + } + + private UUID[] toUUIDArray(List mailboxIds) { + return mailboxIds.stream() + .map(PostgresMailboxId.class::cast) + .map(PostgresMailboxId::asUuid) + .toArray(UUID[]::new); + } + + private MailboxChange readRecord(Record record, AccountId accountId) { + return MailboxChange.builder() + .accountId(accountId) + .state(State.of(record.get(STATE))) + .date(record.get(DATE).toZonedDateTime()) + .isCountChange(record.get(IS_COUNT_CHANGE)) + .shared(record.get(IS_SHARED)) + .created(toMailboxIds(record.get(CREATED))) + .updated(toMailboxIds(record.get(UPDATED))) + .destroyed(toMailboxIds(record.get(DESTROYED))) + .build(); + } + + private List toMailboxIds(UUID[] uuids) { + return Arrays.stream(uuids) + .map(PostgresMailboxId::of) + .collect(ImmutableList.toImmutableList()); + } +} diff --git a/server/data/data-jmap-postgres/src/main/java/org/apache/james/jmap/postgres/change/PostgresMailboxChangeModule.java b/server/data/data-jmap-postgres/src/main/java/org/apache/james/jmap/postgres/change/PostgresMailboxChangeModule.java new file mode 100644 index 00000000000..bf6851e97b8 --- /dev/null +++ b/server/data/data-jmap-postgres/src/main/java/org/apache/james/jmap/postgres/change/PostgresMailboxChangeModule.java @@ -0,0 +1,73 @@ +/**************************************************************** + * 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.jmap.postgres.change; + +import static org.apache.james.jmap.postgres.change.PostgresMailboxChangeModule.PostgresMailboxChangeTable.INDEX; +import static org.apache.james.jmap.postgres.change.PostgresMailboxChangeModule.PostgresMailboxChangeTable.TABLE; + +import java.time.OffsetDateTime; +import java.util.UUID; + +import org.apache.james.backends.postgres.PostgresIndex; +import org.apache.james.backends.postgres.PostgresModule; +import org.apache.james.backends.postgres.PostgresTable; +import org.jooq.Field; +import org.jooq.Record; +import org.jooq.Table; +import org.jooq.impl.DSL; +import org.jooq.impl.SQLDataType; + +public interface PostgresMailboxChangeModule { + interface PostgresMailboxChangeTable { + Table TABLE_NAME = DSL.table("mailbox_change"); + + Field ACCOUNT_ID = DSL.field("account_id", SQLDataType.VARCHAR.notNull()); + Field STATE = DSL.field("state", SQLDataType.UUID.notNull()); + Field DATE = DSL.field("date", SQLDataType.TIMESTAMPWITHTIMEZONE.notNull()); + Field IS_SHARED = DSL.field("is_shared", SQLDataType.BOOLEAN.notNull()); + Field IS_COUNT_CHANGE = DSL.field("is_count_change", SQLDataType.BOOLEAN.notNull()); + Field CREATED = DSL.field("created", SQLDataType.UUID.getArrayDataType().notNull()); + Field UPDATED = DSL.field("updated", SQLDataType.UUID.getArrayDataType().notNull()); + Field DESTROYED = DSL.field("destroyed", SQLDataType.UUID.getArrayDataType().notNull()); + + PostgresTable TABLE = PostgresTable.name(TABLE_NAME.getName()) + .createTableStep(((dsl, tableName) -> dsl.createTableIfNotExists(tableName) + .column(ACCOUNT_ID) + .column(STATE) + .column(DATE) + .column(IS_SHARED) + .column(IS_COUNT_CHANGE) + .column(CREATED) + .column(UPDATED) + .column(DESTROYED) + .constraint(DSL.primaryKey(ACCOUNT_ID, STATE, IS_SHARED)))) + .supportsRowLevelSecurity() + .build(); + + PostgresIndex INDEX = PostgresIndex.name("index_mailbox_change_date") + .createIndexStep((dslContext, indexName) -> dslContext.createIndexIfNotExists(indexName) + .on(TABLE_NAME, DATE)); + } + + PostgresModule MODULE = PostgresModule.builder() + .addTable(TABLE) + .addIndex(INDEX) + .build(); +} diff --git a/server/data/data-jmap-postgres/src/main/java/org/apache/james/jmap/postgres/change/PostgresMailboxChangeRepository.java b/server/data/data-jmap-postgres/src/main/java/org/apache/james/jmap/postgres/change/PostgresMailboxChangeRepository.java new file mode 100644 index 00000000000..84d586ea9cd --- /dev/null +++ b/server/data/data-jmap-postgres/src/main/java/org/apache/james/jmap/postgres/change/PostgresMailboxChangeRepository.java @@ -0,0 +1,115 @@ +/**************************************************************** + * 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.jmap.postgres.change; + +import java.util.Optional; + +import jakarta.inject.Inject; +import jakarta.inject.Named; + +import org.apache.james.backends.postgres.utils.PostgresExecutor; +import org.apache.james.core.Username; +import org.apache.james.jmap.api.change.Limit; +import org.apache.james.jmap.api.change.MailboxChange; +import org.apache.james.jmap.api.change.MailboxChangeRepository; +import org.apache.james.jmap.api.change.MailboxChanges; +import org.apache.james.jmap.api.change.State; +import org.apache.james.jmap.api.exception.ChangeNotFoundException; +import org.apache.james.jmap.api.model.AccountId; + +import com.google.common.base.Preconditions; + +import reactor.core.publisher.Flux; +import reactor.core.publisher.Mono; + +public class PostgresMailboxChangeRepository implements MailboxChangeRepository { + public static final String LIMIT_NAME = "mailboxChangeDefaultLimit"; + + private final PostgresExecutor.Factory executorFactory; + private final Limit defaultLimit; + + @Inject + public PostgresMailboxChangeRepository(PostgresExecutor.Factory executorFactory, @Named(LIMIT_NAME) Limit defaultLimit) { + this.executorFactory = executorFactory; + this.defaultLimit = defaultLimit; + } + + @Override + public Mono save(MailboxChange change) { + PostgresMailboxChangeDAO mailboxChangeDAO = createPostgresMailboxChangeDAO(change.getAccountId()); + return mailboxChangeDAO.insert(change); + } + + @Override + public Mono getSinceState(AccountId accountId, State state, Optional maxChanges) { + Preconditions.checkNotNull(accountId); + Preconditions.checkNotNull(state); + maxChanges.ifPresent(limit -> Preconditions.checkArgument(limit.getValue() > 0, "maxChanges must be a positive integer")); + + PostgresMailboxChangeDAO mailboxChangeDAO = createPostgresMailboxChangeDAO(accountId); + if (state.equals(State.INITIAL)) { + return mailboxChangeDAO.getAllChanges(accountId) + .filter(change -> !change.isShared()) + .collect(new MailboxChanges.MailboxChangesBuilder.MailboxChangeCollector(state, maxChanges.orElse(defaultLimit))); + } + + return mailboxChangeDAO.getChangesSince(accountId, state) + .switchIfEmpty(Flux.error(() -> new ChangeNotFoundException(state, String.format("State '%s' could not be found", state.getValue())))) + .filter(change -> !change.isShared()) + .filter(change -> !change.getState().equals(state)) + .collect(new MailboxChanges.MailboxChangesBuilder.MailboxChangeCollector(state, maxChanges.orElse(defaultLimit))); + } + + @Override + public Mono getSinceStateWithDelegation(AccountId accountId, State state, Optional maxChanges) { + Preconditions.checkNotNull(accountId); + Preconditions.checkNotNull(state); + maxChanges.ifPresent(limit -> Preconditions.checkArgument(limit.getValue() > 0, "maxChanges must be a positive integer")); + + PostgresMailboxChangeDAO mailboxChangeDAO = createPostgresMailboxChangeDAO(accountId); + if (state.equals(State.INITIAL)) { + return mailboxChangeDAO.getAllChanges(accountId) + .collect(new MailboxChanges.MailboxChangesBuilder.MailboxChangeCollector(state, maxChanges.orElse(defaultLimit))); + } + + return mailboxChangeDAO.getChangesSince(accountId, state) + .switchIfEmpty(Flux.error(() -> new ChangeNotFoundException(state, String.format("State '%s' could not be found", state.getValue())))) + .filter(change -> !change.getState().equals(state)) + .collect(new MailboxChanges.MailboxChangesBuilder.MailboxChangeCollector(state, maxChanges.orElse(defaultLimit))); + } + + @Override + public Mono getLatestState(AccountId accountId) { + PostgresMailboxChangeDAO mailboxChangeDAO = createPostgresMailboxChangeDAO(accountId); + return mailboxChangeDAO.latestStateNotDelegated(accountId) + .switchIfEmpty(Mono.just(State.INITIAL)); + } + + @Override + public Mono getLatestStateWithDelegation(AccountId accountId) { + PostgresMailboxChangeDAO mailboxChangeDAO = createPostgresMailboxChangeDAO(accountId); + return mailboxChangeDAO.latestState(accountId) + .switchIfEmpty(Mono.just(State.INITIAL)); + } + + private PostgresMailboxChangeDAO createPostgresMailboxChangeDAO(AccountId accountId) { + return new PostgresMailboxChangeDAO(executorFactory.create(Username.of(accountId.getIdentifier()).getDomainPart())); + } +} diff --git a/server/data/data-jmap-postgres/src/main/java/org/apache/james/jmap/postgres/change/PostgresStateFactory.java b/server/data/data-jmap-postgres/src/main/java/org/apache/james/jmap/postgres/change/PostgresStateFactory.java new file mode 100644 index 00000000000..e4ab2129b4b --- /dev/null +++ b/server/data/data-jmap-postgres/src/main/java/org/apache/james/jmap/postgres/change/PostgresStateFactory.java @@ -0,0 +1,31 @@ +/**************************************************************** + * 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.jmap.postgres.change; + +import org.apache.james.jmap.api.change.State; + +import com.github.f4b6a3.uuid.UuidCreator; + +public class PostgresStateFactory implements State.Factory { + @Override + public State generate() { + return State.of(UuidCreator.getTimeOrderedEpoch()); + } +} diff --git a/server/data/data-jmap-postgres/src/main/java/org/apache/james/jmap/postgres/filtering/PostgresFilteringProjection.java b/server/data/data-jmap-postgres/src/main/java/org/apache/james/jmap/postgres/filtering/PostgresFilteringProjection.java new file mode 100644 index 00000000000..0628ab78a89 --- /dev/null +++ b/server/data/data-jmap-postgres/src/main/java/org/apache/james/jmap/postgres/filtering/PostgresFilteringProjection.java @@ -0,0 +1,65 @@ +/**************************************************************** + * 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.jmap.postgres.filtering; + +import java.util.Optional; + +import jakarta.inject.Inject; + +import org.apache.james.core.Username; +import org.apache.james.eventsourcing.Event; +import org.apache.james.eventsourcing.EventWithState; +import org.apache.james.eventsourcing.ReactiveSubscriber; +import org.apache.james.jmap.api.filtering.Rules; +import org.apache.james.jmap.api.filtering.Version; +import org.apache.james.jmap.api.filtering.impl.EventSourcingFilteringManagement; +import org.apache.james.jmap.api.filtering.impl.FilteringAggregate; +import org.reactivestreams.Publisher; + +public class PostgresFilteringProjection implements EventSourcingFilteringManagement.ReadProjection, ReactiveSubscriber { + private final PostgresFilteringProjectionDAO postgresFilteringProjectionDAO; + + @Inject + public PostgresFilteringProjection(PostgresFilteringProjectionDAO postgresFilteringProjectionDAO) { + this.postgresFilteringProjectionDAO = postgresFilteringProjectionDAO; + } + + @Override + public Publisher handleReactive(EventWithState eventWithState) { + Event event = eventWithState.event(); + FilteringAggregate.FilterState state = (FilteringAggregate.FilterState) eventWithState.state().get(); + return postgresFilteringProjectionDAO.upsert(event.getAggregateId(), event.eventId(), state.getRules()); + } + + @Override + public Publisher listRulesForUser(Username username) { + return postgresFilteringProjectionDAO.listRulesForUser(username); + } + + @Override + public Publisher getLatestVersion(Username username) { + return postgresFilteringProjectionDAO.getVersion(username); + } + + @Override + public Optional subscriber() { + return Optional.of(this); + } +} diff --git a/server/data/data-jmap-postgres/src/main/java/org/apache/james/jmap/postgres/filtering/PostgresFilteringProjectionDAO.java b/server/data/data-jmap-postgres/src/main/java/org/apache/james/jmap/postgres/filtering/PostgresFilteringProjectionDAO.java new file mode 100644 index 00000000000..adba057b250 --- /dev/null +++ b/server/data/data-jmap-postgres/src/main/java/org/apache/james/jmap/postgres/filtering/PostgresFilteringProjectionDAO.java @@ -0,0 +1,109 @@ +/**************************************************************** + * 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.jmap.postgres.filtering; + +import static org.apache.james.jmap.postgres.filtering.PostgresFilteringProjectionModule.PostgresFilteringProjectionTable.AGGREGATE_ID; +import static org.apache.james.jmap.postgres.filtering.PostgresFilteringProjectionModule.PostgresFilteringProjectionTable.EVENT_ID; +import static org.apache.james.jmap.postgres.filtering.PostgresFilteringProjectionModule.PostgresFilteringProjectionTable.RULES; +import static org.apache.james.jmap.postgres.filtering.PostgresFilteringProjectionModule.PostgresFilteringProjectionTable.TABLE_NAME; + +import java.util.List; + +import jakarta.inject.Inject; + +import org.apache.james.backends.postgres.utils.PostgresExecutor; +import org.apache.james.core.Username; +import org.apache.james.eventsourcing.AggregateId; +import org.apache.james.eventsourcing.EventId; +import org.apache.james.jmap.api.filtering.Rule; +import org.apache.james.jmap.api.filtering.RuleDTO; +import org.apache.james.jmap.api.filtering.Rules; +import org.apache.james.jmap.api.filtering.Version; +import org.apache.james.jmap.api.filtering.impl.FilteringAggregateId; +import org.jooq.JSON; +import org.jooq.Record; +import org.reactivestreams.Publisher; + +import com.fasterxml.jackson.core.JsonProcessingException; +import com.fasterxml.jackson.core.type.TypeReference; +import com.fasterxml.jackson.databind.ObjectMapper; +import com.fasterxml.jackson.datatype.jdk8.Jdk8Module; +import com.google.common.collect.ImmutableList; + +import reactor.core.publisher.Mono; + +public class PostgresFilteringProjectionDAO { + private final PostgresExecutor postgresExecutor; + private final ObjectMapper objectMapper; + + @Inject + public PostgresFilteringProjectionDAO(PostgresExecutor postgresExecutor) { + this.postgresExecutor = postgresExecutor; + objectMapper = new ObjectMapper().registerModule(new Jdk8Module()); + } + + public Publisher listRulesForUser(Username username) { + return postgresExecutor.executeRow(dslContext -> dslContext.selectFrom(TABLE_NAME) + .where(AGGREGATE_ID.eq(new FilteringAggregateId(username).asAggregateKey()))) + .handle((row, sink) -> { + try { + Rules rules = parseRules(row); + sink.next(rules); + } catch (JsonProcessingException e) { + sink.error(e); + } + }); + } + + public Mono upsert(AggregateId aggregateId, EventId eventId, ImmutableList rules) { + return postgresExecutor.executeVoid(dslContext -> Mono.from(dslContext.insertInto(TABLE_NAME) + .set(AGGREGATE_ID, aggregateId.asAggregateKey()) + .set(EVENT_ID, eventId.value()) + .set(RULES, convertToJooqJson(rules)) + .onConflict(AGGREGATE_ID) + .doUpdate() + .set(EVENT_ID, eventId.value()) + .set(RULES, convertToJooqJson(rules)))); + } + + public Publisher getVersion(Username username) { + return postgresExecutor.executeRow(dslContext -> Mono.from(dslContext.select(EVENT_ID) + .from(TABLE_NAME) + .where(AGGREGATE_ID.eq(new FilteringAggregateId(username).asAggregateKey())))) + .map(this::parseVersion); + } + + private Rules parseRules(Record record) throws JsonProcessingException { + List ruleDTOS = objectMapper.readValue(record.get(RULES).data(), new TypeReference<>() {}); + return new Rules(RuleDTO.toRules(ruleDTOS), parseVersion(record)); + } + + private Version parseVersion(Record record) { + return new Version(record.get(EVENT_ID)); + } + + private JSON convertToJooqJson(List rules) { + try { + return JSON.json(objectMapper.writeValueAsString(RuleDTO.from(rules))); + } catch (JsonProcessingException e) { + throw new RuntimeException(e); + } + } +} diff --git a/server/data/data-jmap-postgres/src/main/java/org/apache/james/jmap/postgres/filtering/PostgresFilteringProjectionModule.java b/server/data/data-jmap-postgres/src/main/java/org/apache/james/jmap/postgres/filtering/PostgresFilteringProjectionModule.java new file mode 100644 index 00000000000..d87fb603b9c --- /dev/null +++ b/server/data/data-jmap-postgres/src/main/java/org/apache/james/jmap/postgres/filtering/PostgresFilteringProjectionModule.java @@ -0,0 +1,54 @@ +/**************************************************************** + * 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.jmap.postgres.filtering; + +import static org.apache.james.jmap.postgres.filtering.PostgresFilteringProjectionModule.PostgresFilteringProjectionTable.TABLE; + +import org.apache.james.backends.postgres.PostgresModule; +import org.apache.james.backends.postgres.PostgresTable; +import org.jooq.Field; +import org.jooq.JSON; +import org.jooq.Record; +import org.jooq.Table; +import org.jooq.impl.DSL; +import org.jooq.impl.SQLDataType; + +public interface PostgresFilteringProjectionModule { + interface PostgresFilteringProjectionTable { + Table TABLE_NAME = DSL.table("filters_projection"); + + Field AGGREGATE_ID = DSL.field("aggregate_id", SQLDataType.VARCHAR.notNull()); + Field EVENT_ID = DSL.field("event_id", SQLDataType.INTEGER.notNull()); + Field RULES = DSL.field("rules", SQLDataType.JSON.notNull()); + + PostgresTable TABLE = PostgresTable.name(TABLE_NAME.getName()) + .createTableStep(((dsl, tableName) -> dsl.createTableIfNotExists(tableName) + .column(AGGREGATE_ID) + .column(EVENT_ID) + .column(RULES) + .constraint(DSL.primaryKey(AGGREGATE_ID)))) + .disableRowLevelSecurity() + .build(); + } + + PostgresModule MODULE = PostgresModule.builder() + .addTable(TABLE) + .build(); +} diff --git a/server/data/data-jmap-postgres/src/main/java/org/apache/james/jmap/postgres/identity/PostgresCustomIdentityDAO.java b/server/data/data-jmap-postgres/src/main/java/org/apache/james/jmap/postgres/identity/PostgresCustomIdentityDAO.java new file mode 100644 index 00000000000..490bfcdecba --- /dev/null +++ b/server/data/data-jmap-postgres/src/main/java/org/apache/james/jmap/postgres/identity/PostgresCustomIdentityDAO.java @@ -0,0 +1,231 @@ +/**************************************************************** + * 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.jmap.postgres.identity; + +import static org.apache.james.jmap.postgres.identity.PostgresCustomIdentityModule.PostgresCustomIdentityTable.BCC; +import static org.apache.james.jmap.postgres.identity.PostgresCustomIdentityModule.PostgresCustomIdentityTable.EMAIL; +import static org.apache.james.jmap.postgres.identity.PostgresCustomIdentityModule.PostgresCustomIdentityTable.HTML_SIGNATURE; +import static org.apache.james.jmap.postgres.identity.PostgresCustomIdentityModule.PostgresCustomIdentityTable.ID; +import static org.apache.james.jmap.postgres.identity.PostgresCustomIdentityModule.PostgresCustomIdentityTable.MAY_DELETE; +import static org.apache.james.jmap.postgres.identity.PostgresCustomIdentityModule.PostgresCustomIdentityTable.NAME; +import static org.apache.james.jmap.postgres.identity.PostgresCustomIdentityModule.PostgresCustomIdentityTable.REPLY_TO; +import static org.apache.james.jmap.postgres.identity.PostgresCustomIdentityModule.PostgresCustomIdentityTable.SORT_ORDER; +import static org.apache.james.jmap.postgres.identity.PostgresCustomIdentityModule.PostgresCustomIdentityTable.TABLE_NAME; +import static org.apache.james.jmap.postgres.identity.PostgresCustomIdentityModule.PostgresCustomIdentityTable.TEXT_SIGNATURE; +import static org.apache.james.jmap.postgres.identity.PostgresCustomIdentityModule.PostgresCustomIdentityTable.USERNAME; + +import java.util.List; +import java.util.Optional; + +import jakarta.inject.Inject; + +import org.apache.james.backends.postgres.utils.PostgresExecutor; +import org.apache.james.core.MailAddress; +import org.apache.james.core.Username; +import org.apache.james.jmap.api.identity.CustomIdentityDAO; +import org.apache.james.jmap.api.identity.IdentityCreationRequest; +import org.apache.james.jmap.api.identity.IdentityNotFoundException; +import org.apache.james.jmap.api.identity.IdentityUpdate; +import org.apache.james.jmap.api.model.EmailAddress; +import org.apache.james.jmap.api.model.Identity; +import org.apache.james.jmap.api.model.IdentityId; +import org.jooq.JSON; +import org.jooq.Record; +import org.reactivestreams.Publisher; + +import com.fasterxml.jackson.annotation.JsonCreator; +import com.fasterxml.jackson.annotation.JsonProperty; +import com.fasterxml.jackson.core.JsonProcessingException; +import com.fasterxml.jackson.core.type.TypeReference; +import com.fasterxml.jackson.databind.ObjectMapper; +import com.github.fge.lambdas.Throwing; +import com.google.common.collect.ImmutableList; + +import reactor.core.publisher.Flux; +import reactor.core.publisher.Mono; +import reactor.core.scala.publisher.SMono; +import scala.Option; +import scala.collection.immutable.Set; +import scala.jdk.javaapi.CollectionConverters; +import scala.jdk.javaapi.OptionConverters; +import scala.runtime.BoxedUnit; + +public class PostgresCustomIdentityDAO implements CustomIdentityDAO { + static class Email { + private final String name; + private final String email; + + @JsonCreator + public Email(@JsonProperty("name") String name, + @JsonProperty("email") String email) { + this.name = name; + this.email = email; + } + + public String getName() { + return name; + } + + public String getEmail() { + return email; + } + } + + private final PostgresExecutor.Factory executorFactory; + private final ObjectMapper objectMapper = new ObjectMapper(); + + @Inject + public PostgresCustomIdentityDAO(PostgresExecutor.Factory executorFactory) { + this.executorFactory = executorFactory; + } + + @Override + public Publisher save(Username user, IdentityCreationRequest creationRequest) { + return save(user, IdentityId.generate(), creationRequest); + } + + @Override + public Publisher save(Username user, IdentityId identityId, IdentityCreationRequest creationRequest) { + final Identity identity = creationRequest.asIdentity(identityId); + return upsertReturnMono(user, identity); + } + + @Override + public Publisher list(Username user) { + return executorFactory.create(user.getDomainPart()) + .executeRows(dslContext -> Flux.from(dslContext.selectFrom(TABLE_NAME) + .where(USERNAME.eq(user.asString())))) + .map(Throwing.function(this::readRecord)); + } + + @Override + public SMono findByIdentityId(Username user, IdentityId identityId) { + return SMono.fromPublisher(executorFactory.create(user.getDomainPart()) + .executeRow(dslContext -> Mono.from(dslContext.selectFrom(TABLE_NAME) + .where(USERNAME.eq(user.asString())) + .and(ID.eq(identityId.id())))) + .map(Throwing.function(this::readRecord))); + } + + @Override + public Publisher update(Username user, IdentityId identityId, IdentityUpdate identityUpdate) { + return Mono.from(findByIdentityId(user, identityId)) + .switchIfEmpty(Mono.error(new IdentityNotFoundException(identityId))) + .map(identityUpdate::update) + .flatMap(identity -> upsertReturnMono(user, identity)) + .thenReturn(BoxedUnit.UNIT); + } + + @Override + public SMono upsert(Username user, Identity patch) { + return SMono.fromPublisher(upsertReturnMono(user, patch) + .thenReturn(BoxedUnit.UNIT)); + } + + private Mono upsertReturnMono(Username user, Identity identity) { + return executorFactory.create(user.getDomainPart()) + .executeVoid(dslContext -> Mono.from(dslContext.insertInto(TABLE_NAME) + .set(USERNAME, user.asString()) + .set(ID, identity.id().id()) + .set(NAME, identity.name()) + .set(EMAIL, identity.email().asString()) + .set(TEXT_SIGNATURE, identity.textSignature()) + .set(HTML_SIGNATURE, identity.htmlSignature()) + .set(MAY_DELETE, identity.mayDelete()) + .set(SORT_ORDER, identity.sortOrder()) + .set(REPLY_TO, convertToJooqJson(identity.replyTo())) + .set(BCC, convertToJooqJson(identity.bcc())) + .onConflict(USERNAME, ID) + .doUpdate() + .set(NAME, identity.name()) + .set(EMAIL, identity.email().asString()) + .set(TEXT_SIGNATURE, identity.textSignature()) + .set(HTML_SIGNATURE, identity.htmlSignature()) + .set(MAY_DELETE, identity.mayDelete()) + .set(SORT_ORDER, identity.sortOrder()) + .set(REPLY_TO, convertToJooqJson(identity.replyTo())) + .set(BCC, convertToJooqJson(identity.bcc())))) + .thenReturn(identity); + } + + @Override + public Publisher delete(Username username, Set ids) { + if (ids.isEmpty()) { + return Mono.empty(); + } + return executorFactory.create(username.getDomainPart()) + .executeVoid(dslContext -> Mono.from(dslContext.deleteFrom(TABLE_NAME) + .where(USERNAME.eq(username.asString())) + .and(ID.in(CollectionConverters.asJavaCollection(ids).stream().map(IdentityId::id).collect(ImmutableList.toImmutableList()))))) + .thenReturn(BoxedUnit.UNIT); + } + + @Override + public Publisher delete(Username username) { + return executorFactory.create(username.getDomainPart()) + .executeVoid(dslContext -> Mono.from(dslContext.deleteFrom(TABLE_NAME) + .where(USERNAME.eq(username.asString())))) + .thenReturn(BoxedUnit.UNIT); + } + + private Identity readRecord(Record record) throws Exception { + return new Identity(new IdentityId(record.get(ID)), + record.get(SORT_ORDER), + record.get(NAME), + new MailAddress(record.get(EMAIL)), + convertToScala(record.get(REPLY_TO)), + convertToScala(record.get(BCC)), + record.get(TEXT_SIGNATURE), + record.get(HTML_SIGNATURE), + record.get(MAY_DELETE)); + } + + private Option> convertToScala(JSON json) { + return OptionConverters.toScala(Optional.of(CollectionConverters.asScala(convertToObject(json.data()) + .stream() + .map(Throwing.function(email -> EmailAddress.from(Optional.ofNullable(email.getName()), new MailAddress(email.getEmail())))) + .iterator()) + .toList())); + } + + private JSON convertToJooqJson(Option> maybeEmailAddresses) { + return convertToJooqJson(OptionConverters.toJava(maybeEmailAddresses).map(emailAddresses -> + CollectionConverters.asJavaCollection(emailAddresses).stream() + .map(emailAddress -> new Email(emailAddress.nameAsString(), + emailAddress.email().asString())).collect(ImmutableList.toImmutableList())) + .orElse(ImmutableList.of())); + } + + private JSON convertToJooqJson(List list) { + try { + return JSON.json(objectMapper.writeValueAsString(list)); + } catch (JsonProcessingException e) { + throw new RuntimeException(e); + } + } + + private List convertToObject(String json) { + try { + return objectMapper.readValue(json, new TypeReference<>() {}); + } catch (JsonProcessingException e) { + throw new RuntimeException(e); + } + } +} diff --git a/server/data/data-jmap-postgres/src/main/java/org/apache/james/jmap/postgres/identity/PostgresCustomIdentityModule.java b/server/data/data-jmap-postgres/src/main/java/org/apache/james/jmap/postgres/identity/PostgresCustomIdentityModule.java new file mode 100644 index 00000000000..5bd3b627299 --- /dev/null +++ b/server/data/data-jmap-postgres/src/main/java/org/apache/james/jmap/postgres/identity/PostgresCustomIdentityModule.java @@ -0,0 +1,77 @@ +/**************************************************************** + * 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.jmap.postgres.identity; + +import static org.apache.james.jmap.postgres.identity.PostgresCustomIdentityModule.PostgresCustomIdentityTable.TABLE; +import static org.apache.james.jmap.postgres.identity.PostgresCustomIdentityModule.PostgresCustomIdentityTable.USERNAME_INDEX; + +import java.util.UUID; + +import org.apache.james.backends.postgres.PostgresIndex; +import org.apache.james.backends.postgres.PostgresModule; +import org.apache.james.backends.postgres.PostgresTable; +import org.jooq.Field; +import org.jooq.JSON; +import org.jooq.Record; +import org.jooq.Table; +import org.jooq.impl.DSL; +import org.jooq.impl.SQLDataType; + +public interface PostgresCustomIdentityModule { + interface PostgresCustomIdentityTable { + Table TABLE_NAME = DSL.table("custom_identity"); + + Field USERNAME = DSL.field("username", SQLDataType.VARCHAR(255).notNull()); + Field ID = DSL.field("id", SQLDataType.UUID.notNull()); + Field NAME = DSL.field("name", SQLDataType.VARCHAR(255).notNull()); + Field EMAIL = DSL.field("email", SQLDataType.VARCHAR(255).notNull()); + Field REPLY_TO = DSL.field("reply_to", SQLDataType.JSON.notNull()); + Field BCC = DSL.field("bcc", SQLDataType.JSON.notNull()); + Field TEXT_SIGNATURE = DSL.field("text_signature", SQLDataType.VARCHAR(255).notNull()); + Field HTML_SIGNATURE = DSL.field("html_signature", SQLDataType.VARCHAR(255).notNull()); + Field SORT_ORDER = DSL.field("sort_order", SQLDataType.INTEGER.notNull()); + Field MAY_DELETE = DSL.field("may_delete", SQLDataType.BOOLEAN.notNull()); + + PostgresTable TABLE = PostgresTable.name(TABLE_NAME.getName()) + .createTableStep(((dsl, tableName) -> dsl.createTableIfNotExists(tableName) + .column(USERNAME) + .column(ID) + .column(NAME) + .column(EMAIL) + .column(REPLY_TO) + .column(BCC) + .column(TEXT_SIGNATURE) + .column(HTML_SIGNATURE) + .column(SORT_ORDER) + .column(MAY_DELETE) + .constraint(DSL.primaryKey(USERNAME, ID)))) + .supportsRowLevelSecurity() + .build(); + + PostgresIndex USERNAME_INDEX = PostgresIndex.name("custom_identity_username_index") + .createIndexStep((dsl, indexName) -> dsl.createIndexIfNotExists(indexName) + .on(TABLE_NAME, USERNAME)); + } + + PostgresModule MODULE = PostgresModule.builder() + .addTable(TABLE) + .addIndex(USERNAME_INDEX) + .build(); +} diff --git a/server/data/data-jmap-postgres/src/main/java/org/apache/james/jmap/postgres/projections/PostgresEmailQueryView.java b/server/data/data-jmap-postgres/src/main/java/org/apache/james/jmap/postgres/projections/PostgresEmailQueryView.java new file mode 100644 index 00000000000..0f801feecfe --- /dev/null +++ b/server/data/data-jmap-postgres/src/main/java/org/apache/james/jmap/postgres/projections/PostgresEmailQueryView.java @@ -0,0 +1,88 @@ +/**************************************************************** + * 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.jmap.postgres.projections; + +import java.time.ZonedDateTime; + +import jakarta.inject.Inject; + +import org.apache.james.jmap.api.projections.EmailQueryView; +import org.apache.james.mailbox.model.MailboxId; +import org.apache.james.mailbox.model.MessageId; +import org.apache.james.mailbox.postgres.PostgresMailboxId; +import org.apache.james.mailbox.postgres.PostgresMessageId; +import org.apache.james.util.streams.Limit; + +import reactor.core.publisher.Flux; +import reactor.core.publisher.Mono; + +public class PostgresEmailQueryView implements EmailQueryView { + private PostgresEmailQueryViewDAO emailQueryViewDAO; + + @Inject + public PostgresEmailQueryView(PostgresEmailQueryViewDAO emailQueryViewDAO) { + this.emailQueryViewDAO = emailQueryViewDAO; + } + + @Override + public Flux listMailboxContentSortedBySentAt(MailboxId mailboxId, Limit limit) { + return emailQueryViewDAO.listMailboxContentSortedBySentAt(PostgresMailboxId.class.cast(mailboxId), limit); + } + + @Override + public Flux listMailboxContentSortedByReceivedAt(MailboxId mailboxId, Limit limit) { + return emailQueryViewDAO.listMailboxContentSortedByReceivedAt(PostgresMailboxId.class.cast(mailboxId), limit); + } + + @Override + public Flux listMailboxContentSinceAfterSortedBySentAt(MailboxId mailboxId, ZonedDateTime since, Limit limit) { + return emailQueryViewDAO.listMailboxContentSinceAfterSortedBySentAt(PostgresMailboxId.class.cast(mailboxId), since, limit); + } + + @Override + public Flux listMailboxContentSinceAfterSortedByReceivedAt(MailboxId mailboxId, ZonedDateTime since, Limit limit) { + return emailQueryViewDAO.listMailboxContentSinceAfterSortedByReceivedAt(PostgresMailboxId.class.cast(mailboxId), since, limit); + } + + @Override + public Flux listMailboxContentBeforeSortedByReceivedAt(MailboxId mailboxId, ZonedDateTime since, Limit limit) { + return emailQueryViewDAO.listMailboxContentBeforeSortedByReceivedAt(PostgresMailboxId.class.cast(mailboxId), since, limit); + } + + @Override + public Flux listMailboxContentSinceSentAt(MailboxId mailboxId, ZonedDateTime since, Limit limit) { + return emailQueryViewDAO.listMailboxContentSinceSentAt(PostgresMailboxId.class.cast(mailboxId), since, limit); + } + + @Override + public Mono delete(MailboxId mailboxId, MessageId messageId) { + return emailQueryViewDAO.delete(PostgresMailboxId.class.cast(mailboxId), PostgresMessageId.class.cast(messageId)); + } + + @Override + public Mono delete(MailboxId mailboxId) { + return emailQueryViewDAO.delete(PostgresMailboxId.class.cast(mailboxId)); + } + + @Override + public Mono save(MailboxId mailboxId, ZonedDateTime sentAt, ZonedDateTime receivedAt, MessageId messageId) { + return emailQueryViewDAO.save(PostgresMailboxId.class.cast(mailboxId), sentAt, receivedAt, PostgresMessageId.class.cast(messageId)); + } +} diff --git a/server/data/data-jmap-postgres/src/main/java/org/apache/james/jmap/postgres/projections/PostgresEmailQueryViewDAO.java b/server/data/data-jmap-postgres/src/main/java/org/apache/james/jmap/postgres/projections/PostgresEmailQueryViewDAO.java new file mode 100644 index 00000000000..a61146c67ac --- /dev/null +++ b/server/data/data-jmap-postgres/src/main/java/org/apache/james/jmap/postgres/projections/PostgresEmailQueryViewDAO.java @@ -0,0 +1,143 @@ +/**************************************************************** + * 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.jmap.postgres.projections; + +import static org.apache.james.jmap.postgres.projections.PostgresEmailQueryViewModule.PostgresEmailQueryViewTable.MAILBOX_ID; +import static org.apache.james.jmap.postgres.projections.PostgresEmailQueryViewModule.PostgresEmailQueryViewTable.MESSAGE_ID; +import static org.apache.james.jmap.postgres.projections.PostgresEmailQueryViewModule.PostgresEmailQueryViewTable.PK_CONSTRAINT_NAME; +import static org.apache.james.jmap.postgres.projections.PostgresEmailQueryViewModule.PostgresEmailQueryViewTable.RECEIVED_AT; +import static org.apache.james.jmap.postgres.projections.PostgresEmailQueryViewModule.PostgresEmailQueryViewTable.SENT_AT; +import static org.apache.james.jmap.postgres.projections.PostgresEmailQueryViewModule.PostgresEmailQueryViewTable.TABLE_NAME; + +import java.time.ZonedDateTime; + +import jakarta.inject.Inject; +import jakarta.inject.Named; + +import org.apache.james.backends.postgres.utils.PostgresExecutor; +import org.apache.james.mailbox.model.MessageId; +import org.apache.james.mailbox.postgres.PostgresMailboxId; +import org.apache.james.mailbox.postgres.PostgresMessageId; +import org.apache.james.util.streams.Limit; + +import com.google.common.base.Preconditions; + +import reactor.core.publisher.Flux; +import reactor.core.publisher.Mono; + +public class PostgresEmailQueryViewDAO { + private PostgresExecutor postgresExecutor; + + @Inject + public PostgresEmailQueryViewDAO(@Named(PostgresExecutor.BY_PASS_RLS_INJECT) PostgresExecutor postgresExecutor) { + this.postgresExecutor = postgresExecutor; + } + + public Flux listMailboxContentSortedBySentAt(PostgresMailboxId mailboxId, Limit limit) { + Preconditions.checkArgument(!limit.isUnlimited(), "Limit should be defined"); + + return postgresExecutor.executeRows(dslContext -> Flux.from(dslContext.select(MESSAGE_ID) + .from(TABLE_NAME) + .where(MAILBOX_ID.eq(mailboxId.asUuid())) + .orderBy(SENT_AT.desc()) + .limit(limit.getLimit().get()))) + .map(record -> PostgresMessageId.Factory.of(record.get(MESSAGE_ID))); + } + + public Flux listMailboxContentSortedByReceivedAt(PostgresMailboxId mailboxId, Limit limit) { + Preconditions.checkArgument(!limit.isUnlimited(), "Limit should be defined"); + + return postgresExecutor.executeRows(dslContext -> Flux.from(dslContext.select(MESSAGE_ID) + .from(TABLE_NAME) + .where(MAILBOX_ID.eq(mailboxId.asUuid())) + .orderBy(RECEIVED_AT.desc()) + .limit(limit.getLimit().get()))) + .map(record -> PostgresMessageId.Factory.of(record.get(MESSAGE_ID))); + } + + public Flux listMailboxContentSinceAfterSortedBySentAt(PostgresMailboxId mailboxId, ZonedDateTime since, Limit limit) { + Preconditions.checkArgument(!limit.isUnlimited(), "Limit should be defined"); + + return postgresExecutor.executeRows(dslContext -> Flux.from(dslContext.select(MESSAGE_ID) + .from(TABLE_NAME) + .where(MAILBOX_ID.eq(mailboxId.asUuid())) + .and(RECEIVED_AT.greaterOrEqual(since.toOffsetDateTime())) + .orderBy(SENT_AT.desc()) + .limit(limit.getLimit().get()))) + .map(record -> PostgresMessageId.Factory.of(record.get(MESSAGE_ID))); + } + + public Flux listMailboxContentSinceAfterSortedByReceivedAt(PostgresMailboxId mailboxId, ZonedDateTime since, Limit limit) { + Preconditions.checkArgument(!limit.isUnlimited(), "Limit should be defined"); + + return postgresExecutor.executeRows(dslContext -> Flux.from(dslContext.select(MESSAGE_ID) + .from(TABLE_NAME) + .where(MAILBOX_ID.eq(mailboxId.asUuid())) + .and(RECEIVED_AT.greaterOrEqual(since.toOffsetDateTime())) + .orderBy(RECEIVED_AT.desc()) + .limit(limit.getLimit().get()))) + .map(record -> PostgresMessageId.Factory.of(record.get(MESSAGE_ID))); + } + + public Flux listMailboxContentBeforeSortedByReceivedAt(PostgresMailboxId mailboxId, ZonedDateTime since, Limit limit) { + Preconditions.checkArgument(!limit.isUnlimited(), "Limit should be defined"); + + return postgresExecutor.executeRows(dslContext -> Flux.from(dslContext.select(MESSAGE_ID) + .from(TABLE_NAME) + .where(MAILBOX_ID.eq(mailboxId.asUuid())) + .and(RECEIVED_AT.lessOrEqual(since.toOffsetDateTime())) + .orderBy(RECEIVED_AT.desc()) + .limit(limit.getLimit().get()))) + .map(record -> PostgresMessageId.Factory.of(record.get(MESSAGE_ID))); + } + + public Flux listMailboxContentSinceSentAt(PostgresMailboxId mailboxId, ZonedDateTime since, Limit limit) { + Preconditions.checkArgument(!limit.isUnlimited(), "Limit should be defined"); + + return postgresExecutor.executeRows(dslContext -> Flux.from(dslContext.select(MESSAGE_ID) + .from(TABLE_NAME) + .where(MAILBOX_ID.eq(mailboxId.asUuid())) + .and(SENT_AT.greaterOrEqual(since.toOffsetDateTime())) + .orderBy(SENT_AT.desc()) + .limit(limit.getLimit().get()))) + .map(record -> PostgresMessageId.Factory.of(record.get(MESSAGE_ID))); + } + + public Mono delete(PostgresMailboxId mailboxId, PostgresMessageId messageId) { + return postgresExecutor.executeVoid(dslContext -> Mono.from(dslContext.deleteFrom(TABLE_NAME) + .where(MAILBOX_ID.eq(mailboxId.asUuid())) + .and(MESSAGE_ID.eq(messageId.asUuid())))); + } + + public Mono delete(PostgresMailboxId mailboxId) { + return postgresExecutor.executeVoid(dslContext -> Mono.from(dslContext.deleteFrom(TABLE_NAME) + .where(MAILBOX_ID.eq(mailboxId.asUuid())))); + } + + public Mono save(PostgresMailboxId mailboxId, ZonedDateTime sentAt, ZonedDateTime receivedAt, PostgresMessageId messageId) { + return postgresExecutor.executeVoid(dslContext -> Mono.from(dslContext.insertInto(TABLE_NAME) + .set(MAILBOX_ID, mailboxId.asUuid()) + .set(MESSAGE_ID, messageId.asUuid()) + .set(SENT_AT, sentAt.toOffsetDateTime()) + .set(RECEIVED_AT, receivedAt.toOffsetDateTime()) + .onConflictOnConstraint(PK_CONSTRAINT_NAME) + .doNothing())); + } +} diff --git a/server/data/data-jmap-postgres/src/main/java/org/apache/james/jmap/postgres/projections/PostgresEmailQueryViewManager.java b/server/data/data-jmap-postgres/src/main/java/org/apache/james/jmap/postgres/projections/PostgresEmailQueryViewManager.java new file mode 100644 index 00000000000..3095d530587 --- /dev/null +++ b/server/data/data-jmap-postgres/src/main/java/org/apache/james/jmap/postgres/projections/PostgresEmailQueryViewManager.java @@ -0,0 +1,41 @@ +/**************************************************************** + * 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.jmap.postgres.projections; + +import jakarta.inject.Inject; + +import org.apache.james.backends.postgres.utils.PostgresExecutor; +import org.apache.james.core.Username; +import org.apache.james.jmap.api.projections.EmailQueryView; +import org.apache.james.jmap.api.projections.EmailQueryViewManager; + +public class PostgresEmailQueryViewManager implements EmailQueryViewManager { + private final PostgresExecutor.Factory executorFactory; + + @Inject + public PostgresEmailQueryViewManager(PostgresExecutor.Factory executorFactory) { + this.executorFactory = executorFactory; + } + + @Override + public EmailQueryView getEmailQueryView(Username username) { + return new PostgresEmailQueryView(new PostgresEmailQueryViewDAO(executorFactory.create(username.getDomainPart()))); + } +} diff --git a/server/data/data-jmap-postgres/src/main/java/org/apache/james/jmap/postgres/projections/PostgresEmailQueryViewModule.java b/server/data/data-jmap-postgres/src/main/java/org/apache/james/jmap/postgres/projections/PostgresEmailQueryViewModule.java new file mode 100644 index 00000000000..cd413128faf --- /dev/null +++ b/server/data/data-jmap-postgres/src/main/java/org/apache/james/jmap/postgres/projections/PostgresEmailQueryViewModule.java @@ -0,0 +1,81 @@ +/**************************************************************** + * 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.jmap.postgres.projections; + +import static org.apache.james.jmap.postgres.projections.PostgresEmailQueryViewModule.PostgresEmailQueryViewTable.MAILBOX_ID_INDEX; +import static org.apache.james.jmap.postgres.projections.PostgresEmailQueryViewModule.PostgresEmailQueryViewTable.MAILBOX_ID_RECEIVED_AT_INDEX; +import static org.apache.james.jmap.postgres.projections.PostgresEmailQueryViewModule.PostgresEmailQueryViewTable.MAILBOX_ID_SENT_AT_INDEX; +import static org.apache.james.jmap.postgres.projections.PostgresEmailQueryViewModule.PostgresEmailQueryViewTable.TABLE; + +import java.time.OffsetDateTime; +import java.util.UUID; + +import org.apache.james.backends.postgres.PostgresIndex; +import org.apache.james.backends.postgres.PostgresModule; +import org.apache.james.backends.postgres.PostgresTable; +import org.apache.james.mailbox.postgres.mail.PostgresMessageModule; +import org.jooq.Field; +import org.jooq.Name; +import org.jooq.Record; +import org.jooq.Table; +import org.jooq.impl.DSL; +import org.jooq.impl.SQLDataType; + +public interface PostgresEmailQueryViewModule { + interface PostgresEmailQueryViewTable { + Table TABLE_NAME = DSL.table("email_query_view"); + + Field MAILBOX_ID = DSL.field("mailbox_id", SQLDataType.UUID.notNull()); + Field MESSAGE_ID = PostgresMessageModule.MESSAGE_ID; + Field RECEIVED_AT = DSL.field("received_at", SQLDataType.TIMESTAMPWITHTIMEZONE.notNull()); + Field SENT_AT = DSL.field("sent_at", SQLDataType.TIMESTAMPWITHTIMEZONE.notNull()); + + Name PK_CONSTRAINT_NAME = DSL.name("email_query_view_pkey"); + + PostgresTable TABLE = PostgresTable.name(TABLE_NAME.getName()) + .createTableStep(((dsl, tableName) -> dsl.createTableIfNotExists(tableName) + .column(MAILBOX_ID) + .column(MESSAGE_ID) + .column(RECEIVED_AT) + .column(SENT_AT) + .constraint(DSL.constraint(PK_CONSTRAINT_NAME).primaryKey(MAILBOX_ID, MESSAGE_ID)))) + .supportsRowLevelSecurity() + .build(); + + PostgresIndex MAILBOX_ID_INDEX = PostgresIndex.name("email_query_view_mailbox_id_index") + .createIndexStep((dslContext, indexName) -> dslContext.createIndexIfNotExists(indexName) + .on(TABLE_NAME, MAILBOX_ID)); + + PostgresIndex MAILBOX_ID_RECEIVED_AT_INDEX = PostgresIndex.name("email_query_view_mailbox_id__received_at_index") + .createIndexStep((dslContext, indexName) -> dslContext.createIndexIfNotExists(indexName) + .on(TABLE_NAME, MAILBOX_ID, RECEIVED_AT)); + + PostgresIndex MAILBOX_ID_SENT_AT_INDEX = PostgresIndex.name("email_query_view_mailbox_id_sent_at_index") + .createIndexStep((dslContext, indexName) -> dslContext.createIndexIfNotExists(indexName) + .on(TABLE_NAME, MAILBOX_ID, SENT_AT)); + } + + PostgresModule MODULE = PostgresModule.builder() + .addTable(TABLE) + .addIndex(MAILBOX_ID_INDEX) + .addIndex(MAILBOX_ID_RECEIVED_AT_INDEX) + .addIndex(MAILBOX_ID_SENT_AT_INDEX) + .build(); +} \ No newline at end of file diff --git a/server/data/data-jmap-postgres/src/main/java/org/apache/james/jmap/postgres/projections/PostgresMessageFastViewProjection.java b/server/data/data-jmap-postgres/src/main/java/org/apache/james/jmap/postgres/projections/PostgresMessageFastViewProjection.java new file mode 100644 index 00000000000..68d173c31cd --- /dev/null +++ b/server/data/data-jmap-postgres/src/main/java/org/apache/james/jmap/postgres/projections/PostgresMessageFastViewProjection.java @@ -0,0 +1,105 @@ +/**************************************************************** + * 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.jmap.postgres.projections; + +import static org.apache.james.jmap.postgres.projections.PostgresMessageFastViewProjectionModule.MessageFastViewProjectionTable.HAS_ATTACHMENT; +import static org.apache.james.jmap.postgres.projections.PostgresMessageFastViewProjectionModule.MessageFastViewProjectionTable.MESSAGE_ID; +import static org.apache.james.jmap.postgres.projections.PostgresMessageFastViewProjectionModule.MessageFastViewProjectionTable.PREVIEW; +import static org.apache.james.jmap.postgres.projections.PostgresMessageFastViewProjectionModule.MessageFastViewProjectionTable.TABLE_NAME; + +import jakarta.inject.Inject; + +import org.apache.james.backends.postgres.utils.PostgresExecutor; +import org.apache.james.jmap.api.model.Preview; +import org.apache.james.jmap.api.projections.MessageFastViewPrecomputedProperties; +import org.apache.james.jmap.api.projections.MessageFastViewProjection; +import org.apache.james.mailbox.model.MessageId; +import org.apache.james.mailbox.postgres.PostgresMessageId; +import org.apache.james.metrics.api.Metric; +import org.apache.james.metrics.api.MetricFactory; +import org.jooq.Record; +import org.reactivestreams.Publisher; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + +import com.google.common.base.Preconditions; + +import reactor.core.publisher.Mono; + +public class PostgresMessageFastViewProjection implements MessageFastViewProjection { + public static final Logger LOGGER = LoggerFactory.getLogger(PostgresMessageFastViewProjection.class); + + private final PostgresExecutor postgresExecutor; + private final Metric metricRetrieveHitCount; + private final Metric metricRetrieveMissCount; + + @Inject + public PostgresMessageFastViewProjection(PostgresExecutor postgresExecutor, MetricFactory metricFactory) { + this.postgresExecutor = postgresExecutor; + this.metricRetrieveHitCount = metricFactory.generate(METRIC_RETRIEVE_HIT_COUNT); + this.metricRetrieveMissCount = metricFactory.generate(METRIC_RETRIEVE_MISS_COUNT); + } + + @Override + public Publisher store(MessageId messageId, MessageFastViewPrecomputedProperties precomputedProperties) { + return postgresExecutor.executeVoid(dslContext -> Mono.from(dslContext.insertInto(TABLE_NAME) + .set(MESSAGE_ID, ((PostgresMessageId) messageId).asUuid()) + .set(PREVIEW, precomputedProperties.getPreview().getValue()) + .set(HAS_ATTACHMENT, precomputedProperties.hasAttachment()) + .onConflict(MESSAGE_ID) + .doUpdate() + .set(PREVIEW, precomputedProperties.getPreview().getValue()) + .set(HAS_ATTACHMENT, precomputedProperties.hasAttachment()))); + } + + @Override + public Publisher retrieve(MessageId messageId) { + Preconditions.checkNotNull(messageId); + + return postgresExecutor.executeRow(dslContext -> Mono.from(dslContext.select(PREVIEW, HAS_ATTACHMENT) + .from(TABLE_NAME) + .where(MESSAGE_ID.eq(((PostgresMessageId) messageId).asUuid())))) + .doOnNext(preview -> metricRetrieveHitCount.increment()) + .switchIfEmpty(Mono.fromRunnable(metricRetrieveMissCount::increment)) + .map(this::toMessageFastViewPrecomputedProperties) + .onErrorResume(e -> { + LOGGER.error("Error while retrieving MessageFastView projection item for {}", messageId, e); + return Mono.empty(); + }); + } + + private MessageFastViewPrecomputedProperties toMessageFastViewPrecomputedProperties(Record record) { + return MessageFastViewPrecomputedProperties.builder() + .preview(Preview.from(record.get(PREVIEW))) + .hasAttachment(record.get(HAS_ATTACHMENT)) + .build(); + } + + @Override + public Publisher delete(MessageId messageId) { + return postgresExecutor.executeVoid(dslContext -> Mono.from(dslContext.deleteFrom(TABLE_NAME) + .where(MESSAGE_ID.eq(((PostgresMessageId) messageId).asUuid())))); + } + + @Override + public Publisher clear() { + return postgresExecutor.executeVoid(dslContext -> Mono.from(dslContext.truncate(TABLE_NAME))); + } +} diff --git a/server/data/data-jmap-postgres/src/main/java/org/apache/james/jmap/postgres/projections/PostgresMessageFastViewProjectionModule.java b/server/data/data-jmap-postgres/src/main/java/org/apache/james/jmap/postgres/projections/PostgresMessageFastViewProjectionModule.java new file mode 100644 index 00000000000..ef1e0cb885d --- /dev/null +++ b/server/data/data-jmap-postgres/src/main/java/org/apache/james/jmap/postgres/projections/PostgresMessageFastViewProjectionModule.java @@ -0,0 +1,56 @@ +/**************************************************************** + * 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.jmap.postgres.projections; + +import static org.apache.james.jmap.postgres.projections.PostgresMessageFastViewProjectionModule.MessageFastViewProjectionTable.TABLE; + +import java.util.UUID; + +import org.apache.james.backends.postgres.PostgresModule; +import org.apache.james.backends.postgres.PostgresTable; +import org.jooq.Field; +import org.jooq.Record; +import org.jooq.Table; +import org.jooq.impl.DSL; +import org.jooq.impl.SQLDataType; + +public interface PostgresMessageFastViewProjectionModule { + interface MessageFastViewProjectionTable { + Table TABLE_NAME = DSL.table("message_fast_view_projection"); + + Field MESSAGE_ID = DSL.field("messageId", SQLDataType.UUID.notNull()); + Field PREVIEW = DSL.field("preview", SQLDataType.VARCHAR.notNull()); + Field HAS_ATTACHMENT = DSL.field("has_attachment", SQLDataType.BOOLEAN.notNull()); + + PostgresTable TABLE = PostgresTable.name(TABLE_NAME.getName()) + .createTableStep(((dsl, tableName) -> dsl.createTableIfNotExists(tableName) + .column(MESSAGE_ID) + .column(PREVIEW) + .column(HAS_ATTACHMENT) + .primaryKey(MESSAGE_ID) + .comment("Storing the JMAP projections for MessageFastView, an aggregation of JMAP properties expected to be fast to fetch."))) + .disableRowLevelSecurity() + .build(); + } + + PostgresModule MODULE = PostgresModule.builder() + .addTable(TABLE) + .build(); +} diff --git a/server/data/data-jmap-postgres/src/main/java/org/apache/james/jmap/postgres/pushsubscription/PostgresPushSubscriptionDAO.java b/server/data/data-jmap-postgres/src/main/java/org/apache/james/jmap/postgres/pushsubscription/PostgresPushSubscriptionDAO.java new file mode 100644 index 00000000000..f94f514d314 --- /dev/null +++ b/server/data/data-jmap-postgres/src/main/java/org/apache/james/jmap/postgres/pushsubscription/PostgresPushSubscriptionDAO.java @@ -0,0 +1,172 @@ +/**************************************************************** + * 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.jmap.postgres.pushsubscription; + +import static org.apache.james.backends.postgres.PostgresCommons.IN_CLAUSE_MAX_SIZE; +import static org.apache.james.backends.postgres.PostgresCommons.OFFSET_DATE_TIME_ZONED_DATE_TIME_FUNCTION; +import static org.apache.james.backends.postgres.utils.PostgresUtils.UNIQUE_CONSTRAINT_VIOLATION_PREDICATE; +import static org.apache.james.jmap.postgres.pushsubscription.PostgresPushSubscriptionModule.PushSubscriptionTable.PRIMARY_KEY_CONSTRAINT; + +import java.time.ZonedDateTime; +import java.util.Arrays; +import java.util.Collection; +import java.util.Optional; +import java.util.Set; +import java.util.function.Function; +import java.util.function.Predicate; +import java.util.stream.Collectors; + +import org.apache.james.backends.postgres.utils.PostgresExecutor; +import org.apache.james.core.Username; +import org.apache.james.jmap.api.change.TypeStateFactory; +import org.apache.james.jmap.api.model.DeviceClientIdInvalidException; +import org.apache.james.jmap.api.model.PushSubscription; +import org.apache.james.jmap.api.model.PushSubscriptionExpiredTime; +import org.apache.james.jmap.api.model.PushSubscriptionId; +import org.apache.james.jmap.api.model.PushSubscriptionKeys; +import org.apache.james.jmap.api.model.PushSubscriptionServerURL; +import org.apache.james.jmap.api.model.TypeName; +import org.apache.james.jmap.postgres.pushsubscription.PostgresPushSubscriptionModule.PushSubscriptionTable; +import org.jooq.Record; + +import com.google.common.base.Preconditions; +import com.google.common.collect.Iterables; + +import reactor.core.publisher.Flux; +import reactor.core.publisher.Mono; +import scala.jdk.javaapi.CollectionConverters; +import scala.jdk.javaapi.OptionConverters; + +public class PostgresPushSubscriptionDAO { + private static final Predicate IS_PRIMARY_KEY_UNIQUE_CONSTRAINT = throwable -> throwable.getMessage().contains(PRIMARY_KEY_CONSTRAINT); + + private final PostgresExecutor postgresExecutor; + private final TypeStateFactory typeStateFactory; + + public PostgresPushSubscriptionDAO(PostgresExecutor postgresExecutor, TypeStateFactory typeStateFactory) { + this.postgresExecutor = postgresExecutor; + this.typeStateFactory = typeStateFactory; + } + + public Mono save(Username username, PushSubscription pushSubscription) { + return postgresExecutor.executeVoid(dslContext -> Mono.from(dslContext.insertInto(PushSubscriptionTable.TABLE_NAME) + .set(PushSubscriptionTable.USER, username.asString()) + .set(PushSubscriptionTable.DEVICE_CLIENT_ID, pushSubscription.deviceClientId()) + .set(PushSubscriptionTable.ID, pushSubscription.id().value()) + .set(PushSubscriptionTable.EXPIRES, pushSubscription.expires().value().toOffsetDateTime()) + .set(PushSubscriptionTable.TYPES, CollectionConverters.asJava(pushSubscription.types()) + .stream().map(TypeName::asString).toArray(String[]::new)) + .set(PushSubscriptionTable.URL, pushSubscription.url().value().toString()) + .set(PushSubscriptionTable.VERIFICATION_CODE, pushSubscription.verificationCode()) + .set(PushSubscriptionTable.VALIDATED, pushSubscription.validated()) + .set(PushSubscriptionTable.ENCRYPT_PUBLIC_KEY, OptionConverters.toJava(pushSubscription.keys().map(PushSubscriptionKeys::p256dh)).orElse(null)) + .set(PushSubscriptionTable.ENCRYPT_AUTH_SECRET, OptionConverters.toJava(pushSubscription.keys().map(PushSubscriptionKeys::auth)).orElse(null)))) + .onErrorMap(UNIQUE_CONSTRAINT_VIOLATION_PREDICATE.and(IS_PRIMARY_KEY_UNIQUE_CONSTRAINT), + e -> new DeviceClientIdInvalidException(pushSubscription.deviceClientId(), "deviceClientId must be unique")); + } + + public Flux listByUsername(Username username) { + return postgresExecutor.executeRows(dslContext -> Flux.from(dslContext.selectFrom(PushSubscriptionTable.TABLE_NAME) + .where(PushSubscriptionTable.USER.eq(username.asString())))) + .map(this::recordAsPushSubscription); + } + + public Flux getByUsernameAndIds(Username username, Collection ids) { + if (ids.isEmpty()) { + return Flux.empty(); + } + Function, Flux> queryPublisherFunction = idsMatching -> postgresExecutor.executeRows(dslContext -> Flux.from(dslContext.selectFrom(PushSubscriptionTable.TABLE_NAME) + .where(PushSubscriptionTable.USER.eq(username.asString())) + .and(PushSubscriptionTable.ID.in(idsMatching.stream().map(PushSubscriptionId::value).collect(Collectors.toList()))))) + .map(this::recordAsPushSubscription); + + if (ids.size() <= IN_CLAUSE_MAX_SIZE) { + return queryPublisherFunction.apply(ids); + } else { + return Flux.fromIterable(Iterables.partition(ids, IN_CLAUSE_MAX_SIZE)) + .flatMap(queryPublisherFunction); + } + } + + public Mono deleteByUsername(Username username) { + return postgresExecutor.executeVoid(dslContext -> Mono.from(dslContext.deleteFrom(PushSubscriptionTable.TABLE_NAME) + .where(PushSubscriptionTable.USER.eq(username.asString())))); + } + + public Mono deleteByUsernameAndId(Username username, PushSubscriptionId id) { + return postgresExecutor.executeVoid(dslContext -> Mono.from(dslContext.deleteFrom(PushSubscriptionTable.TABLE_NAME) + .where(PushSubscriptionTable.USER.eq(username.asString())) + .and(PushSubscriptionTable.ID.eq(id.value())))); + } + + public Mono> updateType(Username username, PushSubscriptionId id, Set newTypes) { + Preconditions.checkNotNull(newTypes, "newTypes should not be null"); + return postgresExecutor.executeRow(dslContext -> Mono.from(dslContext.update(PushSubscriptionTable.TABLE_NAME) + .set(PushSubscriptionTable.TYPES, newTypes.stream().map(TypeName::asString).toArray(String[]::new)) + .where(PushSubscriptionTable.USER.eq(username.asString())) + .and(PushSubscriptionTable.ID.eq(id.value())) + .returning(PushSubscriptionTable.TYPES))) + .map(this::extractTypes); + } + + public Mono updateValidated(Username username, PushSubscriptionId id, boolean validated) { + return postgresExecutor.executeRow(dslContext -> Mono.from(dslContext.update(PushSubscriptionTable.TABLE_NAME) + .set(PushSubscriptionTable.VALIDATED, validated) + .where(PushSubscriptionTable.USER.eq(username.asString())) + .and(PushSubscriptionTable.ID.eq(id.value())) + .returning(PushSubscriptionTable.VALIDATED))) + .map(record -> record.get(PushSubscriptionTable.VALIDATED)); + } + + public Mono updateExpireTime(Username username, PushSubscriptionId id, ZonedDateTime newExpire) { + Preconditions.checkNotNull(newExpire, "newExpire should not be null"); + return postgresExecutor.executeRow(dslContext -> Mono.from(dslContext.update(PushSubscriptionTable.TABLE_NAME) + .set(PushSubscriptionTable.EXPIRES, newExpire.toOffsetDateTime()) + .where(PushSubscriptionTable.USER.eq(username.asString())) + .and(PushSubscriptionTable.ID.eq(id.value())) + .returning(PushSubscriptionTable.EXPIRES))) + .map(record -> OFFSET_DATE_TIME_ZONED_DATE_TIME_FUNCTION.apply(record.get(PushSubscriptionTable.EXPIRES))); + } + + private PushSubscription recordAsPushSubscription(Record record) { + try { + return new PushSubscription(new PushSubscriptionId(record.get(PushSubscriptionTable.ID)), + record.get(PushSubscriptionTable.DEVICE_CLIENT_ID), + PushSubscriptionServerURL.from(record.get(PushSubscriptionTable.URL)).get(), + scala.jdk.javaapi.OptionConverters.toScala(Optional.ofNullable(record.get(PushSubscriptionTable.ENCRYPT_PUBLIC_KEY)) + .flatMap(key -> Optional.ofNullable(record.get(PushSubscriptionTable.ENCRYPT_AUTH_SECRET)) + .map(secret -> new PushSubscriptionKeys(key, secret)))), + record.get(PushSubscriptionTable.VERIFICATION_CODE), + record.get(PushSubscriptionTable.VALIDATED), + Optional.ofNullable(record.get(PushSubscriptionTable.EXPIRES)) + .map(OFFSET_DATE_TIME_ZONED_DATE_TIME_FUNCTION) + .map(PushSubscriptionExpiredTime::new).get(), + CollectionConverters.asScala(extractTypes(record)).toSeq()); + } catch (Exception e) { + throw new RuntimeException("Error while parsing PushSubscription from database", e); + } + } + + private Set extractTypes(Record record) { + return Arrays.stream(record.get(PushSubscriptionTable.TYPES)) + .map(string -> typeStateFactory.strictParse(string).right().get()) + .collect(Collectors.toSet()); + } +} \ No newline at end of file diff --git a/server/data/data-jmap-postgres/src/main/java/org/apache/james/jmap/postgres/pushsubscription/PostgresPushSubscriptionModule.java b/server/data/data-jmap-postgres/src/main/java/org/apache/james/jmap/postgres/pushsubscription/PostgresPushSubscriptionModule.java new file mode 100644 index 00000000000..ebe3c552ee8 --- /dev/null +++ b/server/data/data-jmap-postgres/src/main/java/org/apache/james/jmap/postgres/pushsubscription/PostgresPushSubscriptionModule.java @@ -0,0 +1,82 @@ +/**************************************************************** + * 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.jmap.postgres.pushsubscription; + +import java.time.OffsetDateTime; +import java.util.UUID; + +import org.apache.james.backends.postgres.PostgresCommons; +import org.apache.james.backends.postgres.PostgresIndex; +import org.apache.james.backends.postgres.PostgresModule; +import org.apache.james.backends.postgres.PostgresTable; +import org.jooq.Field; +import org.jooq.Record; +import org.jooq.Table; +import org.jooq.impl.DSL; +import org.jooq.impl.SQLDataType; + +public interface PostgresPushSubscriptionModule { + + interface PushSubscriptionTable { + Table TABLE_NAME = DSL.table("push_subscription"); + + String PRIMARY_KEY_CONSTRAINT = "push_subscription_primary_key_constraint"; + + Field USER = DSL.field("username", SQLDataType.VARCHAR.notNull()); + Field DEVICE_CLIENT_ID = DSL.field("device_client_id", SQLDataType.VARCHAR.notNull()); + Field ID = DSL.field("id", SQLDataType.UUID.notNull()); + Field EXPIRES = DSL.field("expires", PostgresCommons.DataTypes.TIMESTAMP_WITH_TIMEZONE); + Field TYPES = DSL.field("types", PostgresCommons.DataTypes.STRING_ARRAY.notNull()); + Field URL = DSL.field("url", SQLDataType.VARCHAR.notNull()); + Field VERIFICATION_CODE = DSL.field("verification_code", SQLDataType.VARCHAR); + Field ENCRYPT_PUBLIC_KEY = DSL.field("encrypt_public_key", SQLDataType.VARCHAR); + Field ENCRYPT_AUTH_SECRET = DSL.field("encrypt_auth_secret", SQLDataType.VARCHAR); + Field VALIDATED = DSL.field("validated", SQLDataType.BOOLEAN.notNull()); + + PostgresTable TABLE = PostgresTable.name(TABLE_NAME.getName()) + .createTableStep(((dsl, tableName) -> dsl.createTableIfNotExists(tableName) + .column(USER) + .column(DEVICE_CLIENT_ID) + .column(ID) + .column(EXPIRES) + .column(TYPES) + .column(URL) + .column(VERIFICATION_CODE) + .column(ENCRYPT_PUBLIC_KEY) + .column(ENCRYPT_AUTH_SECRET) + .column(VALIDATED) + .constraint(DSL.constraint(PRIMARY_KEY_CONSTRAINT) + .primaryKey(USER, DEVICE_CLIENT_ID)))) + .supportsRowLevelSecurity() + .build(); + + PostgresIndex USERNAME_INDEX = PostgresIndex.name("push_subscription_username_index") + .createIndexStep((dslContext, indexName) -> dslContext.createIndexIfNotExists(indexName) + .on(TABLE_NAME, USER)); + PostgresIndex USERNAME_ID_INDEX = PostgresIndex.name("push_subscription_username_id_index") + .createIndexStep((dslContext, indexName) -> dslContext.createIndexIfNotExists(indexName) + .on(TABLE_NAME, USER, ID)); + } + + PostgresModule MODULE = PostgresModule.builder() + .addTable(PushSubscriptionTable.TABLE) + .addIndex(PushSubscriptionTable.USERNAME_INDEX, PushSubscriptionTable.USERNAME_ID_INDEX) + .build(); +} diff --git a/server/data/data-jmap-postgres/src/main/java/org/apache/james/jmap/postgres/pushsubscription/PostgresPushSubscriptionRepository.java b/server/data/data-jmap-postgres/src/main/java/org/apache/james/jmap/postgres/pushsubscription/PostgresPushSubscriptionRepository.java new file mode 100644 index 00000000000..9e48a2d421a --- /dev/null +++ b/server/data/data-jmap-postgres/src/main/java/org/apache/james/jmap/postgres/pushsubscription/PostgresPushSubscriptionRepository.java @@ -0,0 +1,142 @@ +/**************************************************************** + * 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.jmap.postgres.pushsubscription; + +import static org.apache.james.jmap.api.pushsubscription.PushSubscriptionHelpers.evaluateExpiresTime; +import static org.apache.james.jmap.api.pushsubscription.PushSubscriptionHelpers.isInThePast; +import static org.apache.james.jmap.api.pushsubscription.PushSubscriptionHelpers.isInvalidPushSubscriptionKey; + +import java.time.Clock; +import java.time.ZonedDateTime; +import java.util.Optional; +import java.util.Set; + +import jakarta.inject.Inject; +import jakarta.inject.Singleton; + +import org.apache.james.backends.postgres.utils.PostgresExecutor; +import org.apache.james.core.Username; +import org.apache.james.jmap.api.change.TypeStateFactory; +import org.apache.james.jmap.api.model.ExpireTimeInvalidException; +import org.apache.james.jmap.api.model.InvalidPushSubscriptionKeys; +import org.apache.james.jmap.api.model.PushSubscription; +import org.apache.james.jmap.api.model.PushSubscriptionCreationRequest; +import org.apache.james.jmap.api.model.PushSubscriptionExpiredTime; +import org.apache.james.jmap.api.model.PushSubscriptionId; +import org.apache.james.jmap.api.model.PushSubscriptionNotFoundException; +import org.apache.james.jmap.api.model.TypeName; +import org.apache.james.jmap.api.pushsubscription.PushSubscriptionRepository; + +import reactor.core.publisher.Flux; +import reactor.core.publisher.Mono; +import scala.jdk.javaapi.OptionConverters; + +public class PostgresPushSubscriptionRepository implements PushSubscriptionRepository { + private final Clock clock; + private final TypeStateFactory typeStateFactory; + private final PostgresExecutor.Factory executorFactory; + + @Inject + @Singleton + public PostgresPushSubscriptionRepository(Clock clock, TypeStateFactory typeStateFactory, PostgresExecutor.Factory executorFactory) { + this.clock = clock; + this.typeStateFactory = typeStateFactory; + this.executorFactory = executorFactory; + } + + @Override + public Mono save(Username username, PushSubscriptionCreationRequest request) { + PostgresPushSubscriptionDAO pushSubscriptionDAO = getDAO(username); + + return validateCreationRequest(request) + .then(Mono.defer(() -> { + PushSubscription pushSubscription = PushSubscription.from(request, + evaluateExpiresTime(OptionConverters.toJava(request.expires().map(PushSubscriptionExpiredTime::value)), clock)); + + return pushSubscriptionDAO.save(username, pushSubscription) + .thenReturn(pushSubscription); + })); + } + + private Mono validateCreationRequest(PushSubscriptionCreationRequest request) { + return Mono.just(request) + .handle((creationRequest, sink) -> { + if (isInThePast(request.expires(), clock)) { + sink.error(new ExpireTimeInvalidException(request.expires().get().value(), "expires must be greater than now")); + return; + } + if (isInvalidPushSubscriptionKey(request.keys())) { + sink.error(new InvalidPushSubscriptionKeys(request.keys().get())); + } + }); + } + + @Override + public Mono updateExpireTime(Username username, PushSubscriptionId id, ZonedDateTime newExpire) { + return Mono.just(newExpire) + .handle((inputTime, sink) -> { + if (newExpire.isBefore(ZonedDateTime.now(clock))) { + sink.error(new ExpireTimeInvalidException(inputTime, "expires must be greater than now")); + } + }) + .then(getDAO(username).updateExpireTime(username, id, evaluateExpiresTime(Optional.of(newExpire), clock).value()) + .map(PushSubscriptionExpiredTime::new) + .switchIfEmpty(Mono.error(() -> new PushSubscriptionNotFoundException(id)))); + } + + @Override + public Mono updateTypes(Username username, PushSubscriptionId id, Set types) { + return getDAO(username).updateType(username, id, types) + .switchIfEmpty(Mono.error(() -> new PushSubscriptionNotFoundException(id))) + .then(); + } + + @Override + public Mono validateVerificationCode(Username username, PushSubscriptionId id) { + return getDAO(username) + .updateValidated(username, id, true) + .switchIfEmpty(Mono.error(() -> new PushSubscriptionNotFoundException(id))) + .then(); + } + + @Override + public Mono revoke(Username username, PushSubscriptionId id) { + return getDAO(username).deleteByUsernameAndId(username, id); + } + + @Override + public Mono delete(Username username) { + return getDAO(username).deleteByUsername(username); + } + + @Override + public Flux get(Username username, Set ids) { + return getDAO(username).getByUsernameAndIds(username, ids); + } + + @Override + public Flux list(Username username) { + return getDAO(username).listByUsername(username); + } + + private PostgresPushSubscriptionDAO getDAO(Username username) { + return new PostgresPushSubscriptionDAO(executorFactory.create(username.getDomainPart()), typeStateFactory); + } +} diff --git a/server/data/data-jmap-postgres/src/main/java/org/apache/james/jmap/postgres/upload/PostgresUploadDAO.java b/server/data/data-jmap-postgres/src/main/java/org/apache/james/jmap/postgres/upload/PostgresUploadDAO.java new file mode 100644 index 00000000000..345865c7833 --- /dev/null +++ b/server/data/data-jmap-postgres/src/main/java/org/apache/james/jmap/postgres/upload/PostgresUploadDAO.java @@ -0,0 +1,126 @@ +/**************************************************************** + * 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.jmap.postgres.upload; + +import static org.apache.james.backends.postgres.PostgresCommons.INSTANT_TO_LOCAL_DATE_TIME; +import static org.apache.james.jmap.postgres.upload.PostgresUploadModule.PostgresUploadTable; + +import java.time.LocalDateTime; +import java.util.Optional; + +import jakarta.inject.Inject; +import jakarta.inject.Named; +import jakarta.inject.Singleton; + +import org.apache.commons.lang3.tuple.Pair; +import org.apache.james.backends.postgres.PostgresCommons; +import org.apache.james.backends.postgres.utils.PostgresExecutor; +import org.apache.james.blob.api.BlobId; +import org.apache.james.core.Domain; +import org.apache.james.core.Username; +import org.apache.james.jmap.api.model.UploadId; +import org.apache.james.jmap.api.model.UploadMetaData; +import org.apache.james.mailbox.model.ContentType; +import org.jooq.Record; + +import reactor.core.publisher.Flux; +import reactor.core.publisher.Mono; + +public class PostgresUploadDAO { + public static class Factory { + private final BlobId.Factory blobIdFactory; + private final PostgresExecutor.Factory executorFactory; + + @Inject + @Singleton + public Factory(BlobId.Factory blobIdFactory, PostgresExecutor.Factory executorFactory) { + this.blobIdFactory = blobIdFactory; + this.executorFactory = executorFactory; + } + + public PostgresUploadDAO create(Optional domain) { + return new PostgresUploadDAO(executorFactory.create(domain), blobIdFactory); + } + } + + private final PostgresExecutor postgresExecutor; + + private final BlobId.Factory blobIdFactory; + + @Singleton + @Inject + public PostgresUploadDAO(@Named(PostgresExecutor.BY_PASS_RLS_INJECT) PostgresExecutor postgresExecutor, BlobId.Factory blobIdFactory) { + this.postgresExecutor = postgresExecutor; + this.blobIdFactory = blobIdFactory; + } + + public Mono insert(UploadMetaData upload, Username user) { + return postgresExecutor.executeRow(dslContext -> Mono.from(dslContext.insertInto(PostgresUploadTable.TABLE_NAME) + .set(PostgresUploadTable.ID, upload.uploadId().getId()) + .set(PostgresUploadTable.CONTENT_TYPE, upload.contentType().asString()) + .set(PostgresUploadTable.SIZE, upload.sizeAsLong()) + .set(PostgresUploadTable.BLOB_ID, upload.blobId().asString()) + .set(PostgresUploadTable.USER_NAME, user.asString()) + .set(PostgresUploadTable.UPLOAD_DATE, INSTANT_TO_LOCAL_DATE_TIME.apply(upload.uploadDate())) + .returning(PostgresUploadTable.ID, + PostgresUploadTable.CONTENT_TYPE, + PostgresUploadTable.SIZE, + PostgresUploadTable.BLOB_ID, + PostgresUploadTable.UPLOAD_DATE, + PostgresUploadTable.USER_NAME))) + .map(this::uploadMetaDataFromRow); + } + + public Flux list(Username user) { + return postgresExecutor.executeRows(dslContext -> Flux.from(dslContext.selectFrom(PostgresUploadTable.TABLE_NAME) + .where(PostgresUploadTable.USER_NAME.eq(user.asString())))) + .map(this::uploadMetaDataFromRow); + } + + public Mono get(UploadId uploadId, Username user) { + return postgresExecutor.executeRow(dslContext -> Mono.from(dslContext.selectFrom(PostgresUploadTable.TABLE_NAME) + .where(PostgresUploadTable.ID.eq(uploadId.getId())) + .and(PostgresUploadTable.USER_NAME.eq(user.asString())))) + .map(this::uploadMetaDataFromRow); + } + + public Mono delete(UploadId uploadId, Username user) { + return postgresExecutor.executeRow(dslContext -> Mono.from(dslContext.deleteFrom(PostgresUploadTable.TABLE_NAME) + .where(PostgresUploadTable.ID.eq(uploadId.getId())) + .and(PostgresUploadTable.USER_NAME.eq(user.asString())) + .returning(PostgresUploadTable.ID))) + .hasElement(); + } + + public Flux> listByUploadDateBefore(LocalDateTime before) { + return postgresExecutor.executeRows(dslContext -> Flux.from(dslContext.selectFrom(PostgresUploadTable.TABLE_NAME) + .where(PostgresUploadTable.UPLOAD_DATE.lessThan(before)))) + .map(record -> Pair.of(uploadMetaDataFromRow(record), Username.of(record.get(PostgresUploadTable.USER_NAME)))); + } + + private UploadMetaData uploadMetaDataFromRow(Record record) { + return UploadMetaData.from( + UploadId.from(record.get(PostgresUploadTable.ID)), + Optional.ofNullable(record.get(PostgresUploadTable.CONTENT_TYPE)).map(ContentType::of).orElse(null), + record.get(PostgresUploadTable.SIZE), + blobIdFactory.parse(record.get(PostgresUploadTable.BLOB_ID)), + PostgresCommons.LOCAL_DATE_TIME_INSTANT_FUNCTION.apply(record.get(PostgresUploadTable.UPLOAD_DATE, LocalDateTime.class))); + } +} diff --git a/server/data/data-jmap-postgres/src/main/java/org/apache/james/jmap/postgres/upload/PostgresUploadModule.java b/server/data/data-jmap-postgres/src/main/java/org/apache/james/jmap/postgres/upload/PostgresUploadModule.java new file mode 100644 index 00000000000..cfc9d097a5e --- /dev/null +++ b/server/data/data-jmap-postgres/src/main/java/org/apache/james/jmap/postgres/upload/PostgresUploadModule.java @@ -0,0 +1,78 @@ +/**************************************************************** + * 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.jmap.postgres.upload; + + +import static org.apache.james.jmap.postgres.upload.PostgresUploadModule.PostgresUploadTable.TABLE; + +import java.time.LocalDateTime; +import java.util.UUID; + +import org.apache.james.backends.postgres.PostgresCommons; +import org.apache.james.backends.postgres.PostgresIndex; +import org.apache.james.backends.postgres.PostgresModule; +import org.apache.james.backends.postgres.PostgresTable; +import org.jooq.Field; +import org.jooq.Record; +import org.jooq.Table; +import org.jooq.impl.DSL; +import org.jooq.impl.SQLDataType; + +public interface PostgresUploadModule { + interface PostgresUploadTable { + + Table TABLE_NAME = DSL.table("uploads"); + + Field ID = DSL.field("id", SQLDataType.UUID.notNull()); + Field CONTENT_TYPE = DSL.field("content_type", SQLDataType.VARCHAR); + Field SIZE = DSL.field("size", SQLDataType.BIGINT.notNull()); + Field BLOB_ID = DSL.field("blob_id", SQLDataType.VARCHAR.notNull()); + Field USER_NAME = DSL.field("user_name", SQLDataType.VARCHAR.notNull()); + Field UPLOAD_DATE = DSL.field("upload_date", PostgresCommons.DataTypes.TIMESTAMP.notNull()); + + PostgresTable TABLE = PostgresTable.name(TABLE_NAME.getName()) + .createTableStep(((dsl, tableName) -> dsl.createTableIfNotExists(tableName) + .column(ID) + .column(CONTENT_TYPE) + .column(SIZE) + .column(BLOB_ID) + .column(USER_NAME) + .column(UPLOAD_DATE) + .primaryKey(ID))) + .supportsRowLevelSecurity() + .build(); + + PostgresIndex USER_NAME_INDEX = PostgresIndex.name("uploads_user_name_index") + .createIndexStep((dslContext, indexName) -> dslContext.createIndexIfNotExists(indexName) + .on(TABLE_NAME, USER_NAME)); + PostgresIndex ID_USERNAME_INDEX = PostgresIndex.name("uploads_id_user_name_index") + .createIndexStep((dslContext, indexName) -> dslContext.createIndexIfNotExists(indexName) + .on(TABLE_NAME, ID, USER_NAME)); + PostgresIndex UPLOAD_DATE_INDEX = PostgresIndex.name("uploads_upload_date_index") + .createIndexStep((dslContext, indexName) -> dslContext.createIndexIfNotExists(indexName) + .on(TABLE_NAME, UPLOAD_DATE)); + + } + + PostgresModule MODULE = PostgresModule.builder() + .addTable(TABLE) + .addIndex(PostgresUploadTable.USER_NAME_INDEX, PostgresUploadTable.ID_USERNAME_INDEX, PostgresUploadTable.UPLOAD_DATE_INDEX) + .build(); +} diff --git a/server/data/data-jmap-postgres/src/main/java/org/apache/james/jmap/postgres/upload/PostgresUploadRepository.java b/server/data/data-jmap-postgres/src/main/java/org/apache/james/jmap/postgres/upload/PostgresUploadRepository.java new file mode 100644 index 00000000000..35d2c7b86c0 --- /dev/null +++ b/server/data/data-jmap-postgres/src/main/java/org/apache/james/jmap/postgres/upload/PostgresUploadRepository.java @@ -0,0 +1,113 @@ +/**************************************************************** + * 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.jmap.postgres.upload; + +import static org.apache.james.backends.postgres.PostgresCommons.INSTANT_TO_LOCAL_DATE_TIME; +import static org.apache.james.blob.api.BlobStore.StoragePolicy.LOW_COST; +import static org.apache.james.util.ReactorUtils.DEFAULT_CONCURRENCY; + +import java.io.InputStream; +import java.time.Clock; +import java.time.Duration; +import java.time.LocalDateTime; + +import jakarta.inject.Inject; +import jakarta.inject.Singleton; + +import org.apache.james.blob.api.BlobStore; +import org.apache.james.blob.api.BucketName; +import org.apache.james.core.Username; +import org.apache.james.jmap.api.model.Upload; +import org.apache.james.jmap.api.model.UploadId; +import org.apache.james.jmap.api.model.UploadMetaData; +import org.apache.james.jmap.api.model.UploadNotFoundException; +import org.apache.james.jmap.api.upload.UploadRepository; +import org.apache.james.mailbox.model.ContentType; + +import com.github.f4b6a3.uuid.UuidCreator; +import com.google.common.io.CountingInputStream; + +import reactor.core.publisher.Flux; +import reactor.core.publisher.Mono; + +public class PostgresUploadRepository implements UploadRepository { + public static final BucketName UPLOAD_BUCKET = BucketName.of("jmap-uploads"); + private final BlobStore blobStore; + private final Clock clock; + private final PostgresUploadDAO.Factory uploadDAOFactory; + private final PostgresUploadDAO byPassRLSUploadDAO; + + @Inject + @Singleton + public PostgresUploadRepository(BlobStore blobStore, Clock clock, + PostgresUploadDAO.Factory uploadDAOFactory, + PostgresUploadDAO byPassRLSUploadDAO) { + this.blobStore = blobStore; + this.clock = clock; + this.uploadDAOFactory = uploadDAOFactory; + this.byPassRLSUploadDAO = byPassRLSUploadDAO; + } + + @Override + public Mono upload(InputStream data, ContentType contentType, Username user) { + UploadId uploadId = generateId(); + PostgresUploadDAO uploadDAO = uploadDAOFactory.create(user.getDomainPart()); + return Mono.fromCallable(() -> new CountingInputStream(data)) + .flatMap(countingInputStream -> Mono.from(blobStore.save(UPLOAD_BUCKET, countingInputStream, LOW_COST)) + .map(blobId -> UploadMetaData.from(uploadId, contentType, countingInputStream.getCount(), blobId, clock.instant())) + .flatMap(uploadMetaData -> uploadDAO.insert(uploadMetaData, user))); + } + + @Override + public Mono retrieve(UploadId id, Username user) { + return uploadDAOFactory.create(user.getDomainPart()).get(id, user) + .flatMap(upload -> Mono.from(blobStore.readReactive(UPLOAD_BUCKET, upload.blobId(), LOW_COST)) + .map(inputStream -> Upload.from(upload, () -> inputStream))) + .switchIfEmpty(Mono.error(() -> new UploadNotFoundException(id))); + } + + @Override + public Mono delete(UploadId id, Username user) { + return uploadDAOFactory.create(user.getDomainPart()).delete(id, user); + } + + @Override + public Flux listUploads(Username user) { + return uploadDAOFactory.create(user.getDomainPart()).list(user); + } + + @Override + public Mono deleteByUploadDateBefore(Duration expireDuration) { + LocalDateTime expirationTime = INSTANT_TO_LOCAL_DATE_TIME.apply(clock.instant().minus(expireDuration)); + + return Flux.from(byPassRLSUploadDAO.listByUploadDateBefore(expirationTime)) + .flatMap(uploadPair -> { + Username username = uploadPair.getRight(); + UploadMetaData upload = uploadPair.getLeft(); + return Mono.from(blobStore.delete(UPLOAD_BUCKET, upload.blobId())) + .then(byPassRLSUploadDAO.delete(upload.uploadId(), username)); + }, DEFAULT_CONCURRENCY) + .then(); + } + + private UploadId generateId() { + return UploadId.from(UuidCreator.getTimeOrderedEpoch()); + } +} diff --git a/server/data/data-jmap-postgres/src/main/java/org/apache/james/jmap/postgres/upload/PostgresUploadUsageRepository.java b/server/data/data-jmap-postgres/src/main/java/org/apache/james/jmap/postgres/upload/PostgresUploadUsageRepository.java new file mode 100644 index 00000000000..58993e1ec5c --- /dev/null +++ b/server/data/data-jmap-postgres/src/main/java/org/apache/james/jmap/postgres/upload/PostgresUploadUsageRepository.java @@ -0,0 +1,69 @@ +/**************************************************************** + * 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.jmap.postgres.upload; + +import jakarta.inject.Inject; +import jakarta.inject.Singleton; + +import org.apache.james.backends.postgres.quota.PostgresQuotaCurrentValueDAO; +import org.apache.james.core.Username; +import org.apache.james.core.quota.QuotaComponent; +import org.apache.james.core.quota.QuotaCurrentValue; +import org.apache.james.core.quota.QuotaSizeUsage; +import org.apache.james.core.quota.QuotaType; +import org.apache.james.jmap.api.upload.UploadUsageRepository; + +import reactor.core.publisher.Mono; + +public class PostgresUploadUsageRepository implements UploadUsageRepository { + private static final QuotaSizeUsage DEFAULT_QUOTA_SIZE_USAGE = QuotaSizeUsage.size(0); + + private final PostgresQuotaCurrentValueDAO quotaCurrentValueDAO; + + @Inject + @Singleton + public PostgresUploadUsageRepository(PostgresQuotaCurrentValueDAO quotaCurrentValueDAO) { + this.quotaCurrentValueDAO = quotaCurrentValueDAO; + } + + @Override + public Mono increaseSpace(Username username, QuotaSizeUsage usage) { + return quotaCurrentValueDAO.increase(QuotaCurrentValue.Key.of(QuotaComponent.JMAP_UPLOADS, username.asString(), QuotaType.SIZE), + usage.asLong()); + } + + @Override + public Mono decreaseSpace(Username username, QuotaSizeUsage usage) { + return quotaCurrentValueDAO.decrease(QuotaCurrentValue.Key.of(QuotaComponent.JMAP_UPLOADS, username.asString(), QuotaType.SIZE), + usage.asLong()); + } + + @Override + public Mono getSpaceUsage(Username username) { + return quotaCurrentValueDAO.getQuotaCurrentValue(QuotaCurrentValue.Key.of(QuotaComponent.JMAP_UPLOADS, username.asString(), QuotaType.SIZE)) + .map(quotaCurrentValue -> QuotaSizeUsage.size(quotaCurrentValue.getCurrentValue())).defaultIfEmpty(DEFAULT_QUOTA_SIZE_USAGE); + } + + @Override + public Mono resetSpace(Username username, QuotaSizeUsage newUsage) { + return quotaCurrentValueDAO.upsert(QuotaCurrentValue.Key.of(QuotaComponent.JMAP_UPLOADS, username.asString(), QuotaType.SIZE), newUsage.asLong()) + .then(); + } +} diff --git a/server/data/data-jmap-postgres/src/test/java/org/apache/james/jmap/postgres/change/PostgresEmailChangeRepositoryTest.java b/server/data/data-jmap-postgres/src/test/java/org/apache/james/jmap/postgres/change/PostgresEmailChangeRepositoryTest.java new file mode 100644 index 00000000000..7b2865102b5 --- /dev/null +++ b/server/data/data-jmap-postgres/src/test/java/org/apache/james/jmap/postgres/change/PostgresEmailChangeRepositoryTest.java @@ -0,0 +1,58 @@ +/**************************************************************** + * 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.jmap.postgres.change; + +import java.util.UUID; + +import org.apache.james.backends.postgres.PostgresExtension; +import org.apache.james.jmap.api.change.EmailChangeRepository; +import org.apache.james.jmap.api.change.EmailChangeRepositoryContract; +import org.apache.james.jmap.api.change.State; +import org.apache.james.mailbox.model.MessageId; +import org.apache.james.mailbox.postgres.PostgresMessageId; +import org.junit.jupiter.api.BeforeEach; +import org.junit.jupiter.api.extension.RegisterExtension; + +public class PostgresEmailChangeRepositoryTest implements EmailChangeRepositoryContract { + @RegisterExtension + static PostgresExtension postgresExtension = PostgresExtension.withRowLevelSecurity(PostgresEmailChangeModule.MODULE); + + PostgresEmailChangeRepository postgresEmailChangeRepository; + + @BeforeEach + public void setUp() { + postgresEmailChangeRepository = new PostgresEmailChangeRepository(postgresExtension.getExecutorFactory(), DEFAULT_NUMBER_OF_CHANGES); + } + + @Override + public EmailChangeRepository emailChangeRepository() { + return postgresEmailChangeRepository; + } + + @Override + public MessageId generateNewMessageId() { + return PostgresMessageId.Factory.of(UUID.randomUUID()); + } + + @Override + public State generateNewState() { + return new PostgresStateFactory().generate(); + } +} diff --git a/server/data/data-jmap-postgres/src/test/java/org/apache/james/jmap/postgres/change/PostgresMailboxChangeRepositoryTest.java b/server/data/data-jmap-postgres/src/test/java/org/apache/james/jmap/postgres/change/PostgresMailboxChangeRepositoryTest.java new file mode 100644 index 00000000000..d6b1dba21ff --- /dev/null +++ b/server/data/data-jmap-postgres/src/test/java/org/apache/james/jmap/postgres/change/PostgresMailboxChangeRepositoryTest.java @@ -0,0 +1,58 @@ +/**************************************************************** + * 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.jmap.postgres.change; + +import java.util.UUID; + +import org.apache.james.backends.postgres.PostgresExtension; +import org.apache.james.jmap.api.change.MailboxChangeRepository; +import org.apache.james.jmap.api.change.MailboxChangeRepositoryContract; +import org.apache.james.jmap.api.change.State; +import org.apache.james.mailbox.model.MailboxId; +import org.apache.james.mailbox.postgres.PostgresMailboxId; +import org.junit.jupiter.api.BeforeEach; +import org.junit.jupiter.api.extension.RegisterExtension; + +class PostgresMailboxChangeRepositoryTest implements MailboxChangeRepositoryContract { + @RegisterExtension + static PostgresExtension postgresExtension = PostgresExtension.withRowLevelSecurity(PostgresMailboxChangeModule.MODULE); + + PostgresMailboxChangeRepository postgresMailboxChangeRepository; + + @BeforeEach + public void setUp() { + postgresMailboxChangeRepository = new PostgresMailboxChangeRepository(postgresExtension.getExecutorFactory(), DEFAULT_NUMBER_OF_CHANGES); + } + + @Override + public State.Factory stateFactory() { + return new PostgresStateFactory(); + } + + @Override + public MailboxChangeRepository mailboxChangeRepository() { + return postgresMailboxChangeRepository; + } + + @Override + public MailboxId generateNewMailboxId() { + return PostgresMailboxId.of(UUID.randomUUID()); + } +} diff --git a/server/data/data-jmap-postgres/src/test/java/org/apache/james/jmap/postgres/filtering/PostgresEventSourcingFilteringManagementNoProjectionTest.java b/server/data/data-jmap-postgres/src/test/java/org/apache/james/jmap/postgres/filtering/PostgresEventSourcingFilteringManagementNoProjectionTest.java new file mode 100644 index 00000000000..fc66484ae6a --- /dev/null +++ b/server/data/data-jmap-postgres/src/test/java/org/apache/james/jmap/postgres/filtering/PostgresEventSourcingFilteringManagementNoProjectionTest.java @@ -0,0 +1,46 @@ +/**************************************************************** + * 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.jmap.postgres.filtering; + +import org.apache.james.backends.postgres.PostgresExtension; +import org.apache.james.eventsourcing.eventstore.EventStore; +import org.apache.james.eventsourcing.eventstore.JsonEventSerializer; +import org.apache.james.eventsourcing.eventstore.postgres.PostgresEventStore; +import org.apache.james.eventsourcing.eventstore.postgres.PostgresEventStoreDAO; +import org.apache.james.eventsourcing.eventstore.postgres.PostgresEventStoreModule; +import org.apache.james.jmap.api.filtering.FilteringManagement; +import org.apache.james.jmap.api.filtering.FilteringManagementContract; +import org.apache.james.jmap.api.filtering.FilteringRuleSetDefineDTOModules; +import org.apache.james.jmap.api.filtering.impl.EventSourcingFilteringManagement; +import org.junit.jupiter.api.extension.RegisterExtension; + +public class PostgresEventSourcingFilteringManagementNoProjectionTest implements FilteringManagementContract { + @RegisterExtension + static PostgresExtension postgresExtension = PostgresExtension.withoutRowLevelSecurity(PostgresEventStoreModule.MODULE); + + @Override + public FilteringManagement instantiateFilteringManagement() { + EventStore eventStore = new PostgresEventStore(new PostgresEventStoreDAO(postgresExtension.getDefaultPostgresExecutor(), + JsonEventSerializer.forModules(FilteringRuleSetDefineDTOModules.FILTERING_RULE_SET_DEFINED, + FilteringRuleSetDefineDTOModules.FILTERING_INCREMENT).withoutNestedType())); + return new EventSourcingFilteringManagement(eventStore, + new EventSourcingFilteringManagement.NoReadProjection(eventStore)); + } +} diff --git a/server/data/data-jmap-postgres/src/test/java/org/apache/james/jmap/postgres/filtering/PostgresEventSourcingFilteringManagementTest.java b/server/data/data-jmap-postgres/src/test/java/org/apache/james/jmap/postgres/filtering/PostgresEventSourcingFilteringManagementTest.java new file mode 100644 index 00000000000..4cb286c21da --- /dev/null +++ b/server/data/data-jmap-postgres/src/test/java/org/apache/james/jmap/postgres/filtering/PostgresEventSourcingFilteringManagementTest.java @@ -0,0 +1,46 @@ +/**************************************************************** + * 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.jmap.postgres.filtering; + +import org.apache.james.backends.postgres.PostgresExtension; +import org.apache.james.backends.postgres.PostgresModule; +import org.apache.james.eventsourcing.eventstore.JsonEventSerializer; +import org.apache.james.eventsourcing.eventstore.postgres.PostgresEventStore; +import org.apache.james.eventsourcing.eventstore.postgres.PostgresEventStoreDAO; +import org.apache.james.eventsourcing.eventstore.postgres.PostgresEventStoreModule; +import org.apache.james.jmap.api.filtering.FilteringManagement; +import org.apache.james.jmap.api.filtering.FilteringManagementContract; +import org.apache.james.jmap.api.filtering.FilteringRuleSetDefineDTOModules; +import org.apache.james.jmap.api.filtering.impl.EventSourcingFilteringManagement; +import org.junit.jupiter.api.extension.RegisterExtension; + +public class PostgresEventSourcingFilteringManagementTest implements FilteringManagementContract { + @RegisterExtension + static PostgresExtension postgresExtension = PostgresExtension.withoutRowLevelSecurity(PostgresModule.aggregateModules(PostgresFilteringProjectionModule.MODULE, + PostgresEventStoreModule.MODULE)); + + @Override + public FilteringManagement instantiateFilteringManagement() { + return new EventSourcingFilteringManagement(new PostgresEventStore(new PostgresEventStoreDAO(postgresExtension.getDefaultPostgresExecutor(), + JsonEventSerializer.forModules(FilteringRuleSetDefineDTOModules.FILTERING_RULE_SET_DEFINED, + FilteringRuleSetDefineDTOModules.FILTERING_INCREMENT).withoutNestedType())), + new PostgresFilteringProjection(new PostgresFilteringProjectionDAO(postgresExtension.getDefaultPostgresExecutor()))); + } +} diff --git a/server/data/data-jmap-postgres/src/test/java/org/apache/james/jmap/postgres/identity/PostgresCustomIdentityDAOTest.java b/server/data/data-jmap-postgres/src/test/java/org/apache/james/jmap/postgres/identity/PostgresCustomIdentityDAOTest.java new file mode 100644 index 00000000000..7c72f9cceb3 --- /dev/null +++ b/server/data/data-jmap-postgres/src/test/java/org/apache/james/jmap/postgres/identity/PostgresCustomIdentityDAOTest.java @@ -0,0 +1,35 @@ +/**************************************************************** + * 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.jmap.postgres.identity; + +import org.apache.james.backends.postgres.PostgresExtension; +import org.apache.james.jmap.api.identity.CustomIdentityDAO; +import org.apache.james.jmap.api.identity.CustomIdentityDAOContract; +import org.junit.jupiter.api.extension.RegisterExtension; + +public class PostgresCustomIdentityDAOTest implements CustomIdentityDAOContract { + @RegisterExtension + static PostgresExtension postgresExtension = PostgresExtension.withRowLevelSecurity(PostgresCustomIdentityModule.MODULE); + + @Override + public CustomIdentityDAO testee() { + return new PostgresCustomIdentityDAO(postgresExtension.getExecutorFactory()); + } +} diff --git a/server/data/data-jmap-postgres/src/test/java/org/apache/james/jmap/postgres/projections/PostgresEmailQueryViewManagerRLSTest.java b/server/data/data-jmap-postgres/src/test/java/org/apache/james/jmap/postgres/projections/PostgresEmailQueryViewManagerRLSTest.java new file mode 100644 index 00000000000..a0a48d734f2 --- /dev/null +++ b/server/data/data-jmap-postgres/src/test/java/org/apache/james/jmap/postgres/projections/PostgresEmailQueryViewManagerRLSTest.java @@ -0,0 +1,73 @@ +/**************************************************************** + * 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.jmap.postgres.projections; + +import static org.assertj.core.api.Assertions.assertThat; + +import java.time.ZonedDateTime; + +import org.apache.james.backends.postgres.PostgresExtension; +import org.apache.james.core.Username; +import org.apache.james.jmap.api.projections.EmailQueryViewManager; +import org.apache.james.mailbox.postgres.PostgresMailboxId; +import org.apache.james.mailbox.postgres.PostgresMessageId; +import org.apache.james.util.streams.Limit; +import org.junit.jupiter.api.BeforeEach; +import org.junit.jupiter.api.Test; +import org.junit.jupiter.api.extension.RegisterExtension; + +public class PostgresEmailQueryViewManagerRLSTest { + public static final PostgresMailboxId MAILBOX_ID_1 = PostgresMailboxId.generate(); + public static final PostgresMessageId.Factory MESSAGE_ID_FACTORY = new PostgresMessageId.Factory(); + public static final PostgresMessageId MESSAGE_ID_1 = MESSAGE_ID_FACTORY.generate(); + private static final ZonedDateTime DATE_1 = ZonedDateTime.parse("2010-10-30T15:12:00Z"); + private static final ZonedDateTime DATE_2 = ZonedDateTime.parse("2010-10-30T16:12:00Z"); + + @RegisterExtension + static PostgresExtension postgresExtension = PostgresExtension.withRowLevelSecurity(PostgresEmailQueryViewModule.MODULE); + + private EmailQueryViewManager emailQueryViewManager; + + @BeforeEach + public void setUp() { + emailQueryViewManager = new PostgresEmailQueryViewManager(postgresExtension.getExecutorFactory()); + } + + @Test + void emailQueryViewCanBeAccessedAtTheDataLevelByMembersOfTheSameDomain() { + Username username = Username.of("alice@domain1"); + + emailQueryViewManager.getEmailQueryView(username).save(MAILBOX_ID_1, DATE_1, DATE_2, MESSAGE_ID_1).block(); + + assertThat(emailQueryViewManager.getEmailQueryView(username).listMailboxContentSortedByReceivedAt(MAILBOX_ID_1, Limit.limit(1)).collectList().block()) + .isNotEmpty(); + } + + @Test + void emailQueryViewShouldBeIsolatedByDomain() { + Username username = Username.of("alice@domain1"); + Username username2 = Username.of("bob@domain2"); + + emailQueryViewManager.getEmailQueryView(username).save(MAILBOX_ID_1, DATE_1, DATE_2, MESSAGE_ID_1).block(); + + assertThat(emailQueryViewManager.getEmailQueryView(username2).listMailboxContentSortedByReceivedAt(MAILBOX_ID_1, Limit.limit(1)).collectList().block()) + .isEmpty(); + } +} diff --git a/server/data/data-jmap-postgres/src/test/java/org/apache/james/jmap/postgres/projections/PostgresEmailQueryViewTest.java b/server/data/data-jmap-postgres/src/test/java/org/apache/james/jmap/postgres/projections/PostgresEmailQueryViewTest.java new file mode 100644 index 00000000000..0b4218a2ad1 --- /dev/null +++ b/server/data/data-jmap-postgres/src/test/java/org/apache/james/jmap/postgres/projections/PostgresEmailQueryViewTest.java @@ -0,0 +1,71 @@ +/**************************************************************** + * 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.jmap.postgres.projections; + +import org.apache.james.backends.postgres.PostgresExtension; +import org.apache.james.jmap.api.projections.EmailQueryView; +import org.apache.james.jmap.api.projections.EmailQueryViewContract; +import org.apache.james.mailbox.model.MailboxId; +import org.apache.james.mailbox.model.MessageId; +import org.apache.james.mailbox.postgres.PostgresMailboxId; +import org.apache.james.mailbox.postgres.PostgresMessageId; +import org.junit.jupiter.api.extension.RegisterExtension; + +public class PostgresEmailQueryViewTest implements EmailQueryViewContract { + public static final PostgresMailboxId MAILBOX_ID_1 = PostgresMailboxId.generate(); + public static final PostgresMessageId.Factory MESSAGE_ID_FACTORY = new PostgresMessageId.Factory(); + public static final PostgresMessageId MESSAGE_ID_1 = MESSAGE_ID_FACTORY.generate(); + public static final PostgresMessageId MESSAGE_ID_2 = MESSAGE_ID_FACTORY.generate(); + public static final PostgresMessageId MESSAGE_ID_3 = MESSAGE_ID_FACTORY.generate(); + public static final PostgresMessageId MESSAGE_ID_4 = MESSAGE_ID_FACTORY.generate(); + + @RegisterExtension + static PostgresExtension postgresExtension = PostgresExtension.withoutRowLevelSecurity(PostgresEmailQueryViewModule.MODULE); + + @Override + public EmailQueryView testee() { + return new PostgresEmailQueryView(new PostgresEmailQueryViewDAO(postgresExtension.getDefaultPostgresExecutor())); + } + + @Override + public MailboxId mailboxId1() { + return MAILBOX_ID_1; + } + + @Override + public MessageId messageId1() { + return MESSAGE_ID_1; + } + + @Override + public MessageId messageId2() { + return MESSAGE_ID_2; + } + + @Override + public MessageId messageId3() { + return MESSAGE_ID_3; + } + + @Override + public MessageId messageId4() { + return MESSAGE_ID_4; + } +} diff --git a/server/data/data-jmap-postgres/src/test/java/org/apache/james/jmap/postgres/projections/PostgresMessageFastViewProjectionTest.java b/server/data/data-jmap-postgres/src/test/java/org/apache/james/jmap/postgres/projections/PostgresMessageFastViewProjectionTest.java new file mode 100644 index 00000000000..d436cb6677d --- /dev/null +++ b/server/data/data-jmap-postgres/src/test/java/org/apache/james/jmap/postgres/projections/PostgresMessageFastViewProjectionTest.java @@ -0,0 +1,62 @@ +/**************************************************************** + * 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.jmap.postgres.projections; + +import org.apache.james.backends.postgres.PostgresExtension; +import org.apache.james.backends.postgres.PostgresModule; +import org.apache.james.jmap.api.projections.MessageFastViewProjection; +import org.apache.james.jmap.api.projections.MessageFastViewProjectionContract; +import org.apache.james.mailbox.model.MessageId; +import org.apache.james.mailbox.postgres.PostgresMessageId; +import org.apache.james.metrics.tests.RecordingMetricFactory; +import org.junit.jupiter.api.BeforeEach; +import org.junit.jupiter.api.extension.RegisterExtension; + +class PostgresMessageFastViewProjectionTest implements MessageFastViewProjectionContract { + @RegisterExtension + static PostgresExtension postgresExtension = PostgresExtension.withoutRowLevelSecurity( + PostgresModule.aggregateModules(PostgresMessageFastViewProjectionModule.MODULE)); + + private PostgresMessageFastViewProjection testee; + private PostgresMessageId.Factory postgresMessageIdFactory; + private RecordingMetricFactory metricFactory; + + @BeforeEach + void setUp() { + metricFactory = new RecordingMetricFactory(); + postgresMessageIdFactory = new PostgresMessageId.Factory(); + testee = new PostgresMessageFastViewProjection(postgresExtension.getDefaultPostgresExecutor(), metricFactory); + } + + @Override + public MessageFastViewProjection testee() { + return testee; + } + + @Override + public MessageId newMessageId() { + return postgresMessageIdFactory.generate(); + } + + @Override + public RecordingMetricFactory metricFactory() { + return metricFactory; + } +} \ No newline at end of file diff --git a/server/data/data-jmap-postgres/src/test/java/org/apache/james/jmap/postgres/pushsubscription/PostgresPushSubscriptionRepositoryTest.java b/server/data/data-jmap-postgres/src/test/java/org/apache/james/jmap/postgres/pushsubscription/PostgresPushSubscriptionRepositoryTest.java new file mode 100644 index 00000000000..7a471569dfb --- /dev/null +++ b/server/data/data-jmap-postgres/src/test/java/org/apache/james/jmap/postgres/pushsubscription/PostgresPushSubscriptionRepositoryTest.java @@ -0,0 +1,62 @@ +/**************************************************************** + * 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.jmap.postgres.pushsubscription; + +import java.util.Set; + +import org.apache.james.backends.postgres.PostgresExtension; +import org.apache.james.backends.postgres.PostgresModule; +import org.apache.james.jmap.api.change.TypeStateFactory; +import org.apache.james.jmap.api.model.TypeName; +import org.apache.james.jmap.api.pushsubscription.PushSubscriptionRepository; +import org.apache.james.jmap.api.pushsubscription.PushSubscriptionRepositoryContract; +import org.apache.james.utils.UpdatableTickingClock; +import org.junit.jupiter.api.BeforeEach; +import org.junit.jupiter.api.extension.RegisterExtension; + +import scala.jdk.javaapi.CollectionConverters; + +class PostgresPushSubscriptionRepositoryTest implements PushSubscriptionRepositoryContract { + + @RegisterExtension + static PostgresExtension postgresExtension = PostgresExtension.withRowLevelSecurity( + PostgresModule.aggregateModules(PostgresPushSubscriptionModule.MODULE)); + + UpdatableTickingClock clock; + PushSubscriptionRepository pushSubscriptionRepository; + + @BeforeEach + void setup() { + clock = new UpdatableTickingClock(PushSubscriptionRepositoryContract.NOW()); + pushSubscriptionRepository = new PostgresPushSubscriptionRepository(clock, + new TypeStateFactory((Set) CollectionConverters.asJava(PushSubscriptionRepositoryContract.TYPE_NAME_SET())), + postgresExtension.getExecutorFactory()); + } + + @Override + public UpdatableTickingClock clock() { + return clock; + } + + @Override + public PushSubscriptionRepository testee() { + return pushSubscriptionRepository; + } +} diff --git a/server/data/data-jmap-postgres/src/test/java/org/apache/james/jmap/postgres/upload/PostgresUploadRepositoryTest.java b/server/data/data-jmap-postgres/src/test/java/org/apache/james/jmap/postgres/upload/PostgresUploadRepositoryTest.java new file mode 100644 index 00000000000..9cf3a9f9843 --- /dev/null +++ b/server/data/data-jmap-postgres/src/test/java/org/apache/james/jmap/postgres/upload/PostgresUploadRepositoryTest.java @@ -0,0 +1,65 @@ +/**************************************************************** + * 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.jmap.postgres.upload; + +import java.time.Clock; + +import org.apache.james.backends.postgres.PostgresExtension; +import org.apache.james.backends.postgres.PostgresModule; +import org.apache.james.blob.api.BlobId; +import org.apache.james.blob.api.BlobStore; +import org.apache.james.blob.api.BucketName; +import org.apache.james.blob.api.PlainBlobId; +import org.apache.james.blob.memory.MemoryBlobStoreDAO; +import org.apache.james.jmap.api.upload.UploadRepository; +import org.apache.james.jmap.api.upload.UploadRepositoryContract; +import org.apache.james.server.blob.deduplication.DeDuplicationBlobStore; +import org.apache.james.utils.UpdatableTickingClock; +import org.junit.jupiter.api.BeforeEach; +import org.junit.jupiter.api.extension.RegisterExtension; + +class PostgresUploadRepositoryTest implements UploadRepositoryContract { + + @RegisterExtension + static PostgresExtension postgresExtension = PostgresExtension.withoutRowLevelSecurity( + PostgresModule.aggregateModules(PostgresUploadModule.MODULE)); + private UploadRepository testee; + private UpdatableTickingClock clock; + + @BeforeEach + void setUp() { + clock = new UpdatableTickingClock(Clock.systemUTC().instant()); + BlobId.Factory blobIdFactory = new PlainBlobId.Factory(); + BlobStore blobStore = new DeDuplicationBlobStore(new MemoryBlobStoreDAO(), BucketName.DEFAULT, blobIdFactory); + PostgresUploadDAO uploadDAO = new PostgresUploadDAO(postgresExtension.getDefaultPostgresExecutor(), blobIdFactory); + PostgresUploadDAO.Factory uploadFactory = new PostgresUploadDAO.Factory(blobIdFactory, postgresExtension.getExecutorFactory()); + testee = new PostgresUploadRepository(blobStore, clock, uploadFactory, uploadDAO); + } + + @Override + public UploadRepository testee() { + return testee; + } + + @Override + public UpdatableTickingClock clock() { + return clock; + } +} diff --git a/server/data/data-jmap-postgres/src/test/java/org/apache/james/jmap/postgres/upload/PostgresUploadServiceTest.java b/server/data/data-jmap-postgres/src/test/java/org/apache/james/jmap/postgres/upload/PostgresUploadServiceTest.java new file mode 100644 index 00000000000..4cd2fe07bc2 --- /dev/null +++ b/server/data/data-jmap-postgres/src/test/java/org/apache/james/jmap/postgres/upload/PostgresUploadServiceTest.java @@ -0,0 +1,79 @@ +/**************************************************************** + * 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.jmap.postgres.upload; + +import java.time.Clock; + +import org.apache.james.backends.postgres.PostgresExtension; +import org.apache.james.backends.postgres.PostgresModule; +import org.apache.james.backends.postgres.quota.PostgresQuotaCurrentValueDAO; +import org.apache.james.backends.postgres.quota.PostgresQuotaModule; +import org.apache.james.blob.api.BlobId; +import org.apache.james.blob.api.BlobStore; +import org.apache.james.blob.api.BucketName; +import org.apache.james.blob.api.PlainBlobId; +import org.apache.james.blob.memory.MemoryBlobStoreDAO; +import org.apache.james.jmap.api.upload.UploadRepository; +import org.apache.james.jmap.api.upload.UploadService; +import org.apache.james.jmap.api.upload.UploadServiceContract; +import org.apache.james.jmap.api.upload.UploadServiceDefaultImpl; +import org.apache.james.jmap.api.upload.UploadUsageRepository; +import org.apache.james.server.blob.deduplication.DeDuplicationBlobStore; +import org.junit.jupiter.api.BeforeEach; +import org.junit.jupiter.api.extension.RegisterExtension; + +public class PostgresUploadServiceTest implements UploadServiceContract { + + @RegisterExtension + static PostgresExtension postgresExtension = PostgresExtension.withoutRowLevelSecurity( + PostgresModule.aggregateModules(PostgresUploadModule.MODULE, PostgresQuotaModule.MODULE)); + + private PostgresUploadRepository uploadRepository; + private PostgresUploadUsageRepository uploadUsageRepository; + private UploadService testee; + + @BeforeEach + void setUp() { + BlobId.Factory blobIdFactory = new PlainBlobId.Factory(); + BlobStore blobStore = new DeDuplicationBlobStore(new MemoryBlobStoreDAO(), BucketName.DEFAULT, blobIdFactory); + PostgresUploadDAO uploadDAO = new PostgresUploadDAO(postgresExtension.getDefaultPostgresExecutor(), blobIdFactory); + PostgresUploadDAO.Factory uploadFactory = new PostgresUploadDAO.Factory(blobIdFactory, postgresExtension.getExecutorFactory()); + uploadRepository = new PostgresUploadRepository(blobStore, Clock.systemUTC(), uploadFactory, uploadDAO); + uploadUsageRepository = new PostgresUploadUsageRepository(new PostgresQuotaCurrentValueDAO(postgresExtension.getDefaultPostgresExecutor())); + testee = new UploadServiceDefaultImpl(uploadRepository, uploadUsageRepository, UploadServiceContract.TEST_CONFIGURATION()); + } + + @Override + public UploadRepository uploadRepository() { + return uploadRepository; + } + + @Override + public UploadUsageRepository uploadUsageRepository() { + return uploadUsageRepository; + } + + @Override + public UploadService testee() { + return testee; + } + + +} diff --git a/server/data/data-jmap-postgres/src/test/java/org/apache/james/jmap/postgres/upload/PostgresUploadUsageRepositoryTest.java b/server/data/data-jmap-postgres/src/test/java/org/apache/james/jmap/postgres/upload/PostgresUploadUsageRepositoryTest.java new file mode 100644 index 00000000000..29aefe323d3 --- /dev/null +++ b/server/data/data-jmap-postgres/src/test/java/org/apache/james/jmap/postgres/upload/PostgresUploadUsageRepositoryTest.java @@ -0,0 +1,49 @@ +/**************************************************************** + * 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.jmap.postgres.upload; + +import org.apache.james.backends.postgres.PostgresExtension; +import org.apache.james.backends.postgres.PostgresModule; +import org.apache.james.backends.postgres.quota.PostgresQuotaCurrentValueDAO; +import org.apache.james.backends.postgres.quota.PostgresQuotaModule; +import org.apache.james.jmap.api.upload.UploadUsageRepository; +import org.apache.james.jmap.api.upload.UploadUsageRepositoryContract; +import org.junit.jupiter.api.BeforeEach; +import org.junit.jupiter.api.extension.RegisterExtension; + +public class PostgresUploadUsageRepositoryTest implements UploadUsageRepositoryContract { + + @RegisterExtension + static PostgresExtension postgresExtension = PostgresExtension.withoutRowLevelSecurity( + PostgresModule.aggregateModules(PostgresUploadModule.MODULE, PostgresQuotaModule.MODULE)); + + private PostgresUploadUsageRepository uploadUsageRepository; + + @BeforeEach + public void setup() { + uploadUsageRepository = new PostgresUploadUsageRepository(new PostgresQuotaCurrentValueDAO(postgresExtension.getDefaultPostgresExecutor())); + resetCounterToZero(); + } + + @Override + public UploadUsageRepository uploadUsageRepository() { + return uploadUsageRepository; + } +} diff --git a/server/data/data-jmap/src/main/java/org/apache/james/jmap/api/filtering/FilteringIncrementalRuleChangeDTO.java b/server/data/data-jmap/src/main/java/org/apache/james/jmap/api/filtering/FilteringIncrementalRuleChangeDTO.java index a1475cf63be..4084226b23a 100644 --- a/server/data/data-jmap/src/main/java/org/apache/james/jmap/api/filtering/FilteringIncrementalRuleChangeDTO.java +++ b/server/data/data-jmap/src/main/java/org/apache/james/jmap/api/filtering/FilteringIncrementalRuleChangeDTO.java @@ -23,6 +23,8 @@ import org.apache.james.eventsourcing.EventId; import org.apache.james.eventsourcing.eventstore.dto.EventDTO; +import org.apache.james.jmap.api.filtering.Rule; +import org.apache.james.jmap.api.filtering.RuleDTO; import org.apache.james.jmap.api.filtering.impl.FilteringAggregateId; import org.apache.james.jmap.api.filtering.impl.IncrementalRuleChange; diff --git a/server/data/data-jmap/src/main/java/org/apache/james/jmap/api/filtering/FilteringRuleSetDefinedDTO.java b/server/data/data-jmap/src/main/java/org/apache/james/jmap/api/filtering/FilteringRuleSetDefinedDTO.java index 73ed1a9805e..c0856c72ab9 100644 --- a/server/data/data-jmap/src/main/java/org/apache/james/jmap/api/filtering/FilteringRuleSetDefinedDTO.java +++ b/server/data/data-jmap/src/main/java/org/apache/james/jmap/api/filtering/FilteringRuleSetDefinedDTO.java @@ -23,6 +23,7 @@ import org.apache.james.eventsourcing.EventId; import org.apache.james.eventsourcing.eventstore.dto.EventDTO; +import org.apache.james.jmap.api.filtering.RuleDTO; import org.apache.james.jmap.api.filtering.impl.FilteringAggregateId; import org.apache.james.jmap.api.filtering.impl.RuleSetDefined; diff --git a/server/data/data-jmap/src/main/java/org/apache/james/jmap/api/projections/DefaultEmailQueryViewManager.java b/server/data/data-jmap/src/main/java/org/apache/james/jmap/api/projections/DefaultEmailQueryViewManager.java new file mode 100644 index 00000000000..2870158c3a9 --- /dev/null +++ b/server/data/data-jmap/src/main/java/org/apache/james/jmap/api/projections/DefaultEmailQueryViewManager.java @@ -0,0 +1,38 @@ +/**************************************************************** + * 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.jmap.api.projections; + +import jakarta.inject.Inject; + +import org.apache.james.core.Username; + +public class DefaultEmailQueryViewManager implements EmailQueryViewManager { + private EmailQueryView emailQueryView; + + @Inject + public DefaultEmailQueryViewManager(EmailQueryView emailQueryView) { + this.emailQueryView = emailQueryView; + } + + @Override + public EmailQueryView getEmailQueryView(Username username) { + return emailQueryView; + } +} diff --git a/server/data/data-jmap/src/main/java/org/apache/james/jmap/api/projections/EmailQueryViewManager.java b/server/data/data-jmap/src/main/java/org/apache/james/jmap/api/projections/EmailQueryViewManager.java new file mode 100644 index 00000000000..e4a281829e9 --- /dev/null +++ b/server/data/data-jmap/src/main/java/org/apache/james/jmap/api/projections/EmailQueryViewManager.java @@ -0,0 +1,26 @@ +/**************************************************************** + * 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.jmap.api.projections; + +import org.apache.james.core.Username; + +public interface EmailQueryViewManager { + EmailQueryView getEmailQueryView(Username username); +} diff --git a/server/data/data-jmap/src/main/java/org/apache/james/jmap/api/projections/MessageFastViewProjection.java b/server/data/data-jmap/src/main/java/org/apache/james/jmap/api/projections/MessageFastViewProjection.java index a0c54e5ad05..c5f0ba15971 100644 --- a/server/data/data-jmap/src/main/java/org/apache/james/jmap/api/projections/MessageFastViewProjection.java +++ b/server/data/data-jmap/src/main/java/org/apache/james/jmap/api/projections/MessageFastViewProjection.java @@ -28,6 +28,7 @@ import org.apache.james.mailbox.model.MessageId; import org.reactivestreams.Publisher; +import com.google.common.annotations.VisibleForTesting; import com.google.common.base.Preconditions; import reactor.core.publisher.Flux; @@ -45,6 +46,7 @@ public interface MessageFastViewProjection { Publisher delete(MessageId messageId); + @VisibleForTesting Publisher clear(); default Publisher> retrieve(Collection messageIds) { diff --git a/server/data/data-jmap/src/main/java/org/apache/james/jmap/api/upload/UploadRepository.java b/server/data/data-jmap/src/main/java/org/apache/james/jmap/api/upload/UploadRepository.java index 2d130b087bd..60c7d207acb 100644 --- a/server/data/data-jmap/src/main/java/org/apache/james/jmap/api/upload/UploadRepository.java +++ b/server/data/data-jmap/src/main/java/org/apache/james/jmap/api/upload/UploadRepository.java @@ -20,6 +20,7 @@ package org.apache.james.jmap.api.upload; import java.io.InputStream; +import java.time.Duration; import org.apache.james.core.Username; import org.apache.james.jmap.api.model.Upload; @@ -36,5 +37,7 @@ public interface UploadRepository { Publisher delete(UploadId id, Username user); Publisher listUploads(Username user); + + Publisher deleteByUploadDateBefore(Duration expireDuration); } diff --git a/server/data/data-jmap/src/main/java/org/apache/james/jmap/memory/upload/InMemoryUploadRepository.java b/server/data/data-jmap/src/main/java/org/apache/james/jmap/memory/upload/InMemoryUploadRepository.java index ae4bce2908e..9823214f98b 100644 --- a/server/data/data-jmap/src/main/java/org/apache/james/jmap/memory/upload/InMemoryUploadRepository.java +++ b/server/data/data-jmap/src/main/java/org/apache/james/jmap/memory/upload/InMemoryUploadRepository.java @@ -22,8 +22,10 @@ import java.io.ByteArrayInputStream; import java.io.InputStream; import java.time.Clock; +import java.time.Duration; import java.time.Instant; import java.util.HashMap; +import java.util.List; import java.util.Map; import jakarta.inject.Inject; @@ -109,6 +111,16 @@ public Publisher listUploads(Username user) { .map(pair -> pair.right); } + @Override + public Publisher deleteByUploadDateBefore(Duration expireDuration) { + Instant expirationTime = clock.instant().minus(expireDuration); + return Flux.fromIterable(List.copyOf(uploadStore.values())) + .filter(pair -> pair.right.uploadDate().isBefore(expirationTime)) + .flatMap(pair -> Mono.from(blobStore.delete(bucketName, pair.right.blobId())) + .then(Mono.fromRunnable(() -> uploadStore.remove(pair.right.uploadId())))) + .then(); + } + private Mono retrieveUpload(UploadMetaData uploadMetaData) { return Mono.from(blobStore.readBytes(bucketName, uploadMetaData.blobId())) .map(content -> Upload.from(uploadMetaData, () -> new ByteArrayInputStream(content))); diff --git a/server/data/data-jmap/src/test/java/org/apache/james/jmap/api/projections/EmailQueryViewContract.java b/server/data/data-jmap/src/test/java/org/apache/james/jmap/api/projections/EmailQueryViewContract.java index 4e1b3abb4ea..ac99142ec40 100644 --- a/server/data/data-jmap/src/test/java/org/apache/james/jmap/api/projections/EmailQueryViewContract.java +++ b/server/data/data-jmap/src/test/java/org/apache/james/jmap/api/projections/EmailQueryViewContract.java @@ -200,7 +200,7 @@ default void datesCanBeDuplicated() { testee().save(mailboxId1(), DATE_1, DATE_2, messageId2()).block(); assertThat(testee().listMailboxContentSortedBySentAt(mailboxId1(), Limit.limit(12)).collectList().block()) - .containsExactly(messageId1(), messageId2()); + .containsExactlyInAnyOrder(messageId1(), messageId2()); } @Test diff --git a/server/data/data-jmap/src/test/java/org/apache/james/jmap/api/upload/UploadRepositoryContract.scala b/server/data/data-jmap/src/test/java/org/apache/james/jmap/api/upload/UploadRepositoryContract.scala index c3993fa4421..f5444089681 100644 --- a/server/data/data-jmap/src/test/java/org/apache/james/jmap/api/upload/UploadRepositoryContract.scala +++ b/server/data/data-jmap/src/test/java/org/apache/james/jmap/api/upload/UploadRepositoryContract.scala @@ -21,6 +21,7 @@ import java.io.InputStream import java.nio.charset.StandardCharsets + import java.time.{Clock, Duration} import java.util.UUID import org.apache.commons.io.IOUtils @@ -29,6 +30,7 @@ import org.apache.james.jmap.api.model.{Upload, UploadId, UploadMetaData, UploadNotFoundException} import org.apache.james.jmap.api.upload.UploadRepositoryContract.{CONTENT_TYPE, DATA_STRING, USER} import org.apache.james.mailbox.model.ContentType + import org.apache.james.utils.UpdatableTickingClock import org.assertj.core.api.Assertions.{assertThat, assertThatCode, assertThatThrownBy} import org.assertj.core.groups.Tuple.tuple import org.junit.jupiter.api.Test @@ -49,6 +51,8 @@ def testee: UploadRepository + def clock: UpdatableTickingClock + def data(): InputStream = IOUtils.toInputStream(DATA_STRING, StandardCharsets.UTF_8) @Test @@ -201,4 +205,17 @@ assertThat(SMono.fromPublisher(testee.delete(uploadIdOfAlice, Username.of("Bob"))).block()).isFalse } + @Test + def deleteByUploadDateBeforeShouldRemoveExpiredUploads(): Unit = { + val uploadId1: UploadId = SMono.fromPublisher(testee.upload(data(), CONTENT_TYPE, USER)).block().uploadId + clock.setInstant(clock.instant().plus(8, java.time.temporal.ChronoUnit.DAYS)) + val uploadId2: UploadId = SMono.fromPublisher(testee.upload(data(), CONTENT_TYPE, USER)).block().uploadId + + SMono(testee.deleteByUploadDateBefore(Duration.ofDays(7))).block(); + + assertThatThrownBy(() => SMono.fromPublisher(testee.retrieve(uploadId1, USER)).block()) + .isInstanceOf(classOf[UploadNotFoundException]) + assertThat(SMono.fromPublisher(testee.retrieve(uploadId2, USER)).block()) + .isNotNull + } } diff --git a/server/data/data-jmap/src/test/java/org/apache/james/jmap/memory/upload/InMemoryUploadRepositoryTest.java b/server/data/data-jmap/src/test/java/org/apache/james/jmap/memory/upload/InMemoryUploadRepositoryTest.java index e949525ac2a..d1f37e01ce5 100644 --- a/server/data/data-jmap/src/test/java/org/apache/james/jmap/memory/upload/InMemoryUploadRepositoryTest.java +++ b/server/data/data-jmap/src/test/java/org/apache/james/jmap/memory/upload/InMemoryUploadRepositoryTest.java @@ -28,20 +28,28 @@ import org.apache.james.jmap.api.upload.UploadRepository; import org.apache.james.jmap.api.upload.UploadRepositoryContract; import org.apache.james.server.blob.deduplication.DeDuplicationBlobStore; +import org.apache.james.utils.UpdatableTickingClock; import org.junit.jupiter.api.BeforeEach; public class InMemoryUploadRepositoryTest implements UploadRepositoryContract { private UploadRepository testee; + private UpdatableTickingClock clock; @BeforeEach void setUp() { BlobStore blobStore = new DeDuplicationBlobStore(new MemoryBlobStoreDAO(), BucketName.DEFAULT, new PlainBlobId.Factory()); - testee = new InMemoryUploadRepository(blobStore, Clock.systemUTC()); + clock = new UpdatableTickingClock(Clock.systemUTC().instant()); + testee = new InMemoryUploadRepository(blobStore, clock); } @Override public UploadRepository testee() { return testee; } + + @Override + public UpdatableTickingClock clock() { + return clock; + } } diff --git a/server/data/data-jmap/src/test/scala/org/apache/james/jmap/api/pushsubscription/PushSubscriptionRepositoryContract.scala b/server/data/data-jmap/src/test/scala/org/apache/james/jmap/api/pushsubscription/PushSubscriptionRepositoryContract.scala index 78a013f5ae0..7b2caba10c4 100644 --- a/server/data/data-jmap/src/test/scala/org/apache/james/jmap/api/pushsubscription/PushSubscriptionRepositoryContract.scala +++ b/server/data/data-jmap/src/test/scala/org/apache/james/jmap/api/pushsubscription/PushSubscriptionRepositoryContract.scala @@ -450,5 +450,24 @@ trait PushSubscriptionRepositoryContract { .isInstanceOf(classOf[InvalidPushSubscriptionKeys]) } + @Test + def updateShouldUpdateCorrectOffsetDateTime(): Unit = { + val validRequest = PushSubscriptionCreationRequest( + deviceClientId = DeviceClientId("1"), + url = PushSubscriptionServerURL(new URL("https://example.com/push")), + types = Seq(CustomTypeName1)) + + val pushSubscriptionId = SMono.fromPublisher(testee.save(ALICE, validRequest)).block().id + + val ZONE_ID: ZoneId = ZoneId.of("Europe/Paris") + val CLOCK: Clock = Clock.fixed(Instant.parse("2021-10-25T07:05:39.160Z"), ZONE_ID) + + val zonedDateTime: ZonedDateTime = ZonedDateTime.now(CLOCK) + SMono.fromPublisher(testee.updateExpireTime(ALICE, pushSubscriptionId, zonedDateTime)).block() + + val updatedSubscription = SFlux.fromPublisher(testee.get(ALICE, Set(pushSubscriptionId).asJava)).blockFirst().get + assertThat(updatedSubscription.expires.value).isEqualTo(zonedDateTime) + } + } diff --git a/server/data/data-jmap/src/test/scala/org/apache/james/jmap/api/upload/UploadServiceContract.scala b/server/data/data-jmap/src/test/scala/org/apache/james/jmap/api/upload/UploadServiceContract.scala index 82ab4495742..fcf17059b20 100644 --- a/server/data/data-jmap/src/test/scala/org/apache/james/jmap/api/upload/UploadServiceContract.scala +++ b/server/data/data-jmap/src/test/scala/org/apache/james/jmap/api/upload/UploadServiceContract.scala @@ -108,6 +108,7 @@ trait UploadServiceContract { .block()) // Exceed 100 bytes limit + Thread.sleep(500) SMono.fromPublisher(testee.upload(asInputStream(TEN_BYTES_DATA_STRING), CONTENT_TYPE, BOB)) .block() @@ -126,6 +127,7 @@ trait UploadServiceContract { .block()) // Exceed 100 bytes limit + Thread.sleep(500) SMono.fromPublisher(testee.upload(asInputStream(TEN_BYTES_DATA_STRING), CONTENT_TYPE, BOB)) .block() @@ -146,6 +148,7 @@ trait UploadServiceContract { .block() // Exceed 100 bytes limit + Thread.sleep(500) SMono.fromPublisher(testee.upload(asInputStream(TEN_BYTES_DATA_STRING), CONTENT_TYPE, BOB)) .block() @@ -173,6 +176,7 @@ trait UploadServiceContract { .block()) // Exceed 100 bytes limit + Thread.sleep(500) SMono.fromPublisher(testee.upload(asInputStream(TEN_BYTES_DATA_STRING), CONTENT_TYPE, BOB)) .block() @@ -192,6 +196,7 @@ trait UploadServiceContract { SMono(uploadUsageRepository.resetSpace(BOB, QuotaSizeUsage.size(105L))).block() // Exceed 100 bytes limit + Thread.sleep(500) SMono.fromPublisher(testee.upload(asInputStream(TEN_BYTES_DATA_STRING), CONTENT_TYPE, BOB)).block() // The current stored usage should be eventually consistent diff --git a/server/data/data-library/src/main/java/org/apache/james/user/lib/UsersRepositoryImpl.java b/server/data/data-library/src/main/java/org/apache/james/user/lib/UsersRepositoryImpl.java index 74bf60bf0dd..b27bc3bc3d4 100644 --- a/server/data/data-library/src/main/java/org/apache/james/user/lib/UsersRepositoryImpl.java +++ b/server/data/data-library/src/main/java/org/apache/james/user/lib/UsersRepositoryImpl.java @@ -87,6 +87,8 @@ public void configure(HierarchicalConfiguration configuration) th verifyFailureDelay = Optional.ofNullable(configuration.getString("verifyFailureDelay")) .map(string -> DurationParser.parse(string, ChronoUnit.SECONDS).toMillis()) .orElse(0L); + LOGGER.debug("Init configure users repository with virtualHosting {}, verifyFailureDelay {}", + virtualHosting, verifyFailureDelay); } private Optional> parseAdministratorId(HierarchicalConfiguration configuration) { diff --git a/server/data/data-library/src/test/java/org/apache/james/sieverepository/lib/SieveRepositoryContract.java b/server/data/data-library/src/test/java/org/apache/james/sieverepository/lib/SieveRepositoryContract.java index 97f58414231..91531749a60 100644 --- a/server/data/data-library/src/test/java/org/apache/james/sieverepository/lib/SieveRepositoryContract.java +++ b/server/data/data-library/src/test/java/org/apache/james/sieverepository/lib/SieveRepositoryContract.java @@ -185,6 +185,17 @@ default void setActiveScriptShouldThrowOnNonExistentScript() { .isInstanceOf(ScriptNotFoundException.class); } + @Test + default void setActiveScriptOnNonExistingScriptShouldNotDeactivateTheCurrentActiveScript() throws Exception { + sieveRepository().putScript(USERNAME, SCRIPT_NAME, SCRIPT_CONTENT); + sieveRepository().setActive(USERNAME, SCRIPT_NAME); + + assertThatThrownBy(() -> sieveRepository().setActive(USERNAME, OTHER_SCRIPT_NAME)) + .isInstanceOf(ScriptNotFoundException.class); + + assertThat(getScriptContent(sieveRepository().getActive(USERNAME))).isEqualTo(SCRIPT_CONTENT); + } + @Test default void setActiveScriptShouldWork() throws Exception { sieveRepository().putScript(USERNAME, SCRIPT_NAME, SCRIPT_CONTENT); diff --git a/server/data/data-memory/src/main/java/org/apache/james/vacation/memory/MemoryNotificationRegistry.java b/server/data/data-memory/src/main/java/org/apache/james/vacation/memory/MemoryNotificationRegistry.java index eb10cff902c..36b449d5777 100644 --- a/server/data/data-memory/src/main/java/org/apache/james/vacation/memory/MemoryNotificationRegistry.java +++ b/server/data/data-memory/src/main/java/org/apache/james/vacation/memory/MemoryNotificationRegistry.java @@ -84,7 +84,7 @@ public Mono isRegistered(AccountId accountId, RecipientId recipientId) } private boolean isStrictlyBefore(ZonedDateTime currentTime, ZonedDateTime registrationEnd) { - return ! currentTime.isAfter(registrationEnd); + return currentTime.isBefore(registrationEnd); } @Override diff --git a/server/data/data-postgres/pom.xml b/server/data/data-postgres/pom.xml new file mode 100644 index 00000000000..be376372532 --- /dev/null +++ b/server/data/data-postgres/pom.xml @@ -0,0 +1,154 @@ + + + 4.0.0 + + org.apache.james + james-server + 3.9.0-SNAPSHOT + ../../pom.xml + + + james-server-data-postgres + Apache James :: Server :: Data :: Postgres + + + + ${james.groupId} + apache-james-backends-postgres + + + ${james.groupId} + apache-james-backends-postgres + test-jar + test + + + ${james.groupId} + blob-api + test-jar + test + + + ${james.groupId} + blob-memory + test + + + ${james.groupId} + james-server-core + + + ${james.groupId} + james-server-data-api + + + ${james.groupId} + james-server-data-api + test-jar + test + + + ${james.groupId} + james-server-data-library + + + ${james.groupId} + james-server-data-library + test-jar + test + + + ${james.groupId} + james-server-dnsservice-api + + + ${james.groupId} + james-server-dnsservice-test + test + + + ${james.groupId} + james-server-guice-common + test-jar + test + + + ${james.groupId} + james-server-lifecycle-api + + + ${james.groupId} + james-server-mail-store + + + ${james.groupId} + james-server-mailrepository-api + + + ${james.groupId} + james-server-mailrepository-api + test-jar + test + + + ${james.groupId} + james-server-testing + test + + + ${james.groupId} + testing-base + test + + + com.google.guava + guava + + + io.cucumber + cucumber-java + test + + + io.cucumber + cucumber-junit-platform-engine + test + + + io.cucumber + cucumber-picocontainer + test + + + org.apache.commons + commons-configuration2 + + + org.junit.platform + junit-platform-suite + test + + + org.mockito + mockito-core + test + + + org.slf4j + jcl-over-slf4j + + + org.slf4j + log4j-over-slf4j + + + org.slf4j + slf4j-api + + + org.testcontainers + postgresql + test + + + diff --git a/server/data/data-postgres/src/main/java/org/apache/james/domainlist/postgres/PostgresDomainList.java b/server/data/data-postgres/src/main/java/org/apache/james/domainlist/postgres/PostgresDomainList.java new file mode 100644 index 00000000000..dcdfb3e2a31 --- /dev/null +++ b/server/data/data-postgres/src/main/java/org/apache/james/domainlist/postgres/PostgresDomainList.java @@ -0,0 +1,88 @@ +/**************************************************************** + * 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.domainlist.postgres; + +import static org.apache.james.backends.postgres.utils.PostgresExecutor.DEFAULT_INJECT; +import static org.apache.james.domainlist.postgres.PostgresDomainModule.PostgresDomainTable.DOMAIN; + +import java.util.List; + +import jakarta.inject.Inject; +import jakarta.inject.Named; + +import org.apache.james.backends.postgres.utils.PostgresExecutor; +import org.apache.james.core.Domain; +import org.apache.james.dnsservice.api.DNSService; +import org.apache.james.domainlist.api.DomainListException; +import org.apache.james.domainlist.lib.AbstractDomainList; +import org.jooq.exception.DataAccessException; + +import reactor.core.publisher.Flux; +import reactor.core.publisher.Mono; + +public class PostgresDomainList extends AbstractDomainList { + private final PostgresExecutor postgresExecutor; + + @Inject + public PostgresDomainList(DNSService dnsService, @Named(DEFAULT_INJECT) PostgresExecutor postgresExecutor) { + super(dnsService); + this.postgresExecutor = postgresExecutor; + } + + @Override + public void addDomain(Domain domain) throws DomainListException { + try { + postgresExecutor.executeVoid(dslContext -> Mono.from(dslContext.insertInto(PostgresDomainModule.PostgresDomainTable.TABLE_NAME, DOMAIN) + .values(domain.asString()))) + .block(); + } catch (DataAccessException exception) { + throw new DomainListException(domain.name() + " already exists."); + } + } + + @Override + protected List getDomainListInternal() { + return postgresExecutor.executeRows(dsl -> Flux.from(dsl.selectFrom(PostgresDomainModule.PostgresDomainTable.TABLE_NAME))) + .map(record -> Domain.of(record.get(DOMAIN))) + .collectList() + .block(); + } + + @Override + protected boolean containsDomainInternal(Domain domain) { + return postgresExecutor.executeRow(dsl -> Mono.from(dsl.selectFrom(PostgresDomainModule.PostgresDomainTable.TABLE_NAME) + .where(DOMAIN.eq(domain.asString())))) + .blockOptional() + .isPresent(); + } + + @Override + protected void doRemoveDomain(Domain domain) throws DomainListException { + boolean executed = postgresExecutor.executeRow(dslContext -> Mono.from(dslContext.deleteFrom(PostgresDomainModule.PostgresDomainTable.TABLE_NAME) + .where(DOMAIN.eq(domain.asString())) + .returning(DOMAIN))) + .blockOptional() + .isPresent(); + + if (!executed) { + throw new DomainListException(domain.name() + " was not found"); + } + } +} diff --git a/server/data/data-postgres/src/main/java/org/apache/james/domainlist/postgres/PostgresDomainModule.java b/server/data/data-postgres/src/main/java/org/apache/james/domainlist/postgres/PostgresDomainModule.java new file mode 100644 index 00000000000..f0a14669175 --- /dev/null +++ b/server/data/data-postgres/src/main/java/org/apache/james/domainlist/postgres/PostgresDomainModule.java @@ -0,0 +1,47 @@ +/**************************************************************** + * 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.domainlist.postgres; + +import org.apache.james.backends.postgres.PostgresModule; +import org.apache.james.backends.postgres.PostgresTable; +import org.jooq.Field; +import org.jooq.Record; +import org.jooq.Table; +import org.jooq.impl.DSL; +import org.jooq.impl.SQLDataType; + +public interface PostgresDomainModule { + interface PostgresDomainTable { + Table TABLE_NAME = DSL.table("domains"); + + Field DOMAIN = DSL.field("domain", SQLDataType.VARCHAR.notNull()); + + PostgresTable TABLE = PostgresTable.name(TABLE_NAME.getName()) + .createTableStep(((dsl, tableName) -> dsl.createTableIfNotExists(tableName) + .column(DOMAIN) + .primaryKey(DOMAIN))) + .disableRowLevelSecurity() + .build(); + } + + PostgresModule MODULE = PostgresModule.builder() + .addTable(PostgresDomainTable.TABLE) + .build(); +} diff --git a/server/data/data-postgres/src/main/java/org/apache/james/droplists/postgres/PostgresDropList.java b/server/data/data-postgres/src/main/java/org/apache/james/droplists/postgres/PostgresDropList.java new file mode 100644 index 00000000000..ff46ee735c5 --- /dev/null +++ b/server/data/data-postgres/src/main/java/org/apache/james/droplists/postgres/PostgresDropList.java @@ -0,0 +1,127 @@ +/**************************************************************** + * 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.droplists.postgres; + +import static org.apache.james.backends.postgres.utils.PostgresExecutor.DEFAULT_INJECT; +import static org.apache.james.droplists.api.DeniedEntityType.DOMAIN; +import static org.apache.james.droplists.postgres.PostgresDropListModule.PostgresDropListsTable.DENIED_ENTITY; +import static org.apache.james.droplists.postgres.PostgresDropListModule.PostgresDropListsTable.DENIED_ENTITY_TYPE; +import static org.apache.james.droplists.postgres.PostgresDropListModule.PostgresDropListsTable.DROPLIST_ID; +import static org.apache.james.droplists.postgres.PostgresDropListModule.PostgresDropListsTable.OWNER; +import static org.apache.james.droplists.postgres.PostgresDropListModule.PostgresDropListsTable.OWNER_SCOPE; +import static org.apache.james.droplists.postgres.PostgresDropListModule.PostgresDropListsTable.TABLE_NAME; + +import java.util.List; +import java.util.UUID; + +import jakarta.inject.Inject; +import jakarta.inject.Named; +import jakarta.mail.internet.AddressException; + +import org.apache.james.backends.postgres.utils.PostgresExecutor; +import org.apache.james.core.Domain; +import org.apache.james.core.MailAddress; +import org.apache.james.droplists.api.DropList; +import org.apache.james.droplists.api.DropListEntry; +import org.apache.james.droplists.api.OwnerScope; +import org.jooq.Record; + +import com.google.common.base.Preconditions; + +import reactor.core.publisher.Flux; +import reactor.core.publisher.Mono; + +public class PostgresDropList implements DropList { + private final PostgresExecutor postgresExecutor; + + @Inject + public PostgresDropList(@Named(DEFAULT_INJECT) PostgresExecutor postgresExecutor) { + this.postgresExecutor = postgresExecutor; + } + + @Override + public Mono add(DropListEntry entry) { + Preconditions.checkArgument(entry != null); + String specifiedOwner = entry.getOwnerScope().equals(OwnerScope.GLOBAL) ? "" : entry.getOwner(); + return postgresExecutor.executeVoid(dslContext -> + Mono.from(dslContext.insertInto(TABLE_NAME, DROPLIST_ID, OWNER_SCOPE, OWNER, DENIED_ENTITY_TYPE, DENIED_ENTITY) + .values(UUID.randomUUID(), + entry.getOwnerScope().name(), + specifiedOwner, + entry.getDeniedEntityType().name(), + entry.getDeniedEntity()) + ) + ); + } + + @Override + public Mono remove(DropListEntry entry) { + Preconditions.checkArgument(entry != null); + return postgresExecutor.executeVoid(dsl -> Mono.from(dsl.deleteFrom(TABLE_NAME) + .where(OWNER_SCOPE.eq(entry.getOwnerScope().name())) + .and(OWNER.eq(entry.getOwner())) + .and(DENIED_ENTITY.eq(entry.getDeniedEntity())))); + } + + @Override + public Flux list(OwnerScope ownerScope, String owner) { + Preconditions.checkArgument(ownerScope != null); + Preconditions.checkArgument(owner != null); + return postgresExecutor.executeRows(dsl -> Flux.from(dsl.selectFrom(TABLE_NAME) + .where(OWNER_SCOPE.eq(ownerScope.name())) + .and(OWNER.eq(owner)))) + .map(PostgresDropList::mapRecordToDropListEntry); + } + + @Override + public Mono query(OwnerScope ownerScope, String owner, MailAddress sender) { + Preconditions.checkArgument(ownerScope != null); + Preconditions.checkArgument(owner != null); + Preconditions.checkArgument(sender != null); + String specifiedOwner = ownerScope.equals(OwnerScope.GLOBAL) ? "" : owner; + return postgresExecutor.executeExists(dsl -> dsl.selectOne().from(TABLE_NAME) + .where(OWNER_SCOPE.eq(ownerScope.name())) + .and(OWNER.eq(specifiedOwner)) + .and(DENIED_ENTITY.in(List.of(sender.asString(), sender.getDomain().asString())))) + .map(isExist -> Boolean.TRUE.equals(isExist) ? DropList.Status.BLOCKED : DropList.Status.ALLOWED); + } + + private static DropListEntry mapRecordToDropListEntry(Record dropListRecord) { + String deniedEntity = dropListRecord.get(DENIED_ENTITY); + String deniedEntityType = dropListRecord.get(DENIED_ENTITY_TYPE); + OwnerScope ownerScope = OwnerScope.valueOf(dropListRecord.get(OWNER_SCOPE)); + try { + DropListEntry.Builder builder = DropListEntry.builder(); + switch (ownerScope) { + case USER -> builder.userOwner(new MailAddress(dropListRecord.get(OWNER))); + case DOMAIN -> builder.domainOwner(Domain.of(dropListRecord.get(OWNER))); + case GLOBAL -> builder.forAll(); + } + if (DOMAIN.name().equals(deniedEntityType)) { + builder.denyDomain(Domain.of(deniedEntity)); + } else { + builder.denyAddress(new MailAddress(deniedEntity)); + } + return builder.build(); + } catch (AddressException e) { + throw new IllegalArgumentException("Entity could not be parsed as a MailAddress", e); + } + } +} diff --git a/server/data/data-postgres/src/main/java/org/apache/james/droplists/postgres/PostgresDropListModule.java b/server/data/data-postgres/src/main/java/org/apache/james/droplists/postgres/PostgresDropListModule.java new file mode 100644 index 00000000000..6d1d50a7521 --- /dev/null +++ b/server/data/data-postgres/src/main/java/org/apache/james/droplists/postgres/PostgresDropListModule.java @@ -0,0 +1,68 @@ +/**************************************************************** + * 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.droplists.postgres; + +import java.util.UUID; + +import org.apache.james.backends.postgres.PostgresIndex; +import org.apache.james.backends.postgres.PostgresModule; +import org.apache.james.backends.postgres.PostgresTable; +import org.jooq.Field; +import org.jooq.Record; +import org.jooq.Table; +import org.jooq.impl.DSL; +import org.jooq.impl.SQLDataType; + +public interface PostgresDropListModule { + interface PostgresDropListsTable { + Table TABLE_NAME = DSL.table("droplist"); + + Field DROPLIST_ID = DSL.field("droplist_id", SQLDataType.UUID.notNull()); + Field OWNER_SCOPE = DSL.field("owner_scope", SQLDataType.VARCHAR); + Field OWNER = DSL.field("owner", SQLDataType.VARCHAR); + Field DENIED_ENTITY_TYPE = DSL.field("denied_entity_type", SQLDataType.VARCHAR); + Field DENIED_ENTITY = DSL.field("denied_entity", SQLDataType.VARCHAR); + + PostgresTable TABLE = PostgresTable.name(TABLE_NAME.getName()) + .createTableStep(((dsl, tableName) -> dsl.createTableIfNotExists(tableName) + .column(DROPLIST_ID) + .column(OWNER_SCOPE) + .column(OWNER) + .column(DENIED_ENTITY_TYPE) + .column(DENIED_ENTITY) + .constraint(DSL.primaryKey(DROPLIST_ID)))) + .disableRowLevelSecurity() + .build(); + + PostgresIndex IDX_OWNER_SCOPE_OWNER = PostgresIndex.name("idx_owner_scope_owner") + .createIndexStep((dslContext, indexName) -> dslContext.createIndexIfNotExists(indexName) + .on(TABLE_NAME, OWNER_SCOPE, OWNER)); + + PostgresIndex IDX_OWNER_SCOPE_OWNER_DENIED_ENTITY = PostgresIndex.name("idx_owner_scope_owner_denied_entity") + .createIndexStep((dslContext, indexName) -> dslContext.createIndexIfNotExists(indexName) + .on(TABLE_NAME, OWNER_SCOPE, OWNER, DENIED_ENTITY)); + } + + PostgresModule MODULE = PostgresModule.builder() + .addTable(PostgresDropListsTable.TABLE) + .addIndex(PostgresDropListsTable.IDX_OWNER_SCOPE_OWNER) + .addIndex(PostgresDropListsTable.IDX_OWNER_SCOPE_OWNER_DENIED_ENTITY) + .build(); +} diff --git a/server/data/data-postgres/src/main/java/org/apache/james/mailrepository/postgres/PostgresMailRepository.java b/server/data/data-postgres/src/main/java/org/apache/james/mailrepository/postgres/PostgresMailRepository.java new file mode 100644 index 00000000000..cc640377a19 --- /dev/null +++ b/server/data/data-postgres/src/main/java/org/apache/james/mailrepository/postgres/PostgresMailRepository.java @@ -0,0 +1,85 @@ +/**************************************************************** + * 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.mailrepository.postgres; + +import java.util.Collection; +import java.util.Iterator; + +import jakarta.inject.Inject; +import jakarta.mail.MessagingException; + +import org.apache.james.mailrepository.api.MailKey; +import org.apache.james.mailrepository.api.MailRepository; +import org.apache.james.mailrepository.api.MailRepositoryUrl; +import org.apache.mailet.Mail; + +import reactor.core.publisher.Mono; + +public class PostgresMailRepository implements MailRepository { + private final MailRepositoryUrl url; + private final PostgresMailRepositoryContentDAO postgresMailRepositoryContentDAO; + + @Inject + public PostgresMailRepository(MailRepositoryUrl url, + PostgresMailRepositoryContentDAO postgresMailRepositoryContentDAO) { + this.url = url; + this.postgresMailRepositoryContentDAO = postgresMailRepositoryContentDAO; + } + + @Override + public long size() throws MessagingException { + return postgresMailRepositoryContentDAO.size(url); + } + + @Override + public Mono sizeReactive() { + return postgresMailRepositoryContentDAO.sizeReactive(url); + } + + @Override + public MailKey store(Mail mail) throws MessagingException { + return postgresMailRepositoryContentDAO.store(mail, url); + } + + @Override + public Iterator list() throws MessagingException { + return postgresMailRepositoryContentDAO.list(url); + } + + @Override + public Mail retrieve(MailKey key) { + return postgresMailRepositoryContentDAO.retrieve(key, url); + } + + @Override + public void remove(MailKey key) { + postgresMailRepositoryContentDAO.remove(key, url); + } + + @Override + public void remove(Collection keys) { + postgresMailRepositoryContentDAO.remove(keys, url); + } + + @Override + public void removeAll() { + postgresMailRepositoryContentDAO.removeAll(url); + } +} diff --git a/server/data/data-postgres/src/main/java/org/apache/james/mailrepository/postgres/PostgresMailRepositoryBlobReferenceSource.java b/server/data/data-postgres/src/main/java/org/apache/james/mailrepository/postgres/PostgresMailRepositoryBlobReferenceSource.java new file mode 100644 index 00000000000..f287d0fcde4 --- /dev/null +++ b/server/data/data-postgres/src/main/java/org/apache/james/mailrepository/postgres/PostgresMailRepositoryBlobReferenceSource.java @@ -0,0 +1,41 @@ +/**************************************************************** + * 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.mailrepository.postgres; + +import jakarta.inject.Inject; + +import org.apache.james.blob.api.BlobId; +import org.apache.james.blob.api.BlobReferenceSource; + +import reactor.core.publisher.Flux; + +public class PostgresMailRepositoryBlobReferenceSource implements BlobReferenceSource { + private final PostgresMailRepositoryContentDAO postgresMailRepositoryContentDAO; + + @Inject + public PostgresMailRepositoryBlobReferenceSource(PostgresMailRepositoryContentDAO postgresMailRepositoryContentDAO) { + this.postgresMailRepositoryContentDAO = postgresMailRepositoryContentDAO; + } + + @Override + public Flux listReferencedBlobs() { + return postgresMailRepositoryContentDAO.listBlobs(); + } +} diff --git a/server/data/data-postgres/src/main/java/org/apache/james/mailrepository/postgres/PostgresMailRepositoryContentDAO.java b/server/data/data-postgres/src/main/java/org/apache/james/mailrepository/postgres/PostgresMailRepositoryContentDAO.java new file mode 100644 index 00000000000..113b9862737 --- /dev/null +++ b/server/data/data-postgres/src/main/java/org/apache/james/mailrepository/postgres/PostgresMailRepositoryContentDAO.java @@ -0,0 +1,354 @@ +/**************************************************************** + * 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.mailrepository.postgres; + +import static org.apache.james.backends.postgres.PostgresCommons.DATE_TO_LOCAL_DATE_TIME; +import static org.apache.james.backends.postgres.PostgresCommons.LOCAL_DATE_TIME_DATE_FUNCTION; +import static org.apache.james.mailrepository.postgres.PostgresMailRepositoryModule.PostgresMailRepositoryContentTable.ATTRIBUTES; +import static org.apache.james.mailrepository.postgres.PostgresMailRepositoryModule.PostgresMailRepositoryContentTable.BODY_BLOB_ID; +import static org.apache.james.mailrepository.postgres.PostgresMailRepositoryModule.PostgresMailRepositoryContentTable.ERROR; +import static org.apache.james.mailrepository.postgres.PostgresMailRepositoryModule.PostgresMailRepositoryContentTable.HEADER_BLOB_ID; +import static org.apache.james.mailrepository.postgres.PostgresMailRepositoryModule.PostgresMailRepositoryContentTable.KEY; +import static org.apache.james.mailrepository.postgres.PostgresMailRepositoryModule.PostgresMailRepositoryContentTable.LAST_UPDATED; +import static org.apache.james.mailrepository.postgres.PostgresMailRepositoryModule.PostgresMailRepositoryContentTable.PER_RECIPIENT_SPECIFIC_HEADERS; +import static org.apache.james.mailrepository.postgres.PostgresMailRepositoryModule.PostgresMailRepositoryContentTable.RECIPIENTS; +import static org.apache.james.mailrepository.postgres.PostgresMailRepositoryModule.PostgresMailRepositoryContentTable.REMOTE_ADDRESS; +import static org.apache.james.mailrepository.postgres.PostgresMailRepositoryModule.PostgresMailRepositoryContentTable.REMOTE_HOST; +import static org.apache.james.mailrepository.postgres.PostgresMailRepositoryModule.PostgresMailRepositoryContentTable.SENDER; +import static org.apache.james.mailrepository.postgres.PostgresMailRepositoryModule.PostgresMailRepositoryContentTable.STATE; +import static org.apache.james.mailrepository.postgres.PostgresMailRepositoryModule.PostgresMailRepositoryContentTable.TABLE_NAME; +import static org.apache.james.mailrepository.postgres.PostgresMailRepositoryModule.PostgresMailRepositoryContentTable.URL; +import static org.apache.james.util.ReactorUtils.DEFAULT_CONCURRENCY; + +import java.time.LocalDateTime; +import java.util.Arrays; +import java.util.Collection; +import java.util.Iterator; +import java.util.LinkedHashMap; +import java.util.List; +import java.util.Map; +import java.util.Optional; +import java.util.function.Consumer; +import java.util.stream.Stream; + +import jakarta.inject.Inject; +import jakarta.mail.MessagingException; +import jakarta.mail.internet.MimeMessage; + +import org.apache.commons.lang3.StringUtils; +import org.apache.commons.lang3.tuple.Pair; +import org.apache.james.backends.postgres.utils.PostgresExecutor; +import org.apache.james.backends.postgres.utils.PostgresUtils; +import org.apache.james.blob.api.BlobId; +import org.apache.james.blob.api.Store; +import org.apache.james.blob.mail.MimeMessagePartsId; +import org.apache.james.blob.mail.MimeMessageStore; +import org.apache.james.core.MailAddress; +import org.apache.james.core.MaybeSender; +import org.apache.james.mailrepository.api.MailKey; +import org.apache.james.mailrepository.api.MailRepositoryUrl; +import org.apache.james.server.core.MailImpl; +import org.apache.james.server.core.MimeMessageWrapper; +import org.apache.james.util.AuditTrail; +import org.apache.mailet.Attribute; +import org.apache.mailet.AttributeName; +import org.apache.mailet.AttributeValue; +import org.apache.mailet.Mail; +import org.apache.mailet.PerRecipientHeaders; +import org.jooq.Record; +import org.jooq.postgres.extensions.types.Hstore; + +import com.fasterxml.jackson.databind.JsonNode; +import com.github.fge.lambdas.Throwing; +import com.google.common.base.Splitter; +import com.google.common.collect.ImmutableList; +import com.google.common.collect.ImmutableMap; +import com.google.common.collect.Multimap; + +import reactor.core.publisher.Flux; +import reactor.core.publisher.Mono; + +public class PostgresMailRepositoryContentDAO { + private static final String HEADERS_SEPARATOR = "; "; + + private final PostgresExecutor postgresExecutor; + private final Store mimeMessageStore; + private final BlobId.Factory blobIdFactory; + + @Inject + public PostgresMailRepositoryContentDAO(PostgresExecutor postgresExecutor, + MimeMessageStore.Factory mimeMessageStoreFactory, + BlobId.Factory blobIdFactory) { + this.postgresExecutor = postgresExecutor; + this.mimeMessageStore = mimeMessageStoreFactory.mimeMessageStore(); + this.blobIdFactory = blobIdFactory; + } + + public long size(MailRepositoryUrl url) throws MessagingException { + return sizeReactive(url).block(); + } + + public Mono sizeReactive(MailRepositoryUrl url) { + return postgresExecutor.executeCount(context -> Mono.from(context.selectCount() + .from(TABLE_NAME) + .where(URL.eq(url.asString())))) + .map(Integer::longValue); + } + + public MailKey store(Mail mail, MailRepositoryUrl url) throws MessagingException { + MailKey mailKey = MailKey.forMail(mail); + + return storeMailBlob(mail) + .flatMap(mimeMessagePartsId -> storeMailMetadata(mail, mailKey, mimeMessagePartsId, url) + .doOnSuccess(auditTrailStoredMail(mail)) + .onErrorResume(PostgresUtils.UNIQUE_CONSTRAINT_VIOLATION_PREDICATE, e -> Mono.from(mimeMessageStore.delete(mimeMessagePartsId)) + .thenReturn(mailKey))) + .block(); + } + + private Mono storeMailBlob(Mail mail) throws MessagingException { + return mimeMessageStore.save(mail.getMessage()); + } + + private Mono storeMailMetadata(Mail mail, MailKey mailKey, MimeMessagePartsId mimeMessagePartsId, MailRepositoryUrl url) { + return postgresExecutor.executeVoid(context -> Mono.from(context.insertInto(TABLE_NAME) + .set(URL, url.asString()) + .set(KEY, mailKey.asString()) + .set(HEADER_BLOB_ID, mimeMessagePartsId.getHeaderBlobId().asString()) + .set(BODY_BLOB_ID, mimeMessagePartsId.getBodyBlobId().asString()) + .set(STATE, mail.getState()) + .set(ERROR, mail.getErrorMessage()) + .set(SENDER, mail.getMaybeSender().asString()) + .set(RECIPIENTS, asStringArray(mail.getRecipients())) + .set(REMOTE_ADDRESS, mail.getRemoteAddr()) + .set(REMOTE_HOST, mail.getRemoteHost()) + .set(LAST_UPDATED, DATE_TO_LOCAL_DATE_TIME.apply(mail.getLastUpdated())) + .set(ATTRIBUTES, asHstore(mail.attributes())) + .set(PER_RECIPIENT_SPECIFIC_HEADERS, asHstore(mail.getPerRecipientSpecificHeaders().getHeadersByRecipient())) + .onConflict(URL, KEY) + .doUpdate() + .set(HEADER_BLOB_ID, mimeMessagePartsId.getHeaderBlobId().asString()) + .set(BODY_BLOB_ID, mimeMessagePartsId.getBodyBlobId().asString()) + .set(STATE, mail.getState()) + .set(ERROR, mail.getErrorMessage()) + .set(SENDER, mail.getMaybeSender().asString()) + .set(RECIPIENTS, asStringArray(mail.getRecipients())) + .set(REMOTE_ADDRESS, mail.getRemoteAddr()) + .set(REMOTE_HOST, mail.getRemoteHost()) + .set(LAST_UPDATED, DATE_TO_LOCAL_DATE_TIME.apply(mail.getLastUpdated())) + .set(ATTRIBUTES, asHstore(mail.attributes())) + .set(PER_RECIPIENT_SPECIFIC_HEADERS, asHstore(mail.getPerRecipientSpecificHeaders().getHeadersByRecipient())) + )) + .thenReturn(mailKey); + } + + private Consumer auditTrailStoredMail(Mail mail) { + return Throwing.consumer(any -> AuditTrail.entry() + .protocol("mailrepository") + .action("store") + .parameters(Throwing.supplier(() -> ImmutableMap.of("mailId", mail.getName(), + "mimeMessageId", Optional.ofNullable(mail.getMessage()) + .map(Throwing.function(MimeMessage::getMessageID)) + .orElse(""), + "sender", mail.getMaybeSender().asString(), + "recipients", StringUtils.join(mail.getRecipients())))) + .log("PostgresMailRepository stored mail.")); + } + + private String[] asStringArray(Collection mailAddresses) { + return mailAddresses.stream() + .map(MailAddress::asString) + .toArray(String[]::new); + } + + private Hstore asHstore(Multimap multimap) { + return Hstore.hstore(multimap + .asMap() + .entrySet() + .stream() + .map(recipientToHeaders -> Pair.of(recipientToHeaders.getKey().asString(), + asString(recipientToHeaders.getValue()))) + .collect(ImmutableMap.toImmutableMap(Pair::getLeft, Pair::getRight))); + } + + private String asString(Collection headers) { + return StringUtils.join(headers.stream() + .map(PerRecipientHeaders.Header::asString) + .collect(ImmutableList.toImmutableList()), HEADERS_SEPARATOR); + } + + private Hstore asHstore(Stream attributes) { + return Hstore.hstore(attributes + .flatMap(attribute -> attribute.getValue() + .toJson() + .map(JsonNode::toString) + .map(value -> Pair.of(attribute.getName().asString(), value)).stream()) + .collect(ImmutableMap.toImmutableMap(Pair::getLeft, Pair::getRight))); + } + + public Iterator list(MailRepositoryUrl url) throws MessagingException { + return listMailKeys(url) + .toStream() + .iterator(); + } + + private Flux listMailKeys(MailRepositoryUrl url) { + return postgresExecutor.executeRows(context -> Flux.from(context.select(KEY) + .from(TABLE_NAME) + .where(URL.eq(url.asString())))) + .map(record -> new MailKey(record.get(KEY))); + } + + public Mail retrieve(MailKey key, MailRepositoryUrl url) { + return postgresExecutor.executeRow(context -> Mono.from(context.select() + .from(TABLE_NAME) + .where(URL.eq(url.asString())) + .and(KEY.eq(key.asString())))) + .flatMap(this::toMail) + .blockOptional() + .orElse(null); + } + + private Mono toMail(Record record) { + return mimeMessageStore.read(toMimeMessagePartsId(record)) + .map(Throwing.function(mimeMessage -> toMail(record, mimeMessage))); + } + + private Mail toMail(Record record, MimeMessage mimeMessage) throws MessagingException { + List recipients = Arrays.stream(record.get(RECIPIENTS)) + .map(Throwing.function(MailAddress::new)) + .collect(ImmutableList.toImmutableList()); + + PerRecipientHeaders perRecipientHeaders = getPerRecipientHeaders(record); + + List attributes = ((LinkedHashMap) record.get(ATTRIBUTES, LinkedHashMap.class)) + .entrySet() + .stream() + .map(Throwing.function(entry -> new Attribute(AttributeName.of(entry.getKey()), + AttributeValue.fromJsonString(entry.getValue())))) + .collect(ImmutableList.toImmutableList()); + + MailImpl mail = MailImpl.builder() + .name(record.get(KEY)) + .sender(MaybeSender.getMailSender(record.get(SENDER))) + .addRecipients(recipients) + .lastUpdated(LOCAL_DATE_TIME_DATE_FUNCTION.apply(record.get(LAST_UPDATED, LocalDateTime.class))) + .errorMessage(record.get(ERROR)) + .remoteHost(record.get(REMOTE_HOST)) + .remoteAddr(record.get(REMOTE_ADDRESS)) + .state(record.get(STATE)) + .addAllHeadersForRecipients(perRecipientHeaders) + .addAttributes(attributes) + .build(); + + if (mimeMessage instanceof MimeMessageWrapper) { + mail.setMessageNoCopy((MimeMessageWrapper) mimeMessage); + } else { + mail.setMessage(mimeMessage); + } + + return mail; + } + + private PerRecipientHeaders getPerRecipientHeaders(Record record) { + PerRecipientHeaders perRecipientHeaders = new PerRecipientHeaders(); + + ((LinkedHashMap) record.get(PER_RECIPIENT_SPECIFIC_HEADERS, LinkedHashMap.class)) + .entrySet() + .stream() + .flatMap(this::recipientToHeaderStream) + .forEach(recipientToHeaderPair -> perRecipientHeaders.addHeaderForRecipient( + recipientToHeaderPair.getRight(), + recipientToHeaderPair.getLeft())); + + return perRecipientHeaders; + } + + private Stream> recipientToHeaderStream(Map.Entry recipientToHeadersString) { + List headers = Splitter.on(HEADERS_SEPARATOR) + .splitToList(recipientToHeadersString.getValue()); + + return headers + .stream() + .map(headerAsString -> Pair.of( + asMailAddress(recipientToHeadersString.getKey()), + PerRecipientHeaders.Header.fromString(headerAsString))); + } + + private MailAddress asMailAddress(String mailAddress) { + return Throwing.supplier(() -> new MailAddress(mailAddress)) + .get(); + } + + private MimeMessagePartsId toMimeMessagePartsId(Record record) { + return MimeMessagePartsId.builder() + .headerBlobId(blobIdFactory.parse(record.get(HEADER_BLOB_ID))) + .bodyBlobId(blobIdFactory.parse(record.get(BODY_BLOB_ID))) + .build(); + } + + public void remove(MailKey key, MailRepositoryUrl url) { + removeReactive(key, url).block(); + } + + private Mono removeReactive(MailKey key, MailRepositoryUrl url) { + return getMimeMessagePartsId(key, url) + .flatMap(mimeMessagePartsId -> deleteMailMetadata(key, url) + .then(deleteMailBlob(mimeMessagePartsId))); + } + + private Mono getMimeMessagePartsId(MailKey key, MailRepositoryUrl url) { + return postgresExecutor.executeRow(context -> Mono.from(context.select(HEADER_BLOB_ID, BODY_BLOB_ID) + .from(TABLE_NAME) + .where(URL.eq(url.asString())) + .and(KEY.eq(key.asString())))) + .map(this::toMimeMessagePartsId); + } + + private Mono deleteMailMetadata(MailKey key, MailRepositoryUrl url) { + return postgresExecutor.executeVoid(context -> Mono.from(context.deleteFrom(TABLE_NAME) + .where(URL.eq(url.asString())) + .and(KEY.eq(key.asString())))); + } + + private Mono deleteMailBlob(MimeMessagePartsId mimeMessagePartsId) { + return Mono.from(mimeMessageStore.delete(mimeMessagePartsId)); + } + + public void remove(Collection keys, MailRepositoryUrl url) { + Flux.fromIterable(keys) + .concatMap(mailKey -> removeReactive(mailKey, url)) + .then() + .block(); + } + + public void removeAll(MailRepositoryUrl url) { + listMailKeys(url) + .flatMap(mailKey -> removeReactive(mailKey, url), DEFAULT_CONCURRENCY) + .then() + .block(); + } + + public Flux listBlobs() { + return postgresExecutor.executeRows(dslContext -> Flux.from(dslContext.select(HEADER_BLOB_ID, BODY_BLOB_ID) + .from(TABLE_NAME))) + .flatMapIterable(record -> ImmutableList.of(blobIdFactory.parse(record.get(HEADER_BLOB_ID)), blobIdFactory.parse(record.get(BODY_BLOB_ID)))); + } +} diff --git a/server/data/data-postgres/src/main/java/org/apache/james/mailrepository/postgres/PostgresMailRepositoryFactory.java b/server/data/data-postgres/src/main/java/org/apache/james/mailrepository/postgres/PostgresMailRepositoryFactory.java new file mode 100644 index 00000000000..f0b3894368e --- /dev/null +++ b/server/data/data-postgres/src/main/java/org/apache/james/mailrepository/postgres/PostgresMailRepositoryFactory.java @@ -0,0 +1,52 @@ +/**************************************************************** + * 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.mailrepository.postgres; + +import jakarta.inject.Inject; + +import org.apache.james.backends.postgres.utils.PostgresExecutor; +import org.apache.james.blob.api.BlobId; +import org.apache.james.blob.mail.MimeMessageStore; +import org.apache.james.mailrepository.api.MailRepository; +import org.apache.james.mailrepository.api.MailRepositoryFactory; +import org.apache.james.mailrepository.api.MailRepositoryUrl; + +public class PostgresMailRepositoryFactory implements MailRepositoryFactory { + private final PostgresExecutor executor; + private final MimeMessageStore.Factory mimeMessageStoreFactory; + private final BlobId.Factory blobIdFactory; + + @Inject + public PostgresMailRepositoryFactory(PostgresExecutor executor, MimeMessageStore.Factory mimeMessageStoreFactory, BlobId.Factory blobIdFactory) { + this.executor = executor; + this.mimeMessageStoreFactory = mimeMessageStoreFactory; + this.blobIdFactory = blobIdFactory; + } + + @Override + public Class mailRepositoryClass() { + return PostgresMailRepository.class; + } + + @Override + public MailRepository create(MailRepositoryUrl url) { + return new PostgresMailRepository(url, new PostgresMailRepositoryContentDAO(executor, mimeMessageStoreFactory, blobIdFactory)); + } +} diff --git a/server/data/data-postgres/src/main/java/org/apache/james/mailrepository/postgres/PostgresMailRepositoryModule.java b/server/data/data-postgres/src/main/java/org/apache/james/mailrepository/postgres/PostgresMailRepositoryModule.java new file mode 100644 index 00000000000..0ea16f49ccb --- /dev/null +++ b/server/data/data-postgres/src/main/java/org/apache/james/mailrepository/postgres/PostgresMailRepositoryModule.java @@ -0,0 +1,91 @@ +/**************************************************************** + * 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.mailrepository.postgres; + +import static org.apache.james.backends.postgres.PostgresCommons.DataTypes.HSTORE; + +import java.time.LocalDateTime; + +import org.apache.james.backends.postgres.PostgresCommons; +import org.apache.james.backends.postgres.PostgresModule; +import org.apache.james.backends.postgres.PostgresTable; +import org.jooq.Field; +import org.jooq.Record; +import org.jooq.Table; +import org.jooq.impl.DSL; +import org.jooq.impl.SQLDataType; +import org.jooq.postgres.extensions.types.Hstore; + +public interface PostgresMailRepositoryModule { + interface PostgresMailRepositoryUrlTable { + Table TABLE_NAME = DSL.table("mail_repository_url"); + + Field URL = DSL.field("url", SQLDataType.VARCHAR(255).notNull()); + + PostgresTable TABLE = PostgresTable.name(TABLE_NAME.getName()) + .createTableStep(((dsl, tableName) -> dsl.createTableIfNotExists(tableName) + .column(URL) + .primaryKey(URL))) + .disableRowLevelSecurity() + .build(); + } + + interface PostgresMailRepositoryContentTable { + Table TABLE_NAME = DSL.table("mail_repository_content"); + + Field URL = DSL.field("url", SQLDataType.VARCHAR(255).notNull()); + Field KEY = DSL.field("key", SQLDataType.VARCHAR.notNull()); + Field STATE = DSL.field("state", SQLDataType.VARCHAR.notNull()); + Field ERROR = DSL.field("error", SQLDataType.VARCHAR); + Field HEADER_BLOB_ID = DSL.field("header_blob_id", SQLDataType.VARCHAR.notNull()); + Field BODY_BLOB_ID = DSL.field("body_blob_id", SQLDataType.VARCHAR.notNull()); + Field ATTRIBUTES = DSL.field("attributes", HSTORE.notNull()); + Field SENDER = DSL.field("sender", SQLDataType.VARCHAR); + Field RECIPIENTS = DSL.field("recipients", SQLDataType.VARCHAR.getArrayDataType().notNull()); + Field REMOTE_HOST = DSL.field("remote_host", SQLDataType.VARCHAR.notNull()); + Field REMOTE_ADDRESS = DSL.field("remote_address", SQLDataType.VARCHAR.notNull()); + Field LAST_UPDATED = DSL.field("last_updated", PostgresCommons.DataTypes.TIMESTAMP.notNull()); + Field PER_RECIPIENT_SPECIFIC_HEADERS = DSL.field("per_recipient_specific_headers", HSTORE.notNull()); + + PostgresTable TABLE = PostgresTable.name(TABLE_NAME.getName()) + .createTableStep(((dsl, tableName) -> dsl.createTableIfNotExists(tableName) + .column(URL) + .column(KEY) + .column(STATE) + .column(ERROR) + .column(HEADER_BLOB_ID) + .column(BODY_BLOB_ID) + .column(ATTRIBUTES) + .column(SENDER) + .column(RECIPIENTS) + .column(REMOTE_HOST) + .column(REMOTE_ADDRESS) + .column(LAST_UPDATED) + .column(PER_RECIPIENT_SPECIFIC_HEADERS) + .primaryKey(URL, KEY))) + .disableRowLevelSecurity() + .build(); + } + + PostgresModule MODULE = PostgresModule.builder() + .addTable(PostgresMailRepositoryUrlTable.TABLE) + .addTable(PostgresMailRepositoryContentTable.TABLE) + .build(); +} diff --git a/server/data/data-postgres/src/main/java/org/apache/james/mailrepository/postgres/PostgresMailRepositoryUrlStore.java b/server/data/data-postgres/src/main/java/org/apache/james/mailrepository/postgres/PostgresMailRepositoryUrlStore.java new file mode 100644 index 00000000000..a01691f9a92 --- /dev/null +++ b/server/data/data-postgres/src/main/java/org/apache/james/mailrepository/postgres/PostgresMailRepositoryUrlStore.java @@ -0,0 +1,66 @@ +/**************************************************************** + * 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.mailrepository.postgres; + +import static org.apache.james.backends.postgres.utils.PostgresExecutor.DEFAULT_INJECT; +import static org.apache.james.mailrepository.postgres.PostgresMailRepositoryModule.PostgresMailRepositoryUrlTable.TABLE_NAME; +import static org.apache.james.mailrepository.postgres.PostgresMailRepositoryModule.PostgresMailRepositoryUrlTable.URL; + +import java.util.stream.Stream; + +import jakarta.inject.Inject; +import jakarta.inject.Named; + +import org.apache.james.backends.postgres.utils.PostgresExecutor; +import org.apache.james.mailrepository.api.MailRepositoryUrl; +import org.apache.james.mailrepository.api.MailRepositoryUrlStore; + +import reactor.core.publisher.Flux; +import reactor.core.publisher.Mono; + +public class PostgresMailRepositoryUrlStore implements MailRepositoryUrlStore { + private final PostgresExecutor postgresExecutor; + + @Inject + public PostgresMailRepositoryUrlStore(@Named(DEFAULT_INJECT) PostgresExecutor postgresExecutor) { + this.postgresExecutor = postgresExecutor; + } + + @Override + public void add(MailRepositoryUrl url) { + postgresExecutor.executeVoid(context -> Mono.from(context.insertInto(TABLE_NAME, URL) + .values(url.asString()) + .onConflict(URL) + .doNothing())) + .block(); + } + + @Override + public Stream listDistinct() { + return postgresExecutor.executeRows(context -> Flux.from(context.selectFrom(TABLE_NAME))) + .map(record -> MailRepositoryUrl.from(record.get(URL))) + .toStream(); + } + + @Override + public boolean contains(MailRepositoryUrl url) { + return listDistinct().anyMatch(url::equals); + } +} diff --git a/server/data/data-postgres/src/main/java/org/apache/james/rrt/postgres/PostgresRecipientRewriteTable.java b/server/data/data-postgres/src/main/java/org/apache/james/rrt/postgres/PostgresRecipientRewriteTable.java new file mode 100644 index 00000000000..be6cd20ba47 --- /dev/null +++ b/server/data/data-postgres/src/main/java/org/apache/james/rrt/postgres/PostgresRecipientRewriteTable.java @@ -0,0 +1,96 @@ +/**************************************************************** + * 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.rrt.postgres; + +import java.util.Map; +import java.util.function.Predicate; +import java.util.stream.Stream; + +import jakarta.inject.Inject; + +import org.apache.commons.lang3.tuple.Pair; +import org.apache.james.core.Domain; +import org.apache.james.rrt.lib.AbstractRecipientRewriteTable; +import org.apache.james.rrt.lib.Mapping; +import org.apache.james.rrt.lib.MappingSource; +import org.apache.james.rrt.lib.Mappings; +import org.apache.james.rrt.lib.MappingsImpl; + +import com.google.common.base.Preconditions; +import com.google.common.collect.ImmutableMap; + +import reactor.core.publisher.Flux; + +public class PostgresRecipientRewriteTable extends AbstractRecipientRewriteTable { + private PostgresRecipientRewriteTableDAO postgresRecipientRewriteTableDAO; + + @Inject + public PostgresRecipientRewriteTable(PostgresRecipientRewriteTableDAO postgresRecipientRewriteTableDAO) { + this.postgresRecipientRewriteTableDAO = postgresRecipientRewriteTableDAO; + } + + @Override + public void addMapping(MappingSource source, Mapping mapping) { + postgresRecipientRewriteTableDAO.addMapping(source, mapping).block(); + } + + @Override + public void removeMapping(MappingSource source, Mapping mapping) { + postgresRecipientRewriteTableDAO.removeMapping(source, mapping).block(); + } + + @Override + public Mappings getStoredMappings(MappingSource source) { + return postgresRecipientRewriteTableDAO.getMappings(source).block(); + } + + @Override + public Map getAllMappings() { + return postgresRecipientRewriteTableDAO.getAllMappings() + .collect(ImmutableMap.toImmutableMap( + Pair::getLeft, + pair -> MappingsImpl.fromMappings(pair.getRight()), + Mappings::union)) + .block(); + } + + @Override + protected Mappings mapAddress(String user, Domain domain) { + return postgresRecipientRewriteTableDAO.getMappings(MappingSource.fromUser(user, domain)) + .filter(Predicate.not(Mappings::isEmpty)) + .blockOptional() + .orElse(postgresRecipientRewriteTableDAO.getMappings(MappingSource.fromDomain(domain)).block()); + } + + @Override + public Stream listSources(Mapping mapping) { + Preconditions.checkArgument(listSourcesSupportedType.contains(mapping.getType()), + "Not supported mapping of type %s", mapping.getType()); + + return postgresRecipientRewriteTableDAO.getSources(mapping).toStream(); + } + + @Override + public Flux listSourcesReactive(Mapping mapping) { + Preconditions.checkArgument(listSourcesSupportedType.contains(mapping.getType()), + "Not supported mapping of type %s", mapping.getType()); + return postgresRecipientRewriteTableDAO.getSources(mapping); + } +} diff --git a/server/data/data-postgres/src/main/java/org/apache/james/rrt/postgres/PostgresRecipientRewriteTableDAO.java b/server/data/data-postgres/src/main/java/org/apache/james/rrt/postgres/PostgresRecipientRewriteTableDAO.java new file mode 100644 index 00000000000..3e354d42e4b --- /dev/null +++ b/server/data/data-postgres/src/main/java/org/apache/james/rrt/postgres/PostgresRecipientRewriteTableDAO.java @@ -0,0 +1,89 @@ +/**************************************************************** + * 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.rrt.postgres; + +import static org.apache.james.rrt.postgres.PostgresRecipientRewriteTableModule.PostgresRecipientRewriteTableTable.DOMAIN_NAME; +import static org.apache.james.rrt.postgres.PostgresRecipientRewriteTableModule.PostgresRecipientRewriteTableTable.PK_CONSTRAINT_NAME; +import static org.apache.james.rrt.postgres.PostgresRecipientRewriteTableModule.PostgresRecipientRewriteTableTable.TABLE_NAME; +import static org.apache.james.rrt.postgres.PostgresRecipientRewriteTableModule.PostgresRecipientRewriteTableTable.TARGET_ADDRESS; +import static org.apache.james.rrt.postgres.PostgresRecipientRewriteTableModule.PostgresRecipientRewriteTableTable.USERNAME; + +import jakarta.inject.Inject; + +import org.apache.commons.lang3.tuple.Pair; +import org.apache.james.backends.postgres.utils.PostgresExecutor; +import org.apache.james.rrt.lib.Mapping; +import org.apache.james.rrt.lib.MappingSource; +import org.apache.james.rrt.lib.Mappings; +import org.apache.james.rrt.lib.MappingsImpl; + +import com.google.common.collect.ImmutableList; + +import reactor.core.publisher.Flux; +import reactor.core.publisher.Mono; + +public class PostgresRecipientRewriteTableDAO { + private final PostgresExecutor postgresExecutor; + + @Inject + public PostgresRecipientRewriteTableDAO(PostgresExecutor postgresExecutor) { + this.postgresExecutor = postgresExecutor; + } + + public Mono addMapping(MappingSource source, Mapping mapping) { + return postgresExecutor.executeVoid(dslContext -> + Mono.from(dslContext.insertInto(TABLE_NAME, USERNAME, DOMAIN_NAME, TARGET_ADDRESS) + .values(source.getFixedUser(), + source.getFixedDomain(), + mapping.asString()) + .onConflictOnConstraint(PK_CONSTRAINT_NAME) + .doUpdate() + .set(TARGET_ADDRESS, mapping.asString()))); + } + + public Mono removeMapping(MappingSource source, Mapping mapping) { + return postgresExecutor.executeVoid(dsl -> Mono.from(dsl.deleteFrom(TABLE_NAME) + .where(USERNAME.eq(source.getFixedUser())) + .and(DOMAIN_NAME.eq(source.getFixedDomain())) + .and(TARGET_ADDRESS.eq(mapping.asString())))); + } + + public Mono getMappings(MappingSource source) { + return postgresExecutor.executeRows(dsl -> Flux.from(dsl.selectFrom(TABLE_NAME) + .where(USERNAME.eq(source.getFixedUser())) + .and(DOMAIN_NAME.eq(source.getFixedDomain())))) + .map(record -> record.get(TARGET_ADDRESS)) + .collect(ImmutableList.toImmutableList()) + .map(MappingsImpl::fromCollection); + } + + public Flux> getAllMappings() { + return postgresExecutor.executeRows(dsl -> Flux.from(dsl.selectFrom(TABLE_NAME))) + .map(record -> Pair.of( + MappingSource.fromUser(record.get(USERNAME), record.get(DOMAIN_NAME)), + Mapping.of(record.get(TARGET_ADDRESS)))); + } + + public Flux getSources(Mapping mapping) { + return postgresExecutor.executeRows(dsl -> Flux.from(dsl.selectFrom(TABLE_NAME) + .where(TARGET_ADDRESS.eq(mapping.asString())))) + .map(record -> MappingSource.fromUser(record.get(USERNAME), record.get(DOMAIN_NAME))); + } +} diff --git a/server/data/data-postgres/src/main/java/org/apache/james/rrt/postgres/PostgresRecipientRewriteTableModule.java b/server/data/data-postgres/src/main/java/org/apache/james/rrt/postgres/PostgresRecipientRewriteTableModule.java new file mode 100644 index 00000000000..dc64b602221 --- /dev/null +++ b/server/data/data-postgres/src/main/java/org/apache/james/rrt/postgres/PostgresRecipientRewriteTableModule.java @@ -0,0 +1,60 @@ +/**************************************************************** + * 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.rrt.postgres; + +import org.apache.james.backends.postgres.PostgresIndex; +import org.apache.james.backends.postgres.PostgresModule; +import org.apache.james.backends.postgres.PostgresTable; +import org.jooq.Field; +import org.jooq.Name; +import org.jooq.Record; +import org.jooq.Table; +import org.jooq.impl.DSL; +import org.jooq.impl.SQLDataType; + +public interface PostgresRecipientRewriteTableModule { + interface PostgresRecipientRewriteTableTable { + Table TABLE_NAME = DSL.table("rrt"); + + Field USERNAME = DSL.field("username", SQLDataType.VARCHAR(255).notNull()); + Field DOMAIN_NAME = DSL.field("domain_name", SQLDataType.VARCHAR(255).notNull()); + Field TARGET_ADDRESS = DSL.field("target_address", SQLDataType.VARCHAR(255).notNull()); + + Name PK_CONSTRAINT_NAME = DSL.name("rrt_pkey"); + + PostgresTable TABLE = PostgresTable.name(TABLE_NAME.getName()) + .createTableStep(((dsl, tableName) -> dsl.createTableIfNotExists(tableName) + .column(USERNAME) + .column(DOMAIN_NAME) + .column(TARGET_ADDRESS) + .constraint(DSL.constraint(PK_CONSTRAINT_NAME).primaryKey(USERNAME, DOMAIN_NAME, TARGET_ADDRESS)))) + .disableRowLevelSecurity() + .build(); + + PostgresIndex INDEX = PostgresIndex.name("idx_rrt_target_address") + .createIndexStep((dslContext, indexName) -> dslContext.createIndexIfNotExists(indexName) + .on(TABLE_NAME, TARGET_ADDRESS)); + } + + PostgresModule MODULE = PostgresModule.builder() + .addTable(PostgresRecipientRewriteTableTable.TABLE) + .addIndex(PostgresRecipientRewriteTableTable.INDEX) + .build(); +} diff --git a/server/data/data-postgres/src/main/java/org/apache/james/sieve/postgres/PostgresSieveModule.java b/server/data/data-postgres/src/main/java/org/apache/james/sieve/postgres/PostgresSieveModule.java new file mode 100644 index 00000000000..74759c81837 --- /dev/null +++ b/server/data/data-postgres/src/main/java/org/apache/james/sieve/postgres/PostgresSieveModule.java @@ -0,0 +1,70 @@ +/**************************************************************** + * 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.sieve.postgres; + +import java.time.OffsetDateTime; +import java.util.UUID; + +import org.apache.james.backends.postgres.PostgresIndex; +import org.apache.james.backends.postgres.PostgresModule; +import org.apache.james.backends.postgres.PostgresTable; +import org.jooq.Field; +import org.jooq.Record; +import org.jooq.Table; +import org.jooq.impl.DSL; +import org.jooq.impl.SQLDataType; + +public interface PostgresSieveModule { + interface PostgresSieveScriptTable { + Table TABLE_NAME = DSL.table("sieve_scripts"); + + Field USERNAME = DSL.field("username", SQLDataType.VARCHAR(255).notNull()); + Field SCRIPT_NAME = DSL.field("script_name", SQLDataType.VARCHAR.notNull()); + Field SCRIPT_ID = DSL.field("script_id", SQLDataType.UUID.notNull()); + Field SCRIPT_SIZE = DSL.field("script_size", SQLDataType.BIGINT.notNull()); + Field SCRIPT_CONTENT = DSL.field("script_content", SQLDataType.VARCHAR.notNull()); + Field IS_ACTIVE = DSL.field("is_active", SQLDataType.BOOLEAN.notNull()); + Field ACTIVATION_DATE_TIME = DSL.field("activation_date_time", SQLDataType.OFFSETDATETIME); + + PostgresTable TABLE = PostgresTable.name(TABLE_NAME.getName()) + .createTableStep(((dsl, tableName) -> dsl.createTableIfNotExists(tableName) + .column(SCRIPT_ID) + .column(USERNAME) + .column(SCRIPT_NAME) + .column(SCRIPT_SIZE) + .column(SCRIPT_CONTENT) + .column(IS_ACTIVE) + .column(ACTIVATION_DATE_TIME) + .primaryKey(SCRIPT_ID) + .constraint(DSL.unique(USERNAME, SCRIPT_NAME)))) + .disableRowLevelSecurity() + .build(); + + PostgresIndex MAXIMUM_ONE_ACTIVE_SCRIPT_PER_USER_UNIQUE_INDEX = PostgresIndex.name("maximum_one_active_script_per_user") + .createIndexStep(((dsl, indexName) -> dsl.createUniqueIndexIfNotExists(indexName) + .on(TABLE_NAME, USERNAME) + .where(IS_ACTIVE))); + } + + PostgresModule MODULE = PostgresModule.builder() + .addTable(PostgresSieveScriptTable.TABLE) + .addIndex(PostgresSieveScriptTable.MAXIMUM_ONE_ACTIVE_SCRIPT_PER_USER_UNIQUE_INDEX) + .build(); +} diff --git a/server/data/data-postgres/src/main/java/org/apache/james/sieve/postgres/PostgresSieveQuotaDAO.java b/server/data/data-postgres/src/main/java/org/apache/james/sieve/postgres/PostgresSieveQuotaDAO.java new file mode 100644 index 00000000000..647b0de2313 --- /dev/null +++ b/server/data/data-postgres/src/main/java/org/apache/james/sieve/postgres/PostgresSieveQuotaDAO.java @@ -0,0 +1,117 @@ +/**************************************************************** + * 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.sieve.postgres; + +import static org.apache.james.core.quota.QuotaType.SIZE; + +import java.util.Optional; + +import jakarta.inject.Inject; + +import org.apache.james.backends.postgres.quota.PostgresQuotaCurrentValueDAO; +import org.apache.james.backends.postgres.quota.PostgresQuotaLimitDAO; +import org.apache.james.core.Username; +import org.apache.james.core.quota.QuotaComponent; +import org.apache.james.core.quota.QuotaCurrentValue; +import org.apache.james.core.quota.QuotaLimit; +import org.apache.james.core.quota.QuotaScope; +import org.apache.james.core.quota.QuotaSizeLimit; + +import reactor.core.publisher.Mono; + +public class PostgresSieveQuotaDAO { + public static final QuotaComponent QUOTA_COMPONENT = QuotaComponent.of("SIEVE"); + public static final String GLOBAL = "GLOBAL"; + + private final PostgresQuotaCurrentValueDAO currentValueDao; + private final PostgresQuotaLimitDAO limitDao; + + @Inject + public PostgresSieveQuotaDAO(PostgresQuotaCurrentValueDAO currentValueDao, PostgresQuotaLimitDAO limitDao) { + this.currentValueDao = currentValueDao; + this.limitDao = limitDao; + } + + public Mono spaceUsedBy(Username username) { + QuotaCurrentValue.Key quotaKey = asQuotaKey(username); + + return currentValueDao.getQuotaCurrentValue(quotaKey).map(QuotaCurrentValue::getCurrentValue) + .switchIfEmpty(Mono.just(0L)); + } + + private QuotaCurrentValue.Key asQuotaKey(Username username) { + return QuotaCurrentValue.Key.of( + QUOTA_COMPONENT, + username.asString(), + SIZE); + } + + public Mono updateSpaceUsed(Username username, long spaceUsed) { + QuotaCurrentValue.Key quotaKey = asQuotaKey(username); + + return currentValueDao.increase(quotaKey, spaceUsed); + } + + public Mono> getGlobalQuota() { + return limitDao.getQuotaLimit(QuotaLimit.QuotaLimitKey.of(QUOTA_COMPONENT, QuotaScope.GLOBAL, GLOBAL, SIZE)) + .map(v -> v.getQuotaLimit().map(QuotaSizeLimit::size)) + .switchIfEmpty(Mono.just(Optional.empty())); + } + + public Mono setGlobalQuota(QuotaSizeLimit quota) { + return limitDao.setQuotaLimit(QuotaLimit.builder() + .quotaComponent(QUOTA_COMPONENT) + .quotaScope(QuotaScope.GLOBAL) + .quotaType(SIZE) + .identifier(GLOBAL) + .quotaLimit(quota.asLong()) + .build()); + } + + public Mono removeGlobalQuota() { + return limitDao.deleteQuotaLimit(QuotaLimit.QuotaLimitKey.of(QUOTA_COMPONENT, QuotaScope.GLOBAL, GLOBAL, SIZE)); + } + + public Mono> getQuota(Username username) { + return limitDao.getQuotaLimits(QUOTA_COMPONENT, QuotaScope.USER, username.asString()) + .map(v -> v.getQuotaLimit().map(QuotaSizeLimit::size)) + .switchIfEmpty(Mono.just(Optional.empty())) + .single(); + } + + public Mono setQuota(Username username, QuotaSizeLimit quota) { + return limitDao.setQuotaLimit(QuotaLimit.builder() + .quotaComponent(QUOTA_COMPONENT) + .quotaScope(QuotaScope.USER) + .quotaType(SIZE) + .identifier(username.asString()) + .quotaLimit(quota.asLong()) + .build()); + } + + public Mono removeQuota(Username username) { + return limitDao.deleteQuotaLimit(QuotaLimit.QuotaLimitKey.of( + QUOTA_COMPONENT, QuotaScope.USER, username.asString(), SIZE)); + } + + public Mono resetSpaceUsed(Username username, long spaceUsed) { + return spaceUsedBy(username).flatMap(currentSpace -> currentValueDao.increase(asQuotaKey(username), spaceUsed - currentSpace)); + } +} diff --git a/server/data/data-postgres/src/main/java/org/apache/james/sieve/postgres/PostgresSieveRepository.java b/server/data/data-postgres/src/main/java/org/apache/james/sieve/postgres/PostgresSieveRepository.java new file mode 100644 index 00000000000..2fc9a33ba40 --- /dev/null +++ b/server/data/data-postgres/src/main/java/org/apache/james/sieve/postgres/PostgresSieveRepository.java @@ -0,0 +1,268 @@ +/**************************************************************** + * 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.sieve.postgres; + +import static org.apache.james.backends.postgres.utils.PostgresUtils.UNIQUE_CONSTRAINT_VIOLATION_PREDICATE; + +import java.io.InputStream; +import java.nio.charset.StandardCharsets; +import java.time.ZonedDateTime; +import java.util.List; +import java.util.Optional; + +import jakarta.inject.Inject; + +import org.apache.commons.io.IOUtils; +import org.apache.james.core.Username; +import org.apache.james.core.quota.QuotaSizeLimit; +import org.apache.james.core.quota.QuotaSizeUsage; +import org.apache.james.sieve.postgres.model.PostgresSieveScript; +import org.apache.james.sieve.postgres.model.PostgresSieveScriptId; +import org.apache.james.sieverepository.api.ScriptContent; +import org.apache.james.sieverepository.api.ScriptName; +import org.apache.james.sieverepository.api.ScriptSummary; +import org.apache.james.sieverepository.api.SieveRepository; +import org.apache.james.sieverepository.api.exception.DuplicateException; +import org.apache.james.sieverepository.api.exception.IsActiveException; +import org.apache.james.sieverepository.api.exception.QuotaExceededException; +import org.apache.james.sieverepository.api.exception.QuotaNotFoundException; +import org.apache.james.sieverepository.api.exception.ScriptNotFoundException; + +import reactor.core.publisher.Flux; +import reactor.core.publisher.Mono; + +public class PostgresSieveRepository implements SieveRepository { + private final PostgresSieveQuotaDAO postgresSieveQuotaDAO; + private final PostgresSieveScriptDAO postgresSieveScriptDAO; + + @Inject + public PostgresSieveRepository(PostgresSieveQuotaDAO postgresSieveQuotaDAO, + PostgresSieveScriptDAO postgresSieveScriptDAO) { + this.postgresSieveQuotaDAO = postgresSieveQuotaDAO; + this.postgresSieveScriptDAO = postgresSieveScriptDAO; + } + + @Override + public void haveSpace(Username username, ScriptName name, long size) throws QuotaExceededException { + long sizeDifference = spaceThatWillBeUsedByNewScript(username, name, size).block(); + throwOnOverQuota(username, sizeDifference); + } + + @Override + public void putScript(Username username, ScriptName name, ScriptContent content) throws QuotaExceededException { + long sizeDifference = spaceThatWillBeUsedByNewScript(username, name, content.length()).block(); + throwOnOverQuota(username, sizeDifference); + postgresSieveScriptDAO.upsertScript(PostgresSieveScript.builder() + .username(username.asString()) + .scriptName(name.getValue()) + .scriptContent(content.getValue()) + .scriptSize(content.length()) + .isActive(false) + .id(PostgresSieveScriptId.generate()) + .build()) + .flatMap(upsertedScripts -> { + if (upsertedScripts > 0) { + return updateSpaceUsed(username, sizeDifference); + } + return Mono.empty(); + }) + .block(); + } + + private Mono updateSpaceUsed(Username username, long spaceToUse) { + if (spaceToUse == 0) { + return Mono.empty(); + } + return postgresSieveQuotaDAO.updateSpaceUsed(username, spaceToUse); + } + + private Mono spaceThatWillBeUsedByNewScript(Username username, ScriptName name, long scriptSize) { + return postgresSieveScriptDAO.getScriptSize(username, name) + .defaultIfEmpty(0L) + .map(sizeOfStoredScript -> scriptSize - sizeOfStoredScript); + } + + private void throwOnOverQuota(Username username, Long sizeDifference) throws QuotaExceededException { + long spaceUsed = postgresSieveQuotaDAO.spaceUsedBy(username).block(); + QuotaSizeLimit limit = limitToUser(username).block(); + + if (QuotaSizeUsage.size(spaceUsed) + .add(sizeDifference) + .exceedLimit(limit)) { + throw new QuotaExceededException(); + } + } + + private Mono limitToUser(Username username) { + return postgresSieveQuotaDAO.getQuota(username) + .filter(Optional::isPresent) + .switchIfEmpty(postgresSieveQuotaDAO.getGlobalQuota()) + .map(optional -> optional.orElse(QuotaSizeLimit.unlimited())); + } + + @Override + public List listScripts(Username username) { + return listScriptsReactive(username) + .collectList() + .block(); + } + + @Override + public Flux listScriptsReactive(Username username) { + return postgresSieveScriptDAO.getScripts(username) + .map(PostgresSieveScript::toScriptSummary); + } + + @Override + public ZonedDateTime getActivationDateForActiveScript(Username username) throws ScriptNotFoundException { + return postgresSieveScriptDAO.getActiveScript(username) + .blockOptional() + .orElseThrow(ScriptNotFoundException::new) + .getActivationDateTime() + .toZonedDateTime(); + } + + @Override + public InputStream getActive(Username username) throws ScriptNotFoundException { + return IOUtils.toInputStream(postgresSieveScriptDAO.getActiveScript(username) + .blockOptional() + .orElseThrow(ScriptNotFoundException::new) + .getScriptContent(), StandardCharsets.UTF_8); + } + + @Override + public void setActive(Username username, ScriptName name) throws ScriptNotFoundException { + if (SieveRepository.NO_SCRIPT_NAME.equals(name)) { + switchOffCurrentActiveScript(username); + } else { + throwOnScriptNonExistence(username, name); + switchOffCurrentActiveScript(username); + activateScript(username, name); + } + } + + private void throwOnScriptNonExistence(Username username, ScriptName name) throws ScriptNotFoundException { + if (!postgresSieveScriptDAO.scriptExists(username, name).block()) { + throw new ScriptNotFoundException(); + } + } + + private void switchOffCurrentActiveScript(Username username) { + postgresSieveScriptDAO.deactivateCurrentActiveScript(username).block(); + } + + private void activateScript(Username username, ScriptName scriptName) { + postgresSieveScriptDAO.activateScript(username, scriptName).block(); + } + + @Override + public InputStream getScript(Username username, ScriptName name) throws ScriptNotFoundException { + return IOUtils.toInputStream(postgresSieveScriptDAO.getScript(username, name) + .blockOptional() + .orElseThrow(ScriptNotFoundException::new) + .getScriptContent(), StandardCharsets.UTF_8); + } + + @Override + public void deleteScript(Username username, ScriptName name) throws ScriptNotFoundException, IsActiveException { + boolean isActive = postgresSieveScriptDAO.getIsActive(username, name) + .blockOptional() + .orElseThrow(ScriptNotFoundException::new); + + if (isActive) { + throw new IsActiveException(); + } + + postgresSieveScriptDAO.deleteScript(username, name).block(); + } + + @Override + public void renameScript(Username username, ScriptName oldName, ScriptName newName) throws DuplicateException, ScriptNotFoundException { + try { + int renamedScripts = postgresSieveScriptDAO.renameScript(username, oldName, newName).block(); + if (renamedScripts == 0) { + throw new ScriptNotFoundException(); + } + } catch (Exception e) { + if (UNIQUE_CONSTRAINT_VIOLATION_PREDICATE.test(e)) { + throw new DuplicateException(); + } + throw e; + } + } + + @Override + public boolean hasDefaultQuota() { + return postgresSieveQuotaDAO.getGlobalQuota() + .block() + .isPresent(); + } + + @Override + public QuotaSizeLimit getDefaultQuota() throws QuotaNotFoundException { + return postgresSieveQuotaDAO.getGlobalQuota() + .block() + .orElseThrow(() -> new QuotaNotFoundException("Unable to find quota for default user")); + } + + @Override + public void setDefaultQuota(QuotaSizeLimit quota) { + postgresSieveQuotaDAO.setGlobalQuota(quota) + .block(); + } + + @Override + public void removeQuota() { + postgresSieveQuotaDAO.removeGlobalQuota() + .block(); + } + + @Override + public boolean hasQuota(Username username) { + Mono hasUserQuota = postgresSieveQuotaDAO.getQuota(username).map(Optional::isPresent); + Mono hasGlobalQuota = postgresSieveQuotaDAO.getGlobalQuota().map(Optional::isPresent); + + return hasUserQuota.zipWith(hasGlobalQuota, (a, b) -> a || b) + .block(); + } + + @Override + public QuotaSizeLimit getQuota(Username username) throws QuotaNotFoundException { + return postgresSieveQuotaDAO.getQuota(username) + .block() + .orElseThrow(() -> new QuotaNotFoundException("Unable to find quota for user " + username.asString())); + } + + @Override + public void setQuota(Username username, QuotaSizeLimit quota) { + postgresSieveQuotaDAO.setQuota(username, quota) + .block(); + } + + @Override + public void removeQuota(Username username) { + postgresSieveQuotaDAO.removeQuota(username).block(); + } + + @Override + public Mono resetSpaceUsedReactive(Username username, long spaceUsed) { + return Mono.error(new UnsupportedOperationException()); + } +} \ No newline at end of file diff --git a/server/data/data-postgres/src/main/java/org/apache/james/sieve/postgres/PostgresSieveScriptDAO.java b/server/data/data-postgres/src/main/java/org/apache/james/sieve/postgres/PostgresSieveScriptDAO.java new file mode 100644 index 00000000000..61274a36760 --- /dev/null +++ b/server/data/data-postgres/src/main/java/org/apache/james/sieve/postgres/PostgresSieveScriptDAO.java @@ -0,0 +1,155 @@ +/**************************************************************** + * 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.sieve.postgres; + +import static org.apache.james.backends.postgres.utils.PostgresExecutor.DEFAULT_INJECT; +import static org.apache.james.sieve.postgres.PostgresSieveModule.PostgresSieveScriptTable.ACTIVATION_DATE_TIME; +import static org.apache.james.sieve.postgres.PostgresSieveModule.PostgresSieveScriptTable.IS_ACTIVE; +import static org.apache.james.sieve.postgres.PostgresSieveModule.PostgresSieveScriptTable.SCRIPT_CONTENT; +import static org.apache.james.sieve.postgres.PostgresSieveModule.PostgresSieveScriptTable.SCRIPT_ID; +import static org.apache.james.sieve.postgres.PostgresSieveModule.PostgresSieveScriptTable.SCRIPT_NAME; +import static org.apache.james.sieve.postgres.PostgresSieveModule.PostgresSieveScriptTable.SCRIPT_SIZE; +import static org.apache.james.sieve.postgres.PostgresSieveModule.PostgresSieveScriptTable.TABLE_NAME; +import static org.apache.james.sieve.postgres.PostgresSieveModule.PostgresSieveScriptTable.USERNAME; + +import java.time.OffsetDateTime; +import java.util.function.Function; + +import jakarta.inject.Inject; +import jakarta.inject.Named; + +import org.apache.james.backends.postgres.utils.PostgresExecutor; +import org.apache.james.core.Username; +import org.apache.james.sieve.postgres.model.PostgresSieveScript; +import org.apache.james.sieve.postgres.model.PostgresSieveScriptId; +import org.apache.james.sieverepository.api.ScriptName; +import org.jooq.Record; + +import reactor.core.publisher.Flux; +import reactor.core.publisher.Mono; + +public class PostgresSieveScriptDAO { + private final PostgresExecutor postgresExecutor; + + @Inject + public PostgresSieveScriptDAO(@Named(DEFAULT_INJECT) PostgresExecutor postgresExecutor) { + this.postgresExecutor = postgresExecutor; + } + + public Mono upsertScript(PostgresSieveScript sieveScript) { + return postgresExecutor.executeReturnAffectedRowsCount(dslContext -> Mono.from(dslContext.insertInto(TABLE_NAME) + .set(SCRIPT_ID, sieveScript.getId().getValue()) + .set(USERNAME, sieveScript.getUsername()) + .set(SCRIPT_NAME, sieveScript.getScriptName()) + .set(SCRIPT_SIZE, sieveScript.getScriptSize()) + .set(SCRIPT_CONTENT, sieveScript.getScriptContent()) + .set(IS_ACTIVE, sieveScript.isActive()) + .set(ACTIVATION_DATE_TIME, sieveScript.getActivationDateTime()) + .onConflict(USERNAME, SCRIPT_NAME) + .doUpdate() + .set(SCRIPT_SIZE, sieveScript.getScriptSize()) + .set(SCRIPT_CONTENT, sieveScript.getScriptContent()) + .set(IS_ACTIVE, sieveScript.isActive()) + .set(ACTIVATION_DATE_TIME, sieveScript.getActivationDateTime()))); + } + + public Mono getScript(Username username, ScriptName scriptName) { + return postgresExecutor.executeRow(dslContext -> Mono.from(dslContext.selectFrom(TABLE_NAME) + .where(USERNAME.eq(username.asString()), + SCRIPT_NAME.eq(scriptName.getValue())))) + .map(recordToPostgresSieveScript()); + } + + public Mono getScriptSize(Username username, ScriptName scriptName) { + return postgresExecutor.executeRow(dslContext -> Mono.from(dslContext.select(SCRIPT_SIZE) + .from(TABLE_NAME) + .where(USERNAME.eq(username.asString()), + SCRIPT_NAME.eq(scriptName.getValue())))) + .map(record -> record.get(SCRIPT_SIZE)); + } + + public Mono getIsActive(Username username, ScriptName scriptName) { + return postgresExecutor.executeRow(dslContext -> Mono.from(dslContext.select(IS_ACTIVE) + .from(TABLE_NAME) + .where(USERNAME.eq(username.asString()), + SCRIPT_NAME.eq(scriptName.getValue())))) + .map(record -> record.get(IS_ACTIVE)); + } + + public Mono scriptExists(Username username, ScriptName scriptName) { + return postgresExecutor.executeExists(dslContext -> dslContext.selectOne() + .from(TABLE_NAME) + .where(USERNAME.eq(username.asString()), + SCRIPT_NAME.eq(scriptName.getValue()))); + } + + public Flux getScripts(Username username) { + return postgresExecutor.executeRows(dslContext -> Flux.from(dslContext.selectFrom(TABLE_NAME) + .where(USERNAME.eq(username.asString())))) + .map(recordToPostgresSieveScript()); + } + + public Mono getActiveScript(Username username) { + return postgresExecutor.executeRow(dslContext -> Mono.from(dslContext.selectFrom(TABLE_NAME) + .where(USERNAME.eq(username.asString()), + IS_ACTIVE.eq(true)))) + .map(recordToPostgresSieveScript()); + } + + public Mono activateScript(Username username, ScriptName scriptName) { + return postgresExecutor.executeVoid(dslContext -> Mono.from(dslContext.update(TABLE_NAME) + .set(IS_ACTIVE, true) + .set(ACTIVATION_DATE_TIME, OffsetDateTime.now()) + .where(USERNAME.eq(username.asString()), + SCRIPT_NAME.eq(scriptName.getValue())))); + } + + public Mono deactivateCurrentActiveScript(Username username) { + return postgresExecutor.executeVoid(dslContext -> Mono.from(dslContext.update(TABLE_NAME) + .set(IS_ACTIVE, false) + .where(USERNAME.eq(username.asString()), + IS_ACTIVE.eq(true)))); + } + + public Mono renameScript(Username username, ScriptName oldName, ScriptName newName) { + return postgresExecutor.executeReturnAffectedRowsCount(dslContext -> Mono.from(dslContext.update(TABLE_NAME) + .set(SCRIPT_NAME, newName.getValue()) + .where(USERNAME.eq(username.asString()), + SCRIPT_NAME.eq(oldName.getValue())))); + } + + public Mono deleteScript(Username username, ScriptName scriptName) { + return postgresExecutor.executeVoid(dslContext -> Mono.from(dslContext.deleteFrom(TABLE_NAME) + .where(USERNAME.eq(username.asString()), + SCRIPT_NAME.eq(scriptName.getValue())))); + } + + private Function recordToPostgresSieveScript() { + return record -> PostgresSieveScript.builder() + .username(record.get(USERNAME)) + .scriptName(record.get(SCRIPT_NAME)) + .scriptContent(record.get(SCRIPT_CONTENT)) + .scriptSize(record.get(SCRIPT_SIZE)) + .isActive(record.get(IS_ACTIVE)) + .activationDateTime(record.get(ACTIVATION_DATE_TIME)) + .id(new PostgresSieveScriptId(record.get(SCRIPT_ID))) + .build(); + } +} diff --git a/server/data/data-postgres/src/main/java/org/apache/james/sieve/postgres/model/PostgresSieveScript.java b/server/data/data-postgres/src/main/java/org/apache/james/sieve/postgres/model/PostgresSieveScript.java new file mode 100644 index 00000000000..f1a29812ccb --- /dev/null +++ b/server/data/data-postgres/src/main/java/org/apache/james/sieve/postgres/model/PostgresSieveScript.java @@ -0,0 +1,162 @@ +/**************************************************************** + * 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.sieve.postgres.model; + +import java.time.OffsetDateTime; +import java.util.Objects; + +import org.apache.commons.lang3.StringUtils; +import org.apache.james.sieverepository.api.ScriptName; +import org.apache.james.sieverepository.api.ScriptSummary; + +import com.google.common.base.Preconditions; + +public class PostgresSieveScript { + + public static Builder builder() { + return new Builder(); + } + + public static class Builder { + private String username; + private String scriptName; + private String scriptContent; + private long scriptSize; + private boolean isActive; + private OffsetDateTime activationDateTime; + private PostgresSieveScriptId id; + + public Builder username(String username) { + Preconditions.checkNotNull(username); + this.username = username; + return this; + } + + public Builder scriptName(String scriptName) { + Preconditions.checkNotNull(scriptName); + this.scriptName = scriptName; + return this; + } + + public Builder scriptContent(String scriptContent) { + this.scriptContent = scriptContent; + return this; + } + + public Builder scriptSize(long scriptSize) { + this.scriptSize = scriptSize; + return this; + } + + public Builder id(PostgresSieveScriptId id) { + this.id = id; + return this; + } + + public Builder isActive(boolean isActive) { + this.isActive = isActive; + return this; + } + + public Builder activationDateTime(OffsetDateTime offsetDateTime) { + this.activationDateTime = offsetDateTime; + return this; + } + + public PostgresSieveScript build() { + Preconditions.checkState(StringUtils.isNotBlank(username), "'username' is mandatory"); + Preconditions.checkState(StringUtils.isNotBlank(scriptName), "'scriptName' is mandatory"); + Preconditions.checkState(id != null, "'id' is mandatory"); + + return new PostgresSieveScript(id, username, scriptName, scriptContent, scriptSize, isActive, activationDateTime); + } + } + + private final PostgresSieveScriptId id; + private final String username; + private final String scriptName; + private final String scriptContent; + private final long scriptSize; + private final boolean isActive; + private final OffsetDateTime activationDateTime; + + private PostgresSieveScript(PostgresSieveScriptId id, String username, String scriptName, String scriptContent, + long scriptSize, boolean isActive, OffsetDateTime activationDateTime) { + this.id = id; + this.username = username; + this.scriptName = scriptName; + this.scriptContent = scriptContent; + this.scriptSize = scriptSize; + this.isActive = isActive; + this.activationDateTime = activationDateTime; + } + + public PostgresSieveScriptId getId() { + return id; + } + + public String getUsername() { + return username; + } + + public String getScriptName() { + return scriptName; + } + + public String getScriptContent() { + return scriptContent; + } + + public long getScriptSize() { + return scriptSize; + } + + public boolean isActive() { + return isActive; + } + + public OffsetDateTime getActivationDateTime() { + return activationDateTime; + } + + public ScriptSummary toScriptSummary() { + return new ScriptSummary(new ScriptName(scriptName), isActive, scriptSize); + } + + @Override + public final boolean equals(Object o) { + if (o instanceof PostgresSieveScript) { + PostgresSieveScript that = (PostgresSieveScript) o; + + return Objects.equals(this.scriptSize, that.scriptSize) + && Objects.equals(this.isActive, that.isActive) + && Objects.equals(this.username, that.username) + && Objects.equals(this.scriptName, that.scriptName) + && Objects.equals(this.scriptContent, that.scriptContent) + && Objects.equals(this.activationDateTime, that.activationDateTime); + } + return false; + } + + @Override + public final int hashCode() { + return Objects.hash(username, scriptName); + } +} diff --git a/server/data/data-postgres/src/main/java/org/apache/james/sieve/postgres/model/PostgresSieveScriptId.java b/server/data/data-postgres/src/main/java/org/apache/james/sieve/postgres/model/PostgresSieveScriptId.java new file mode 100644 index 00000000000..adb0778f1fc --- /dev/null +++ b/server/data/data-postgres/src/main/java/org/apache/james/sieve/postgres/model/PostgresSieveScriptId.java @@ -0,0 +1,38 @@ +/**************************************************************** + * 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.sieve.postgres.model; + +import java.util.UUID; + +public class PostgresSieveScriptId { + public static PostgresSieveScriptId generate() { + return new PostgresSieveScriptId(UUID.randomUUID()); + } + + private final UUID value; + + public PostgresSieveScriptId(UUID value) { + this.value = value; + } + + public UUID getValue() { + return value; + } +} diff --git a/server/data/data-postgres/src/main/java/org/apache/james/user/postgres/PostgresDelegationStore.java b/server/data/data-postgres/src/main/java/org/apache/james/user/postgres/PostgresDelegationStore.java new file mode 100644 index 00000000000..2b9df60e00c --- /dev/null +++ b/server/data/data-postgres/src/main/java/org/apache/james/user/postgres/PostgresDelegationStore.java @@ -0,0 +1,89 @@ +/**************************************************************** + * 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.user.postgres; + +import jakarta.inject.Inject; + +import org.apache.james.core.Username; +import org.apache.james.user.api.DelegationStore; +import org.apache.james.user.api.UsersRepository; +import org.reactivestreams.Publisher; + +import reactor.core.publisher.Mono; + +public class PostgresDelegationStore implements DelegationStore { + public interface UserExistencePredicate { + Mono exists(Username username); + } + + public static class UserExistencePredicateImplementation implements UserExistencePredicate { + private final UsersRepository usersRepository; + + @Inject + UserExistencePredicateImplementation(UsersRepository usersRepository) { + this.usersRepository = usersRepository; + } + + @Override + public Mono exists(Username username) { + return Mono.from(usersRepository.containsReactive(username)); + } + } + + private final PostgresUsersDAO postgresUsersDAO; + private final UserExistencePredicate userExistencePredicate; + + @Inject + public PostgresDelegationStore(PostgresUsersDAO postgresUsersDAO, UserExistencePredicate userExistencePredicate) { + this.postgresUsersDAO = postgresUsersDAO; + this.userExistencePredicate = userExistencePredicate; + } + + @Override + public Publisher authorizedUsers(Username baseUser) { + return postgresUsersDAO.getAuthorizedUsers(baseUser); + } + + @Override + public Publisher clear(Username baseUser) { + return postgresUsersDAO.removeAllAuthorizedUsers(baseUser); + } + + @Override + public Publisher addAuthorizedUser(Username baseUser, Username userWithAccess) { + return userExistencePredicate.exists(userWithAccess) + .flatMap(targetUserExists -> postgresUsersDAO.addAuthorizedUser(baseUser, userWithAccess, targetUserExists)); + } + + @Override + public Publisher removeAuthorizedUser(Username baseUser, Username userWithAccess) { + return postgresUsersDAO.removeAuthorizedUser(baseUser, userWithAccess); + } + + @Override + public Publisher delegatedUsers(Username baseUser) { + return postgresUsersDAO.getDelegatedToUsers(baseUser); + } + + @Override + public Publisher removeDelegatedUser(Username baseUser, Username delegatedToUser) { + return postgresUsersDAO.removeDelegatedToUser(baseUser, delegatedToUser); + } +} diff --git a/server/data/data-postgres/src/main/java/org/apache/james/user/postgres/PostgresUserModule.java b/server/data/data-postgres/src/main/java/org/apache/james/user/postgres/PostgresUserModule.java new file mode 100644 index 00000000000..f0b67a25fd9 --- /dev/null +++ b/server/data/data-postgres/src/main/java/org/apache/james/user/postgres/PostgresUserModule.java @@ -0,0 +1,58 @@ +/**************************************************************** + * 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.user.postgres; + +import org.apache.james.backends.postgres.PostgresModule; +import org.apache.james.backends.postgres.PostgresTable; +import org.jooq.Field; +import org.jooq.Name; +import org.jooq.Record; +import org.jooq.Table; +import org.jooq.impl.DSL; +import org.jooq.impl.SQLDataType; + +public interface PostgresUserModule { + interface PostgresUserTable { + Table TABLE_NAME = DSL.table("users"); + + Field USERNAME = DSL.field("username", SQLDataType.VARCHAR(255).notNull()); + Field HASHED_PASSWORD = DSL.field("hashed_password", SQLDataType.VARCHAR); + Field ALGORITHM = DSL.field("algorithm", SQLDataType.VARCHAR(100)); + Field AUTHORIZED_USERS = DSL.field("authorized_users", SQLDataType.VARCHAR.getArrayDataType()); + Field DELEGATED_USERS = DSL.field("delegated_users", SQLDataType.VARCHAR.getArrayDataType()); + + Name USERNAME_PRIMARY_KEY = DSL.name("users_username_pk"); + + PostgresTable TABLE = PostgresTable.name(TABLE_NAME.getName()) + .createTableStep(((dsl, tableName) -> dsl.createTableIfNotExists(tableName) + .column(USERNAME) + .column(HASHED_PASSWORD) + .column(ALGORITHM) + .column(AUTHORIZED_USERS) + .column(DELEGATED_USERS) + .constraint(DSL.constraint(USERNAME_PRIMARY_KEY).primaryKey(USERNAME)))) + .disableRowLevelSecurity() + .build(); + } + + PostgresModule MODULE = PostgresModule.builder() + .addTable(PostgresUserTable.TABLE) + .build(); +} diff --git a/server/data/data-postgres/src/main/java/org/apache/james/user/postgres/PostgresUsersDAO.java b/server/data/data-postgres/src/main/java/org/apache/james/user/postgres/PostgresUsersDAO.java new file mode 100644 index 00000000000..b28645f666c --- /dev/null +++ b/server/data/data-postgres/src/main/java/org/apache/james/user/postgres/PostgresUsersDAO.java @@ -0,0 +1,251 @@ +/**************************************************************** + * 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.user.postgres; + +import static org.apache.james.backends.postgres.utils.PostgresExecutor.DEFAULT_INJECT; +import static org.apache.james.backends.postgres.utils.PostgresExecutor.EAGER_FETCH; +import static org.apache.james.user.postgres.PostgresUserModule.PostgresUserTable.ALGORITHM; +import static org.apache.james.user.postgres.PostgresUserModule.PostgresUserTable.AUTHORIZED_USERS; +import static org.apache.james.user.postgres.PostgresUserModule.PostgresUserTable.DELEGATED_USERS; +import static org.apache.james.user.postgres.PostgresUserModule.PostgresUserTable.HASHED_PASSWORD; +import static org.apache.james.user.postgres.PostgresUserModule.PostgresUserTable.TABLE; +import static org.apache.james.user.postgres.PostgresUserModule.PostgresUserTable.TABLE_NAME; +import static org.apache.james.user.postgres.PostgresUserModule.PostgresUserTable.USERNAME; +import static org.apache.james.user.postgres.PostgresUserModule.PostgresUserTable.USERNAME_PRIMARY_KEY; +import static org.jooq.impl.DSL.count; + +import java.util.Iterator; +import java.util.Optional; + +import jakarta.inject.Inject; +import jakarta.inject.Named; + +import org.apache.james.backends.postgres.utils.PostgresExecutor; +import org.apache.james.core.Username; +import org.apache.james.user.api.AlreadyExistInUsersRepositoryException; +import org.apache.james.user.api.UsersRepositoryException; +import org.apache.james.user.api.model.User; +import org.apache.james.user.lib.UsersDAO; +import org.apache.james.user.lib.model.Algorithm; +import org.apache.james.user.lib.model.DefaultUser; +import org.jooq.DSLContext; +import org.jooq.Field; +import org.jooq.Record; +import org.jooq.UpdateConditionStep; +import org.jooq.impl.DSL; + +import com.google.common.base.Preconditions; +import com.google.common.collect.ImmutableList; + +import reactor.core.publisher.Flux; +import reactor.core.publisher.Mono; + +public class PostgresUsersDAO implements UsersDAO { + private final PostgresExecutor postgresExecutor; + private final Algorithm algorithm; + private final Algorithm.HashingMode fallbackHashingMode; + + @Inject + public PostgresUsersDAO(@Named(DEFAULT_INJECT) PostgresExecutor postgresExecutor, + PostgresUsersRepositoryConfiguration postgresUsersRepositoryConfiguration) { + this.postgresExecutor = postgresExecutor; + this.algorithm = postgresUsersRepositoryConfiguration.getPreferredAlgorithm(); + this.fallbackHashingMode = postgresUsersRepositoryConfiguration.getFallbackHashingMode(); + } + + @Override + public Optional getUserByName(Username name) { + return getUserByNameReactive(name).blockOptional(); + } + + private Mono getUserByNameReactive(Username name) { + return postgresExecutor.executeRow(dsl -> Mono.from(dsl.selectFrom(TABLE_NAME) + .where(USERNAME.eq(name.asString())))) + .map(record -> new DefaultUser(name, record.get(HASHED_PASSWORD), + Algorithm.of(record.get(ALGORITHM), fallbackHashingMode), algorithm)); + } + + @Override + public void updateUser(User user) throws UsersRepositoryException { + Preconditions.checkArgument(user instanceof DefaultUser); + DefaultUser defaultUser = (DefaultUser) user; + + boolean executed = postgresExecutor.executeRow(dslContext -> Mono.from(dslContext.update(TABLE_NAME) + .set(HASHED_PASSWORD, defaultUser.getHashedPassword()) + .set(ALGORITHM, defaultUser.getHashAlgorithm().asString()) + .where(USERNAME.eq(user.getUserName().asString())) + .returning(USERNAME))) + .map(record -> record.get(USERNAME)) + .blockOptional() + .isPresent(); + + if (!executed) { + throw new UsersRepositoryException("Unable to update user"); + } + } + + @Override + public void removeUser(Username name) throws UsersRepositoryException { + boolean executed = postgresExecutor.executeRow(dslContext -> Mono.from(dslContext.deleteFrom(TABLE_NAME) + .where(USERNAME.eq(name.asString())) + .returning(USERNAME))) + .map(record -> record.get(USERNAME)) + .blockOptional() + .isPresent(); + + if (!executed) { + throw new UsersRepositoryException("Unable to update user"); + } + } + + @Override + public boolean contains(Username name) { + return containsReactive(name).block(); + } + + @Override + public Mono containsReactive(Username name) { + return postgresExecutor.executeExists(dsl -> dsl.selectOne().from(TABLE_NAME).where(USERNAME.eq(name.asString()))); + } + + @Override + public int countUsers() { + return postgresExecutor.executeRow(dsl -> Mono.from(dsl.select(count()).from(TABLE_NAME))) + .map(record -> record.get(0, Integer.class)) + .block(); + } + + @Override + public Iterator list() throws UsersRepositoryException { + return listReactive() + .toIterable() + .iterator(); + } + + @Override + public Flux listReactive() { + return postgresExecutor.executeRows(dslContext -> Flux.from(dslContext.select(USERNAME) + .from(TABLE_NAME)), EAGER_FETCH) + .map(record -> Username.of(record.get(USERNAME))); + } + + @Override + public void addUser(Username username, String password) { + DefaultUser user = new DefaultUser(username, algorithm, algorithm); + user.setPassword(password); + + postgresExecutor.executeRow(dslContext -> Mono.from(dslContext.insertInto(TABLE_NAME, USERNAME, HASHED_PASSWORD, ALGORITHM) + .values(user.getUserName().asString(), user.getHashedPassword(), user.getHashAlgorithm().asString()) + .onConflictOnConstraint(USERNAME_PRIMARY_KEY) + .doNothing() + .returning(USERNAME))) + .switchIfEmpty(Mono.error(new AlreadyExistInUsersRepositoryException("User with username " + username + " already exist!"))) + .block(); + } + + public Mono addAuthorizedUser(Username baseUser, Username userWithAccess, boolean targetUserExists) { + return addUserToList(AUTHORIZED_USERS, baseUser, userWithAccess) + .then(addDelegatedUser(baseUser, userWithAccess, targetUserExists)); + } + + private Mono addDelegatedUser(Username baseUser, Username userWithAccess, boolean targetUserExists) { + if (targetUserExists) { + return addUserToList(DELEGATED_USERS, userWithAccess, baseUser); + } else { + return Mono.empty(); + } + } + + private Mono addUserToList(Field field, Username baseUser, Username targetUser) { + String fullAuthorizedUsersColumnName = TABLE.getName() + "." + field.getName(); + return postgresExecutor.executeVoid(dslContext -> + Mono.from(dslContext.insertInto(TABLE_NAME) + .set(USERNAME, baseUser.asString()) + .set(field, DSL.array(targetUser.asString())) + .onConflict(USERNAME) + .doUpdate() + .set(DSL.field(field.getName()), + (Object) DSL.field("array_append(coalesce(" + fullAuthorizedUsersColumnName + ", array[]::varchar[]), ?)", + targetUser.asString())) + .where(DSL.field(fullAuthorizedUsersColumnName).isNull() + .or(DSL.field(fullAuthorizedUsersColumnName).notContains(new String[]{targetUser.asString()}))))); + } + + public Mono removeAuthorizedUser(Username baseUser, Username userWithAccess) { + return removeUserInAuthorizedList(baseUser, userWithAccess) + .then(removeUserInDelegatedList(userWithAccess, baseUser)); + } + + public Mono removeDelegatedToUser(Username baseUser, Username delegatedToUser) { + return removeUserInDelegatedList(baseUser, delegatedToUser) + .then(removeUserInAuthorizedList(delegatedToUser, baseUser)); + } + + private Mono removeUserInAuthorizedList(Username baseUser, Username targetUser) { + return removeUserFromList(AUTHORIZED_USERS, baseUser, targetUser); + } + + private Mono removeUserInDelegatedList(Username baseUser, Username targetUser) { + return removeUserFromList(DELEGATED_USERS, baseUser, targetUser); + } + + private Mono removeUserFromList(Field field, Username baseUser, Username targetUser) { + return postgresExecutor.executeVoid(dslContext -> + Mono.from(createQueryRemoveUserFromList(dslContext, field, baseUser, targetUser))); + } + + private UpdateConditionStep createQueryRemoveUserFromList(DSLContext dslContext, Field field, Username baseUser, Username targetUser) { + return dslContext.update(TABLE_NAME) + .set(DSL.field(field.getName()), + (Object) DSL.field("array_remove(" + field.getName() + ", ?)", + targetUser.asString())) + .where(USERNAME.eq(baseUser.asString())) + .and(DSL.field(field.getName()).isNotNull()); + } + + public Mono removeAllAuthorizedUsers(Username baseUser) { + return getAuthorizedUsers(baseUser) + .collect(ImmutableList.toImmutableList()) + .flatMap(usernames -> postgresExecutor.executeVoid(dslContext -> + Mono.from(dslContext.batch(usernames.stream() + .map(username -> createQueryRemoveUserFromList(dslContext, DELEGATED_USERS, username, baseUser)) + .collect(ImmutableList.toImmutableList()))))) + .then(postgresExecutor.executeVoid(dslContext -> Mono.from(dslContext.update(TABLE_NAME) + .setNull(AUTHORIZED_USERS) + .where(USERNAME.eq(baseUser.asString()))))); + } + + public Flux getAuthorizedUsers(Username name) { + return getUsersFromList(AUTHORIZED_USERS, name); + } + + public Flux getDelegatedToUsers(Username name) { + return getUsersFromList(DELEGATED_USERS, name); + } + + public Flux getUsersFromList(Field field, Username name) { + return postgresExecutor.executeRow(dslContext -> Mono.from(dslContext.select(field) + .from(TABLE_NAME) + .where(USERNAME.eq(name.asString())))) + .flatMapMany(record -> Optional.ofNullable(record.get(field)) + .map(Flux::fromArray).orElse(Flux.empty())) + .map(Username::of); + } +} diff --git a/server/data/data-postgres/src/main/java/org/apache/james/user/postgres/PostgresUsersRepository.java b/server/data/data-postgres/src/main/java/org/apache/james/user/postgres/PostgresUsersRepository.java new file mode 100644 index 00000000000..472bb277af2 --- /dev/null +++ b/server/data/data-postgres/src/main/java/org/apache/james/user/postgres/PostgresUsersRepository.java @@ -0,0 +1,32 @@ +/**************************************************************** + * 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.user.postgres; + +import jakarta.inject.Inject; + +import org.apache.james.domainlist.api.DomainList; +import org.apache.james.user.lib.UsersRepositoryImpl; + +public class PostgresUsersRepository extends UsersRepositoryImpl { + @Inject + public PostgresUsersRepository(DomainList domainList, PostgresUsersDAO usersDAO) { + super(domainList, usersDAO); + } +} diff --git a/server/data/data-postgres/src/main/java/org/apache/james/user/postgres/PostgresUsersRepositoryConfiguration.java b/server/data/data-postgres/src/main/java/org/apache/james/user/postgres/PostgresUsersRepositoryConfiguration.java new file mode 100644 index 00000000000..8e891c185ff --- /dev/null +++ b/server/data/data-postgres/src/main/java/org/apache/james/user/postgres/PostgresUsersRepositoryConfiguration.java @@ -0,0 +1,57 @@ +/**************************************************************** + * 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.user.postgres; + +import org.apache.commons.configuration2.HierarchicalConfiguration; +import org.apache.commons.configuration2.ex.ConfigurationException; +import org.apache.commons.configuration2.tree.ImmutableNode; +import org.apache.james.user.lib.model.Algorithm; +import org.apache.james.user.lib.model.Algorithm.HashingMode; + +public class PostgresUsersRepositoryConfiguration { + public static final String DEFAULT_ALGORITHM = "PBKDF2-SHA512"; + public static final String DEFAULT_HASHING_MODE = HashingMode.PLAIN.name(); + + public static final PostgresUsersRepositoryConfiguration DEFAULT = new PostgresUsersRepositoryConfiguration( + Algorithm.of(DEFAULT_ALGORITHM), HashingMode.parse(DEFAULT_HASHING_MODE) + ); + + private final Algorithm preferredAlgorithm; + private final HashingMode fallbackHashingMode; + + public PostgresUsersRepositoryConfiguration(Algorithm preferredAlgorithm, HashingMode fallbackHashingMode) { + this.preferredAlgorithm = preferredAlgorithm; + this.fallbackHashingMode = fallbackHashingMode; + } + + public Algorithm getPreferredAlgorithm() { + return preferredAlgorithm; + } + + public HashingMode getFallbackHashingMode() { + return fallbackHashingMode; + } + + public static PostgresUsersRepositoryConfiguration from(HierarchicalConfiguration config) throws ConfigurationException { + return new PostgresUsersRepositoryConfiguration( + Algorithm.of(config.getString("algorithm", DEFAULT_ALGORITHM)), + HashingMode.parse(config.getString("hashingMode", DEFAULT_HASHING_MODE))); + } +} diff --git a/server/data/data-postgres/src/main/java/org/apache/james/vacation/postgres/PostgresNotificationRegistry.java b/server/data/data-postgres/src/main/java/org/apache/james/vacation/postgres/PostgresNotificationRegistry.java new file mode 100644 index 00000000000..d03ceff00f2 --- /dev/null +++ b/server/data/data-postgres/src/main/java/org/apache/james/vacation/postgres/PostgresNotificationRegistry.java @@ -0,0 +1,79 @@ +/**************************************************************** + * 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.vacation.postgres; + +import java.time.ZonedDateTime; +import java.util.Optional; + +import jakarta.inject.Inject; + +import org.apache.james.backends.postgres.utils.PostgresExecutor; +import org.apache.james.core.Username; +import org.apache.james.util.date.ZonedDateTimeProvider; +import org.apache.james.vacation.api.AccountId; +import org.apache.james.vacation.api.NotificationRegistry; +import org.apache.james.vacation.api.RecipientId; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + +import reactor.core.publisher.Mono; + +public class PostgresNotificationRegistry implements NotificationRegistry { + private static final Logger LOGGER = LoggerFactory.getLogger(PostgresNotificationRegistry.class); + + private final ZonedDateTimeProvider zonedDateTimeProvider; + private final PostgresExecutor.Factory executorFactory; + + @Inject + public PostgresNotificationRegistry(ZonedDateTimeProvider zonedDateTimeProvider, + PostgresExecutor.Factory executorFactory) { + this.zonedDateTimeProvider = zonedDateTimeProvider; + this.executorFactory = executorFactory; + } + + @Override + public Mono register(AccountId accountId, RecipientId recipientId, Optional expiryDate) { + if (isValid(expiryDate)) { + return notificationRegistryDAO(accountId).register(accountId, recipientId, expiryDate); + } else { + LOGGER.warn("Invalid vacation notification expiry date for {} {} : {}", accountId, recipientId, expiryDate); + return Mono.empty(); + } + } + + @Override + public Mono isRegistered(AccountId accountId, RecipientId recipientId) { + return notificationRegistryDAO(accountId).isRegistered(accountId, recipientId); + } + + @Override + public Mono flush(AccountId accountId) { + return notificationRegistryDAO(accountId).flush(accountId); + } + + private boolean isValid(Optional expiryDate) { + return expiryDate.isEmpty() || expiryDate.get().isAfter(zonedDateTimeProvider.get()); + } + + private PostgresNotificationRegistryDAO notificationRegistryDAO(AccountId accountId) { + return new PostgresNotificationRegistryDAO(executorFactory.create(Username.of(accountId.getIdentifier()).getDomainPart()), + zonedDateTimeProvider); + } +} diff --git a/server/data/data-postgres/src/main/java/org/apache/james/vacation/postgres/PostgresNotificationRegistryDAO.java b/server/data/data-postgres/src/main/java/org/apache/james/vacation/postgres/PostgresNotificationRegistryDAO.java new file mode 100644 index 00000000000..8ae01ce36f2 --- /dev/null +++ b/server/data/data-postgres/src/main/java/org/apache/james/vacation/postgres/PostgresNotificationRegistryDAO.java @@ -0,0 +1,71 @@ +/**************************************************************** + * 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.vacation.postgres; + +import static org.apache.james.vacation.postgres.PostgresVacationModule.PostgresVacationNotificationRegistryTable.ACCOUNT_ID; +import static org.apache.james.vacation.postgres.PostgresVacationModule.PostgresVacationNotificationRegistryTable.EXPIRY_DATE; +import static org.apache.james.vacation.postgres.PostgresVacationModule.PostgresVacationNotificationRegistryTable.RECIPIENT_ID; +import static org.apache.james.vacation.postgres.PostgresVacationModule.PostgresVacationNotificationRegistryTable.TABLE_NAME; + +import java.time.LocalDateTime; +import java.time.ZoneOffset; +import java.time.ZonedDateTime; +import java.util.Optional; + +import org.apache.james.backends.postgres.utils.PostgresExecutor; +import org.apache.james.util.date.ZonedDateTimeProvider; +import org.apache.james.vacation.api.AccountId; +import org.apache.james.vacation.api.RecipientId; + +import reactor.core.publisher.Mono; + +public class PostgresNotificationRegistryDAO { + private final PostgresExecutor postgresExecutor; + private final ZonedDateTimeProvider zonedDateTimeProvider; + + public PostgresNotificationRegistryDAO(PostgresExecutor postgresExecutor, + ZonedDateTimeProvider zonedDateTimeProvider) { + this.postgresExecutor = postgresExecutor; + this.zonedDateTimeProvider = zonedDateTimeProvider; + } + + public Mono register(AccountId accountId, RecipientId recipientId, Optional expiryDate) { + return postgresExecutor.executeVoid(dsl -> Mono.from(dsl.insertInto(TABLE_NAME) + .set(ACCOUNT_ID, accountId.getIdentifier()) + .set(RECIPIENT_ID, recipientId.getAsString()) + .set(EXPIRY_DATE, expiryDate.map(zonedDateTime -> zonedDateTime.withZoneSameInstant(ZoneOffset.UTC).toLocalDateTime()) + .orElse(null)))); + } + + public Mono isRegistered(AccountId accountId, RecipientId recipientId) { + LocalDateTime currentUTCTime = zonedDateTimeProvider.get().withZoneSameInstant(ZoneOffset.UTC).toLocalDateTime(); + + return postgresExecutor.executeExists(dsl -> dsl.selectOne() + .from(TABLE_NAME) + .where(ACCOUNT_ID.eq(accountId.getIdentifier()), + RECIPIENT_ID.eq(recipientId.getAsString()), + EXPIRY_DATE.ge(currentUTCTime).or(EXPIRY_DATE.isNull()))); + } + + public Mono flush(AccountId accountId) { + return postgresExecutor.executeVoid(dsl -> Mono.from(dsl.deleteFrom(TABLE_NAME) + .where(ACCOUNT_ID.eq(accountId.getIdentifier())))); + } +} diff --git a/server/data/data-postgres/src/main/java/org/apache/james/vacation/postgres/PostgresVacationModule.java b/server/data/data-postgres/src/main/java/org/apache/james/vacation/postgres/PostgresVacationModule.java new file mode 100644 index 00000000000..14fb05df0a4 --- /dev/null +++ b/server/data/data-postgres/src/main/java/org/apache/james/vacation/postgres/PostgresVacationModule.java @@ -0,0 +1,91 @@ +/**************************************************************** + * 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.vacation.postgres; + +import java.time.LocalDateTime; + +import org.apache.james.backends.postgres.PostgresCommons; +import org.apache.james.backends.postgres.PostgresIndex; +import org.apache.james.backends.postgres.PostgresModule; +import org.apache.james.backends.postgres.PostgresTable; +import org.jooq.Field; +import org.jooq.Record; +import org.jooq.Table; +import org.jooq.impl.DSL; +import org.jooq.impl.SQLDataType; + +public interface PostgresVacationModule { + interface PostgresVacationResponseTable { + Table TABLE_NAME = DSL.table("vacation_response"); + + Field ACCOUNT_ID = DSL.field("account_id", SQLDataType.VARCHAR.notNull()); + Field IS_ENABLED = DSL.field("is_enabled", SQLDataType.BOOLEAN.notNull() + .defaultValue(DSL.field("false", SQLDataType.BOOLEAN))); + Field FROM_DATE = DSL.field("from_date", PostgresCommons.DataTypes.TIMESTAMP); + Field TO_DATE = DSL.field("to_date", PostgresCommons.DataTypes.TIMESTAMP); + Field TEXT = DSL.field("text", SQLDataType.VARCHAR); + Field SUBJECT = DSL.field("subject", SQLDataType.VARCHAR); + Field HTML = DSL.field("html", SQLDataType.VARCHAR); + + PostgresTable TABLE = PostgresTable.name(TABLE_NAME.getName()) + .createTableStep(((dsl, tableName) -> dsl.createTableIfNotExists(tableName) + .column(ACCOUNT_ID) + .column(IS_ENABLED) + .column(FROM_DATE) + .column(TO_DATE) + .column(TEXT) + .column(SUBJECT) + .column(HTML) + .constraint(DSL.primaryKey(ACCOUNT_ID)))) + .supportsRowLevelSecurity() + .build(); + } + + interface PostgresVacationNotificationRegistryTable { + Table TABLE_NAME = DSL.table("vacation_notification_registry"); + + Field ACCOUNT_ID = DSL.field("account_id", SQLDataType.VARCHAR.notNull()); + Field RECIPIENT_ID = DSL.field("recipient_id", SQLDataType.VARCHAR.notNull()); + Field EXPIRY_DATE = DSL.field("expiry_date", PostgresCommons.DataTypes.TIMESTAMP); + + PostgresTable TABLE = PostgresTable.name(TABLE_NAME.getName()) + .createTableStep(((dsl, tableName) -> dsl.createTableIfNotExists(tableName) + .column(ACCOUNT_ID) + .column(RECIPIENT_ID) + .column(EXPIRY_DATE) + .constraint(DSL.primaryKey(ACCOUNT_ID, RECIPIENT_ID)))) + .supportsRowLevelSecurity() + .build(); + + PostgresIndex ACCOUNT_ID_INDEX = PostgresIndex.name("vacation_notification_registry_accountid_index") + .createIndexStep((dsl, indexName) -> dsl.createIndexIfNotExists(indexName) + .on(TABLE_NAME, ACCOUNT_ID)); + + PostgresIndex FULL_COMPOSITE_INDEX = PostgresIndex.name("vnr_accountid_recipientid_expirydate_index") + .createIndexStep((dsl, indexName) -> dsl.createIndexIfNotExists(indexName) + .on(TABLE_NAME, ACCOUNT_ID, RECIPIENT_ID, EXPIRY_DATE)); + } + + PostgresModule MODULE = PostgresModule.builder() + .addTable(PostgresVacationResponseTable.TABLE) + .addTable(PostgresVacationNotificationRegistryTable.TABLE) + .addIndex(PostgresVacationNotificationRegistryTable.ACCOUNT_ID_INDEX, PostgresVacationNotificationRegistryTable.FULL_COMPOSITE_INDEX) + .build(); +} diff --git a/server/data/data-postgres/src/main/java/org/apache/james/vacation/postgres/PostgresVacationRepository.java b/server/data/data-postgres/src/main/java/org/apache/james/vacation/postgres/PostgresVacationRepository.java new file mode 100644 index 00000000000..82bb0ced556 --- /dev/null +++ b/server/data/data-postgres/src/main/java/org/apache/james/vacation/postgres/PostgresVacationRepository.java @@ -0,0 +1,64 @@ +/**************************************************************** + * 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.vacation.postgres; + +import jakarta.inject.Inject; +import jakarta.inject.Singleton; + +import org.apache.james.backends.postgres.utils.PostgresExecutor; +import org.apache.james.core.Username; +import org.apache.james.vacation.api.AccountId; +import org.apache.james.vacation.api.Vacation; +import org.apache.james.vacation.api.VacationPatch; +import org.apache.james.vacation.api.VacationRepository; + +import com.google.common.base.Preconditions; + +import reactor.core.publisher.Mono; + +public class PostgresVacationRepository implements VacationRepository { + private final PostgresExecutor.Factory executorFactory; + + @Inject + @Singleton + public PostgresVacationRepository(PostgresExecutor.Factory executorFactory) { + this.executorFactory = executorFactory; + } + + @Override + public Mono modifyVacation(AccountId accountId, VacationPatch vacationPatch) { + Preconditions.checkNotNull(accountId); + Preconditions.checkNotNull(vacationPatch); + if (vacationPatch.isIdentity()) { + return Mono.empty(); + } else { + return vacationResponseDao(accountId).modifyVacation(accountId, vacationPatch); + } + } + + @Override + public Mono retrieveVacation(AccountId accountId) { + return vacationResponseDao(accountId).retrieveVacation(accountId).map(optional -> optional.orElse(DEFAULT_VACATION)); + } + + private PostgresVacationResponseDAO vacationResponseDao(AccountId accountId) { + return new PostgresVacationResponseDAO(executorFactory.create(Username.of(accountId.getIdentifier()).getDomainPart())); + } +} diff --git a/server/data/data-postgres/src/main/java/org/apache/james/vacation/postgres/PostgresVacationResponseDAO.java b/server/data/data-postgres/src/main/java/org/apache/james/vacation/postgres/PostgresVacationResponseDAO.java new file mode 100644 index 00000000000..52e4328ffa6 --- /dev/null +++ b/server/data/data-postgres/src/main/java/org/apache/james/vacation/postgres/PostgresVacationResponseDAO.java @@ -0,0 +1,156 @@ +/**************************************************************** + * 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.vacation.postgres; + +import static org.apache.james.backends.postgres.PostgresCommons.LOCAL_DATE_TIME_ZONED_DATE_TIME_FUNCTION; +import static org.apache.james.vacation.postgres.PostgresVacationModule.PostgresVacationResponseTable.ACCOUNT_ID; +import static org.apache.james.vacation.postgres.PostgresVacationModule.PostgresVacationResponseTable.FROM_DATE; +import static org.apache.james.vacation.postgres.PostgresVacationModule.PostgresVacationResponseTable.HTML; +import static org.apache.james.vacation.postgres.PostgresVacationModule.PostgresVacationResponseTable.IS_ENABLED; +import static org.apache.james.vacation.postgres.PostgresVacationModule.PostgresVacationResponseTable.SUBJECT; +import static org.apache.james.vacation.postgres.PostgresVacationModule.PostgresVacationResponseTable.TABLE_NAME; +import static org.apache.james.vacation.postgres.PostgresVacationModule.PostgresVacationResponseTable.TEXT; +import static org.apache.james.vacation.postgres.PostgresVacationModule.PostgresVacationResponseTable.TO_DATE; + +import java.time.LocalDateTime; +import java.time.ZoneId; +import java.time.ZonedDateTime; +import java.util.Optional; +import java.util.function.BiFunction; +import java.util.function.Function; +import java.util.stream.Stream; + +import org.apache.james.backends.postgres.utils.PostgresExecutor; +import org.apache.james.util.ValuePatch; +import org.apache.james.vacation.api.AccountId; +import org.apache.james.vacation.api.Vacation; +import org.apache.james.vacation.api.VacationPatch; +import org.jooq.DSLContext; +import org.jooq.Field; +import org.jooq.InsertOnDuplicateSetMoreStep; +import org.jooq.InsertOnDuplicateSetStep; +import org.jooq.InsertSetMoreStep; +import org.jooq.Record; + +import reactor.core.publisher.Mono; + +public class PostgresVacationResponseDAO { + private final PostgresExecutor postgresExecutor; + + public PostgresVacationResponseDAO(PostgresExecutor postgresExecutor) { + this.postgresExecutor = postgresExecutor; + } + + public Mono modifyVacation(AccountId accountId, VacationPatch vacationPatch) { + return postgresExecutor.executeVoid(dsl -> { + if (vacationPatch.isIdentity()) { + return Mono.from(insertVacationQuery(accountId, vacationPatch, dsl) + .onConflictDoNothing()); + } else { + return Mono.from(withUpdateOnConflict(vacationPatch, insertVacationQuery(accountId, vacationPatch, dsl))); + } + }); + } + + private InsertSetMoreStep insertVacationQuery(AccountId accountId, VacationPatch vacationPatch, DSLContext dsl) { + InsertSetMoreStep baseInsert = dsl.insertInto(TABLE_NAME) + .set(ACCOUNT_ID, accountId.getIdentifier()); + + return Stream.of( + applyInsertForField(IS_ENABLED, VacationPatch::getIsEnabled), + applyInsertForField(SUBJECT, VacationPatch::getSubject), + applyInsertForField(HTML, VacationPatch::getHtmlBody), + applyInsertForField(TEXT, VacationPatch::getTextBody), + applyInsertForFieldZonedDateTime(FROM_DATE, VacationPatch::getFromDate), + applyInsertForFieldZonedDateTime(TO_DATE, VacationPatch::getToDate)) + .reduce((vacation, insert) -> insert, + (a, b) -> (vacation, insert) -> b.apply(vacation, a.apply(vacation, insert))) + .apply(vacationPatch, baseInsert); + } + + private InsertOnDuplicateSetMoreStep withUpdateOnConflict(VacationPatch vacationPatch, InsertSetMoreStep insertVacation) { + InsertOnDuplicateSetStep baseUpdateIfConflict = insertVacation.onConflict(ACCOUNT_ID) + .doUpdate(); + + return (InsertOnDuplicateSetMoreStep) Stream.of( + applyUpdateOnConflictForField(IS_ENABLED, VacationPatch::getIsEnabled), + applyUpdateOnConflictForField(SUBJECT, VacationPatch::getSubject), + applyUpdateOnConflictForField(HTML, VacationPatch::getHtmlBody), + applyUpdateOnConflictForField(TEXT, VacationPatch::getTextBody), + applyUpdateOnConflictForFieldZonedDateTime(FROM_DATE, VacationPatch::getFromDate), + applyUpdateOnConflictForFieldZonedDateTime(TO_DATE, VacationPatch::getToDate)) + .reduce((vacation, updateOnConflict) -> updateOnConflict, + (a, b) -> (vacation, updateOnConflict) -> b.apply(vacation, a.apply(vacation, updateOnConflict))) + .apply(vacationPatch, baseUpdateIfConflict); + } + + private BiFunction, InsertSetMoreStep> applyInsertForField(Field field, Function> getter) { + return (vacation, insert) -> + getter.apply(vacation) + .mapNotKeptToOptional(optionalValue -> applyInsertForField(field, optionalValue, insert)) + .orElse(insert); + } + + private BiFunction, InsertSetMoreStep> applyInsertForFieldZonedDateTime(Field field, Function> getter) { + return (vacation, insert) -> + getter.apply(vacation) + .mapNotKeptToOptional(optionalValue -> applyInsertForField(field, + optionalValue.map(zonedDateTime -> zonedDateTime.withZoneSameInstant(ZoneId.of("UTC")).toLocalDateTime()), + insert)) + .orElse(insert); + } + + private InsertSetMoreStep applyInsertForField(Field field, Optional value, InsertSetMoreStep insert) { + return insert.set(field, value.orElse(null)); + } + + private BiFunction, InsertOnDuplicateSetStep> applyUpdateOnConflictForField(Field field, Function> getter) { + return (vacation, update) -> + getter.apply(vacation) + .mapNotKeptToOptional(optionalValue -> applyUpdateOnConflictForField(field, optionalValue, update)) + .orElse(update); + } + + private BiFunction, InsertOnDuplicateSetStep> applyUpdateOnConflictForFieldZonedDateTime(Field field, Function> getter) { + return (vacation, update) -> + getter.apply(vacation) + .mapNotKeptToOptional(optionalValue -> applyUpdateOnConflictForField(field, + optionalValue.map(zonedDateTime -> zonedDateTime.withZoneSameInstant(ZoneId.of("UTC")).toLocalDateTime()), + update)) + .orElse(update); + } + + private InsertOnDuplicateSetStep applyUpdateOnConflictForField(Field field, Optional value, InsertOnDuplicateSetStep updateOnConflict) { + return updateOnConflict.set(field, value.orElse(null)); + } + + public Mono> retrieveVacation(AccountId accountId) { + return postgresExecutor.executeSingleRowOptional(dsl -> dsl.selectFrom(TABLE_NAME) + .where(ACCOUNT_ID.eq(accountId.getIdentifier()))) + .map(recordOptional -> recordOptional.map(record -> Vacation.builder() + .enabled(record.get(IS_ENABLED)) + .fromDate(Optional.ofNullable(LOCAL_DATE_TIME_ZONED_DATE_TIME_FUNCTION.apply(record.get(FROM_DATE, LocalDateTime.class)))) + .toDate(Optional.ofNullable(LOCAL_DATE_TIME_ZONED_DATE_TIME_FUNCTION.apply(record.get(TO_DATE, LocalDateTime.class)))) + .subject(Optional.ofNullable(record.get(SUBJECT))) + .textBody(Optional.ofNullable(record.get(TEXT))) + .htmlBody(Optional.ofNullable(record.get(HTML))) + .build())); + } +} \ No newline at end of file diff --git a/server/data/data-postgres/src/reporting-site/site.xml b/server/data/data-postgres/src/reporting-site/site.xml new file mode 100644 index 00000000000..d9191644908 --- /dev/null +++ b/server/data/data-postgres/src/reporting-site/site.xml @@ -0,0 +1,29 @@ + + + + + + + + + + + + diff --git a/server/data/data-postgres/src/test/java/org/apache/james/domainlist/postgres/PostgresDomainListTest.java b/server/data/data-postgres/src/test/java/org/apache/james/domainlist/postgres/PostgresDomainListTest.java new file mode 100644 index 00000000000..a7136faf16c --- /dev/null +++ b/server/data/data-postgres/src/test/java/org/apache/james/domainlist/postgres/PostgresDomainListTest.java @@ -0,0 +1,48 @@ +/**************************************************************** + * 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.domainlist.postgres; + +import org.apache.james.backends.postgres.PostgresExtension; +import org.apache.james.domainlist.api.DomainList; +import org.apache.james.domainlist.lib.DomainListConfiguration; +import org.apache.james.domainlist.lib.DomainListContract; +import org.junit.jupiter.api.BeforeEach; +import org.junit.jupiter.api.extension.RegisterExtension; + +public class PostgresDomainListTest implements DomainListContract { + @RegisterExtension + static PostgresExtension postgresExtension = PostgresExtension.withoutRowLevelSecurity(PostgresDomainModule.MODULE); + + PostgresDomainList domainList; + + @BeforeEach + public void setup() throws Exception { + domainList = new PostgresDomainList(getDNSServer("localhost"), postgresExtension.getDefaultPostgresExecutor()); + domainList.configure(DomainListConfiguration.builder() + .autoDetect(false) + .autoDetectIp(false) + .build()); + } + + @Override + public DomainList domainList() { + return domainList; + } +} diff --git a/server/data/data-postgres/src/test/java/org/apache/james/droplists/postgres/PostgresDropListsTest.java b/server/data/data-postgres/src/test/java/org/apache/james/droplists/postgres/PostgresDropListsTest.java new file mode 100644 index 00000000000..c0697ed2656 --- /dev/null +++ b/server/data/data-postgres/src/test/java/org/apache/james/droplists/postgres/PostgresDropListsTest.java @@ -0,0 +1,43 @@ +/**************************************************************** + * 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.droplists.postgres; + +import org.apache.james.backends.postgres.PostgresExtension; +import org.apache.james.droplists.api.DropList; +import org.apache.james.droplists.api.DropListContract; +import org.junit.jupiter.api.BeforeEach; +import org.junit.jupiter.api.extension.RegisterExtension; + +class PostgresDropListsTest implements DropListContract { + @RegisterExtension + static PostgresExtension postgresExtension = PostgresExtension.withoutRowLevelSecurity(PostgresDropListModule.MODULE); + + PostgresDropList dropList; + + @BeforeEach + void setup() { + dropList = new PostgresDropList(postgresExtension.getDefaultPostgresExecutor()); + } + + @Override + public DropList dropList() { + return dropList; + } +} \ No newline at end of file diff --git a/server/data/data-postgres/src/test/java/org/apache/james/mailrepository/postgres/PostgresMailRepositoryBlobReferenceSourceTest.java b/server/data/data-postgres/src/test/java/org/apache/james/mailrepository/postgres/PostgresMailRepositoryBlobReferenceSourceTest.java new file mode 100644 index 00000000000..49e5e9a6b68 --- /dev/null +++ b/server/data/data-postgres/src/test/java/org/apache/james/mailrepository/postgres/PostgresMailRepositoryBlobReferenceSourceTest.java @@ -0,0 +1,94 @@ +/**************************************************************** + * 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.mailrepository.postgres; + +import static org.assertj.core.api.Assertions.assertThat; + +import jakarta.mail.MessagingException; + +import org.apache.james.backends.postgres.PostgresExtension; +import org.apache.james.backends.postgres.PostgresModule; +import org.apache.james.blob.api.BlobId; +import org.apache.james.blob.api.BlobStore; +import org.apache.james.blob.api.PlainBlobId; +import org.apache.james.blob.mail.MimeMessageStore; +import org.apache.james.blob.memory.MemoryBlobStoreFactory; +import org.apache.james.core.builder.MimeMessageBuilder; +import org.apache.james.mailrepository.api.MailKey; +import org.apache.james.mailrepository.api.MailRepositoryPath; +import org.apache.james.mailrepository.api.MailRepositoryUrl; +import org.apache.james.mailrepository.api.Protocol; +import org.apache.james.server.core.MailImpl; +import org.apache.mailet.Attribute; +import org.junit.jupiter.api.BeforeEach; +import org.junit.jupiter.api.Test; +import org.junit.jupiter.api.extension.RegisterExtension; + +public class PostgresMailRepositoryBlobReferenceSourceTest { + @RegisterExtension + static PostgresExtension postgresExtension = PostgresExtension.withoutRowLevelSecurity(PostgresModule.aggregateModules(PostgresMailRepositoryModule.MODULE)); + + private static final MailRepositoryUrl URL = MailRepositoryUrl.fromPathAndProtocol(new Protocol("postgres"), MailRepositoryPath.from("testrepo")); + + PostgresMailRepositoryContentDAO postgresMailRepositoryContentDAO; + PostgresMailRepositoryBlobReferenceSource postgresMailRepositoryBlobReferenceSource; + + @BeforeEach + void beforeEach() { + BlobId.Factory factory = new PlainBlobId.Factory(); + BlobStore blobStore = MemoryBlobStoreFactory.builder() + .blobIdFactory(factory) + .defaultBucketName() + .passthrough(); + postgresMailRepositoryContentDAO = new PostgresMailRepositoryContentDAO(postgresExtension.getDefaultPostgresExecutor(), MimeMessageStore.factory(blobStore), factory); + postgresMailRepositoryBlobReferenceSource = new PostgresMailRepositoryBlobReferenceSource(postgresMailRepositoryContentDAO); + } + + @Test + void blobReferencesShouldBeEmptyByDefault() { + assertThat(postgresMailRepositoryBlobReferenceSource.listReferencedBlobs().collectList().block()) + .isEmpty(); + } + + @Test + void blobReferencesShouldReturnAllBlobs() throws Exception { + postgresMailRepositoryContentDAO.store(createMail(new MailKey("mail1")), URL); + postgresMailRepositoryContentDAO.store(createMail(new MailKey("mail2")), URL); + + assertThat(postgresMailRepositoryBlobReferenceSource.listReferencedBlobs().collectList().block()) + .hasSize(4); + } + + private MailImpl createMail(MailKey key) throws MessagingException { + return MailImpl.builder() + .name(key.asString()) + .sender("sender@localhost") + .addRecipient("rec1@domain.com") + .addRecipient("rec2@domain.com") + .addAttribute(Attribute.convertToAttribute("testAttribute", "testValue")) + .mimeMessage(MimeMessageBuilder + .mimeMessageBuilder() + .setSubject("test") + .setText("original body") + .build()) + .build(); + } + +} diff --git a/server/data/data-postgres/src/test/java/org/apache/james/mailrepository/postgres/PostgresMailRepositoryTest.java b/server/data/data-postgres/src/test/java/org/apache/james/mailrepository/postgres/PostgresMailRepositoryTest.java new file mode 100644 index 00000000000..9f2bd8033a8 --- /dev/null +++ b/server/data/data-postgres/src/test/java/org/apache/james/mailrepository/postgres/PostgresMailRepositoryTest.java @@ -0,0 +1,63 @@ +/**************************************************************** + * 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.mailrepository.postgres; + +import org.apache.james.backends.postgres.PostgresExtension; +import org.apache.james.backends.postgres.PostgresModule; +import org.apache.james.blob.api.BlobStore; +import org.apache.james.blob.api.TestBlobId; +import org.apache.james.blob.mail.MimeMessageStore; +import org.apache.james.blob.memory.MemoryBlobStoreFactory; +import org.apache.james.mailrepository.MailRepositoryContract; +import org.apache.james.mailrepository.api.MailRepository; +import org.apache.james.mailrepository.api.MailRepositoryPath; +import org.apache.james.mailrepository.api.MailRepositoryUrl; +import org.apache.james.mailrepository.api.Protocol; +import org.junit.jupiter.api.BeforeEach; +import org.junit.jupiter.api.extension.RegisterExtension; + +public class PostgresMailRepositoryTest implements MailRepositoryContract { + static final TestBlobId.Factory BLOB_ID_FACTORY = new TestBlobId.Factory(); + + @RegisterExtension + static PostgresExtension postgresExtension = PostgresExtension.withoutRowLevelSecurity(PostgresModule.aggregateModules(PostgresMailRepositoryModule.MODULE)); + + private PostgresMailRepository mailRepository; + + @BeforeEach + void setUp() { + mailRepository = retrieveRepository(MailRepositoryPath.from("testrepo")); + } + + @Override + public MailRepository retrieveRepository() { + return mailRepository; + } + + @Override + public PostgresMailRepository retrieveRepository(MailRepositoryPath path) { + MailRepositoryUrl url = MailRepositoryUrl.fromPathAndProtocol(new Protocol("postgres"), path); + BlobStore blobStore = MemoryBlobStoreFactory.builder() + .blobIdFactory(BLOB_ID_FACTORY) + .defaultBucketName() + .passthrough(); + return new PostgresMailRepository(url, new PostgresMailRepositoryContentDAO(postgresExtension.getDefaultPostgresExecutor(), MimeMessageStore.factory(blobStore), BLOB_ID_FACTORY)); + } +} diff --git a/server/data/data-postgres/src/test/java/org/apache/james/mailrepository/postgres/PostgresMailRepositoryUrlStoreExtension.java b/server/data/data-postgres/src/test/java/org/apache/james/mailrepository/postgres/PostgresMailRepositoryUrlStoreExtension.java new file mode 100644 index 00000000000..0454c1dc099 --- /dev/null +++ b/server/data/data-postgres/src/test/java/org/apache/james/mailrepository/postgres/PostgresMailRepositoryUrlStoreExtension.java @@ -0,0 +1,70 @@ +/**************************************************************** + * 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.mailrepository.postgres; + +import org.apache.james.backends.postgres.PostgresExtension; +import org.apache.james.backends.postgres.PostgresModule; +import org.apache.james.mailrepository.api.MailRepositoryUrlStore; +import org.junit.jupiter.api.extension.AfterAllCallback; +import org.junit.jupiter.api.extension.AfterEachCallback; +import org.junit.jupiter.api.extension.BeforeAllCallback; +import org.junit.jupiter.api.extension.BeforeEachCallback; +import org.junit.jupiter.api.extension.ExtensionContext; +import org.junit.jupiter.api.extension.ParameterContext; +import org.junit.jupiter.api.extension.ParameterResolutionException; +import org.junit.jupiter.api.extension.ParameterResolver; + +public class PostgresMailRepositoryUrlStoreExtension implements ParameterResolver, AfterEachCallback, AfterAllCallback, BeforeEachCallback, BeforeAllCallback { + private final PostgresExtension postgresExtension; + + public PostgresMailRepositoryUrlStoreExtension() { + postgresExtension = PostgresExtension.withoutRowLevelSecurity(PostgresModule.aggregateModules(PostgresMailRepositoryModule.MODULE)); + } + + @Override + public void afterEach(ExtensionContext context) { + postgresExtension.afterEach(context); + } + + @Override + public void afterAll(ExtensionContext extensionContext) throws Exception { + postgresExtension.afterAll(extensionContext); + } + + @Override + public void beforeAll(ExtensionContext extensionContext) throws Exception { + postgresExtension.beforeAll(extensionContext); + } + + @Override + public void beforeEach(ExtensionContext extensionContext) throws Exception { + postgresExtension.beforeEach(extensionContext); + } + + @Override + public boolean supportsParameter(ParameterContext parameterContext, ExtensionContext extensionContext) throws ParameterResolutionException { + return (parameterContext.getParameter().getType() == MailRepositoryUrlStore.class); + } + + @Override + public Object resolveParameter(ParameterContext parameterContext, ExtensionContext extensionContext) throws ParameterResolutionException { + return new PostgresMailRepositoryUrlStore(postgresExtension.getDefaultPostgresExecutor()); + } +} diff --git a/server/data/data-postgres/src/test/java/org/apache/james/mailrepository/postgres/PostgresMailRepositoryUrlStoreTest.java b/server/data/data-postgres/src/test/java/org/apache/james/mailrepository/postgres/PostgresMailRepositoryUrlStoreTest.java new file mode 100644 index 00000000000..ea4f034aa16 --- /dev/null +++ b/server/data/data-postgres/src/test/java/org/apache/james/mailrepository/postgres/PostgresMailRepositoryUrlStoreTest.java @@ -0,0 +1,28 @@ +/**************************************************************** + * 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.mailrepository.postgres; + +import org.apache.james.mailrepository.MailRepositoryUrlStoreContract; +import org.junit.jupiter.api.extension.ExtendWith; + +@ExtendWith(PostgresMailRepositoryUrlStoreExtension.class) +public class PostgresMailRepositoryUrlStoreTest implements MailRepositoryUrlStoreContract { + +} diff --git a/server/data/data-postgres/src/test/java/org/apache/james/rrt/postgres/PostgresRecipientRewriteTableTest.java b/server/data/data-postgres/src/test/java/org/apache/james/rrt/postgres/PostgresRecipientRewriteTableTest.java new file mode 100644 index 00000000000..21e8ac45c36 --- /dev/null +++ b/server/data/data-postgres/src/test/java/org/apache/james/rrt/postgres/PostgresRecipientRewriteTableTest.java @@ -0,0 +1,62 @@ +/**************************************************************** + * 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.rrt.postgres; + +import org.apache.james.backends.postgres.PostgresExtension; +import org.apache.james.backends.postgres.PostgresModule; +import org.apache.james.domainlist.api.mock.SimpleDomainList; +import org.apache.james.rrt.lib.AbstractRecipientRewriteTable; +import org.apache.james.rrt.lib.RecipientRewriteTableContract; +import org.apache.james.user.postgres.PostgresUserModule; +import org.apache.james.user.postgres.PostgresUsersDAO; +import org.apache.james.user.postgres.PostgresUsersRepository; +import org.apache.james.user.postgres.PostgresUsersRepositoryConfiguration; +import org.junit.jupiter.api.AfterEach; +import org.junit.jupiter.api.BeforeEach; +import org.junit.jupiter.api.extension.RegisterExtension; + +public class PostgresRecipientRewriteTableTest implements RecipientRewriteTableContract { + @RegisterExtension + static PostgresExtension postgresExtension = PostgresExtension.withoutRowLevelSecurity(PostgresModule.aggregateModules(PostgresRecipientRewriteTableModule.MODULE, PostgresUserModule.MODULE)); + + private PostgresRecipientRewriteTable postgresRecipientRewriteTable; + + @BeforeEach + void setup() throws Exception { + setUp(); + } + + @AfterEach + void teardown() throws Exception { + tearDown(); + } + + @Override + public void createRecipientRewriteTable() { + postgresRecipientRewriteTable = new PostgresRecipientRewriteTable(new PostgresRecipientRewriteTableDAO(postgresExtension.getDefaultPostgresExecutor())); + postgresRecipientRewriteTable.setUsersRepository(new PostgresUsersRepository(new SimpleDomainList(), + new PostgresUsersDAO(postgresExtension.getDefaultPostgresExecutor(), PostgresUsersRepositoryConfiguration.DEFAULT))); + } + + @Override + public AbstractRecipientRewriteTable virtualUserTable() { + return postgresRecipientRewriteTable; + } +} diff --git a/server/data/data-postgres/src/test/java/org/apache/james/rrt/postgres/PostgresStepdefs.java b/server/data/data-postgres/src/test/java/org/apache/james/rrt/postgres/PostgresStepdefs.java new file mode 100644 index 00000000000..fc14db19d65 --- /dev/null +++ b/server/data/data-postgres/src/test/java/org/apache/james/rrt/postgres/PostgresStepdefs.java @@ -0,0 +1,66 @@ +/**************************************************************** + * 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.rrt.postgres; + +import org.apache.james.backends.postgres.PostgresExtension; +import org.apache.james.backends.postgres.PostgresModule; +import org.apache.james.domainlist.api.DomainListException; +import org.apache.james.rrt.lib.AbstractRecipientRewriteTable; +import org.apache.james.rrt.lib.RecipientRewriteTableFixture; +import org.apache.james.rrt.lib.RewriteTablesStepdefs; +import org.apache.james.user.postgres.PostgresUserModule; +import org.apache.james.user.postgres.PostgresUsersDAO; +import org.apache.james.user.postgres.PostgresUsersRepository; +import org.apache.james.user.postgres.PostgresUsersRepositoryConfiguration; + +import com.github.fge.lambdas.Throwing; + +import io.cucumber.java.After; +import io.cucumber.java.Before; + +public class PostgresStepdefs { + static PostgresExtension postgresExtension = PostgresExtension.withoutRowLevelSecurity(PostgresModule.aggregateModules(PostgresRecipientRewriteTableModule.MODULE, PostgresUserModule.MODULE)); + + private final RewriteTablesStepdefs mainStepdefs; + + public PostgresStepdefs(RewriteTablesStepdefs mainStepdefs) { + this.mainStepdefs = mainStepdefs; + } + + @Before + public void setup() throws Throwable { + postgresExtension.beforeAll(null); + postgresExtension.beforeEach(null); + mainStepdefs.setUp(Throwing.supplier(this::getRecipientRewriteTable).sneakyThrow()); + } + + @After + public void tearDown() { + postgresExtension.afterEach(null); + postgresExtension.afterAll(null); + } + + private AbstractRecipientRewriteTable getRecipientRewriteTable() throws DomainListException { + PostgresRecipientRewriteTable postgresRecipientRewriteTable = new PostgresRecipientRewriteTable(new PostgresRecipientRewriteTableDAO(postgresExtension.getDefaultPostgresExecutor())); + postgresRecipientRewriteTable.setUsersRepository(new PostgresUsersRepository(RecipientRewriteTableFixture.domainListForCucumberTests(), + new PostgresUsersDAO(postgresExtension.getDefaultPostgresExecutor(), PostgresUsersRepositoryConfiguration.DEFAULT))); + postgresRecipientRewriteTable.setDomainList(RecipientRewriteTableFixture.domainListForCucumberTests()); + return postgresRecipientRewriteTable; + } +} diff --git a/server/data/data-postgres/src/test/java/org/apache/james/rrt/postgres/RewriteTablesTest.java b/server/data/data-postgres/src/test/java/org/apache/james/rrt/postgres/RewriteTablesTest.java new file mode 100644 index 00000000000..e6f3e2cef24 --- /dev/null +++ b/server/data/data-postgres/src/test/java/org/apache/james/rrt/postgres/RewriteTablesTest.java @@ -0,0 +1,33 @@ +/**************************************************************** + * 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.rrt.postgres; + +import static io.cucumber.core.options.Constants.GLUE_PROPERTY_NAME; + +import org.junit.platform.suite.api.ConfigurationParameter; +import org.junit.platform.suite.api.IncludeEngines; +import org.junit.platform.suite.api.SelectClasspathResource; +import org.junit.platform.suite.api.Suite; + +@Suite(failIfNoTests = false) +@IncludeEngines("cucumber") +@SelectClasspathResource("cucumber") +@ConfigurationParameter(key = GLUE_PROPERTY_NAME, value = "org.apache.james.rrt.lib,org.apache.james.rrt.postgres") +public class RewriteTablesTest { +} diff --git a/server/data/data-postgres/src/test/java/org/apache/james/sieve/postgres/PostgresSieveQuotaDAOTest.java b/server/data/data-postgres/src/test/java/org/apache/james/sieve/postgres/PostgresSieveQuotaDAOTest.java new file mode 100644 index 00000000000..1181e810ba2 --- /dev/null +++ b/server/data/data-postgres/src/test/java/org/apache/james/sieve/postgres/PostgresSieveQuotaDAOTest.java @@ -0,0 +1,163 @@ +/**************************************************************** + * 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.sieve.postgres; + +import static org.assertj.core.api.Assertions.assertThat; + +import org.apache.james.backends.postgres.PostgresExtension; +import org.apache.james.backends.postgres.PostgresModule; +import org.apache.james.backends.postgres.quota.PostgresQuotaCurrentValueDAO; +import org.apache.james.backends.postgres.quota.PostgresQuotaLimitDAO; +import org.apache.james.backends.postgres.quota.PostgresQuotaModule; +import org.apache.james.core.Username; +import org.apache.james.core.quota.QuotaSizeLimit; +import org.junit.jupiter.api.BeforeEach; +import org.junit.jupiter.api.Test; +import org.junit.jupiter.api.extension.RegisterExtension; + +class PostgresSieveQuotaDAOTest { + @RegisterExtension + static PostgresExtension postgresExtension = PostgresExtension.withoutRowLevelSecurity(PostgresModule.aggregateModules(PostgresQuotaModule.MODULE)); + + private static final Username USERNAME = Username.of("user"); + private static final QuotaSizeLimit QUOTA_SIZE = QuotaSizeLimit.size(15L); + + private PostgresSieveQuotaDAO testee; + + @BeforeEach + void setup() { + testee = new PostgresSieveQuotaDAO(new PostgresQuotaCurrentValueDAO(postgresExtension.getDefaultPostgresExecutor()), + new PostgresQuotaLimitDAO(postgresExtension.getDefaultPostgresExecutor())); + } + + @Test + void getQuotaShouldReturnEmptyByDefault() { + assertThat(testee.getGlobalQuota().block()) + .isEmpty(); + } + + @Test + void getQuotaUserShouldReturnEmptyByDefault() { + assertThat(testee.getQuota(USERNAME).block()) + .isEmpty(); + } + + @Test + void getQuotaShouldReturnStoredValue() { + testee.setGlobalQuota(QUOTA_SIZE).block(); + + assertThat(testee.getGlobalQuota().block()) + .contains(QUOTA_SIZE); + } + + @Test + void getQuotaUserShouldReturnStoredValue() { + testee.setQuota(USERNAME, QUOTA_SIZE).block(); + + assertThat(testee.getQuota(USERNAME).block()) + .contains(QUOTA_SIZE); + } + + @Test + void removeQuotaShouldDeleteQuota() { + testee.setGlobalQuota(QUOTA_SIZE).block(); + + testee.removeGlobalQuota().block(); + + assertThat(testee.getGlobalQuota().block()) + .isEmpty(); + } + + @Test + void removeQuotaUserShouldDeleteQuotaUser() { + testee.setQuota(USERNAME, QUOTA_SIZE).block(); + + testee.removeQuota(USERNAME).block(); + + assertThat(testee.getQuota(USERNAME).block()) + .isEmpty(); + } + + @Test + void removeQuotaShouldWorkWhenNoneStore() { + testee.removeGlobalQuota().block(); + + assertThat(testee.getGlobalQuota().block()) + .isEmpty(); + } + + @Test + void removeQuotaUserShouldWorkWhenNoneStore() { + testee.removeQuota(USERNAME).block(); + + assertThat(testee.getQuota(USERNAME).block()) + .isEmpty(); + } + + @Test + void spaceUsedByShouldReturnZeroByDefault() { + assertThat(testee.spaceUsedBy(USERNAME).block()).isZero(); + } + + @Test + void spaceUsedByShouldReturnStoredValue() { + long spaceUsed = 18L; + + testee.updateSpaceUsed(USERNAME, spaceUsed).block(); + + assertThat(testee.spaceUsedBy(USERNAME).block()).isEqualTo(spaceUsed); + } + + @Test + void updateSpaceUsedShouldBeAdditive() { + long spaceUsed = 18L; + + testee.updateSpaceUsed(USERNAME, spaceUsed).block(); + testee.updateSpaceUsed(USERNAME, spaceUsed).block(); + + assertThat(testee.spaceUsedBy(USERNAME).block()).isEqualTo(2 * spaceUsed); + } + + @Test + void updateSpaceUsedShouldWorkWithNegativeValues() { + long spaceUsed = 18L; + + testee.updateSpaceUsed(USERNAME, spaceUsed).block(); + testee.updateSpaceUsed(USERNAME, -1 * spaceUsed).block(); + + assertThat(testee.spaceUsedBy(USERNAME).block()).isZero(); + } + + @Test + void resetSpaceUsedShouldResetSpaceWhenNewSpaceIsGreaterThanCurrentSpace() { + testee.updateSpaceUsed(USERNAME, 10L).block(); + testee.resetSpaceUsed(USERNAME, 15L).block(); + + assertThat(testee.spaceUsedBy(USERNAME).block()).isEqualTo(15L); + } + + @Test + void resetSpaceUsedShouldResetSpaceWhenNewSpaceIsSmallerThanCurrentSpace() { + testee.updateSpaceUsed(USERNAME, 10L).block(); + testee.resetSpaceUsed(USERNAME, 9L).block(); + + assertThat(testee.spaceUsedBy(USERNAME).block()).isEqualTo(9L); + } +} diff --git a/server/data/data-postgres/src/test/java/org/apache/james/sieve/postgres/PostgresSieveRepositoryTest.java b/server/data/data-postgres/src/test/java/org/apache/james/sieve/postgres/PostgresSieveRepositoryTest.java new file mode 100644 index 00000000000..35b0b4a0d54 --- /dev/null +++ b/server/data/data-postgres/src/test/java/org/apache/james/sieve/postgres/PostgresSieveRepositoryTest.java @@ -0,0 +1,49 @@ +/**************************************************************** + * 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.sieve.postgres; + +import org.apache.james.backends.postgres.PostgresExtension; +import org.apache.james.backends.postgres.PostgresModule; +import org.apache.james.backends.postgres.quota.PostgresQuotaCurrentValueDAO; +import org.apache.james.backends.postgres.quota.PostgresQuotaLimitDAO; +import org.apache.james.backends.postgres.quota.PostgresQuotaModule; +import org.apache.james.sieverepository.api.SieveRepository; +import org.apache.james.sieverepository.lib.SieveRepositoryContract; +import org.junit.jupiter.api.BeforeEach; +import org.junit.jupiter.api.extension.RegisterExtension; + +class PostgresSieveRepositoryTest implements SieveRepositoryContract { + @RegisterExtension + static PostgresExtension postgresExtension = PostgresExtension.withoutRowLevelSecurity(PostgresModule.aggregateModules(PostgresQuotaModule.MODULE, + PostgresSieveModule.MODULE)); + + SieveRepository sieveRepository; + + @BeforeEach + void setUp() { + sieveRepository = new PostgresSieveRepository(new PostgresSieveQuotaDAO(new PostgresQuotaCurrentValueDAO(postgresExtension.getDefaultPostgresExecutor()), new PostgresQuotaLimitDAO(postgresExtension.getDefaultPostgresExecutor())), + new PostgresSieveScriptDAO(postgresExtension.getDefaultPostgresExecutor())); + } + + @Override + public SieveRepository sieveRepository() { + return sieveRepository; + } +} diff --git a/server/data/data-postgres/src/test/java/org/apache/james/user/postgres/PostgresDelegationStoreTest.java b/server/data/data-postgres/src/test/java/org/apache/james/user/postgres/PostgresDelegationStoreTest.java new file mode 100644 index 00000000000..6d163a23229 --- /dev/null +++ b/server/data/data-postgres/src/test/java/org/apache/james/user/postgres/PostgresDelegationStoreTest.java @@ -0,0 +1,67 @@ +/**************************************************************** + * 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.user.postgres; + +import static org.assertj.core.api.Assertions.assertThat; + +import org.apache.james.backends.postgres.PostgresExtension; +import org.apache.james.core.Username; +import org.apache.james.user.api.DelegationStore; +import org.apache.james.user.api.DelegationStoreContract; +import org.junit.jupiter.api.BeforeEach; +import org.junit.jupiter.api.Test; +import org.junit.jupiter.api.extension.RegisterExtension; + +import reactor.core.publisher.Mono; + +public class PostgresDelegationStoreTest implements DelegationStoreContract { + @RegisterExtension + static PostgresExtension postgresExtension = PostgresExtension.withoutRowLevelSecurity(PostgresUserModule.MODULE); + + private PostgresUsersDAO postgresUsersDAO; + private PostgresDelegationStore postgresDelegationStore; + + @BeforeEach + void beforeEach() { + postgresUsersDAO = new PostgresUsersDAO(postgresExtension.getDefaultPostgresExecutor(), PostgresUsersRepositoryConfiguration.DEFAULT); + postgresDelegationStore = new PostgresDelegationStore(postgresUsersDAO, any -> Mono.just(true)); + } + + @Override + public DelegationStore testee() { + return postgresDelegationStore; + } + + @Override + public void addUser(Username username) { + postgresUsersDAO.addUser(username, "password"); + } + + @Test + void virtualUsersShouldNotBeListed() { + postgresDelegationStore = new PostgresDelegationStore(postgresUsersDAO, any -> Mono.just(false)); + addUser(BOB); + + Mono.from(testee().addAuthorizedUser(ALICE).forUser(BOB)).block(); + + assertThat(postgresUsersDAO.listReactive().collectList().block()) + .containsOnly(BOB); + } +} diff --git a/server/data/data-postgres/src/test/java/org/apache/james/user/postgres/PostgresUsersRepositoryTest.java b/server/data/data-postgres/src/test/java/org/apache/james/user/postgres/PostgresUsersRepositoryTest.java new file mode 100644 index 00000000000..ef86d6be422 --- /dev/null +++ b/server/data/data-postgres/src/test/java/org/apache/james/user/postgres/PostgresUsersRepositoryTest.java @@ -0,0 +1,149 @@ +/**************************************************************** + * 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.user.postgres; + +import static org.assertj.core.api.Assertions.assertThat; + +import java.util.Optional; +import java.util.Set; + +import org.apache.commons.configuration2.BaseHierarchicalConfiguration; +import org.apache.james.backends.postgres.PostgresExtension; +import org.apache.james.core.Domain; +import org.apache.james.core.Username; +import org.apache.james.domainlist.api.DomainList; +import org.apache.james.user.api.UsersRepository; +import org.apache.james.user.lib.UsersRepositoryContract; +import org.apache.james.user.lib.UsersRepositoryImpl; +import org.junit.jupiter.api.BeforeEach; +import org.junit.jupiter.api.Nested; +import org.junit.jupiter.api.Test; +import org.junit.jupiter.api.extension.RegisterExtension; + +import com.github.fge.lambdas.Throwing; + +import reactor.core.publisher.Flux; +import reactor.core.publisher.Mono; +import reactor.core.scheduler.Schedulers; + +class PostgresUsersRepositoryTest { + + @RegisterExtension + static PostgresExtension postgresExtension = PostgresExtension.withoutRowLevelSecurity(PostgresUserModule.MODULE); + + @Nested + class WhenEnableVirtualHosting implements UsersRepositoryContract.WithVirtualHostingContract { + @RegisterExtension + UserRepositoryExtension extension = UserRepositoryExtension.withVirtualHost(); + + private UsersRepositoryImpl usersRepository; + private TestSystem testSystem; + + @BeforeEach + void setUp(TestSystem testSystem) throws Exception { + usersRepository = getUsersRepository(testSystem.getDomainList(), extension.isSupportVirtualHosting(), Optional.empty()); + this.testSystem = testSystem; + } + + @Override + public UsersRepositoryImpl testee() { + return usersRepository; + } + + @Override + public UsersRepository testee(Optional administrator) throws Exception { + return getUsersRepository(testSystem.getDomainList(), extension.isSupportVirtualHosting(), administrator); + } + + @Override + public UsersRepository testee(Set administrators) throws Exception { + return getUsersRepository(testSystem.getDomainList(), extension.isSupportVirtualHosting(), administrators); + } + + @Test + void listUsersReactiveThenExecuteOtherPostgresQueriesShouldNotHang() throws Exception { + Domain domain = Domain.of("example.com"); + testSystem.getDomainList().addDomain(domain); + + Flux.range(1, 1000) + .flatMap(counter -> Mono.fromRunnable(Throwing.runnable(() -> usersRepository.addUser(Username.fromLocalPartWithDomain(counter.toString(), domain), "password"))), + 128) + .collectList() + .block(); + + assertThat(Flux.from(usersRepository.listReactive()) + .flatMap(username -> Mono.fromCallable(() -> usersRepository.test(username, "password") + .orElseThrow(() -> new RuntimeException("Wrong user credential"))) + .subscribeOn(Schedulers.boundedElastic())) + .collectList() + .block()) + .hasSize(1000); + } + } + + @Nested + class WhenDisableVirtualHosting implements UsersRepositoryContract.WithOutVirtualHostingContract { + @RegisterExtension + UserRepositoryExtension extension = UserRepositoryExtension.withoutVirtualHosting(); + + private UsersRepositoryImpl usersRepository; + private TestSystem testSystem; + + @BeforeEach + void setUp(TestSystem testSystem) throws Exception { + usersRepository = getUsersRepository(testSystem.getDomainList(), extension.isSupportVirtualHosting(), Optional.empty()); + this.testSystem = testSystem; + } + + @Override + public UsersRepositoryImpl testee() { + return usersRepository; + } + + @Override + public UsersRepository testee(Optional administrator) throws Exception { + return getUsersRepository(testSystem.getDomainList(), extension.isSupportVirtualHosting(), administrator); + } + + @Override + public UsersRepository testee(Set administrators) throws Exception { + return getUsersRepository(testSystem.getDomainList(), extension.isSupportVirtualHosting(), administrators); + } + } + + private static UsersRepositoryImpl getUsersRepository(DomainList domainList, boolean enableVirtualHosting, Optional administrator) throws Exception { + Set administrators = administrator.map(Set::of) + .orElse(Set.of()); + + return getUsersRepository(domainList, enableVirtualHosting, administrators); + } + + private static UsersRepositoryImpl getUsersRepository(DomainList domainList, boolean enableVirtualHosting, Set administrators) throws Exception { + PostgresUsersDAO usersDAO = new PostgresUsersDAO(postgresExtension.getDefaultPostgresExecutor(), + PostgresUsersRepositoryConfiguration.DEFAULT); + BaseHierarchicalConfiguration configuration = new BaseHierarchicalConfiguration(); + configuration.addProperty("enableVirtualHosting", String.valueOf(enableVirtualHosting)); + administrators.forEach(admin -> configuration.addProperty("administratorIds.administratorId", admin.asString())); + + UsersRepositoryImpl usersRepository = new PostgresUsersRepository(domainList, usersDAO); + usersRepository.configure(configuration); + return usersRepository; + } +} diff --git a/server/data/data-postgres/src/test/java/org/apache/james/vacation/postgres/PostgresNotificationRegistryTest.java b/server/data/data-postgres/src/test/java/org/apache/james/vacation/postgres/PostgresNotificationRegistryTest.java new file mode 100644 index 00000000000..d9a18faa708 --- /dev/null +++ b/server/data/data-postgres/src/test/java/org/apache/james/vacation/postgres/PostgresNotificationRegistryTest.java @@ -0,0 +1,53 @@ +/**************************************************************** + * 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.vacation.postgres; + +import org.apache.james.backends.postgres.PostgresExtension; +import org.apache.james.backends.postgres.PostgresModule; +import org.apache.james.core.MailAddress; +import org.apache.james.vacation.api.NotificationRegistry; +import org.apache.james.vacation.api.NotificationRegistryContract; +import org.apache.james.vacation.api.RecipientId; +import org.junit.jupiter.api.BeforeEach; +import org.junit.jupiter.api.extension.RegisterExtension; + +class PostgresNotificationRegistryTest implements NotificationRegistryContract { + @RegisterExtension + static PostgresExtension postgresExtension = PostgresExtension.withoutRowLevelSecurity(PostgresModule.aggregateModules(PostgresVacationModule.MODULE)); + + NotificationRegistry notificationRegistry; + RecipientId recipientId; + + @BeforeEach + public void setUp() throws Exception { + notificationRegistry = new PostgresNotificationRegistry(zonedDateTimeProvider, postgresExtension.getExecutorFactory()); + recipientId = RecipientId.fromMailAddress(new MailAddress("benwa@apache.org")); + } + + @Override + public NotificationRegistry notificationRegistry() { + return notificationRegistry; + } + + @Override + public RecipientId recipientId() { + return recipientId; + } +} diff --git a/server/data/data-postgres/src/test/java/org/apache/james/vacation/postgres/PostgresVacationRepositoryTest.java b/server/data/data-postgres/src/test/java/org/apache/james/vacation/postgres/PostgresVacationRepositoryTest.java new file mode 100644 index 00000000000..81488b86777 --- /dev/null +++ b/server/data/data-postgres/src/test/java/org/apache/james/vacation/postgres/PostgresVacationRepositoryTest.java @@ -0,0 +1,44 @@ +/**************************************************************** + * 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.vacation.postgres; + +import org.apache.james.backends.postgres.PostgresExtension; +import org.apache.james.backends.postgres.PostgresModule; +import org.apache.james.vacation.api.VacationRepository; +import org.apache.james.vacation.api.VacationRepositoryContract; +import org.junit.jupiter.api.BeforeEach; +import org.junit.jupiter.api.extension.RegisterExtension; + +class PostgresVacationRepositoryTest implements VacationRepositoryContract { + @RegisterExtension + static PostgresExtension postgresExtension = PostgresExtension.withRowLevelSecurity(PostgresModule.aggregateModules(PostgresVacationModule.MODULE)); + + VacationRepository vacationRepository; + + @BeforeEach + void setUp() { + vacationRepository = new PostgresVacationRepository(postgresExtension.getExecutorFactory()); + } + + @Override + public VacationRepository vacationRepository() { + return vacationRepository; + } +} diff --git a/server/data/data-postgres/src/test/resources/log4j.properties b/server/data/data-postgres/src/test/resources/log4j.properties new file mode 100644 index 00000000000..34f5a5f5c28 --- /dev/null +++ b/server/data/data-postgres/src/test/resources/log4j.properties @@ -0,0 +1,6 @@ +log4j.rootLogger=WARN, A1 +log4j.appender.A1=org.apache.log4j.ConsoleAppender +log4j.appender.A1.layout=org.apache.log4j.PatternLayout + +# Print the date in ISO 8601 format +log4j.appender.A1.layout.ConversionPattern=%d [%t] %-5p %c - %m%n diff --git a/server/pom.xml b/server/pom.xml index a9ee64b77d6..81d7cf120f1 100644 --- a/server/pom.xml +++ b/server/pom.xml @@ -46,6 +46,7 @@ apps/jpa-app apps/jpa-smtp-app apps/memory-app + apps/postgres-app apps/scaling-pulsar-smtp apps/spring-app apps/webadmin-cli @@ -67,10 +68,12 @@ data/data-file data/data-jmap data/data-jmap-cassandra + data/data-jmap-postgres data/data-jpa data/data-ldap data/data-library data/data-memory + data/data-postgres dns-service/dnsservice-api dns-service/dnsservice-dnsjava @@ -120,6 +123,7 @@ task/task-distributed task/task-json task/task-memory + task/task-postgres testing diff --git a/server/protocols/jmap-draft-integration-testing/jmap-draft-integration-testing-common/src/test/java/org/apache/james/jmap/draft/methods/integration/SetMessagesMethodTest.java b/server/protocols/jmap-draft-integration-testing/jmap-draft-integration-testing-common/src/test/java/org/apache/james/jmap/draft/methods/integration/SetMessagesMethodTest.java new file mode 100644 index 00000000000..e69de29bb2d diff --git a/server/protocols/jmap-draft-integration-testing/rabbitmq-jmap-draft-integration-testing/pom.xml b/server/protocols/jmap-draft-integration-testing/rabbitmq-jmap-draft-integration-testing/pom.xml new file mode 100644 index 00000000000..e69de29bb2d diff --git a/server/protocols/jmap-draft/src/main/java/org/apache/james/jmap/draft/methods/BlobManagerImpl.java b/server/protocols/jmap-draft/src/main/java/org/apache/james/jmap/draft/methods/BlobManagerImpl.java new file mode 100644 index 00000000000..e69de29bb2d diff --git a/server/protocols/jmap-draft/src/main/java/org/apache/james/jmap/draft/methods/GetMessageListMethod.java b/server/protocols/jmap-draft/src/main/java/org/apache/james/jmap/draft/methods/GetMessageListMethod.java new file mode 100644 index 00000000000..e69de29bb2d diff --git a/server/protocols/jmap-draft/src/main/java/org/apache/james/jmap/draft/model/BlobId.java b/server/protocols/jmap-draft/src/main/java/org/apache/james/jmap/draft/model/BlobId.java new file mode 100644 index 00000000000..e69de29bb2d diff --git a/server/protocols/jmap-draft/src/test/java/org/apache/james/jmap/draft/methods/BlobManagerImplTest.java b/server/protocols/jmap-draft/src/test/java/org/apache/james/jmap/draft/methods/BlobManagerImplTest.java new file mode 100644 index 00000000000..e69de29bb2d diff --git a/server/protocols/jmap-draft/src/test/java/org/apache/james/jmap/draft/methods/MIMEMessageConverterTest.java b/server/protocols/jmap-draft/src/test/java/org/apache/james/jmap/draft/methods/MIMEMessageConverterTest.java new file mode 100644 index 00000000000..e69de29bb2d diff --git a/server/protocols/jmap-draft/src/test/java/org/apache/james/jmap/draft/model/message/view/MessageFastViewFactoryTest.java b/server/protocols/jmap-draft/src/test/java/org/apache/james/jmap/draft/model/message/view/MessageFastViewFactoryTest.java new file mode 100644 index 00000000000..e69de29bb2d diff --git a/server/protocols/jmap-draft/src/test/java/org/apache/james/jmap/draft/model/message/view/MessageFullViewFactoryTest.java b/server/protocols/jmap-draft/src/test/java/org/apache/james/jmap/draft/model/message/view/MessageFullViewFactoryTest.java new file mode 100644 index 00000000000..e69de29bb2d diff --git a/server/protocols/jmap-draft/src/test/java/org/apache/james/jmap/draft/model/message/view/MessageHeaderViewFactoryTest.java b/server/protocols/jmap-draft/src/test/java/org/apache/james/jmap/draft/model/message/view/MessageHeaderViewFactoryTest.java new file mode 100644 index 00000000000..e69de29bb2d diff --git a/server/protocols/jmap-draft/src/test/java/org/apache/james/jmap/draft/model/message/view/MessageMetadataViewFactoryTest.java b/server/protocols/jmap-draft/src/test/java/org/apache/james/jmap/draft/model/message/view/MessageMetadataViewFactoryTest.java new file mode 100644 index 00000000000..e69de29bb2d diff --git a/server/protocols/jmap-rfc-8621-integration-tests/distributed-jmap-rfc-8621-integration-tests/pom.xml b/server/protocols/jmap-rfc-8621-integration-tests/distributed-jmap-rfc-8621-integration-tests/pom.xml index f030ba3b578..223468947e4 100644 --- a/server/protocols/jmap-rfc-8621-integration-tests/distributed-jmap-rfc-8621-integration-tests/pom.xml +++ b/server/protocols/jmap-rfc-8621-integration-tests/distributed-jmap-rfc-8621-integration-tests/pom.xml @@ -30,6 +30,18 @@ Apache James :: Server :: JMAP RFC-8621 :: Distributed Integration Testing Distributed Integration testing for JMAP RFC-8621 + + + + ${james.groupId} + james-server-guice + ${project.version} + pom + import + + + + ${james.groupId} @@ -110,6 +122,13 @@ test-jar test + + ${james.groupId} + james-server-guice-opensearch + ${project.version} + test-jar + test + ${james.groupId} james-server-testing @@ -120,6 +139,12 @@ jmap-rfc-8621-integration-tests-common test + + ${james.groupId} + queue-rabbitmq-guice + test-jar + test + org.testcontainers pulsar diff --git a/server/protocols/jmap-rfc-8621-integration-tests/jmap-rfc-8621-integration-tests-common/src/main/scala/org/apache/james/jmap/rfc8621/contract/EmailQueryMethodContract.scala b/server/protocols/jmap-rfc-8621-integration-tests/jmap-rfc-8621-integration-tests-common/src/main/scala/org/apache/james/jmap/rfc8621/contract/EmailQueryMethodContract.scala index e7a01c144d4..ca4c8fec630 100644 --- a/server/protocols/jmap-rfc-8621-integration-tests/jmap-rfc-8621-integration-tests-common/src/main/scala/org/apache/james/jmap/rfc8621/contract/EmailQueryMethodContract.scala +++ b/server/protocols/jmap-rfc-8621-integration-tests/jmap-rfc-8621-integration-tests-common/src/main/scala/org/apache/james/jmap/rfc8621/contract/EmailQueryMethodContract.scala @@ -7033,22 +7033,24 @@ trait EmailQueryMethodContract { | "c1"]] |}""".stripMargin - val response = `given` - .header(ACCEPT.toString, ACCEPT_RFC8621_VERSION_HEADER) - .body(request) - .when - .post - .`then` - .statusCode(SC_OK) - .contentType(JSON) - .extract - .body - .asString + awaitAtMostTenSeconds.untilAsserted { () => + val response = `given` + .header(ACCEPT.toString, ACCEPT_RFC8621_VERSION_HEADER) + .body(request) + .when + .post + .`then` + .statusCode(SC_OK) + .contentType(JSON) + .extract + .body + .asString - assertThatJson(response) - .withOptions(IGNORING_ARRAY_ORDER) - .inPath("$.methodResponses[0][1].ids") - .isEqualTo(s"""["${messageId1.serialize}","${messageId2.serialize}"]""") + assertThatJson(response) + .withOptions(IGNORING_ARRAY_ORDER) + .inPath("$.methodResponses[0][1].ids") + .isEqualTo(s"""["${messageId1.serialize}","${messageId2.serialize}"]""") + } } @Test diff --git a/server/protocols/jmap-rfc-8621-integration-tests/jmap-rfc-8621-integration-tests-common/src/main/scala/org/apache/james/jmap/rfc8621/contract/MailboxSetMethodContract.scala b/server/protocols/jmap-rfc-8621-integration-tests/jmap-rfc-8621-integration-tests-common/src/main/scala/org/apache/james/jmap/rfc8621/contract/MailboxSetMethodContract.scala index 92635957550..ac772459597 100644 --- a/server/protocols/jmap-rfc-8621-integration-tests/jmap-rfc-8621-integration-tests-common/src/main/scala/org/apache/james/jmap/rfc8621/contract/MailboxSetMethodContract.scala +++ b/server/protocols/jmap-rfc-8621-integration-tests/jmap-rfc-8621-integration-tests-common/src/main/scala/org/apache/james/jmap/rfc8621/contract/MailboxSetMethodContract.scala @@ -46,7 +46,7 @@ import org.apache.james.modules.{ACLProbeImpl, MailboxProbeImpl} import org.apache.james.util.concurrency.ConcurrentTestRunner import org.apache.james.utils.DataProbeImpl import org.assertj.core.api.Assertions.assertThat -import org.assertj.core.api.{Assertions, SoftAssertions} +import org.assertj.core.api.{Assertions, SoftAssertions, ThrowingConsumer} import org.awaitility.Awaitility import org.hamcrest.Matchers.{equalTo, hasSize, not} import org.junit.jupiter.api.{BeforeEach, RepeatedTest, Tag, Test} @@ -62,6 +62,7 @@ import sttp.monad.MonadError import sttp.ws.WebSocketFrame import scala.collection.mutable.ListBuffer +import scala.concurrent.duration.MILLISECONDS import scala.jdk.CollectionConverters._ @@ -6934,7 +6935,7 @@ trait MailboxSetMethodContract { |}""".stripMargin) } - @RepeatedTest(100) + @RepeatedTest(20) def concurrencyChecksUponParentIdUpdate(server: GuiceJamesServer): Unit = { val mailboxId1: MailboxId = server.getProbe(classOf[MailboxProbeImpl]) .createMailbox(MailboxPath.forUser(BOB, "mailbox1")) @@ -8788,17 +8789,15 @@ trait MailboxSetMethodContract { | } | }, "c1"]] |}""".stripMargin)) - - List(ws.receive().asPayload) + ws.receiveMessageInTimespan(scala.concurrent.duration.Duration(1000, MILLISECONDS)) }) .send(backend) .body - Thread.sleep(200) + val hasMailboxStateChangeConsumer : ThrowingConsumer[String] = (s: String) => assertThat(s) + .startsWith("{\"@type\":\"StateChange\",\"changed\":{\"29883977c13473ae7cb7678ef767cbfbaffc8a44a6e463d971d23a65c1dc4af6\":{\"Mailbox\":") assertThat(response.toOption.get.asJava) - .hasSize(1) - assertThat(response.toOption.get.head) - .startsWith("{\"@type\":\"StateChange\",\"changed\":{\"29883977c13473ae7cb7678ef767cbfbaffc8a44a6e463d971d23a65c1dc4af6\":{\"Mailbox\":") + .anySatisfy(hasMailboxStateChangeConsumer) } @Test diff --git a/server/protocols/jmap-rfc-8621-integration-tests/jmap-rfc-8621-integration-tests-common/src/main/scala/org/apache/james/jmap/rfc8621/contract/PushSubscriptionSetMethodContract.scala b/server/protocols/jmap-rfc-8621-integration-tests/jmap-rfc-8621-integration-tests-common/src/main/scala/org/apache/james/jmap/rfc8621/contract/PushSubscriptionSetMethodContract.scala index 705ff2ea707..d0f706ebd73 100644 --- a/server/protocols/jmap-rfc-8621-integration-tests/jmap-rfc-8621-integration-tests-common/src/main/scala/org/apache/james/jmap/rfc8621/contract/PushSubscriptionSetMethodContract.scala +++ b/server/protocols/jmap-rfc-8621-integration-tests/jmap-rfc-8621-integration-tests-common/src/main/scala/org/apache/james/jmap/rfc8621/contract/PushSubscriptionSetMethodContract.scala @@ -38,6 +38,7 @@ import io.restassured.RestAssured.{`given`, requestSpecification} import io.restassured.http.ContentType.JSON import jakarta.inject.Inject import net.javacrumbs.jsonunit.assertj.JsonAssertions.assertThatJson +import net.javacrumbs.jsonunit.core.Option.IGNORING_ARRAY_ORDER import org.apache.http.HttpStatus.SC_OK import org.apache.james.GuiceJamesServer import org.apache.james.core.Username @@ -610,6 +611,7 @@ trait PushSubscriptionSetMethodContract { .asString assertThatJson(response) + .withOptions(IGNORING_ARRAY_ORDER) .isEqualTo( s"""{ | "sessionState": "${SESSION_STATE.value}", @@ -771,6 +773,7 @@ trait PushSubscriptionSetMethodContract { .asString assertThatJson(response) + .withOptions(IGNORING_ARRAY_ORDER) .isEqualTo( s"""{ | "sessionState": "${SESSION_STATE.value}", @@ -913,6 +916,7 @@ trait PushSubscriptionSetMethodContract { .asString assertThatJson(response) + .withOptions(IGNORING_ARRAY_ORDER) .isEqualTo( s"""{ | "sessionState": "${SESSION_STATE.value}", diff --git a/server/protocols/jmap-rfc-8621-integration-tests/jmap-rfc-8621-integration-tests-common/src/main/scala/org/apache/james/jmap/rfc8621/contract/QuotaGetMethodContract.scala b/server/protocols/jmap-rfc-8621-integration-tests/jmap-rfc-8621-integration-tests-common/src/main/scala/org/apache/james/jmap/rfc8621/contract/QuotaGetMethodContract.scala index 90cc7a112d3..2e96cbbb9f5 100644 --- a/server/protocols/jmap-rfc-8621-integration-tests/jmap-rfc-8621-integration-tests-common/src/main/scala/org/apache/james/jmap/rfc8621/contract/QuotaGetMethodContract.scala +++ b/server/protocols/jmap-rfc-8621-integration-tests/jmap-rfc-8621-integration-tests-common/src/main/scala/org/apache/james/jmap/rfc8621/contract/QuotaGetMethodContract.scala @@ -501,73 +501,75 @@ trait QuotaGetMethodContract { .build)) .getMessageId.serialize() - val response = `given` - .body( - s"""{ - | "using": [ - | "urn:ietf:params:jmap:core", - | "urn:ietf:params:jmap:quota"], - | "methodCalls": [[ - | "Quota/get", - | { - | "accountId": "29883977c13473ae7cb7678ef767cbfbaffc8a44a6e463d971d23a65c1dc4af6", - | "ids": null - | }, - | "c1"]] - |}""".stripMargin) - .when - .post - .`then` - .statusCode(SC_OK) - .contentType(JSON) - .extract - .body - .asString + awaitAtMostTenSeconds.untilAsserted(() => { + val response = `given` + .body( + s"""{ + | "using": [ + | "urn:ietf:params:jmap:core", + | "urn:ietf:params:jmap:quota"], + | "methodCalls": [[ + | "Quota/get", + | { + | "accountId": "29883977c13473ae7cb7678ef767cbfbaffc8a44a6e463d971d23a65c1dc4af6", + | "ids": null + | }, + | "c1"]] + |}""".stripMargin) + .when + .post + .`then` + .statusCode(SC_OK) + .contentType(JSON) + .extract + .body + .asString - assertThatJson(response) - .withOptions(IGNORING_ARRAY_ORDER) - .isEqualTo( - s"""{ - | "sessionState": "${SESSION_STATE.value}", - | "methodResponses": [ - | [ - | "Quota/get", - | { - | "accountId": "29883977c13473ae7cb7678ef767cbfbaffc8a44a6e463d971d23a65c1dc4af6", - | "notFound": [ ], - | "state": "3c51d50a-d766-38b7-9fa4-c9ff12de87a4", - | "list": [ - | { - | "used": 1, - | "name": "#private&bob@domain.tld@domain.tld:account:count:Mail", - | "id": "08417be420b6dd6fa77d48fb2438e0d19108cd29424844bb109b52d356fab528", - | "types": [ - | "Mail" - | ], - | "hardLimit": 100, - | "warnLimit": 90, - | "resourceType": "count", - | "scope": "account" - | }, - | { - | "used": 85, - | "name": "#private&bob@domain.tld@domain.tld:account:octets:Mail", - | "id": "eab6ce8ac5d9730a959e614854410cf39df98ff3760a623b8e540f36f5184947", - | "types": [ - | "Mail" - | ], - | "hardLimit": 900, - | "warnLimit": 810, - | "resourceType": "octets", - | "scope": "account" - | } - | ] - | }, - | "c1" - | ] - | ] - |} - |""".stripMargin) + assertThatJson(response) + .withOptions(IGNORING_ARRAY_ORDER) + .isEqualTo( + s"""{ + | "sessionState": "${SESSION_STATE.value}", + | "methodResponses": [ + | [ + | "Quota/get", + | { + | "accountId": "29883977c13473ae7cb7678ef767cbfbaffc8a44a6e463d971d23a65c1dc4af6", + | "notFound": [ ], + | "state": "3c51d50a-d766-38b7-9fa4-c9ff12de87a4", + | "list": [ + | { + | "used": 1, + | "name": "#private&bob@domain.tld@domain.tld:account:count:Mail", + | "id": "08417be420b6dd6fa77d48fb2438e0d19108cd29424844bb109b52d356fab528", + | "types": [ + | "Mail" + | ], + | "hardLimit": 100, + | "warnLimit": 90, + | "resourceType": "count", + | "scope": "account" + | }, + | { + | "used": 85, + | "name": "#private&bob@domain.tld@domain.tld:account:octets:Mail", + | "id": "eab6ce8ac5d9730a959e614854410cf39df98ff3760a623b8e540f36f5184947", + | "types": [ + | "Mail" + | ], + | "hardLimit": 900, + | "warnLimit": 810, + | "resourceType": "octets", + | "scope": "account" + | } + | ] + | }, + | "c1" + | ] + | ] + |} + |""".stripMargin) + }) } diff --git a/server/protocols/jmap-rfc-8621-integration-tests/jmap-rfc-8621-integration-tests-common/src/main/scala/org/apache/james/jmap/rfc8621/contract/UploadContract.scala b/server/protocols/jmap-rfc-8621-integration-tests/jmap-rfc-8621-integration-tests-common/src/main/scala/org/apache/james/jmap/rfc8621/contract/UploadContract.scala index 87b63790378..ef9047858f5 100644 --- a/server/protocols/jmap-rfc-8621-integration-tests/jmap-rfc-8621-integration-tests-common/src/main/scala/org/apache/james/jmap/rfc8621/contract/UploadContract.scala +++ b/server/protocols/jmap-rfc-8621-integration-tests/jmap-rfc-8621-integration-tests-common/src/main/scala/org/apache/james/jmap/rfc8621/contract/UploadContract.scala @@ -58,7 +58,7 @@ trait UploadContract { .build } - @RepeatedTest(50) + @RepeatedTest(20) def shouldUploadFileAndAllowToDownloadIt(): Unit = { val uploadResponse: String = `given` .basePath("") diff --git a/server/protocols/jmap-rfc-8621-integration-tests/jmap-rfc-8621-integration-tests-common/src/main/scala/org/apache/james/jmap/rfc8621/contract/package.scala b/server/protocols/jmap-rfc-8621-integration-tests/jmap-rfc-8621-integration-tests-common/src/main/scala/org/apache/james/jmap/rfc8621/contract/package.scala index 4b2a41999ab..a004d608dca 100644 --- a/server/protocols/jmap-rfc-8621-integration-tests/jmap-rfc-8621-integration-tests-common/src/main/scala/org/apache/james/jmap/rfc8621/contract/package.scala +++ b/server/protocols/jmap-rfc-8621-integration-tests/jmap-rfc-8621-integration-tests-common/src/main/scala/org/apache/james/jmap/rfc8621/contract/package.scala @@ -19,10 +19,17 @@ package org.apache.james.jmap.rfc8621 +import java.util.concurrent.TimeoutException + import cats.implicits.toFunctorOps +import reactor.core.publisher.Flux +import reactor.core.scala.publisher.SMono +import reactor.core.scheduler.Schedulers import sttp.client3.Identity -import sttp.ws.WebSocketFrame import sttp.ws.WebSocketFrame.Text +import sttp.ws.{WebSocket, WebSocketFrame} + +import scala.concurrent.duration.{Duration, MILLISECONDS} package object contract { @@ -32,4 +39,19 @@ package object contract { case _ => throw new RuntimeException("Not a text frame") } } + + + implicit class receiveMessageInTimespan(val ws: WebSocket[Identity]) { + def receiveMessageInTimespan(timeout: Duration = scala.concurrent.duration.Duration(1000, MILLISECONDS)): List[Identity[String]] = + SMono.fromCallable(() => ws.receive().asPayload) + .publishOn(Schedulers.boundedElastic()) + .repeat() + .take(timeout) + .onErrorResume { + case _: TimeoutException => + Flux.empty[String] + } + .collectSeq() + .block().toList + } } diff --git a/server/protocols/jmap-rfc-8621-integration-tests/pom.xml b/server/protocols/jmap-rfc-8621-integration-tests/pom.xml index 8eed57415a7..e1e2432213d 100644 --- a/server/protocols/jmap-rfc-8621-integration-tests/pom.xml +++ b/server/protocols/jmap-rfc-8621-integration-tests/pom.xml @@ -34,6 +34,7 @@ distributed-jmap-rfc-8621-integration-tests jmap-rfc-8621-integration-tests-common memory-jmap-rfc-8621-integration-tests + postgres-jmap-rfc-8621-integration-tests diff --git a/server/protocols/jmap-rfc-8621-integration-tests/postgres-jmap-rfc-8621-integration-tests/pom.xml b/server/protocols/jmap-rfc-8621-integration-tests/postgres-jmap-rfc-8621-integration-tests/pom.xml new file mode 100644 index 00000000000..95a55007679 --- /dev/null +++ b/server/protocols/jmap-rfc-8621-integration-tests/postgres-jmap-rfc-8621-integration-tests/pom.xml @@ -0,0 +1,111 @@ + + + + + 4.0.0 + + org.apache.james + jmap-rfc-8621-integration-tests + 3.9.0-SNAPSHOT + + postgres-jmap-rfc-8621-integration-tests + Apache James :: Server :: JMAP RFC-8621 :: Postgres Integration Testing + JMAP RFC-8621 integration test for postgres product + + + + ${james.groupId} + apache-james-backends-opensearch + test-jar + test + + + ${james.groupId} + apache-james-backends-postgres + test-jar + test + + + ${james.groupId} + apache-james-backends-rabbitmq + test-jar + test + + + ${james.groupId} + james-server-guice-common + test-jar + test + + + ${james.groupId} + james-server-guice-jmap + test-jar + test + + + ${james.groupId} + james-server-guice-opensearch + ${project.version} + test-jar + test + + + ${james.groupId} + james-server-postgres-app + test + + + ${james.groupId} + james-server-postgres-app + test-jar + test + + + ${project.groupId} + james-server-testing + test + + + ${project.groupId} + jmap-rfc-8621-integration-tests-common + test + + + org.testcontainers + postgresql + test + + + + + + + org.apache.maven.plugins + maven-surefire-plugin + + 3600 + true + 2 + + + + + diff --git a/server/protocols/jmap-rfc-8621-integration-tests/postgres-jmap-rfc-8621-integration-tests/src/test/java/org/apache/james/jmap/rfc8621/postgres/PostgresAuthenticationTest.java b/server/protocols/jmap-rfc-8621-integration-tests/postgres-jmap-rfc-8621-integration-tests/src/test/java/org/apache/james/jmap/rfc8621/postgres/PostgresAuthenticationTest.java new file mode 100644 index 00000000000..57e4f56dcac --- /dev/null +++ b/server/protocols/jmap-rfc-8621-integration-tests/postgres-jmap-rfc-8621-integration-tests/src/test/java/org/apache/james/jmap/rfc8621/postgres/PostgresAuthenticationTest.java @@ -0,0 +1,57 @@ +/**************************************************************** + * 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.jmap.rfc8621.postgres; + +import static org.apache.james.data.UsersRepositoryModuleChooser.Implementation.DEFAULT; + +import org.apache.james.JamesServerBuilder; +import org.apache.james.JamesServerExtension; +import org.apache.james.PostgresJamesConfiguration; +import org.apache.james.PostgresJamesServerMain; +import org.apache.james.SearchConfiguration; +import org.apache.james.backends.postgres.PostgresExtension; +import org.apache.james.jmap.rfc8621.contract.AuthenticationContract; +import org.apache.james.modules.RabbitMQExtension; +import org.apache.james.modules.TestJMAPServerModule; +import org.apache.james.modules.blobstore.BlobStoreConfiguration; +import org.junit.jupiter.api.extension.RegisterExtension; + +public class PostgresAuthenticationTest implements AuthenticationContract { + @RegisterExtension + static JamesServerExtension testExtension = new JamesServerBuilder(tmpDir -> + PostgresJamesConfiguration.builder() + .workingDirectory(tmpDir) + .configurationFromClasspath() + .searchConfiguration(SearchConfiguration.scanning()) + .usersRepository(DEFAULT) + .eventBusImpl(PostgresJamesConfiguration.EventBusImpl.RABBITMQ) + .blobStore(BlobStoreConfiguration.builder() + .postgres() + .disableCache() + .deduplication() + .noCryptoConfig()) + .build()) + .extension(PostgresExtension.empty()) + .extension(new RabbitMQExtension()) + .server(configuration -> PostgresJamesServerMain.createServer(configuration) + .overrideWith(new TestJMAPServerModule())) + .lifeCycle(JamesServerExtension.Lifecycle.PER_ENCLOSING_CLASS) + .build(); +} diff --git a/server/protocols/jmap-rfc-8621-integration-tests/postgres-jmap-rfc-8621-integration-tests/src/test/java/org/apache/james/jmap/rfc8621/postgres/PostgresBase.java b/server/protocols/jmap-rfc-8621-integration-tests/postgres-jmap-rfc-8621-integration-tests/src/test/java/org/apache/james/jmap/rfc8621/postgres/PostgresBase.java new file mode 100644 index 00000000000..3c33d39221d --- /dev/null +++ b/server/protocols/jmap-rfc-8621-integration-tests/postgres-jmap-rfc-8621-integration-tests/src/test/java/org/apache/james/jmap/rfc8621/postgres/PostgresBase.java @@ -0,0 +1,59 @@ +/**************************************************************** + * 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.jmap.rfc8621.postgres; + +import static org.apache.james.data.UsersRepositoryModuleChooser.Implementation.DEFAULT; + +import org.apache.james.JamesServerBuilder; +import org.apache.james.JamesServerExtension; +import org.apache.james.PostgresJamesConfiguration; +import org.apache.james.PostgresJamesServerMain; +import org.apache.james.SearchConfiguration; +import org.apache.james.backends.postgres.PostgresExtension; +import org.apache.james.jmap.rfc8621.contract.IdentityProbeModule; +import org.apache.james.jmap.rfc8621.contract.probe.DelegationProbeModule; +import org.apache.james.modules.RabbitMQExtension; +import org.apache.james.modules.TestJMAPServerModule; +import org.apache.james.modules.blobstore.BlobStoreConfiguration; +import org.junit.jupiter.api.extension.RegisterExtension; + +public class PostgresBase { + @RegisterExtension + static JamesServerExtension testExtension = new JamesServerBuilder(tmpDir -> + PostgresJamesConfiguration.builder() + .workingDirectory(tmpDir) + .configurationFromClasspath() + .searchConfiguration(SearchConfiguration.scanning()) + .usersRepository(DEFAULT) + .eventBusImpl(PostgresJamesConfiguration.EventBusImpl.RABBITMQ) + .blobStore(BlobStoreConfiguration.builder() + .postgres() + .disableCache() + .deduplication() + .noCryptoConfig()) + .build()) + .extension(PostgresExtension.empty()) + .extension(new RabbitMQExtension()) + .server(configuration -> PostgresJamesServerMain.createServer(configuration) + .overrideWith(new TestJMAPServerModule()) + .overrideWith(new DelegationProbeModule()) + .overrideWith(new IdentityProbeModule())) + .build(); +} diff --git a/server/protocols/jmap-rfc-8621-integration-tests/postgres-jmap-rfc-8621-integration-tests/src/test/java/org/apache/james/jmap/rfc8621/postgres/PostgresCustomMethodTest.java b/server/protocols/jmap-rfc-8621-integration-tests/postgres-jmap-rfc-8621-integration-tests/src/test/java/org/apache/james/jmap/rfc8621/postgres/PostgresCustomMethodTest.java new file mode 100644 index 00000000000..37f55fe9e12 --- /dev/null +++ b/server/protocols/jmap-rfc-8621-integration-tests/postgres-jmap-rfc-8621-integration-tests/src/test/java/org/apache/james/jmap/rfc8621/postgres/PostgresCustomMethodTest.java @@ -0,0 +1,58 @@ +/**************************************************************** + * 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.jmap.rfc8621.postgres; + +import static org.apache.james.data.UsersRepositoryModuleChooser.Implementation.DEFAULT; + +import org.apache.james.JamesServerBuilder; +import org.apache.james.JamesServerExtension; +import org.apache.james.PostgresJamesConfiguration; +import org.apache.james.PostgresJamesServerMain; +import org.apache.james.SearchConfiguration; +import org.apache.james.backends.postgres.PostgresExtension; +import org.apache.james.jmap.rfc8621.contract.CustomMethodContract; +import org.apache.james.jmap.rfc8621.contract.CustomMethodModule; +import org.apache.james.modules.RabbitMQExtension; +import org.apache.james.modules.TestJMAPServerModule; +import org.apache.james.modules.blobstore.BlobStoreConfiguration; +import org.junit.jupiter.api.extension.RegisterExtension; + +public class PostgresCustomMethodTest implements CustomMethodContract { + @RegisterExtension + static JamesServerExtension testExtension = new JamesServerBuilder(tmpDir -> + PostgresJamesConfiguration.builder() + .workingDirectory(tmpDir) + .configurationFromClasspath() + .searchConfiguration(SearchConfiguration.scanning()) + .usersRepository(DEFAULT) + .eventBusImpl(PostgresJamesConfiguration.EventBusImpl.RABBITMQ) + .blobStore(BlobStoreConfiguration.builder() + .postgres() + .disableCache() + .deduplication() + .noCryptoConfig()) + .build()) + .extension(PostgresExtension.empty()) + .extension(new RabbitMQExtension()) + .server(configuration -> PostgresJamesServerMain.createServer(configuration) + .overrideWith(new TestJMAPServerModule()) + .overrideWith(new CustomMethodModule())) + .build(); +} diff --git a/server/protocols/jmap-rfc-8621-integration-tests/postgres-jmap-rfc-8621-integration-tests/src/test/java/org/apache/james/jmap/rfc8621/postgres/PostgresCustomNamespaceTest.java b/server/protocols/jmap-rfc-8621-integration-tests/postgres-jmap-rfc-8621-integration-tests/src/test/java/org/apache/james/jmap/rfc8621/postgres/PostgresCustomNamespaceTest.java new file mode 100644 index 00000000000..f6bef51a269 --- /dev/null +++ b/server/protocols/jmap-rfc-8621-integration-tests/postgres-jmap-rfc-8621-integration-tests/src/test/java/org/apache/james/jmap/rfc8621/postgres/PostgresCustomNamespaceTest.java @@ -0,0 +1,58 @@ +/**************************************************************** + * 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.jmap.rfc8621.postgres; + +import static org.apache.james.data.UsersRepositoryModuleChooser.Implementation.DEFAULT; + +import org.apache.james.JamesServerBuilder; +import org.apache.james.JamesServerExtension; +import org.apache.james.PostgresJamesConfiguration; +import org.apache.james.PostgresJamesServerMain; +import org.apache.james.SearchConfiguration; +import org.apache.james.backends.postgres.PostgresExtension; +import org.apache.james.jmap.rfc8621.contract.CustomNamespaceContract; +import org.apache.james.jmap.rfc8621.contract.CustomNamespaceModule; +import org.apache.james.modules.RabbitMQExtension; +import org.apache.james.modules.TestJMAPServerModule; +import org.apache.james.modules.blobstore.BlobStoreConfiguration; +import org.junit.jupiter.api.extension.RegisterExtension; + +public class PostgresCustomNamespaceTest implements CustomNamespaceContract { + @RegisterExtension + static JamesServerExtension testExtension = new JamesServerBuilder(tmpDir -> + PostgresJamesConfiguration.builder() + .workingDirectory(tmpDir) + .configurationFromClasspath() + .searchConfiguration(SearchConfiguration.scanning()) + .usersRepository(DEFAULT) + .eventBusImpl(PostgresJamesConfiguration.EventBusImpl.RABBITMQ) + .blobStore(BlobStoreConfiguration.builder() + .postgres() + .disableCache() + .deduplication() + .noCryptoConfig()) + .build()) + .extension(PostgresExtension.empty()) + .extension(new RabbitMQExtension()) + .server(configuration -> PostgresJamesServerMain.createServer(configuration) + .overrideWith(new TestJMAPServerModule()) + .overrideWith(new CustomNamespaceModule())) + .build(); +} diff --git a/server/protocols/jmap-rfc-8621-integration-tests/postgres-jmap-rfc-8621-integration-tests/src/test/java/org/apache/james/jmap/rfc8621/postgres/PostgresDelegatedAccountGetMethodTest.java b/server/protocols/jmap-rfc-8621-integration-tests/postgres-jmap-rfc-8621-integration-tests/src/test/java/org/apache/james/jmap/rfc8621/postgres/PostgresDelegatedAccountGetMethodTest.java new file mode 100644 index 00000000000..b95cb50b1d4 --- /dev/null +++ b/server/protocols/jmap-rfc-8621-integration-tests/postgres-jmap-rfc-8621-integration-tests/src/test/java/org/apache/james/jmap/rfc8621/postgres/PostgresDelegatedAccountGetMethodTest.java @@ -0,0 +1,25 @@ +/**************************************************************** + * 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.jmap.rfc8621.postgres; + +import org.apache.james.jmap.rfc8621.contract.DelegatedAccountGetMethodContract; + +public class PostgresDelegatedAccountGetMethodTest extends PostgresBase implements DelegatedAccountGetMethodContract { +} diff --git a/server/protocols/jmap-rfc-8621-integration-tests/postgres-jmap-rfc-8621-integration-tests/src/test/java/org/apache/james/jmap/rfc8621/postgres/PostgresDelegatedAccountSetTest.java b/server/protocols/jmap-rfc-8621-integration-tests/postgres-jmap-rfc-8621-integration-tests/src/test/java/org/apache/james/jmap/rfc8621/postgres/PostgresDelegatedAccountSetTest.java new file mode 100644 index 00000000000..82b0505a47e --- /dev/null +++ b/server/protocols/jmap-rfc-8621-integration-tests/postgres-jmap-rfc-8621-integration-tests/src/test/java/org/apache/james/jmap/rfc8621/postgres/PostgresDelegatedAccountSetTest.java @@ -0,0 +1,25 @@ +/**************************************************************** + * 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.jmap.rfc8621.postgres; + +import org.apache.james.jmap.rfc8621.contract.DelegatedAccountSetContract; + +public class PostgresDelegatedAccountSetTest extends PostgresBase implements DelegatedAccountSetContract { +} diff --git a/server/protocols/jmap-rfc-8621-integration-tests/postgres-jmap-rfc-8621-integration-tests/src/test/java/org/apache/james/jmap/rfc8621/postgres/PostgresDownloadTest.java b/server/protocols/jmap-rfc-8621-integration-tests/postgres-jmap-rfc-8621-integration-tests/src/test/java/org/apache/james/jmap/rfc8621/postgres/PostgresDownloadTest.java new file mode 100644 index 00000000000..d26b104bc02 --- /dev/null +++ b/server/protocols/jmap-rfc-8621-integration-tests/postgres-jmap-rfc-8621-integration-tests/src/test/java/org/apache/james/jmap/rfc8621/postgres/PostgresDownloadTest.java @@ -0,0 +1,33 @@ +/**************************************************************** + * 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.jmap.rfc8621.postgres; + +import org.apache.james.jmap.rfc8621.contract.DownloadContract; +import org.apache.james.mailbox.model.MessageId; +import org.apache.james.mailbox.postgres.PostgresMessageId; + +public class PostgresDownloadTest extends PostgresBase implements DownloadContract { + public static final PostgresMessageId.Factory MESSAGE_ID_FACTORY = new PostgresMessageId.Factory(); + + @Override + public MessageId randomMessageId() { + return MESSAGE_ID_FACTORY.generate(); + } +} diff --git a/server/protocols/jmap-rfc-8621-integration-tests/postgres-jmap-rfc-8621-integration-tests/src/test/java/org/apache/james/jmap/rfc8621/postgres/PostgresEchoMethodTest.java b/server/protocols/jmap-rfc-8621-integration-tests/postgres-jmap-rfc-8621-integration-tests/src/test/java/org/apache/james/jmap/rfc8621/postgres/PostgresEchoMethodTest.java new file mode 100644 index 00000000000..83bef32ee2a --- /dev/null +++ b/server/protocols/jmap-rfc-8621-integration-tests/postgres-jmap-rfc-8621-integration-tests/src/test/java/org/apache/james/jmap/rfc8621/postgres/PostgresEchoMethodTest.java @@ -0,0 +1,25 @@ +/**************************************************************** + * 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.jmap.rfc8621.postgres; + +import org.apache.james.jmap.rfc8621.contract.EchoMethodContract; + +public class PostgresEchoMethodTest extends PostgresBase implements EchoMethodContract { +} diff --git a/server/protocols/jmap-rfc-8621-integration-tests/postgres-jmap-rfc-8621-integration-tests/src/test/java/org/apache/james/jmap/rfc8621/postgres/PostgresEmailChangesMethodTest.java b/server/protocols/jmap-rfc-8621-integration-tests/postgres-jmap-rfc-8621-integration-tests/src/test/java/org/apache/james/jmap/rfc8621/postgres/PostgresEmailChangesMethodTest.java new file mode 100644 index 00000000000..0bc5bdae280 --- /dev/null +++ b/server/protocols/jmap-rfc-8621-integration-tests/postgres-jmap-rfc-8621-integration-tests/src/test/java/org/apache/james/jmap/rfc8621/postgres/PostgresEmailChangesMethodTest.java @@ -0,0 +1,69 @@ +/**************************************************************** + * 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.jmap.rfc8621.postgres; + +import static org.apache.james.data.UsersRepositoryModuleChooser.Implementation.DEFAULT; + +import org.apache.james.JamesServerBuilder; +import org.apache.james.JamesServerExtension; +import org.apache.james.PostgresJamesConfiguration; +import org.apache.james.PostgresJamesServerMain; +import org.apache.james.SearchConfiguration; +import org.apache.james.backends.postgres.PostgresExtension; +import org.apache.james.jmap.api.change.Limit; +import org.apache.james.jmap.api.change.State; +import org.apache.james.jmap.postgres.change.PostgresEmailChangeRepository; +import org.apache.james.jmap.postgres.change.PostgresStateFactory; +import org.apache.james.jmap.rfc8621.contract.EmailChangesMethodContract; +import org.apache.james.modules.RabbitMQExtension; +import org.apache.james.modules.TestJMAPServerModule; +import org.apache.james.modules.blobstore.BlobStoreConfiguration; +import org.junit.jupiter.api.extension.RegisterExtension; + +import com.google.inject.name.Names; + +public class PostgresEmailChangesMethodTest implements EmailChangesMethodContract { + @RegisterExtension + static JamesServerExtension testExtension = new JamesServerBuilder(tmpDir -> + PostgresJamesConfiguration.builder() + .workingDirectory(tmpDir) + .configurationFromClasspath() + .searchConfiguration(SearchConfiguration.scanning()) + .usersRepository(DEFAULT) + .eventBusImpl(PostgresJamesConfiguration.EventBusImpl.RABBITMQ) + .blobStore(BlobStoreConfiguration.builder() + .postgres() + .disableCache() + .deduplication() + .noCryptoConfig()) + .build()) + .extension(PostgresExtension.empty()) + .extension(new RabbitMQExtension()) + .server(configuration -> PostgresJamesServerMain.createServer(configuration) + .overrideWith(new TestJMAPServerModule()) + .overrideWith(binder -> binder.bind(Limit.class).annotatedWith(Names.named(PostgresEmailChangeRepository.LIMIT_NAME)).toInstance(Limit.of(5))) + .overrideWith(binder -> binder.bind(Limit.class).annotatedWith(Names.named(PostgresEmailChangeRepository.LIMIT_NAME)).toInstance(Limit.of(5)))) + .build(); + + @Override + public State.Factory stateFactory() { + return new PostgresStateFactory(); + } +} diff --git a/server/protocols/jmap-rfc-8621-integration-tests/postgres-jmap-rfc-8621-integration-tests/src/test/java/org/apache/james/jmap/rfc8621/postgres/PostgresEmailGetMethodTest.java b/server/protocols/jmap-rfc-8621-integration-tests/postgres-jmap-rfc-8621-integration-tests/src/test/java/org/apache/james/jmap/rfc8621/postgres/PostgresEmailGetMethodTest.java new file mode 100644 index 00000000000..43e5c293fc1 --- /dev/null +++ b/server/protocols/jmap-rfc-8621-integration-tests/postgres-jmap-rfc-8621-integration-tests/src/test/java/org/apache/james/jmap/rfc8621/postgres/PostgresEmailGetMethodTest.java @@ -0,0 +1,33 @@ +/**************************************************************** + * 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.jmap.rfc8621.postgres; + +import org.apache.james.jmap.rfc8621.contract.EmailGetMethodContract; +import org.apache.james.mailbox.model.MessageId; +import org.apache.james.mailbox.postgres.PostgresMessageId; + +public class PostgresEmailGetMethodTest extends PostgresBase implements EmailGetMethodContract { + public static final PostgresMessageId.Factory MESSAGE_ID_FACTORY = new PostgresMessageId.Factory(); + + @Override + public MessageId randomMessageId() { + return MESSAGE_ID_FACTORY.generate(); + } +} diff --git a/server/protocols/jmap-rfc-8621-integration-tests/postgres-jmap-rfc-8621-integration-tests/src/test/java/org/apache/james/jmap/rfc8621/postgres/PostgresEmailQueryMethodTest.java b/server/protocols/jmap-rfc-8621-integration-tests/postgres-jmap-rfc-8621-integration-tests/src/test/java/org/apache/james/jmap/rfc8621/postgres/PostgresEmailQueryMethodTest.java new file mode 100644 index 00000000000..094d01701fa --- /dev/null +++ b/server/protocols/jmap-rfc-8621-integration-tests/postgres-jmap-rfc-8621-integration-tests/src/test/java/org/apache/james/jmap/rfc8621/postgres/PostgresEmailQueryMethodTest.java @@ -0,0 +1,62 @@ +/**************************************************************** + * 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.jmap.rfc8621.postgres; + +import static org.apache.james.data.UsersRepositoryModuleChooser.Implementation.DEFAULT; + +import org.apache.james.DockerOpenSearchExtension; +import org.apache.james.JamesServerBuilder; +import org.apache.james.JamesServerExtension; +import org.apache.james.PostgresJamesConfiguration; +import org.apache.james.PostgresJamesServerMain; +import org.apache.james.SearchConfiguration; +import org.apache.james.backends.postgres.PostgresExtension; +import org.apache.james.jmap.rfc8621.contract.EmailQueryMethodContract; +import org.apache.james.jmap.rfc8621.contract.IdentityProbeModule; +import org.apache.james.jmap.rfc8621.contract.probe.DelegationProbeModule; +import org.apache.james.modules.RabbitMQExtension; +import org.apache.james.modules.TestJMAPServerModule; +import org.apache.james.modules.blobstore.BlobStoreConfiguration; +import org.junit.jupiter.api.extension.RegisterExtension; + +public class PostgresEmailQueryMethodTest implements EmailQueryMethodContract { + @RegisterExtension + static JamesServerExtension testExtension = new JamesServerBuilder(tmpDir -> + PostgresJamesConfiguration.builder() + .workingDirectory(tmpDir) + .configurationFromClasspath() + .searchConfiguration(SearchConfiguration.openSearch()) + .usersRepository(DEFAULT) + .eventBusImpl(PostgresJamesConfiguration.EventBusImpl.RABBITMQ) + .blobStore(BlobStoreConfiguration.builder() + .postgres() + .disableCache() + .deduplication() + .noCryptoConfig()) + .build()) + .extension(PostgresExtension.empty()) + .extension(new RabbitMQExtension()) + .extension(new DockerOpenSearchExtension()) + .server(configuration -> PostgresJamesServerMain.createServer(configuration) + .overrideWith(new TestJMAPServerModule()) + .overrideWith(new DelegationProbeModule()) + .overrideWith(new IdentityProbeModule())) + .build(); +} diff --git a/server/protocols/jmap-rfc-8621-integration-tests/postgres-jmap-rfc-8621-integration-tests/src/test/java/org/apache/james/jmap/rfc8621/postgres/PostgresEmailSetMethodTest.java b/server/protocols/jmap-rfc-8621-integration-tests/postgres-jmap-rfc-8621-integration-tests/src/test/java/org/apache/james/jmap/rfc8621/postgres/PostgresEmailSetMethodTest.java new file mode 100644 index 00000000000..f7b87a67ce1 --- /dev/null +++ b/server/protocols/jmap-rfc-8621-integration-tests/postgres-jmap-rfc-8621-integration-tests/src/test/java/org/apache/james/jmap/rfc8621/postgres/PostgresEmailSetMethodTest.java @@ -0,0 +1,53 @@ +/**************************************************************** + * 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.jmap.rfc8621.postgres; + +import org.apache.james.GuiceJamesServer; +import org.apache.james.jmap.rfc8621.contract.EmailSetMethodContract; +import org.apache.james.mailbox.model.MessageId; +import org.apache.james.mailbox.postgres.PostgresMessageId; +import org.junit.jupiter.api.Disabled; +import org.junit.jupiter.api.Test; + +public class PostgresEmailSetMethodTest extends PostgresBase implements EmailSetMethodContract { + public static final PostgresMessageId.Factory MESSAGE_ID_FACTORY = new PostgresMessageId.Factory(); + + @Override + public MessageId randomMessageId() { + return MESSAGE_ID_FACTORY.generate(); + } + + @Override + public String invalidMessageIdMessage(String invalid) { + return String.format("Invalid UUID string: %s", invalid); + } + + @Override + @Test + @Disabled("Distributed event bus is asynchronous, we cannot expect the newState to be returned immediately after Email/set call") + public void newStateShouldBeUpToDate(GuiceJamesServer server) { + } + + @Override + @Test + @Disabled("Distributed event bus is asynchronous, we cannot expect the newState to be returned immediately after Email/set call") + public void oldStateShouldIncludeSetChanges(GuiceJamesServer server) { + } +} diff --git a/server/protocols/jmap-rfc-8621-integration-tests/postgres-jmap-rfc-8621-integration-tests/src/test/java/org/apache/james/jmap/rfc8621/postgres/PostgresEmailSubmissionSetMethodFutureReleaseTest.java b/server/protocols/jmap-rfc-8621-integration-tests/postgres-jmap-rfc-8621-integration-tests/src/test/java/org/apache/james/jmap/rfc8621/postgres/PostgresEmailSubmissionSetMethodFutureReleaseTest.java new file mode 100644 index 00000000000..4d189374bc9 --- /dev/null +++ b/server/protocols/jmap-rfc-8621-integration-tests/postgres-jmap-rfc-8621-integration-tests/src/test/java/org/apache/james/jmap/rfc8621/postgres/PostgresEmailSubmissionSetMethodFutureReleaseTest.java @@ -0,0 +1,95 @@ +/**************************************************************** + * 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.jmap.rfc8621.postgres; + +import static org.apache.james.data.UsersRepositoryModuleChooser.Implementation.DEFAULT; + +import org.apache.james.ClockExtension; +import org.apache.james.GuiceJamesServer; +import org.apache.james.JamesServerBuilder; +import org.apache.james.JamesServerExtension; +import org.apache.james.PostgresJamesConfiguration; +import org.apache.james.PostgresJamesServerMain; +import org.apache.james.SearchConfiguration; +import org.apache.james.backends.postgres.PostgresExtension; +import org.apache.james.jmap.rfc8621.contract.EmailSubmissionSetMethodFutureReleaseContract; +import org.apache.james.jmap.rfc8621.contract.probe.DelegationProbeModule; +import org.apache.james.mailbox.model.MessageId; +import org.apache.james.mailbox.postgres.PostgresMessageId; +import org.apache.james.modules.RabbitMQExtension; +import org.apache.james.modules.TestJMAPServerModule; +import org.apache.james.modules.blobstore.BlobStoreConfiguration; +import org.apache.james.utils.UpdatableTickingClock; +import org.junit.jupiter.api.Disabled; +import org.junit.jupiter.api.extension.RegisterExtension; + +import com.google.inject.name.Names; + +public class PostgresEmailSubmissionSetMethodFutureReleaseTest implements EmailSubmissionSetMethodFutureReleaseContract { + public static final PostgresMessageId.Factory MESSAGE_ID_FACTORY = new PostgresMessageId.Factory(); + + @RegisterExtension + static JamesServerExtension testExtension = new JamesServerBuilder(tmpDir -> + PostgresJamesConfiguration.builder() + .workingDirectory(tmpDir) + .configurationFromClasspath() + .searchConfiguration(SearchConfiguration.scanning()) + .usersRepository(DEFAULT) + .eventBusImpl(PostgresJamesConfiguration.EventBusImpl.RABBITMQ) + .blobStore(BlobStoreConfiguration.builder() + .postgres() + .disableCache() + .deduplication() + .noCryptoConfig()) + .build()) + .extension(PostgresExtension.empty()) + .extension(new RabbitMQExtension()) + .extension(new ClockExtension()) + .server(configuration -> PostgresJamesServerMain.createServer(configuration) + .overrideWith(new TestJMAPServerModule()) + .overrideWith(new DelegationProbeModule())) + .overrideServerModule(binder -> binder.bind(Boolean.class).annotatedWith(Names.named("supportsDelaySends")).toInstance(true)) + .build(); + + @Override + public MessageId randomMessageId() { + return MESSAGE_ID_FACTORY.generate(); + } + + @Disabled("Not work for postgres test") + @Override + public void emailSubmissionSetCreateShouldDeliverEmailWhenHoldForExpired(GuiceJamesServer server, UpdatableTickingClock updatableTickingClock){ + } + + @Disabled("Not work for postgres test") + @Override + public void emailSubmissionSetCreateShouldDeliverEmailWhenHoldUntilExpired(GuiceJamesServer server, UpdatableTickingClock updatableTickingClock){ + } + + @Disabled("Not work for postgres test") + @Override + public void emailSubmissionSetCreateShouldDelayEmailWithHoldFor(GuiceJamesServer server, UpdatableTickingClock updatableTickingClock){ + } + + @Disabled("Not work for postgres test") + @Override + public void emailSubmissionSetCreateShouldDelayEmailWithHoldUntil(GuiceJamesServer server, UpdatableTickingClock updatableTickingClock){ + } +} diff --git a/server/protocols/jmap-rfc-8621-integration-tests/postgres-jmap-rfc-8621-integration-tests/src/test/java/org/apache/james/jmap/rfc8621/postgres/PostgresEmailSubmissionSetMethodTest.java b/server/protocols/jmap-rfc-8621-integration-tests/postgres-jmap-rfc-8621-integration-tests/src/test/java/org/apache/james/jmap/rfc8621/postgres/PostgresEmailSubmissionSetMethodTest.java new file mode 100644 index 00000000000..536a5928d3f --- /dev/null +++ b/server/protocols/jmap-rfc-8621-integration-tests/postgres-jmap-rfc-8621-integration-tests/src/test/java/org/apache/james/jmap/rfc8621/postgres/PostgresEmailSubmissionSetMethodTest.java @@ -0,0 +1,33 @@ +/**************************************************************** + * 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.jmap.rfc8621.postgres; + +import org.apache.james.jmap.rfc8621.contract.EmailSubmissionSetMethodContract; +import org.apache.james.mailbox.model.MessageId; +import org.apache.james.mailbox.postgres.PostgresMessageId; + +public class PostgresEmailSubmissionSetMethodTest extends PostgresBase implements EmailSubmissionSetMethodContract { + public static final PostgresMessageId.Factory MESSAGE_ID_FACTORY = new PostgresMessageId.Factory(); + + @Override + public MessageId randomMessageId() { + return MESSAGE_ID_FACTORY.generate(); + } +} diff --git a/server/protocols/jmap-rfc-8621-integration-tests/postgres-jmap-rfc-8621-integration-tests/src/test/java/org/apache/james/jmap/rfc8621/postgres/PostgresIdentityGetTest.java b/server/protocols/jmap-rfc-8621-integration-tests/postgres-jmap-rfc-8621-integration-tests/src/test/java/org/apache/james/jmap/rfc8621/postgres/PostgresIdentityGetTest.java new file mode 100644 index 00000000000..6bbddd16a9c --- /dev/null +++ b/server/protocols/jmap-rfc-8621-integration-tests/postgres-jmap-rfc-8621-integration-tests/src/test/java/org/apache/james/jmap/rfc8621/postgres/PostgresIdentityGetTest.java @@ -0,0 +1,25 @@ +/**************************************************************** + * 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.jmap.rfc8621.postgres; + +import org.apache.james.jmap.rfc8621.contract.IdentityGetContract; + +public class PostgresIdentityGetTest extends PostgresBase implements IdentityGetContract { +} diff --git a/server/protocols/jmap-rfc-8621-integration-tests/postgres-jmap-rfc-8621-integration-tests/src/test/java/org/apache/james/jmap/rfc8621/postgres/PostgresIdentitySetTest.java b/server/protocols/jmap-rfc-8621-integration-tests/postgres-jmap-rfc-8621-integration-tests/src/test/java/org/apache/james/jmap/rfc8621/postgres/PostgresIdentitySetTest.java new file mode 100644 index 00000000000..b00cd3e2438 --- /dev/null +++ b/server/protocols/jmap-rfc-8621-integration-tests/postgres-jmap-rfc-8621-integration-tests/src/test/java/org/apache/james/jmap/rfc8621/postgres/PostgresIdentitySetTest.java @@ -0,0 +1,25 @@ +/**************************************************************** + * 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.jmap.rfc8621.postgres; + +import org.apache.james.jmap.rfc8621.contract.IdentitySetContract; + +public class PostgresIdentitySetTest extends PostgresBase implements IdentitySetContract { +} diff --git a/server/protocols/jmap-rfc-8621-integration-tests/postgres-jmap-rfc-8621-integration-tests/src/test/java/org/apache/james/jmap/rfc8621/postgres/PostgresMDNParseMethodTest.java b/server/protocols/jmap-rfc-8621-integration-tests/postgres-jmap-rfc-8621-integration-tests/src/test/java/org/apache/james/jmap/rfc8621/postgres/PostgresMDNParseMethodTest.java new file mode 100644 index 00000000000..135c9073507 --- /dev/null +++ b/server/protocols/jmap-rfc-8621-integration-tests/postgres-jmap-rfc-8621-integration-tests/src/test/java/org/apache/james/jmap/rfc8621/postgres/PostgresMDNParseMethodTest.java @@ -0,0 +1,33 @@ +/**************************************************************** + * 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.jmap.rfc8621.postgres; + +import org.apache.james.jmap.rfc8621.contract.MDNParseMethodContract; +import org.apache.james.mailbox.model.MessageId; +import org.apache.james.mailbox.postgres.PostgresMessageId; + +public class PostgresMDNParseMethodTest extends PostgresBase implements MDNParseMethodContract { + public static final PostgresMessageId.Factory MESSAGE_ID_FACTORY = new PostgresMessageId.Factory(); + + @Override + public MessageId randomMessageId() { + return MESSAGE_ID_FACTORY.generate(); + } +} diff --git a/server/protocols/jmap-rfc-8621-integration-tests/postgres-jmap-rfc-8621-integration-tests/src/test/java/org/apache/james/jmap/rfc8621/postgres/PostgresMDNSendMethodTest.java b/server/protocols/jmap-rfc-8621-integration-tests/postgres-jmap-rfc-8621-integration-tests/src/test/java/org/apache/james/jmap/rfc8621/postgres/PostgresMDNSendMethodTest.java new file mode 100644 index 00000000000..1c57e5682d4 --- /dev/null +++ b/server/protocols/jmap-rfc-8621-integration-tests/postgres-jmap-rfc-8621-integration-tests/src/test/java/org/apache/james/jmap/rfc8621/postgres/PostgresMDNSendMethodTest.java @@ -0,0 +1,33 @@ +/**************************************************************** + * 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.jmap.rfc8621.postgres; + +import org.apache.james.jmap.rfc8621.contract.MDNSendMethodContract; +import org.apache.james.mailbox.model.MessageId; +import org.apache.james.mailbox.postgres.PostgresMessageId; + +public class PostgresMDNSendMethodTest extends PostgresBase implements MDNSendMethodContract { + public static final PostgresMessageId.Factory MESSAGE_ID_FACTORY = new PostgresMessageId.Factory(); + + @Override + public MessageId randomMessageId() { + return MESSAGE_ID_FACTORY.generate(); + } +} diff --git a/server/protocols/jmap-rfc-8621-integration-tests/postgres-jmap-rfc-8621-integration-tests/src/test/java/org/apache/james/jmap/rfc8621/postgres/PostgresMailboxChangesMethodTest.java b/server/protocols/jmap-rfc-8621-integration-tests/postgres-jmap-rfc-8621-integration-tests/src/test/java/org/apache/james/jmap/rfc8621/postgres/PostgresMailboxChangesMethodTest.java new file mode 100644 index 00000000000..e2b013b15e4 --- /dev/null +++ b/server/protocols/jmap-rfc-8621-integration-tests/postgres-jmap-rfc-8621-integration-tests/src/test/java/org/apache/james/jmap/rfc8621/postgres/PostgresMailboxChangesMethodTest.java @@ -0,0 +1,76 @@ +/**************************************************************** + * 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.jmap.rfc8621.postgres; + +import static org.apache.james.data.UsersRepositoryModuleChooser.Implementation.DEFAULT; + +import org.apache.james.JamesServerBuilder; +import org.apache.james.JamesServerExtension; +import org.apache.james.PostgresJamesConfiguration; +import org.apache.james.PostgresJamesServerMain; +import org.apache.james.SearchConfiguration; +import org.apache.james.backends.postgres.PostgresExtension; +import org.apache.james.jmap.api.change.Limit; +import org.apache.james.jmap.api.change.State; +import org.apache.james.jmap.postgres.change.PostgresMailboxChangeRepository; +import org.apache.james.jmap.postgres.change.PostgresStateFactory; +import org.apache.james.jmap.rfc8621.contract.MailboxChangesMethodContract; +import org.apache.james.mailbox.model.MailboxId; +import org.apache.james.mailbox.postgres.PostgresMailboxId; +import org.apache.james.modules.RabbitMQExtension; +import org.apache.james.modules.TestJMAPServerModule; +import org.apache.james.modules.blobstore.BlobStoreConfiguration; +import org.junit.jupiter.api.extension.RegisterExtension; + +import com.google.inject.name.Names; + +public class PostgresMailboxChangesMethodTest implements MailboxChangesMethodContract { + @RegisterExtension + static JamesServerExtension testExtension = new JamesServerBuilder(tmpDir -> + PostgresJamesConfiguration.builder() + .workingDirectory(tmpDir) + .configurationFromClasspath() + .searchConfiguration(SearchConfiguration.scanning()) + .usersRepository(DEFAULT) + .eventBusImpl(PostgresJamesConfiguration.EventBusImpl.RABBITMQ) + .blobStore(BlobStoreConfiguration.builder() + .postgres() + .disableCache() + .deduplication() + .noCryptoConfig()) + .build()) + .extension(PostgresExtension.empty()) + .extension(new RabbitMQExtension()) + .server(configuration -> PostgresJamesServerMain.createServer(configuration) + .overrideWith(new TestJMAPServerModule()) + .overrideWith(binder -> binder.bind(Limit.class).annotatedWith(Names.named(PostgresMailboxChangeRepository.LIMIT_NAME)).toInstance(Limit.of(5))) + .overrideWith(binder -> binder.bind(Limit.class).annotatedWith(Names.named(PostgresMailboxChangeRepository.LIMIT_NAME)).toInstance(Limit.of(5)))) + .build(); + + @Override + public State.Factory stateFactory() { + return new PostgresStateFactory(); + } + + @Override + public MailboxId generateMailboxId() { + return PostgresMailboxId.generate(); + } +} diff --git a/server/protocols/jmap-rfc-8621-integration-tests/postgres-jmap-rfc-8621-integration-tests/src/test/java/org/apache/james/jmap/rfc8621/postgres/PostgresMailboxGetMethodTest.java b/server/protocols/jmap-rfc-8621-integration-tests/postgres-jmap-rfc-8621-integration-tests/src/test/java/org/apache/james/jmap/rfc8621/postgres/PostgresMailboxGetMethodTest.java new file mode 100644 index 00000000000..8632344dc44 --- /dev/null +++ b/server/protocols/jmap-rfc-8621-integration-tests/postgres-jmap-rfc-8621-integration-tests/src/test/java/org/apache/james/jmap/rfc8621/postgres/PostgresMailboxGetMethodTest.java @@ -0,0 +1,31 @@ +/**************************************************************** + * 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.jmap.rfc8621.postgres; + +import org.apache.james.jmap.rfc8621.contract.MailboxGetMethodContract; +import org.apache.james.mailbox.model.MailboxId; +import org.apache.james.mailbox.postgres.PostgresMailboxId; + +public class PostgresMailboxGetMethodTest extends PostgresBase implements MailboxGetMethodContract { + @Override + public MailboxId randomMailboxId() { + return PostgresMailboxId.generate(); + } +} diff --git a/server/protocols/jmap-rfc-8621-integration-tests/postgres-jmap-rfc-8621-integration-tests/src/test/java/org/apache/james/jmap/rfc8621/postgres/PostgresMailboxQueryChangesTest.java b/server/protocols/jmap-rfc-8621-integration-tests/postgres-jmap-rfc-8621-integration-tests/src/test/java/org/apache/james/jmap/rfc8621/postgres/PostgresMailboxQueryChangesTest.java new file mode 100644 index 00000000000..47a3abdc567 --- /dev/null +++ b/server/protocols/jmap-rfc-8621-integration-tests/postgres-jmap-rfc-8621-integration-tests/src/test/java/org/apache/james/jmap/rfc8621/postgres/PostgresMailboxQueryChangesTest.java @@ -0,0 +1,25 @@ +/**************************************************************** + * 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.jmap.rfc8621.postgres; + +import org.apache.james.jmap.rfc8621.contract.MailboxQueryChangesContract; + +public class PostgresMailboxQueryChangesTest extends PostgresBase implements MailboxQueryChangesContract { +} diff --git a/server/protocols/jmap-rfc-8621-integration-tests/postgres-jmap-rfc-8621-integration-tests/src/test/java/org/apache/james/jmap/rfc8621/postgres/PostgresMailboxQueryMethodTest.java b/server/protocols/jmap-rfc-8621-integration-tests/postgres-jmap-rfc-8621-integration-tests/src/test/java/org/apache/james/jmap/rfc8621/postgres/PostgresMailboxQueryMethodTest.java new file mode 100644 index 00000000000..f64a44f89c3 --- /dev/null +++ b/server/protocols/jmap-rfc-8621-integration-tests/postgres-jmap-rfc-8621-integration-tests/src/test/java/org/apache/james/jmap/rfc8621/postgres/PostgresMailboxQueryMethodTest.java @@ -0,0 +1,25 @@ +/**************************************************************** + * 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.jmap.rfc8621.postgres; + +import org.apache.james.jmap.rfc8621.contract.MailboxQueryMethodContract; + +public class PostgresMailboxQueryMethodTest extends PostgresBase implements MailboxQueryMethodContract { +} diff --git a/server/protocols/jmap-rfc-8621-integration-tests/postgres-jmap-rfc-8621-integration-tests/src/test/java/org/apache/james/jmap/rfc8621/postgres/PostgresMailboxSetMethodTest.java b/server/protocols/jmap-rfc-8621-integration-tests/postgres-jmap-rfc-8621-integration-tests/src/test/java/org/apache/james/jmap/rfc8621/postgres/PostgresMailboxSetMethodTest.java new file mode 100644 index 00000000000..8346421fb1e --- /dev/null +++ b/server/protocols/jmap-rfc-8621-integration-tests/postgres-jmap-rfc-8621-integration-tests/src/test/java/org/apache/james/jmap/rfc8621/postgres/PostgresMailboxSetMethodTest.java @@ -0,0 +1,51 @@ +/**************************************************************** + * 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.jmap.rfc8621.postgres; + +import org.apache.james.GuiceJamesServer; +import org.apache.james.jmap.rfc8621.contract.MailboxSetMethodContract; +import org.apache.james.mailbox.model.MailboxId; +import org.apache.james.mailbox.postgres.PostgresMailboxId; +import org.junit.jupiter.api.Disabled; +import org.junit.jupiter.api.Test; + +public class PostgresMailboxSetMethodTest extends PostgresBase implements MailboxSetMethodContract { + @Override + public MailboxId randomMailboxId() { + return PostgresMailboxId.generate(); + } + + @Override + public String errorInvalidMailboxIdMessage(String value) { + return String.format("%s is not a mailboxId: Invalid UUID string: %s", value, value); + } + + @Override + @Test + @Disabled("Distributed event bus is asynchronous, we cannot expect the newState to be returned immediately after Mailbox/set call") + public void newStateShouldBeUpToDate(GuiceJamesServer server) { + } + + @Override + @Test + @Disabled("Distributed event bus is asynchronous, we cannot expect the newState to be returned immediately after Mailbox/set call") + public void oldStateShouldIncludeSetChanges(GuiceJamesServer server) { + } +} diff --git a/server/protocols/jmap-rfc-8621-integration-tests/postgres-jmap-rfc-8621-integration-tests/src/test/java/org/apache/james/jmap/rfc8621/postgres/PostgresProvisioningTest.java b/server/protocols/jmap-rfc-8621-integration-tests/postgres-jmap-rfc-8621-integration-tests/src/test/java/org/apache/james/jmap/rfc8621/postgres/PostgresProvisioningTest.java new file mode 100644 index 00000000000..83877ba90e3 --- /dev/null +++ b/server/protocols/jmap-rfc-8621-integration-tests/postgres-jmap-rfc-8621-integration-tests/src/test/java/org/apache/james/jmap/rfc8621/postgres/PostgresProvisioningTest.java @@ -0,0 +1,25 @@ +/**************************************************************** + * 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.jmap.rfc8621.postgres; + +import org.apache.james.jmap.rfc8621.contract.ProvisioningContract; + +public class PostgresProvisioningTest extends PostgresBase implements ProvisioningContract { +} diff --git a/server/protocols/jmap-rfc-8621-integration-tests/postgres-jmap-rfc-8621-integration-tests/src/test/java/org/apache/james/jmap/rfc8621/postgres/PostgresPushSubscriptionSetMethodTest.java b/server/protocols/jmap-rfc-8621-integration-tests/postgres-jmap-rfc-8621-integration-tests/src/test/java/org/apache/james/jmap/rfc8621/postgres/PostgresPushSubscriptionSetMethodTest.java new file mode 100644 index 00000000000..06ba0f85e90 --- /dev/null +++ b/server/protocols/jmap-rfc-8621-integration-tests/postgres-jmap-rfc-8621-integration-tests/src/test/java/org/apache/james/jmap/rfc8621/postgres/PostgresPushSubscriptionSetMethodTest.java @@ -0,0 +1,66 @@ +/**************************************************************** + * 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.jmap.rfc8621.postgres; + +import static org.apache.james.data.UsersRepositoryModuleChooser.Implementation.DEFAULT; + +import org.apache.james.ClockExtension; +import org.apache.james.JamesServerBuilder; +import org.apache.james.JamesServerExtension; +import org.apache.james.PostgresJamesConfiguration; +import org.apache.james.PostgresJamesServerMain; +import org.apache.james.SearchConfiguration; +import org.apache.james.backends.postgres.PostgresExtension; +import org.apache.james.jmap.pushsubscription.PushClientConfiguration; +import org.apache.james.jmap.rfc8621.contract.PushServerExtension; +import org.apache.james.jmap.rfc8621.contract.PushSubscriptionProbeModule; +import org.apache.james.jmap.rfc8621.contract.PushSubscriptionSetMethodContract; +import org.apache.james.modules.RabbitMQExtension; +import org.apache.james.modules.TestJMAPServerModule; +import org.apache.james.modules.blobstore.BlobStoreConfiguration; +import org.junit.jupiter.api.extension.RegisterExtension; + +public class PostgresPushSubscriptionSetMethodTest implements PushSubscriptionSetMethodContract { + @RegisterExtension + static JamesServerExtension testExtension = new JamesServerBuilder(tmpDir -> + PostgresJamesConfiguration.builder() + .workingDirectory(tmpDir) + .configurationFromClasspath() + .searchConfiguration(SearchConfiguration.scanning()) + .usersRepository(DEFAULT) + .eventBusImpl(PostgresJamesConfiguration.EventBusImpl.RABBITMQ) + .blobStore(BlobStoreConfiguration.builder() + .postgres() + .disableCache() + .deduplication() + .noCryptoConfig()) + .build()) + .extension(PostgresExtension.empty()) + .extension(new RabbitMQExtension()) + .extension(new ClockExtension()) + .server(configuration -> PostgresJamesServerMain.createServer(configuration) + .overrideWith(new TestJMAPServerModule()) + .overrideWith(new PushSubscriptionProbeModule()) + .overrideWith(binder -> binder.bind(PushClientConfiguration.class).toInstance(PushClientConfiguration.UNSAFE_DEFAULT()))) + .build(); + + @RegisterExtension + static PushServerExtension pushServerExtension = new PushServerExtension(); +} diff --git a/server/protocols/jmap-rfc-8621-integration-tests/postgres-jmap-rfc-8621-integration-tests/src/test/java/org/apache/james/jmap/rfc8621/postgres/PostgresQuotaChangesMethodTest.java b/server/protocols/jmap-rfc-8621-integration-tests/postgres-jmap-rfc-8621-integration-tests/src/test/java/org/apache/james/jmap/rfc8621/postgres/PostgresQuotaChangesMethodTest.java new file mode 100644 index 00000000000..f83c7619274 --- /dev/null +++ b/server/protocols/jmap-rfc-8621-integration-tests/postgres-jmap-rfc-8621-integration-tests/src/test/java/org/apache/james/jmap/rfc8621/postgres/PostgresQuotaChangesMethodTest.java @@ -0,0 +1,25 @@ +/**************************************************************** + * 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.jmap.rfc8621.postgres; + +import org.apache.james.jmap.rfc8621.contract.QuotaChangesMethodContract; + +public class PostgresQuotaChangesMethodTest extends PostgresBase implements QuotaChangesMethodContract { +} diff --git a/server/protocols/jmap-rfc-8621-integration-tests/postgres-jmap-rfc-8621-integration-tests/src/test/java/org/apache/james/jmap/rfc8621/postgres/PostgresQuotaGetMethodTest.java b/server/protocols/jmap-rfc-8621-integration-tests/postgres-jmap-rfc-8621-integration-tests/src/test/java/org/apache/james/jmap/rfc8621/postgres/PostgresQuotaGetMethodTest.java new file mode 100644 index 00000000000..a64d8e683ca --- /dev/null +++ b/server/protocols/jmap-rfc-8621-integration-tests/postgres-jmap-rfc-8621-integration-tests/src/test/java/org/apache/james/jmap/rfc8621/postgres/PostgresQuotaGetMethodTest.java @@ -0,0 +1,25 @@ +/**************************************************************** + * 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.jmap.rfc8621.postgres; + +import org.apache.james.jmap.rfc8621.contract.QuotaGetMethodContract; + +public class PostgresQuotaGetMethodTest extends PostgresBase implements QuotaGetMethodContract { +} diff --git a/server/protocols/jmap-rfc-8621-integration-tests/postgres-jmap-rfc-8621-integration-tests/src/test/java/org/apache/james/jmap/rfc8621/postgres/PostgresQuotaQueryMethodTest.java b/server/protocols/jmap-rfc-8621-integration-tests/postgres-jmap-rfc-8621-integration-tests/src/test/java/org/apache/james/jmap/rfc8621/postgres/PostgresQuotaQueryMethodTest.java new file mode 100644 index 00000000000..558709ab5e5 --- /dev/null +++ b/server/protocols/jmap-rfc-8621-integration-tests/postgres-jmap-rfc-8621-integration-tests/src/test/java/org/apache/james/jmap/rfc8621/postgres/PostgresQuotaQueryMethodTest.java @@ -0,0 +1,25 @@ +/**************************************************************** + * 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.jmap.rfc8621.postgres; + +import org.apache.james.jmap.rfc8621.contract.QuotaQueryMethodContract; + +public class PostgresQuotaQueryMethodTest extends PostgresBase implements QuotaQueryMethodContract { +} diff --git a/server/protocols/jmap-rfc-8621-integration-tests/postgres-jmap-rfc-8621-integration-tests/src/test/java/org/apache/james/jmap/rfc8621/postgres/PostgresSessionRoutesTest.java b/server/protocols/jmap-rfc-8621-integration-tests/postgres-jmap-rfc-8621-integration-tests/src/test/java/org/apache/james/jmap/rfc8621/postgres/PostgresSessionRoutesTest.java new file mode 100644 index 00000000000..9957ff3cf59 --- /dev/null +++ b/server/protocols/jmap-rfc-8621-integration-tests/postgres-jmap-rfc-8621-integration-tests/src/test/java/org/apache/james/jmap/rfc8621/postgres/PostgresSessionRoutesTest.java @@ -0,0 +1,25 @@ +/**************************************************************** + * 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.jmap.rfc8621.postgres; + +import org.apache.james.jmap.rfc8621.contract.SessionRoutesContract; + +public class PostgresSessionRoutesTest extends PostgresBase implements SessionRoutesContract { +} diff --git a/server/protocols/jmap-rfc-8621-integration-tests/postgres-jmap-rfc-8621-integration-tests/src/test/java/org/apache/james/jmap/rfc8621/postgres/PostgresThreadGetTest.java b/server/protocols/jmap-rfc-8621-integration-tests/postgres-jmap-rfc-8621-integration-tests/src/test/java/org/apache/james/jmap/rfc8621/postgres/PostgresThreadGetTest.java new file mode 100644 index 00000000000..10d3c11f717 --- /dev/null +++ b/server/protocols/jmap-rfc-8621-integration-tests/postgres-jmap-rfc-8621-integration-tests/src/test/java/org/apache/james/jmap/rfc8621/postgres/PostgresThreadGetTest.java @@ -0,0 +1,70 @@ +/**************************************************************** + * 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.jmap.rfc8621.postgres; + +import static org.apache.james.data.UsersRepositoryModuleChooser.Implementation.DEFAULT; + +import java.util.List; + +import org.apache.james.DockerOpenSearchExtension; +import org.apache.james.JamesServerBuilder; +import org.apache.james.JamesServerExtension; +import org.apache.james.PostgresJamesConfiguration; +import org.apache.james.PostgresJamesServerMain; +import org.apache.james.SearchConfiguration; +import org.apache.james.backends.postgres.PostgresExtension; +import org.apache.james.jmap.rfc8621.contract.ThreadGetContract; +import org.apache.james.mailbox.model.MailboxId; +import org.apache.james.mailbox.model.SearchQuery; +import org.apache.james.modules.RabbitMQExtension; +import org.apache.james.modules.TestJMAPServerModule; +import org.apache.james.modules.blobstore.BlobStoreConfiguration; +import org.junit.jupiter.api.extension.RegisterExtension; + +public class PostgresThreadGetTest extends PostgresBase implements ThreadGetContract { + @RegisterExtension + static JamesServerExtension testExtension = new JamesServerBuilder(tmpDir -> + PostgresJamesConfiguration.builder() + .workingDirectory(tmpDir) + .configurationFromClasspath() + .searchConfiguration(SearchConfiguration.openSearch()) + .usersRepository(DEFAULT) + .eventBusImpl(PostgresJamesConfiguration.EventBusImpl.RABBITMQ) + .blobStore(BlobStoreConfiguration.builder() + .postgres() + .disableCache() + .deduplication() + .noCryptoConfig()) + .build()) + .extension(PostgresExtension.empty()) + .extension(new RabbitMQExtension()) + .extension(new DockerOpenSearchExtension()) + .server(configuration -> PostgresJamesServerMain.createServer(configuration) + .overrideWith(new TestJMAPServerModule())) + .build(); + + @Override + public void awaitMessageCount(List mailboxIds, SearchQuery query, long messageCount) { + } + + @Override + public void initOpenSearchClient() { + } +} diff --git a/server/protocols/jmap-rfc-8621-integration-tests/postgres-jmap-rfc-8621-integration-tests/src/test/java/org/apache/james/jmap/rfc8621/postgres/PostgresUploadTest.java b/server/protocols/jmap-rfc-8621-integration-tests/postgres-jmap-rfc-8621-integration-tests/src/test/java/org/apache/james/jmap/rfc8621/postgres/PostgresUploadTest.java new file mode 100644 index 00000000000..b280238f956 --- /dev/null +++ b/server/protocols/jmap-rfc-8621-integration-tests/postgres-jmap-rfc-8621-integration-tests/src/test/java/org/apache/james/jmap/rfc8621/postgres/PostgresUploadTest.java @@ -0,0 +1,25 @@ +/**************************************************************** + * 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.jmap.rfc8621.postgres; + +import org.apache.james.jmap.rfc8621.contract.UploadContract; + +public class PostgresUploadTest extends PostgresBase implements UploadContract { +} diff --git a/server/protocols/jmap-rfc-8621-integration-tests/postgres-jmap-rfc-8621-integration-tests/src/test/java/org/apache/james/jmap/rfc8621/postgres/PostgresVacationResponseGetMethodTest.java b/server/protocols/jmap-rfc-8621-integration-tests/postgres-jmap-rfc-8621-integration-tests/src/test/java/org/apache/james/jmap/rfc8621/postgres/PostgresVacationResponseGetMethodTest.java new file mode 100644 index 00000000000..98aa5ade206 --- /dev/null +++ b/server/protocols/jmap-rfc-8621-integration-tests/postgres-jmap-rfc-8621-integration-tests/src/test/java/org/apache/james/jmap/rfc8621/postgres/PostgresVacationResponseGetMethodTest.java @@ -0,0 +1,25 @@ +/**************************************************************** + * 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.jmap.rfc8621.postgres; + +import org.apache.james.jmap.rfc8621.contract.VacationResponseGetMethodContract; + +public class PostgresVacationResponseGetMethodTest extends PostgresBase implements VacationResponseGetMethodContract { +} diff --git a/server/protocols/jmap-rfc-8621-integration-tests/postgres-jmap-rfc-8621-integration-tests/src/test/java/org/apache/james/jmap/rfc8621/postgres/PostgresVacationResponseSetMethodTest.java b/server/protocols/jmap-rfc-8621-integration-tests/postgres-jmap-rfc-8621-integration-tests/src/test/java/org/apache/james/jmap/rfc8621/postgres/PostgresVacationResponseSetMethodTest.java new file mode 100644 index 00000000000..4ecf2ab5793 --- /dev/null +++ b/server/protocols/jmap-rfc-8621-integration-tests/postgres-jmap-rfc-8621-integration-tests/src/test/java/org/apache/james/jmap/rfc8621/postgres/PostgresVacationResponseSetMethodTest.java @@ -0,0 +1,25 @@ +/**************************************************************** + * 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.jmap.rfc8621.postgres; + +import org.apache.james.jmap.rfc8621.contract.VacationResponseSetMethodContract; + +public class PostgresVacationResponseSetMethodTest extends PostgresBase implements VacationResponseSetMethodContract { +} diff --git a/server/protocols/jmap-rfc-8621-integration-tests/postgres-jmap-rfc-8621-integration-tests/src/test/java/org/apache/james/jmap/rfc8621/postgres/PostgresWebPushTest.java b/server/protocols/jmap-rfc-8621-integration-tests/postgres-jmap-rfc-8621-integration-tests/src/test/java/org/apache/james/jmap/rfc8621/postgres/PostgresWebPushTest.java new file mode 100644 index 00000000000..919bb3fecd2 --- /dev/null +++ b/server/protocols/jmap-rfc-8621-integration-tests/postgres-jmap-rfc-8621-integration-tests/src/test/java/org/apache/james/jmap/rfc8621/postgres/PostgresWebPushTest.java @@ -0,0 +1,66 @@ +/**************************************************************** + * 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.jmap.rfc8621.postgres; + +import static org.apache.james.data.UsersRepositoryModuleChooser.Implementation.DEFAULT; + +import org.apache.james.ClockExtension; +import org.apache.james.JamesServerBuilder; +import org.apache.james.JamesServerExtension; +import org.apache.james.PostgresJamesConfiguration; +import org.apache.james.PostgresJamesServerMain; +import org.apache.james.SearchConfiguration; +import org.apache.james.backends.postgres.PostgresExtension; +import org.apache.james.jmap.pushsubscription.PushClientConfiguration; +import org.apache.james.jmap.rfc8621.contract.PushServerExtension; +import org.apache.james.jmap.rfc8621.contract.PushSubscriptionProbeModule; +import org.apache.james.jmap.rfc8621.contract.WebPushContract; +import org.apache.james.modules.RabbitMQExtension; +import org.apache.james.modules.TestJMAPServerModule; +import org.apache.james.modules.blobstore.BlobStoreConfiguration; +import org.junit.jupiter.api.extension.RegisterExtension; + +public class PostgresWebPushTest implements WebPushContract { + @RegisterExtension + static JamesServerExtension testExtension = new JamesServerBuilder(tmpDir -> + PostgresJamesConfiguration.builder() + .workingDirectory(tmpDir) + .configurationFromClasspath() + .searchConfiguration(SearchConfiguration.scanning()) + .usersRepository(DEFAULT) + .eventBusImpl(PostgresJamesConfiguration.EventBusImpl.RABBITMQ) + .blobStore(BlobStoreConfiguration.builder() + .postgres() + .disableCache() + .deduplication() + .noCryptoConfig()) + .build()) + .extension(PostgresExtension.empty()) + .extension(new RabbitMQExtension()) + .extension(new ClockExtension()) + .server(configuration -> PostgresJamesServerMain.createServer(configuration) + .overrideWith(new TestJMAPServerModule()) + .overrideWith(new PushSubscriptionProbeModule()) + .overrideWith(binder -> binder.bind(PushClientConfiguration.class).toInstance(PushClientConfiguration.UNSAFE_DEFAULT()))) + .build(); + + @RegisterExtension + static PushServerExtension pushServerExtension = new PushServerExtension(); +} diff --git a/server/protocols/jmap-rfc-8621-integration-tests/postgres-jmap-rfc-8621-integration-tests/src/test/java/org/apache/james/jmap/rfc8621/postgres/PostgresWebSocketTest.java b/server/protocols/jmap-rfc-8621-integration-tests/postgres-jmap-rfc-8621-integration-tests/src/test/java/org/apache/james/jmap/rfc8621/postgres/PostgresWebSocketTest.java new file mode 100644 index 00000000000..c16d808925c --- /dev/null +++ b/server/protocols/jmap-rfc-8621-integration-tests/postgres-jmap-rfc-8621-integration-tests/src/test/java/org/apache/james/jmap/rfc8621/postgres/PostgresWebSocketTest.java @@ -0,0 +1,25 @@ +/**************************************************************** + * 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.jmap.rfc8621.postgres; + +import org.apache.james.jmap.rfc8621.contract.WebSocketContract; + +public class PostgresWebSocketTest extends PostgresBase implements WebSocketContract { +} diff --git a/server/protocols/jmap-rfc-8621-integration-tests/postgres-jmap-rfc-8621-integration-tests/src/test/resources/dnsservice.xml b/server/protocols/jmap-rfc-8621-integration-tests/postgres-jmap-rfc-8621-integration-tests/src/test/resources/dnsservice.xml new file mode 100644 index 00000000000..6e4fbd2efb5 --- /dev/null +++ b/server/protocols/jmap-rfc-8621-integration-tests/postgres-jmap-rfc-8621-integration-tests/src/test/resources/dnsservice.xml @@ -0,0 +1,25 @@ + + + + + true + false + 50000 + diff --git a/server/protocols/jmap-rfc-8621-integration-tests/postgres-jmap-rfc-8621-integration-tests/src/test/resources/domainlist.xml b/server/protocols/jmap-rfc-8621-integration-tests/postgres-jmap-rfc-8621-integration-tests/src/test/resources/domainlist.xml new file mode 100644 index 00000000000..fe17431a1ea --- /dev/null +++ b/server/protocols/jmap-rfc-8621-integration-tests/postgres-jmap-rfc-8621-integration-tests/src/test/resources/domainlist.xml @@ -0,0 +1,24 @@ + + + + + false + false + diff --git a/server/protocols/jmap-rfc-8621-integration-tests/postgres-jmap-rfc-8621-integration-tests/src/test/resources/imapserver.xml b/server/protocols/jmap-rfc-8621-integration-tests/postgres-jmap-rfc-8621-integration-tests/src/test/resources/imapserver.xml new file mode 100644 index 00000000000..ead2b342f34 --- /dev/null +++ b/server/protocols/jmap-rfc-8621-integration-tests/postgres-jmap-rfc-8621-integration-tests/src/test/resources/imapserver.xml @@ -0,0 +1,24 @@ + + + + + + + diff --git a/server/protocols/jmap-rfc-8621-integration-tests/postgres-jmap-rfc-8621-integration-tests/src/test/resources/jmap.properties b/server/protocols/jmap-rfc-8621-integration-tests/postgres-jmap-rfc-8621-integration-tests/src/test/resources/jmap.properties new file mode 100644 index 00000000000..519703e204c --- /dev/null +++ b/server/protocols/jmap-rfc-8621-integration-tests/postgres-jmap-rfc-8621-integration-tests/src/test/resources/jmap.properties @@ -0,0 +1,7 @@ +# Configuration urlPrefix for JMAP routes. +url.prefix=http://domain.com +websocket.url.prefix=ws://domain.com +upload.max.size=20M +webpush.maxTimeoutSeconds=10 +webpush.maxConnections=10 +dynamic.jmap.prefix.resolution.enabled=true \ No newline at end of file diff --git a/server/protocols/jmap-rfc-8621-integration-tests/postgres-jmap-rfc-8621-integration-tests/src/test/resources/keystore b/server/protocols/jmap-rfc-8621-integration-tests/postgres-jmap-rfc-8621-integration-tests/src/test/resources/keystore new file mode 100644 index 00000000000..536a6c792b0 Binary files /dev/null and b/server/protocols/jmap-rfc-8621-integration-tests/postgres-jmap-rfc-8621-integration-tests/src/test/resources/keystore differ diff --git a/server/protocols/jmap-rfc-8621-integration-tests/postgres-jmap-rfc-8621-integration-tests/src/test/resources/listeners.xml b/server/protocols/jmap-rfc-8621-integration-tests/postgres-jmap-rfc-8621-integration-tests/src/test/resources/listeners.xml new file mode 100644 index 00000000000..ddc4d9d1522 --- /dev/null +++ b/server/protocols/jmap-rfc-8621-integration-tests/postgres-jmap-rfc-8621-integration-tests/src/test/resources/listeners.xml @@ -0,0 +1,26 @@ + + + + + + org.apache.james.jmap.event.PopulateEmailQueryViewListener + true + + \ No newline at end of file diff --git a/server/protocols/jmap-rfc-8621-integration-tests/postgres-jmap-rfc-8621-integration-tests/src/test/resources/mailetcontainer.xml b/server/protocols/jmap-rfc-8621-integration-tests/postgres-jmap-rfc-8621-integration-tests/src/test/resources/mailetcontainer.xml new file mode 100644 index 00000000000..f429a43156b --- /dev/null +++ b/server/protocols/jmap-rfc-8621-integration-tests/postgres-jmap-rfc-8621-integration-tests/src/test/resources/mailetcontainer.xml @@ -0,0 +1,98 @@ + + + + + + + + postmaster + + + + 2 + postgres://var/mail/error/ + + + + + + transport + + + + + + ignore + + + + + + ignore + + + + + + + + + + + + + + + + + bcc + + + error + + + ignore + + + ignore + + + ignore + + + + + outgoing + 5000, 100000, 500000 + 3 + 0 + 10 + true + error + + + + error + + + + + + diff --git a/server/protocols/jmap-rfc-8621-integration-tests/postgres-jmap-rfc-8621-integration-tests/src/test/resources/mailrepositorystore.xml b/server/protocols/jmap-rfc-8621-integration-tests/postgres-jmap-rfc-8621-integration-tests/src/test/resources/mailrepositorystore.xml new file mode 100644 index 00000000000..573ec24ad3e --- /dev/null +++ b/server/protocols/jmap-rfc-8621-integration-tests/postgres-jmap-rfc-8621-integration-tests/src/test/resources/mailrepositorystore.xml @@ -0,0 +1,30 @@ + + + + + + + + + postgres + + + + diff --git a/server/protocols/jmap-rfc-8621-integration-tests/postgres-jmap-rfc-8621-integration-tests/src/test/resources/managesieveserver.xml b/server/protocols/jmap-rfc-8621-integration-tests/postgres-jmap-rfc-8621-integration-tests/src/test/resources/managesieveserver.xml new file mode 100644 index 00000000000..f136a432b8a --- /dev/null +++ b/server/protocols/jmap-rfc-8621-integration-tests/postgres-jmap-rfc-8621-integration-tests/src/test/resources/managesieveserver.xml @@ -0,0 +1,32 @@ + + + + + + + + + + + + diff --git a/server/protocols/jmap-rfc-8621-integration-tests/postgres-jmap-rfc-8621-integration-tests/src/test/resources/pop3server.xml b/server/protocols/jmap-rfc-8621-integration-tests/postgres-jmap-rfc-8621-integration-tests/src/test/resources/pop3server.xml new file mode 100644 index 00000000000..bec385ae306 --- /dev/null +++ b/server/protocols/jmap-rfc-8621-integration-tests/postgres-jmap-rfc-8621-integration-tests/src/test/resources/pop3server.xml @@ -0,0 +1,23 @@ + + + + + + diff --git a/server/protocols/jmap-rfc-8621-integration-tests/postgres-jmap-rfc-8621-integration-tests/src/test/resources/rabbitmq.properties b/server/protocols/jmap-rfc-8621-integration-tests/postgres-jmap-rfc-8621-integration-tests/src/test/resources/rabbitmq.properties new file mode 100644 index 00000000000..25d0dd6a976 --- /dev/null +++ b/server/protocols/jmap-rfc-8621-integration-tests/postgres-jmap-rfc-8621-integration-tests/src/test/resources/rabbitmq.properties @@ -0,0 +1,2 @@ +uri=amqp://james:james@rabbitmq_host:5672 +management.uri=http://james:james@rabbitmq_host:15672/api/ \ No newline at end of file diff --git a/server/protocols/jmap-rfc-8621-integration-tests/postgres-jmap-rfc-8621-integration-tests/src/test/resources/smtpserver.xml b/server/protocols/jmap-rfc-8621-integration-tests/postgres-jmap-rfc-8621-integration-tests/src/test/resources/smtpserver.xml new file mode 100644 index 00000000000..21dc0a9af9c --- /dev/null +++ b/server/protocols/jmap-rfc-8621-integration-tests/postgres-jmap-rfc-8621-integration-tests/src/test/resources/smtpserver.xml @@ -0,0 +1,53 @@ + + + + + + + smtpserver-global + 0.0.0.0:0 + 200 + + file://conf/keystore + james72laBalle + org.bouncycastle.jce.provider.BouncyCastleProvider + SunX509 + + 360 + 0 + 0 + false + + never + false + true + + 0 + true + Apache JAMES awesome SMTP Server + + + + + false + + + + diff --git a/server/protocols/jmap-rfc-8621-integration-tests/postgres-jmap-rfc-8621-integration-tests/src/test/resources/usersrepository.xml b/server/protocols/jmap-rfc-8621-integration-tests/postgres-jmap-rfc-8621-integration-tests/src/test/resources/usersrepository.xml new file mode 100644 index 00000000000..f8c8a258722 --- /dev/null +++ b/server/protocols/jmap-rfc-8621-integration-tests/postgres-jmap-rfc-8621-integration-tests/src/test/resources/usersrepository.xml @@ -0,0 +1,25 @@ + + + + + + true + SHA-1 + diff --git a/server/protocols/jmap-rfc-8621/src/main/java/org/apache/james/jmap/event/PopulateEmailQueryViewListener.java b/server/protocols/jmap-rfc-8621/src/main/java/org/apache/james/jmap/event/PopulateEmailQueryViewListener.java index f7b78c69c5f..af9faf78c24 100644 --- a/server/protocols/jmap-rfc-8621/src/main/java/org/apache/james/jmap/event/PopulateEmailQueryViewListener.java +++ b/server/protocols/jmap-rfc-8621/src/main/java/org/apache/james/jmap/event/PopulateEmailQueryViewListener.java @@ -30,10 +30,11 @@ import jakarta.inject.Inject; +import org.apache.james.core.Username; import org.apache.james.events.Event; import org.apache.james.events.EventListener.ReactiveGroupEventListener; import org.apache.james.events.Group; -import org.apache.james.jmap.api.projections.EmailQueryView; +import org.apache.james.jmap.api.projections.EmailQueryViewManager; import org.apache.james.mailbox.MailboxSession; import org.apache.james.mailbox.MessageIdManager; import org.apache.james.mailbox.Role; @@ -72,13 +73,13 @@ public static class PopulateEmailQueryViewListenerGroup extends Group { private static final int CONCURRENCY = 5; private final MessageIdManager messageIdManager; - private final EmailQueryView view; + private final EmailQueryViewManager viewManager; private final SessionProvider sessionProvider; @Inject - public PopulateEmailQueryViewListener(MessageIdManager messageIdManager, EmailQueryView view, SessionProvider sessionProvider) { + public PopulateEmailQueryViewListener(MessageIdManager messageIdManager, EmailQueryViewManager viewManager, SessionProvider sessionProvider) { this.messageIdManager = messageIdManager; - this.view = view; + this.viewManager = viewManager; this.sessionProvider = sessionProvider; } @@ -113,13 +114,13 @@ public Publisher reactiveEvent(Event event) { } private Publisher handleMailboxDeletion(MailboxDeletion mailboxDeletion) { - return view.delete(mailboxDeletion.getMailboxId()); + return viewManager.getEmailQueryView(mailboxDeletion.getUsername()).delete(mailboxDeletion.getMailboxId()); } private Publisher handleExpunged(Expunged expunged) { return Flux.fromStream(expunged.getUids().stream() .map(uid -> expunged.getMetaData(uid).getMessageId())) - .concatMap(messageId -> view.delete(expunged.getMailboxId(), messageId)) + .concatMap(messageId -> viewManager.getEmailQueryView(expunged.getUsername()).delete(expunged.getMailboxId(), messageId)) .then(); } @@ -131,7 +132,7 @@ private Publisher handleFlagsUpdated(FlagsUpdated flagsUpdated) { .filter(updatedFlags -> updatedFlags.isModifiedToSet(DELETED)) .map(UpdatedFlags::getMessageId) .handle(publishIfPresent()) - .concatMap(messageId -> view.delete(flagsUpdated.getMailboxId(), messageId)) + .concatMap(messageId -> viewManager.getEmailQueryView(flagsUpdated.getUsername()).delete(flagsUpdated.getMailboxId(), messageId)) .then(); Mono addMessagesNoLongerMarkedAsDeleted = Flux.fromIterable(flagsUpdated.getUpdatedFlags()) @@ -141,7 +142,7 @@ private Publisher handleFlagsUpdated(FlagsUpdated flagsUpdated) { .concatMap(messageId -> Flux.from(messageIdManager.getMessagesReactive(ImmutableList.of(messageId), FetchGroup.HEADERS, session)) .next()) - .concatMap(message -> handleAdded(flagsUpdated.getMailboxId(), message)) + .concatMap(message -> handleAdded(flagsUpdated.getMailboxId(), message, flagsUpdated.getUsername())) .then(); return removeMessagesMarkedAsDeleted @@ -163,7 +164,7 @@ private Mono handleAdded(Added added, MessageMetaData messageMetaData, Mai Mono doHandleAdded = Flux.from(messageIdManager.getMessagesReactive(ImmutableList.of(messageId), FetchGroup.HEADERS, session)) .next() .filter(message -> !message.getFlags().contains(DELETED)) - .flatMap(messageResult -> handleAdded(added.getMailboxId(), messageResult)); + .flatMap(messageResult -> handleAdded(added.getMailboxId(), messageResult, added.getUsername())); if (Role.from(added.getMailboxPath().getName()).equals(Optional.of(Role.OUTBOX))) { return checkMessageStillInOriginMailbox(messageId, session, mailboxId) .filter(FunctionalUtils.identityPredicate()) @@ -178,13 +179,13 @@ private Mono checkMessageStillInOriginMailbox(MessageId messageId, Mail .hasElements(); } - public Mono handleAdded(MailboxId mailboxId, MessageResult messageResult) { + public Mono handleAdded(MailboxId mailboxId, MessageResult messageResult, Username username) { ZonedDateTime receivedAt = ZonedDateTime.ofInstant(messageResult.getInternalDate().toInstant(), ZoneOffset.UTC); return Mono.fromCallable(() -> parseMessage(messageResult)) .map(header -> date(header).orElse(messageResult.getInternalDate())) .map(date -> ZonedDateTime.ofInstant(date.toInstant(), ZoneOffset.UTC)) - .flatMap(sentAt -> view.save(mailboxId, sentAt, receivedAt, messageResult.getMessageId())) + .flatMap(sentAt -> viewManager.getEmailQueryView(username).save(mailboxId, sentAt, receivedAt, messageResult.getMessageId())) .then(); } diff --git a/server/protocols/jmap-rfc-8621/src/main/scala/org/apache/james/jmap/method/EmailQueryMethod.scala b/server/protocols/jmap-rfc-8621/src/main/scala/org/apache/james/jmap/method/EmailQueryMethod.scala index 26d2869ef81..a24a7e225c0 100644 --- a/server/protocols/jmap-rfc-8621/src/main/scala/org/apache/james/jmap/method/EmailQueryMethod.scala +++ b/server/protocols/jmap-rfc-8621/src/main/scala/org/apache/james/jmap/method/EmailQueryMethod.scala @@ -25,7 +25,7 @@ import eu.timepit.refined.auto._ import jakarta.inject.Inject import jakarta.mail.Flags.Flag.DELETED import org.apache.james.jmap.JMAPConfiguration -import org.apache.james.jmap.api.projections.EmailQueryView +import org.apache.james.jmap.api.projections.{EmailQueryView, EmailQueryViewManager} import org.apache.james.jmap.core.CapabilityIdentifier.{CapabilityIdentifier, JMAP_CORE, JMAP_MAIL} import org.apache.james.jmap.core.Invocation.{Arguments, MethodName} import org.apache.james.jmap.core.Limit.Limit @@ -52,7 +52,7 @@ class EmailQueryMethod @Inject() (serializer: EmailQuerySerializer, val sessionSupplier: SessionSupplier, val sessionTranslator: SessionTranslator, val configuration: JMAPConfiguration, - val emailQueryView: EmailQueryView) extends MethodRequiringAccountId[EmailQueryRequest] { + val emailQueryViewManager: EmailQueryViewManager) extends MethodRequiringAccountId[EmailQueryRequest] { override val methodName: MethodName = MethodName("Email/query") override val requiredCapabilities: Set[CapabilityIdentifier] = Set(JMAP_CORE, JMAP_MAIL) @@ -114,7 +114,8 @@ class EmailQueryMethod @Inject() (serializer: EmailQuerySerializer, val mailboxId: MailboxId = condition.inMailbox.get val after: ZonedDateTime = condition.after.get.asUTC - val queryViewEntries: SFlux[MessageId] = SFlux.fromPublisher(emailQueryView.listMailboxContentSinceAfterSortedBySentAt(mailboxId, after, JavaLimit.from(limitToUse.value + position.value))) + val queryViewEntries: SFlux[MessageId] = SFlux.fromPublisher(emailQueryViewManager.getEmailQueryView(mailboxSession.getUser) + .listMailboxContentSinceAfterSortedBySentAt(mailboxId, after, JavaLimit.from(limitToUse.value + position.value))) fromQueryViewEntries(mailboxId, queryViewEntries, mailboxSession, position, limitToUse, namespace) } @@ -123,7 +124,8 @@ class EmailQueryMethod @Inject() (serializer: EmailQuerySerializer, val condition: FilterCondition = request.filter.get.asInstanceOf[FilterCondition] val mailboxId: MailboxId = condition.inMailbox.get val after: ZonedDateTime = condition.after.get.asUTC - val queryViewEntries: SFlux[MessageId] = SFlux.fromPublisher(emailQueryView.listMailboxContentSinceAfterSortedByReceivedAt(mailboxId, after, JavaLimit.from(limitToUse.value + position.value))) + val queryViewEntries: SFlux[MessageId] = SFlux.fromPublisher(emailQueryViewManager.getEmailQueryView(mailboxSession.getUser) + .listMailboxContentSinceAfterSortedByReceivedAt(mailboxId, after, JavaLimit.from(limitToUse.value + position.value))) fromQueryViewEntries(mailboxId, queryViewEntries, mailboxSession, position, limitToUse, namespace) } @@ -132,21 +134,24 @@ class EmailQueryMethod @Inject() (serializer: EmailQuerySerializer, val condition: FilterCondition = request.filter.get.asInstanceOf[FilterCondition] val mailboxId: MailboxId = condition.inMailbox.get val before: ZonedDateTime = condition.before.get.asUTC - val queryViewEntries: SFlux[MessageId] = SFlux.fromPublisher(emailQueryView.listMailboxContentBeforeSortedByReceivedAt(mailboxId, before, JavaLimit.from(limitToUse.value + position.value))) + val queryViewEntries: SFlux[MessageId] = SFlux.fromPublisher(emailQueryViewManager.getEmailQueryView(mailboxSession.getUser) + .listMailboxContentBeforeSortedByReceivedAt(mailboxId, before, JavaLimit.from(limitToUse.value + position.value))) fromQueryViewEntries(mailboxId, queryViewEntries, mailboxSession, position, limitToUse, namespace) } private def queryViewForListingSortedBySentAt(mailboxSession: MailboxSession, position: Position, limitToUse: Limit, request: EmailQueryRequest, namespace: Namespace): SMono[Seq[MessageId]] = { val mailboxId: MailboxId = request.filter.get.asInstanceOf[FilterCondition].inMailbox.get - val queryViewEntries: SFlux[MessageId] = SFlux.fromPublisher(emailQueryView.listMailboxContentSortedBySentAt(mailboxId, JavaLimit.from(limitToUse.value + position.value))) + val queryViewEntries: SFlux[MessageId] = SFlux.fromPublisher(emailQueryViewManager.getEmailQueryView(mailboxSession.getUser) + .listMailboxContentSortedBySentAt(mailboxId, JavaLimit.from(limitToUse.value + position.value))) fromQueryViewEntries(mailboxId, queryViewEntries, mailboxSession, position, limitToUse, namespace) } private def queryViewForListingSortedByReceivedAt(mailboxSession: MailboxSession, position: Position, limitToUse: Limit, request: EmailQueryRequest, namespace: Namespace): SMono[Seq[MessageId]] = { val mailboxId: MailboxId = request.filter.get.asInstanceOf[FilterCondition].inMailbox.get - val queryViewEntries: SFlux[MessageId] = SFlux.fromPublisher(emailQueryView.listMailboxContentSortedByReceivedAt(mailboxId, JavaLimit.from(limitToUse.value + position.value))) + val queryViewEntries: SFlux[MessageId] = SFlux.fromPublisher(emailQueryViewManager + .getEmailQueryView(mailboxSession.getUser).listMailboxContentSortedByReceivedAt(mailboxId, JavaLimit.from(limitToUse.value + position.value))) fromQueryViewEntries(mailboxId, queryViewEntries, mailboxSession, position, limitToUse, namespace) } diff --git a/server/protocols/jmap-rfc-8621/src/test/java/org/apache/james/jmap/event/PopulateEmailQueryViewListenerTest.java b/server/protocols/jmap-rfc-8621/src/test/java/org/apache/james/jmap/event/PopulateEmailQueryViewListenerTest.java index 97446748900..bdd6d8e532f 100644 --- a/server/protocols/jmap-rfc-8621/src/test/java/org/apache/james/jmap/event/PopulateEmailQueryViewListenerTest.java +++ b/server/protocols/jmap-rfc-8621/src/test/java/org/apache/james/jmap/event/PopulateEmailQueryViewListenerTest.java @@ -39,6 +39,8 @@ import org.apache.james.events.MemoryEventDeadLetters; import org.apache.james.events.RetryBackoffConfiguration; import org.apache.james.events.delivery.InVmEventDelivery; +import org.apache.james.jmap.api.projections.DefaultEmailQueryViewManager; +import org.apache.james.jmap.api.projections.EmailQueryViewManager; import org.apache.james.jmap.memory.projections.MemoryEmailQueryView; import org.apache.james.mailbox.MailboxSession; import org.apache.james.mailbox.MailboxSessionUtil; @@ -82,7 +84,7 @@ public class PopulateEmailQueryViewListenerTest { PopulateEmailQueryViewListener listener; MessageIdManager messageIdManager; SessionProviderImpl sessionProvider; - private MemoryEmailQueryView view; + private EmailQueryViewManager viewManager; private MailboxId inboxId; @BeforeEach @@ -112,8 +114,8 @@ void setup() throws Exception { authenticator.addUser(BOB, "12345"); sessionProvider = new SessionProviderImpl(authenticator, FakeAuthorizator.defaultReject()); - view = new MemoryEmailQueryView(); - listener = new PopulateEmailQueryViewListener(messageIdManager, view, sessionProvider); + viewManager = new DefaultEmailQueryViewManager(new MemoryEmailQueryView()); + listener = new PopulateEmailQueryViewListener(messageIdManager, viewManager, sessionProvider); resources.getEventBus().register(listener); @@ -141,7 +143,7 @@ void appendingAMessageShouldAddItToTheView() throws Exception { .build(emptyMessage(Date.from(ZonedDateTime.parse("2014-10-30T14:12:00Z").toInstant()))), mailboxSession).getId(); - assertThat(view.listMailboxContentSortedBySentAt(inboxId, Limit.limit(12)).collectList().block()) + assertThat(viewManager.getEmailQueryView(mailboxSession.getUser()).listMailboxContentSortedBySentAt(inboxId, Limit.limit(12)).collectList().block()) .containsOnly(composedId.getMessageId()); } @@ -154,13 +156,13 @@ void appendingADeletedMessageShouldNotAddItToTheView() throws Exception { .build(emptyMessage(Date.from(ZonedDateTime.parse("2014-10-30T14:12:00Z").toInstant()))), mailboxSession).getId(); - assertThat(view.listMailboxContentSortedBySentAt(inboxId, Limit.limit(12)).collectList().block()) + assertThat(viewManager.getEmailQueryView(mailboxSession.getUser()).listMailboxContentSortedBySentAt(inboxId, Limit.limit(12)).collectList().block()) .isEmpty(); } @Test void appendingAOutdatedMessageInOutBoxShouldNotAddItToTheView() throws Exception { - MemoryEmailQueryView emailQueryView = new MemoryEmailQueryView(); + EmailQueryViewManager emailQueryView = new DefaultEmailQueryViewManager(new MemoryEmailQueryView()); PopulateEmailQueryViewListener queryViewListener = new PopulateEmailQueryViewListener(messageIdManager, emailQueryView, sessionProvider); MailboxPath outboxPath = MailboxPath.forUser(BOB, "Outbox"); MailboxId outboxId = mailboxManager.createMailbox(outboxPath, mailboxSession).orElseThrow(); @@ -193,7 +195,7 @@ void appendingAOutdatedMessageInOutBoxShouldNotAddItToTheView() throws Exception Mono.from(queryViewListener.reactiveEvent(addedOutDatedEvent)).block(); - assertThat(emailQueryView.listMailboxContentSortedBySentAt(outboxId, Limit.limit(12)).collectList().block()) + assertThat(viewManager.getEmailQueryView(mailboxSession.getUser()).listMailboxContentSortedBySentAt(outboxId, Limit.limit(12)).collectList().block()) .isEmpty(); } @@ -209,7 +211,7 @@ void removingDeletedFlagsShouldAddItToTheView() throws Exception { inboxMessageManager.setFlags(new Flags(), MessageManager.FlagsUpdateMode.REPLACE, MessageRange.all(), mailboxSession); - assertThat(view.listMailboxContentSortedBySentAt(inboxId, Limit.limit(12)).collectList().block()) + assertThat(viewManager.getEmailQueryView(mailboxSession.getUser()).listMailboxContentSortedBySentAt(inboxId, Limit.limit(12)).collectList().block()) .containsOnly(composedId.getMessageId()); } @@ -223,7 +225,7 @@ void addingDeletedFlagsShouldRemoveItToTheView() throws Exception { inboxMessageManager.setFlags(new Flags(DELETED), MessageManager.FlagsUpdateMode.REPLACE, MessageRange.all(), mailboxSession); - assertThat(view.listMailboxContentSortedBySentAt(inboxId, Limit.limit(12)).collectList().block()) + assertThat(viewManager.getEmailQueryView(mailboxSession.getUser()).listMailboxContentSortedBySentAt(inboxId, Limit.limit(12)).collectList().block()) .isEmpty(); } @@ -237,7 +239,7 @@ void deletingMailboxShouldClearTheView() throws Exception { mailboxManager.deleteMailbox(inboxId, mailboxSession); - assertThat(view.listMailboxContentSortedBySentAt(inboxId, Limit.limit(12)).collectList().block()) + assertThat(viewManager.getEmailQueryView(mailboxSession.getUser()).listMailboxContentSortedBySentAt(inboxId, Limit.limit(12)).collectList().block()) .isEmpty(); } @@ -251,7 +253,7 @@ void deletingEmailShouldClearTheView() throws Exception { inboxMessageManager.delete(ImmutableList.of(composedMessageId.getUid()), mailboxSession); - assertThat(view.listMailboxContentSortedBySentAt(inboxId, Limit.limit(12)).collectList().block()) + assertThat(viewManager.getEmailQueryView(mailboxSession.getUser()).listMailboxContentSortedBySentAt(inboxId, Limit.limit(12)).collectList().block()) .isEmpty(); } diff --git a/server/protocols/webadmin-integration-test/distributed-webadmin-integration-test/pom.xml b/server/protocols/webadmin-integration-test/distributed-webadmin-integration-test/pom.xml index e2f36b1d573..5274ec86d52 100644 --- a/server/protocols/webadmin-integration-test/distributed-webadmin-integration-test/pom.xml +++ b/server/protocols/webadmin-integration-test/distributed-webadmin-integration-test/pom.xml @@ -32,6 +32,18 @@ Apache James :: Server :: Web Admin server integration tests :: Distributed + + + + ${james.groupId} + james-server-guice + ${project.version} + pom + import + + + + ${james.groupId} @@ -80,6 +92,13 @@ test-jar test + + ${james.groupId} + james-server-guice-opensearch + ${project.version} + test-jar + test + ${james.groupId} james-server-webadmin-cassandra-data @@ -90,6 +109,12 @@ james-server-webadmin-integration-test-common test + + ${james.groupId} + queue-rabbitmq-guice + test-jar + test + @@ -123,7 +148,7 @@ org.apache.maven.plugins maven-surefire-plugin - + unstable diff --git a/server/protocols/webadmin-integration-test/pom.xml b/server/protocols/webadmin-integration-test/pom.xml index ea9509f5154..f3bd3187991 100644 --- a/server/protocols/webadmin-integration-test/pom.xml +++ b/server/protocols/webadmin-integration-test/pom.xml @@ -35,6 +35,7 @@ distributed-webadmin-integration-test memory-webadmin-integration-test + postgres-webadmin-integration-test webadmin-integration-test-common diff --git a/server/protocols/webadmin-integration-test/postgres-webadmin-integration-test/pom.xml b/server/protocols/webadmin-integration-test/postgres-webadmin-integration-test/pom.xml new file mode 100644 index 00000000000..3bed95cec39 --- /dev/null +++ b/server/protocols/webadmin-integration-test/postgres-webadmin-integration-test/pom.xml @@ -0,0 +1,114 @@ + + + + 4.0.0 + + + org.apache.james + webadmin-integration-test + 3.9.0-SNAPSHOT + ../pom.xml + + + postgres-webadmin-integration-test + jar + + Apache James :: Server :: Web Admin server integration tests :: Postgres App + + + + + ${james.groupId} + james-server-guice + ${project.version} + pom + import + + + + + + + ${james.groupId} + apache-james-backends-postgres + test-jar + test + + + ${james.groupId} + apache-james-mailbox-opensearch + test-jar + test + + + ${james.groupId} + blob-s3 + test-jar + test + + + ${james.groupId} + blob-s3-guice + test-jar + test + + + ${james.groupId} + james-server-guice-opensearch + ${project.version} + test-jar + test + + + ${james.groupId} + james-server-postgres-app + test + + + ${james.groupId} + james-server-postgres-app + test-jar + test + + + ${james.groupId} + james-server-webadmin-integration-test-common + test + + + org.testcontainers + postgresql + test + + + + + + + org.apache.maven.plugins + maven-surefire-plugin + + true + 1800 + + + + + diff --git a/server/protocols/webadmin-integration-test/postgres-webadmin-integration-test/src/test/java/org/apache/james/webadmin/integration/postgres/PostgresAuthorizedEndpointsTest.java b/server/protocols/webadmin-integration-test/postgres-webadmin-integration-test/src/test/java/org/apache/james/webadmin/integration/postgres/PostgresAuthorizedEndpointsTest.java new file mode 100644 index 00000000000..9e0a2d5ebae --- /dev/null +++ b/server/protocols/webadmin-integration-test/postgres-webadmin-integration-test/src/test/java/org/apache/james/webadmin/integration/postgres/PostgresAuthorizedEndpointsTest.java @@ -0,0 +1,49 @@ +/**************************************************************** + * 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.integration.postgres; + +import static org.apache.james.data.UsersRepositoryModuleChooser.Implementation.DEFAULT; + +import org.apache.james.JamesServerBuilder; +import org.apache.james.JamesServerExtension; +import org.apache.james.PostgresJamesConfiguration; +import org.apache.james.PostgresJamesServerMain; +import org.apache.james.SearchConfiguration; +import org.apache.james.backends.postgres.PostgresExtension; +import org.apache.james.webadmin.integration.AuthorizedEndpointsTest; +import org.apache.james.webadmin.integration.UnauthorizedModule; +import org.junit.jupiter.api.extension.RegisterExtension; + +public class PostgresAuthorizedEndpointsTest extends AuthorizedEndpointsTest { + @RegisterExtension + static JamesServerExtension jamesServerExtension = new JamesServerBuilder(tmpDir -> + PostgresJamesConfiguration.builder() + .workingDirectory(tmpDir) + .configurationFromClasspath() + .searchConfiguration(SearchConfiguration.scanning()) + .usersRepository(DEFAULT) + .eventBusImpl(PostgresJamesConfiguration.EventBusImpl.IN_MEMORY) + .build()) + .extension(PostgresExtension.empty()) + .server(configuration -> PostgresJamesServerMain.createServer(configuration) + .overrideWith(new UnauthorizedModule())) + .lifeCycle(JamesServerExtension.Lifecycle.PER_CLASS) + .build(); +} diff --git a/server/protocols/webadmin-integration-test/postgres-webadmin-integration-test/src/test/java/org/apache/james/webadmin/integration/postgres/PostgresFastViewProjectionHealthCheckIntegrationTest.java b/server/protocols/webadmin-integration-test/postgres-webadmin-integration-test/src/test/java/org/apache/james/webadmin/integration/postgres/PostgresFastViewProjectionHealthCheckIntegrationTest.java new file mode 100644 index 00000000000..6061f0665f4 --- /dev/null +++ b/server/protocols/webadmin-integration-test/postgres-webadmin-integration-test/src/test/java/org/apache/james/webadmin/integration/postgres/PostgresFastViewProjectionHealthCheckIntegrationTest.java @@ -0,0 +1,48 @@ +/**************************************************************** + * 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.integration.postgres; + +import static org.apache.james.data.UsersRepositoryModuleChooser.Implementation.DEFAULT; + +import org.apache.james.JamesServerBuilder; +import org.apache.james.JamesServerExtension; +import org.apache.james.PostgresJamesConfiguration; +import org.apache.james.PostgresJamesServerMain; +import org.apache.james.SearchConfiguration; +import org.apache.james.backends.postgres.PostgresExtension; +import org.apache.james.modules.TestJMAPServerModule; +import org.apache.james.webadmin.integration.FastViewProjectionHealthCheckIntegrationContract; +import org.junit.jupiter.api.extension.RegisterExtension; + +public class PostgresFastViewProjectionHealthCheckIntegrationTest extends FastViewProjectionHealthCheckIntegrationContract { + @RegisterExtension + static JamesServerExtension jamesServerExtension = new JamesServerBuilder(tmpDir -> + PostgresJamesConfiguration.builder() + .workingDirectory(tmpDir) + .configurationFromClasspath() + .searchConfiguration(SearchConfiguration.scanning()) + .usersRepository(DEFAULT) + .eventBusImpl(PostgresJamesConfiguration.EventBusImpl.IN_MEMORY) + .build()) + .extension(PostgresExtension.empty()) + .server(configuration -> PostgresJamesServerMain.createServer(configuration) + .overrideWith(new TestJMAPServerModule())) + .build(); +} diff --git a/server/protocols/webadmin-integration-test/postgres-webadmin-integration-test/src/test/java/org/apache/james/webadmin/integration/postgres/PostgresForwardIntegrationTest.java b/server/protocols/webadmin-integration-test/postgres-webadmin-integration-test/src/test/java/org/apache/james/webadmin/integration/postgres/PostgresForwardIntegrationTest.java new file mode 100644 index 00000000000..66f36aee4ab --- /dev/null +++ b/server/protocols/webadmin-integration-test/postgres-webadmin-integration-test/src/test/java/org/apache/james/webadmin/integration/postgres/PostgresForwardIntegrationTest.java @@ -0,0 +1,48 @@ +/**************************************************************** + * 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.integration.postgres; + +import static org.apache.james.data.UsersRepositoryModuleChooser.Implementation.DEFAULT; + +import org.apache.james.JamesServerBuilder; +import org.apache.james.JamesServerExtension; +import org.apache.james.PostgresJamesConfiguration; +import org.apache.james.PostgresJamesServerMain; +import org.apache.james.SearchConfiguration; +import org.apache.james.backends.postgres.PostgresExtension; +import org.apache.james.modules.TestJMAPServerModule; +import org.apache.james.webadmin.integration.ForwardIntegrationTest; +import org.junit.jupiter.api.extension.RegisterExtension; + +public class PostgresForwardIntegrationTest extends ForwardIntegrationTest { + @RegisterExtension + static JamesServerExtension jamesServerExtension = new JamesServerBuilder(tmpDir -> + PostgresJamesConfiguration.builder() + .workingDirectory(tmpDir) + .configurationFromClasspath() + .searchConfiguration(SearchConfiguration.scanning()) + .usersRepository(DEFAULT) + .eventBusImpl(PostgresJamesConfiguration.EventBusImpl.IN_MEMORY) + .build()) + .extension(PostgresExtension.empty()) + .server(configuration -> PostgresJamesServerMain.createServer(configuration) + .overrideWith(new TestJMAPServerModule())) + .build(); +} diff --git a/server/protocols/webadmin-integration-test/postgres-webadmin-integration-test/src/test/java/org/apache/james/webadmin/integration/postgres/PostgresJwtFilterIntegrationTest.java b/server/protocols/webadmin-integration-test/postgres-webadmin-integration-test/src/test/java/org/apache/james/webadmin/integration/postgres/PostgresJwtFilterIntegrationTest.java new file mode 100644 index 00000000000..49aa9ed55d4 --- /dev/null +++ b/server/protocols/webadmin-integration-test/postgres-webadmin-integration-test/src/test/java/org/apache/james/webadmin/integration/postgres/PostgresJwtFilterIntegrationTest.java @@ -0,0 +1,57 @@ +/**************************************************************** + * 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.integration.postgres; + +import static org.apache.james.JamesServerExtension.Lifecycle.PER_CLASS; +import static org.apache.james.data.UsersRepositoryModuleChooser.Implementation.DEFAULT; + +import org.apache.james.JamesServerBuilder; +import org.apache.james.JamesServerExtension; +import org.apache.james.PostgresJamesConfiguration; +import org.apache.james.PostgresJamesServerMain; +import org.apache.james.SearchConfiguration; +import org.apache.james.backends.postgres.PostgresExtension; +import org.apache.james.jwt.JwtTokenVerifier; +import org.apache.james.webadmin.authentication.AuthenticationFilter; +import org.apache.james.webadmin.authentication.JwtFilter; +import org.apache.james.webadmin.integration.JwtFilterIntegrationTest; +import org.junit.jupiter.api.extension.RegisterExtension; + +import com.google.inject.name.Names; + +public class PostgresJwtFilterIntegrationTest extends JwtFilterIntegrationTest { + @RegisterExtension + static JamesServerExtension jamesServerExtension = new JamesServerBuilder(tmpDir -> + PostgresJamesConfiguration.builder() + .workingDirectory(tmpDir) + .configurationFromClasspath() + .searchConfiguration(SearchConfiguration.scanning()) + .usersRepository(DEFAULT) + .eventBusImpl(PostgresJamesConfiguration.EventBusImpl.IN_MEMORY) + .build()) + .extension(PostgresExtension.empty()) + .server(configuration -> PostgresJamesServerMain.createServer(configuration) + .overrideWith(binder -> binder.bind(AuthenticationFilter.class).to(JwtFilter.class)) + .overrideWith(binder -> binder.bind(JwtTokenVerifier.Factory.class) + .annotatedWith(Names.named("webadmin")) + .toInstance(() -> JwtTokenVerifier.create(jwtConfiguration())))) + .lifeCycle(PER_CLASS) + .build(); +} diff --git a/server/protocols/webadmin-integration-test/postgres-webadmin-integration-test/src/test/java/org/apache/james/webadmin/integration/postgres/PostgresPopulateEmailQueryViewTaskIntegrationTest.java b/server/protocols/webadmin-integration-test/postgres-webadmin-integration-test/src/test/java/org/apache/james/webadmin/integration/postgres/PostgresPopulateEmailQueryViewTaskIntegrationTest.java new file mode 100644 index 00000000000..69518da4b3d --- /dev/null +++ b/server/protocols/webadmin-integration-test/postgres-webadmin-integration-test/src/test/java/org/apache/james/webadmin/integration/postgres/PostgresPopulateEmailQueryViewTaskIntegrationTest.java @@ -0,0 +1,155 @@ +/**************************************************************** + * 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.integration.postgres; + +import static io.restassured.RestAssured.given; +import static io.restassured.RestAssured.with; +import static org.apache.james.data.UsersRepositoryModuleChooser.Implementation.DEFAULT; +import static org.hamcrest.Matchers.is; + +import java.nio.charset.StandardCharsets; +import java.time.Duration; +import java.util.stream.IntStream; + +import org.apache.james.GuiceJamesServer; +import org.apache.james.JamesServerBuilder; +import org.apache.james.JamesServerExtension; +import org.apache.james.PostgresJamesConfiguration; +import org.apache.james.PostgresJamesServerMain; +import org.apache.james.SearchConfiguration; +import org.apache.james.backends.postgres.PostgresExtension; +import org.apache.james.core.Username; +import org.apache.james.mailbox.MessageManager; +import org.apache.james.mailbox.model.MailboxPath; +import org.apache.james.mime4j.dom.Message; +import org.apache.james.modules.MailboxProbeImpl; +import org.apache.james.modules.TestJMAPServerModule; +import org.apache.james.probe.DataProbe; +import org.apache.james.utils.DataProbeImpl; +import org.apache.james.utils.WebAdminGuiceProbe; +import org.apache.james.webadmin.WebAdminUtils; +import org.apache.james.webadmin.routes.TasksRoutes; +import org.awaitility.Awaitility; +import org.awaitility.core.ConditionFactory; +import org.junit.jupiter.api.BeforeEach; +import org.junit.jupiter.api.Test; +import org.junit.jupiter.api.extension.RegisterExtension; + +import com.github.fge.lambdas.Throwing; + +import io.restassured.RestAssured; +import reactor.core.publisher.Flux; +import reactor.core.publisher.Mono; + +class PostgresPopulateEmailQueryViewTaskIntegrationTest { + @RegisterExtension + static JamesServerExtension jamesServerExtension = new JamesServerBuilder(tmpDir -> + PostgresJamesConfiguration.builder() + .workingDirectory(tmpDir) + .configurationFromClasspath() + .searchConfiguration(SearchConfiguration.scanning()) + .usersRepository(DEFAULT) + .eventBusImpl(PostgresJamesConfiguration.EventBusImpl.IN_MEMORY) + .build()) + .extension(PostgresExtension.empty()) + .server(configuration -> PostgresJamesServerMain.createServer(configuration) + .overrideWith(new TestJMAPServerModule())) + .build(); + + private static final String DOMAIN = "domain.tld"; + private static final Username BOB = Username.of("bob@" + DOMAIN); + private static final String PASSWORD = "password"; + private static final MailboxPath BOB_INBOX_PATH = MailboxPath.inbox(Username.of(BOB.asString())); + private static final Username ALICE = Username.of("alice@" + DOMAIN); + private static final MailboxPath ALICE_INBOX_PATH = MailboxPath.inbox(Username.of(ALICE.asString())); + private static final Username CEDRIC = Username.of("cedric@" + DOMAIN); + private static final MailboxPath CEDRIC_INBOX_PATH = MailboxPath.inbox(Username.of(CEDRIC.asString())); + + ConditionFactory calmlyAwait = Awaitility.with() + .pollInterval(Duration.ofMillis(200)) + .and().with() + .await(); + + private MailboxProbeImpl mailboxProbe; + + @BeforeEach + void setUp(GuiceJamesServer guiceJamesServer) throws Exception { + DataProbe dataProbe = guiceJamesServer.getProbe(DataProbeImpl.class); + mailboxProbe = guiceJamesServer.getProbe(MailboxProbeImpl.class); + WebAdminGuiceProbe webAdminGuiceProbe = guiceJamesServer.getProbe(WebAdminGuiceProbe.class); + + RestAssured.requestSpecification = WebAdminUtils.buildRequestSpecification(webAdminGuiceProbe.getWebAdminPort()) + .build(); + + dataProbe.addDomain(DOMAIN); + dataProbe.addUser(BOB.asString(), PASSWORD); + dataProbe.addUser(ALICE.asString(), PASSWORD); + dataProbe.addUser(CEDRIC.asString(), PASSWORD); + + // Provision 1000 dummy users. A good users amount is needed to trigger the hanging scenario. + Flux.range(1, 1000) + .flatMap(counter -> Mono.fromRunnable(Throwing.runnable(() -> dataProbe.addUser(counter + "@" + DOMAIN, "password"))), + 128) + .collectList() + .block(); + + mailboxProbe.createMailbox(BOB_INBOX_PATH); + addMessagesToMailbox(BOB, BOB_INBOX_PATH); + + mailboxProbe.createMailbox(ALICE_INBOX_PATH); + addMessagesToMailbox(ALICE, ALICE_INBOX_PATH); + + mailboxProbe.createMailbox(CEDRIC_INBOX_PATH); + addMessagesToMailbox(CEDRIC, CEDRIC_INBOX_PATH); + } + + @Test + void populateEmailQueryViewTaskShouldNotHang() { + String taskId = with() + .post("/mailboxes?task=populateEmailQueryView") + .jsonPath() + .get("taskId"); + + calmlyAwait.atMost(Duration.ofSeconds(30)) + .untilAsserted(() -> + given() + .basePath(TasksRoutes.BASE) + .when() + .get(taskId) + .then() + .body("status", is("completed")) + .body("type", is("PopulateEmailQueryViewTask")) + .body("additionalInformation.processedUserCount", is(1003)) + .body("additionalInformation.failedUserCount", is(0)) + .body("additionalInformation.processedMessageCount", is(30)) + .body("additionalInformation.failedMessageCount", is(0))); + } + + private void addMessagesToMailbox(Username username, MailboxPath mailbox) { + IntStream.rangeClosed(1, 10) + .forEach(Throwing.intConsumer(ignored -> + mailboxProbe.appendMessage(username.asString(), mailbox, + MessageManager.AppendCommand.builder() + .build(Message.Builder.of() + .setSubject("small message") + .setBody("small message for postgres", StandardCharsets.UTF_8) + .build())))); + } +} \ No newline at end of file diff --git a/server/protocols/webadmin-integration-test/postgres-webadmin-integration-test/src/test/java/org/apache/james/webadmin/integration/postgres/PostgresQuotaSearchIntegrationTest.java b/server/protocols/webadmin-integration-test/postgres-webadmin-integration-test/src/test/java/org/apache/james/webadmin/integration/postgres/PostgresQuotaSearchIntegrationTest.java new file mode 100644 index 00000000000..d1b027a2a5b --- /dev/null +++ b/server/protocols/webadmin-integration-test/postgres-webadmin-integration-test/src/test/java/org/apache/james/webadmin/integration/postgres/PostgresQuotaSearchIntegrationTest.java @@ -0,0 +1,51 @@ +/**************************************************************** + * 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.integration.postgres; + +import static org.apache.james.data.UsersRepositoryModuleChooser.Implementation.DEFAULT; + +import org.apache.james.JamesServerBuilder; +import org.apache.james.JamesServerExtension; +import org.apache.james.PostgresJamesConfiguration; +import org.apache.james.PostgresJamesServerMain; +import org.apache.james.SearchConfiguration; +import org.apache.james.backends.postgres.PostgresExtension; +import org.apache.james.modules.TestJMAPServerModule; +import org.apache.james.webadmin.integration.QuotaSearchIntegrationTest; +import org.junit.jupiter.api.extension.RegisterExtension; + +public class PostgresQuotaSearchIntegrationTest extends QuotaSearchIntegrationTest { + @RegisterExtension + static JamesServerExtension jamesServerExtension = new JamesServerBuilder(tmpDir -> + PostgresJamesConfiguration.builder() + .workingDirectory(tmpDir) + .configurationFromClasspath() + .searchConfiguration(SearchConfiguration.scanning()) + .usersRepository(DEFAULT) + .eventBusImpl(PostgresJamesConfiguration.EventBusImpl.IN_MEMORY) + .build()) + .extension(PostgresExtension.empty()) + .server(configuration -> PostgresJamesServerMain.createServer(configuration) + .overrideWith(new TestJMAPServerModule())) + .build(); + + @Override + protected void awaitSearchUpToDate() { + } +} diff --git a/server/protocols/webadmin-integration-test/postgres-webadmin-integration-test/src/test/java/org/apache/james/webadmin/integration/postgres/PostgresUnauthorizedEndpointsTest.java b/server/protocols/webadmin-integration-test/postgres-webadmin-integration-test/src/test/java/org/apache/james/webadmin/integration/postgres/PostgresUnauthorizedEndpointsTest.java new file mode 100644 index 00000000000..526418f22d2 --- /dev/null +++ b/server/protocols/webadmin-integration-test/postgres-webadmin-integration-test/src/test/java/org/apache/james/webadmin/integration/postgres/PostgresUnauthorizedEndpointsTest.java @@ -0,0 +1,51 @@ +/**************************************************************** + * 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.integration.postgres; + +import static org.apache.james.data.UsersRepositoryModuleChooser.Implementation.DEFAULT; + +import org.apache.james.JamesServerBuilder; +import org.apache.james.JamesServerExtension; +import org.apache.james.PostgresJamesConfiguration; +import org.apache.james.PostgresJamesServerMain; +import org.apache.james.SearchConfiguration; +import org.apache.james.backends.postgres.PostgresExtension; +import org.apache.james.vault.VaultConfiguration; +import org.apache.james.webadmin.integration.UnauthorizedEndpointsTest; +import org.apache.james.webadmin.integration.UnauthorizedModule; +import org.junit.jupiter.api.extension.RegisterExtension; + +public class PostgresUnauthorizedEndpointsTest extends UnauthorizedEndpointsTest { + @RegisterExtension + static JamesServerExtension jamesServerExtension = new JamesServerBuilder(tmpDir -> + PostgresJamesConfiguration.builder() + .workingDirectory(tmpDir) + .configurationFromClasspath() + .searchConfiguration(SearchConfiguration.scanning()) + .usersRepository(DEFAULT) + .eventBusImpl(PostgresJamesConfiguration.EventBusImpl.IN_MEMORY) + .deletedMessageVaultConfiguration(VaultConfiguration.ENABLED_DEFAULT) + .build()) + .extension(PostgresExtension.empty()) + .server(configuration -> PostgresJamesServerMain.createServer(configuration) + .overrideWith(new UnauthorizedModule())) + .lifeCycle(JamesServerExtension.Lifecycle.PER_CLASS) + .build(); +} diff --git a/server/protocols/webadmin-integration-test/postgres-webadmin-integration-test/src/test/java/org/apache/james/webadmin/integration/postgres/PostgresWebAdminServerBlobGCIntegrationTest.java b/server/protocols/webadmin-integration-test/postgres-webadmin-integration-test/src/test/java/org/apache/james/webadmin/integration/postgres/PostgresWebAdminServerBlobGCIntegrationTest.java new file mode 100644 index 00000000000..94c7de66f5e --- /dev/null +++ b/server/protocols/webadmin-integration-test/postgres-webadmin-integration-test/src/test/java/org/apache/james/webadmin/integration/postgres/PostgresWebAdminServerBlobGCIntegrationTest.java @@ -0,0 +1,280 @@ +/**************************************************************** + * 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.integration.postgres; + +import static io.restassured.RestAssured.given; +import static io.restassured.RestAssured.with; +import static org.apache.james.data.UsersRepositoryModuleChooser.Implementation.DEFAULT; +import static org.hamcrest.Matchers.is; + +import java.time.Clock; +import java.time.ZonedDateTime; +import java.util.Date; + +import jakarta.mail.Flags; +import jakarta.mail.util.SharedByteArrayInputStream; + +import org.apache.james.GuiceJamesServer; +import org.apache.james.GuiceModuleTestExtension; +import org.apache.james.JamesServerBuilder; +import org.apache.james.JamesServerExtension; +import org.apache.james.PostgresJamesConfiguration; +import org.apache.james.PostgresJamesServerMain; +import org.apache.james.SearchConfiguration; +import org.apache.james.backends.postgres.PostgresExtension; +import org.apache.james.core.Username; +import org.apache.james.mailbox.exception.MailboxException; +import org.apache.james.mailbox.model.MailboxConstants; +import org.apache.james.mailbox.model.MailboxPath; +import org.apache.james.mailbox.probe.MailboxProbe; +import org.apache.james.modules.MailboxProbeImpl; +import org.apache.james.modules.TestJMAPServerModule; +import org.apache.james.modules.blobstore.BlobStoreConfiguration; +import org.apache.james.probe.DataProbe; +import org.apache.james.task.TaskManager; +import org.apache.james.util.ClassLoaderUtils; +import org.apache.james.utils.DataProbeImpl; +import org.apache.james.utils.UpdatableTickingClock; +import org.apache.james.utils.WebAdminGuiceProbe; +import org.apache.james.webadmin.WebAdminUtils; +import org.apache.james.webadmin.routes.TasksRoutes; +import org.junit.jupiter.api.BeforeEach; +import org.junit.jupiter.api.Test; +import org.junit.jupiter.api.extension.ExtensionContext; +import org.junit.jupiter.api.extension.ParameterContext; +import org.junit.jupiter.api.extension.ParameterResolutionException; +import org.junit.jupiter.api.extension.RegisterExtension; + +import com.google.inject.Module; + +import io.restassured.RestAssured; + +public class PostgresWebAdminServerBlobGCIntegrationTest { + private static final ZonedDateTime TIMESTAMP = ZonedDateTime.parse("2015-10-30T16:12:00Z"); + + public static class ClockExtension implements GuiceModuleTestExtension { + private UpdatableTickingClock clock; + + @Override + public void beforeEach(ExtensionContext extensionContext) { + clock = new UpdatableTickingClock(TIMESTAMP.toInstant()); + } + + @Override + public Module getModule() { + return binder -> binder.bind(Clock.class).toInstance(clock); + } + + @Override + public boolean supportsParameter(ParameterContext parameterContext, ExtensionContext extensionContext) throws ParameterResolutionException { + return parameterContext.getParameter().getType() == UpdatableTickingClock.class; + } + + @Override + public Object resolveParameter(ParameterContext parameterContext, ExtensionContext extensionContext) throws ParameterResolutionException { + return clock; + } + } + + @RegisterExtension + static JamesServerExtension jamesServerExtension = new JamesServerBuilder(tmpDir -> + PostgresJamesConfiguration.builder() + .workingDirectory(tmpDir) + .configurationFromClasspath() + .searchConfiguration(SearchConfiguration.scanning()) + .usersRepository(DEFAULT) + .eventBusImpl(PostgresJamesConfiguration.EventBusImpl.IN_MEMORY) + .blobStore(BlobStoreConfiguration.builder() + .postgres() + .disableCache() + .deduplication() + .noCryptoConfig()) + .build()) + .extension(PostgresExtension.empty()) + .extension(new ClockExtension()) + .server(configuration -> PostgresJamesServerMain.createServer(configuration) + .overrideWith(new TestJMAPServerModule())) + .build(); + + private static final String DOMAIN = "domain"; + private static final String USERNAME = "username@" + DOMAIN; + + private DataProbe dataProbe; + private MailboxProbe mailboxProbe; + + @BeforeEach + void setUp(GuiceJamesServer guiceJamesServer, UpdatableTickingClock clock) throws Exception { + clock.setInstant(TIMESTAMP.toInstant()); + + WebAdminGuiceProbe webAdminGuiceProbe = guiceJamesServer.getProbe(WebAdminGuiceProbe.class); + dataProbe = guiceJamesServer.getProbe(DataProbeImpl.class); + mailboxProbe = guiceJamesServer.getProbe(MailboxProbeImpl.class); + + dataProbe.addDomain(DOMAIN); + dataProbe.addUser(USERNAME, "secret"); + mailboxProbe.createMailbox(MailboxConstants.USER_NAMESPACE, USERNAME, MailboxConstants.INBOX); + + RestAssured.requestSpecification = WebAdminUtils.buildRequestSpecification(webAdminGuiceProbe.getWebAdminPort()) + .build(); + RestAssured.enableLoggingOfRequestAndResponseIfValidationFails(); + } + + @Test + void blobGCShouldRemoveUnreferencedAndInactiveBlobId(UpdatableTickingClock clock) throws MailboxException { + SharedByteArrayInputStream mailInputStream = ClassLoaderUtils.getSystemResourceAsSharedStream("eml/emailWithOnlyAttachment.eml"); + mailboxProbe.appendMessage( + USERNAME, + MailboxPath.inbox(Username.of(USERNAME)), + mailInputStream.newStream(0, -1), + new Date(), + false, + new Flags()); + + mailboxProbe.deleteMailbox(MailboxConstants.USER_NAMESPACE, USERNAME, MailboxConstants.INBOX); + clock.setInstant(TIMESTAMP.plusMonths(2).toInstant()); + + String taskId = given() + .queryParam("scope", "unreferenced") + .delete("blobs") + .jsonPath() + .getString("taskId"); + + with() + .basePath(TasksRoutes.BASE) + .when() + .get(taskId + "/await") + .then() + .body("status", is(TaskManager.Status.COMPLETED.getValue())) + .body("taskId", is(taskId)) + .body("type", is("BlobGCTask")) + .body("additionalInformation.referenceSourceCount", is(0)) + .body("additionalInformation.blobCount", is(2)) + .body("additionalInformation.gcedBlobCount", is(2)) + .body("additionalInformation.errorCount", is(0)); + } + + @Test + void blobGCShouldNotRemoveActiveBlobId() throws MailboxException { + SharedByteArrayInputStream mailInputStream = ClassLoaderUtils.getSystemResourceAsSharedStream("eml/emailWithOnlyAttachment.eml"); + mailboxProbe.appendMessage( + USERNAME, + MailboxPath.inbox(Username.of(USERNAME)), + mailInputStream.newStream(0, -1), + new Date(), + false, + new Flags()); + + mailboxProbe.deleteMailbox(MailboxConstants.USER_NAMESPACE, USERNAME, MailboxConstants.INBOX); + + String taskId = given() + .queryParam("scope", "unreferenced") + .delete("blobs") + .jsonPath() + .getString("taskId"); + + with() + .basePath(TasksRoutes.BASE) + .when() + .get(taskId + "/await") + .then() + .body("status", is(TaskManager.Status.COMPLETED.getValue())) + .body("taskId", is(taskId)) + .body("type", is("BlobGCTask")) + .body("additionalInformation.referenceSourceCount", is(0)) + .body("additionalInformation.blobCount", is(2)) + .body("additionalInformation.gcedBlobCount", is(0)) + .body("additionalInformation.errorCount", is(0)); + } + + @Test + void blobGCShouldNotRemoveReferencedBlobId(UpdatableTickingClock clock) throws MailboxException { + SharedByteArrayInputStream mailInputStream = ClassLoaderUtils.getSystemResourceAsSharedStream("eml/emailWithOnlyAttachment.eml"); + mailboxProbe.appendMessage( + USERNAME, + MailboxPath.inbox(Username.of(USERNAME)), + mailInputStream.newStream(0, -1), + new Date(), + false, + new Flags()); + clock.setInstant(TIMESTAMP.plusMonths(2).toInstant()); + + String taskId = given() + .queryParam("scope", "unreferenced") + .delete("blobs") + .jsonPath() + .getString("taskId"); + + with() + .basePath(TasksRoutes.BASE) + .when() + .get(taskId + "/await") + .then() + .body("status", is(TaskManager.Status.COMPLETED.getValue())) + .body("taskId", is(taskId)) + .body("type", is("BlobGCTask")) + .body("additionalInformation.referenceSourceCount", is(2)) + .body("additionalInformation.blobCount", is(2)) + .body("additionalInformation.gcedBlobCount", is(0)) + .body("additionalInformation.errorCount", is(0)); + } + + @Test + void blobGCShouldNotRemoveReferencedBlobIdToAnotherMailbox(UpdatableTickingClock clock) throws Exception { + SharedByteArrayInputStream mailInputStream = ClassLoaderUtils.getSystemResourceAsSharedStream("eml/emailWithOnlyAttachment.eml"); + mailboxProbe.appendMessage( + USERNAME, + MailboxPath.inbox(Username.of(USERNAME)), + mailInputStream.newStream(0, -1), + new Date(), + false, + new Flags()); + + mailboxProbe.createMailbox(MailboxConstants.USER_NAMESPACE, USERNAME, "CustomBox"); + mailboxProbe.appendMessage( + USERNAME, + MailboxPath.forUser(Username.of(USERNAME), "CustomBox"), + mailInputStream.newStream(0, -1), + new Date(), + false, + new Flags()); + + mailboxProbe.deleteMailbox(MailboxConstants.USER_NAMESPACE, USERNAME, MailboxConstants.INBOX); + clock.setInstant(TIMESTAMP.plusMonths(2).toInstant()); + + String taskId = given() + .queryParam("scope", "unreferenced") + .delete("blobs") + .jsonPath() + .getString("taskId"); + + with() + .basePath(TasksRoutes.BASE) + .when() + .get(taskId + "/await") + .then() + .body("status", is(TaskManager.Status.COMPLETED.getValue())) + .body("taskId", is(taskId)) + .body("type", is("BlobGCTask")) + .body("additionalInformation.referenceSourceCount", is(2)) + .body("additionalInformation.blobCount", is(2)) + .body("additionalInformation.gcedBlobCount", is(0)) + .body("additionalInformation.errorCount", is(0)); + } +} diff --git a/server/protocols/webadmin-integration-test/postgres-webadmin-integration-test/src/test/java/org/apache/james/webadmin/integration/postgres/PostgresWebAdminServerIntegrationImmutableTest.java b/server/protocols/webadmin-integration-test/postgres-webadmin-integration-test/src/test/java/org/apache/james/webadmin/integration/postgres/PostgresWebAdminServerIntegrationImmutableTest.java new file mode 100644 index 00000000000..a8afc6b52ae --- /dev/null +++ b/server/protocols/webadmin-integration-test/postgres-webadmin-integration-test/src/test/java/org/apache/james/webadmin/integration/postgres/PostgresWebAdminServerIntegrationImmutableTest.java @@ -0,0 +1,48 @@ +/**************************************************************** + * 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.integration.postgres; + +import static org.apache.james.JamesServerExtension.Lifecycle.PER_CLASS; +import static org.apache.james.data.UsersRepositoryModuleChooser.Implementation.DEFAULT; + +import org.apache.james.JamesServerBuilder; +import org.apache.james.JamesServerExtension; +import org.apache.james.PostgresJamesConfiguration; +import org.apache.james.PostgresJamesServerMain; +import org.apache.james.SearchConfiguration; +import org.apache.james.backends.postgres.PostgresExtension; +import org.apache.james.webadmin.integration.WebAdminServerIntegrationImmutableTest; +import org.junit.jupiter.api.extension.RegisterExtension; + +public class PostgresWebAdminServerIntegrationImmutableTest extends WebAdminServerIntegrationImmutableTest { + @RegisterExtension + static JamesServerExtension jamesServerExtension = new JamesServerBuilder(tmpDir -> + PostgresJamesConfiguration.builder() + .workingDirectory(tmpDir) + .configurationFromClasspath() + .searchConfiguration(SearchConfiguration.scanning()) + .usersRepository(DEFAULT) + .eventBusImpl(PostgresJamesConfiguration.EventBusImpl.IN_MEMORY) + .build()) + .extension(PostgresExtension.empty()) + .server(PostgresJamesServerMain::createServer) + .lifeCycle(PER_CLASS) + .build(); +} diff --git a/server/protocols/webadmin-integration-test/postgres-webadmin-integration-test/src/test/java/org/apache/james/webadmin/integration/postgres/PostgresWebAdminServerIntegrationTest.java b/server/protocols/webadmin-integration-test/postgres-webadmin-integration-test/src/test/java/org/apache/james/webadmin/integration/postgres/PostgresWebAdminServerIntegrationTest.java new file mode 100644 index 00000000000..40ccdfe683b --- /dev/null +++ b/server/protocols/webadmin-integration-test/postgres-webadmin-integration-test/src/test/java/org/apache/james/webadmin/integration/postgres/PostgresWebAdminServerIntegrationTest.java @@ -0,0 +1,71 @@ +/**************************************************************** + * 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.integration.postgres; + +import static io.restassured.RestAssured.given; +import static io.restassured.RestAssured.with; +import static org.apache.james.data.UsersRepositoryModuleChooser.Implementation.DEFAULT; +import static org.hamcrest.Matchers.is; + +import org.apache.james.JamesServerBuilder; +import org.apache.james.JamesServerExtension; +import org.apache.james.PostgresJamesConfiguration; +import org.apache.james.PostgresJamesServerMain; +import org.apache.james.SearchConfiguration; +import org.apache.james.backends.postgres.PostgresExtension; +import org.apache.james.task.TaskManager; +import org.apache.james.webadmin.integration.WebAdminServerIntegrationTest; +import org.apache.james.webadmin.routes.TasksRoutes; +import org.junit.jupiter.api.Test; +import org.junit.jupiter.api.extension.RegisterExtension; + +public class PostgresWebAdminServerIntegrationTest extends WebAdminServerIntegrationTest { + @RegisterExtension + static JamesServerExtension jamesServerExtension = new JamesServerBuilder(tmpDir -> + PostgresJamesConfiguration.builder() + .workingDirectory(tmpDir) + .configurationFromClasspath() + .searchConfiguration(SearchConfiguration.scanning()) + .usersRepository(DEFAULT) + .eventBusImpl(PostgresJamesConfiguration.EventBusImpl.IN_MEMORY) + .build()) + .extension(PostgresExtension.empty()) + .server(PostgresJamesServerMain::createServer) + .build(); + + @Test + void cleanUploadRepositoryShouldComplete() { + String taskId = given() + .queryParam("scope", "expired") + .delete("jmap/uploads") + .jsonPath() + .getString("taskId"); + + with() + .basePath(TasksRoutes.BASE) + .when() + .get(taskId + "/await") + .then() + .body("status", is(TaskManager.Status.COMPLETED.getValue())) + .body("taskId", is(taskId)) + .body("type", is("UploadRepositoryCleanupTask")) + .body("additionalInformation.scope", is("expired")); + } +} diff --git a/server/protocols/webadmin-integration-test/postgres-webadmin-integration-test/src/test/java/org/apache/james/webadmin/integration/vault/PostgresDeletedMessageVaultIntegrationTest.java b/server/protocols/webadmin-integration-test/postgres-webadmin-integration-test/src/test/java/org/apache/james/webadmin/integration/vault/PostgresDeletedMessageVaultIntegrationTest.java new file mode 100644 index 00000000000..e7bcb0daaa2 --- /dev/null +++ b/server/protocols/webadmin-integration-test/postgres-webadmin-integration-test/src/test/java/org/apache/james/webadmin/integration/vault/PostgresDeletedMessageVaultIntegrationTest.java @@ -0,0 +1,130 @@ +/**************************************************************** + * 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.integration.vault; + +import static org.apache.james.data.UsersRepositoryModuleChooser.Implementation.DEFAULT; +import static org.awaitility.Durations.FIVE_HUNDRED_MILLISECONDS; +import static org.awaitility.Durations.ONE_MINUTE; + +import org.apache.james.GuiceJamesServer; +import org.apache.james.JamesServerBuilder; +import org.apache.james.JamesServerExtension; +import org.apache.james.PostgresJamesConfiguration; +import org.apache.james.PostgresJamesServerMain; +import org.apache.james.SearchConfiguration; +import org.apache.james.backends.postgres.PostgresExtension; +import org.apache.james.mailbox.DefaultMailboxes; +import org.apache.james.modules.protocols.ImapGuiceProbe; +import org.apache.james.modules.protocols.SmtpGuiceProbe; +import org.apache.james.utils.DataProbeImpl; +import org.apache.james.utils.SMTPMessageSender; +import org.apache.james.utils.TestIMAPClient; +import org.apache.james.utils.WebAdminGuiceProbe; +import org.apache.james.vault.VaultConfiguration; +import org.apache.james.webadmin.WebAdminUtils; +import org.awaitility.Awaitility; +import org.awaitility.core.ConditionFactory; +import org.junit.jupiter.api.BeforeEach; +import org.junit.jupiter.api.Test; +import org.junit.jupiter.api.extension.RegisterExtension; + +import io.restassured.config.ParamConfig; +import io.restassured.specification.RequestSpecification; + +class PostgresDeletedMessageVaultIntegrationTest { + @RegisterExtension + static JamesServerExtension jamesServerExtension = new JamesServerBuilder(tmpDir -> + PostgresJamesConfiguration.builder() + .workingDirectory(tmpDir) + .configurationFromClasspath() + .searchConfiguration(SearchConfiguration.scanning()) + .usersRepository(DEFAULT) + .eventBusImpl(PostgresJamesConfiguration.EventBusImpl.IN_MEMORY) + .deletedMessageVaultConfiguration(VaultConfiguration.ENABLED_DEFAULT) + .build()) + .server(PostgresJamesServerMain::createServer) + .extension(PostgresExtension.empty()) + .lifeCycle(JamesServerExtension.Lifecycle.PER_CLASS) + .build(); + + private static final ConditionFactory AWAIT = Awaitility.await() + .atMost(ONE_MINUTE) + .with() + .pollInterval(FIVE_HUNDRED_MILLISECONDS); + private static final String DOMAIN = "james.local"; + private static final String USER = "toto@" + DOMAIN; + private static final String PASSWORD = "123456"; + private static final String JAMES_SERVER_HOST = "127.0.0.1"; + + private TestIMAPClient testIMAPClient; + private SMTPMessageSender smtpMessageSender; + private RequestSpecification webAdminApi; + + @BeforeEach + void setUp(GuiceJamesServer jamesServer) throws Exception { + this.testIMAPClient = new TestIMAPClient(); + this.smtpMessageSender = new SMTPMessageSender(DOMAIN); + this.webAdminApi = WebAdminUtils.spec(jamesServer.getProbe(WebAdminGuiceProbe.class).getWebAdminPort()) + .config(WebAdminUtils.defaultConfig() + .paramConfig(new ParamConfig().replaceAllParameters())); + + jamesServer.getProbe(DataProbeImpl.class) + .fluent() + .addDomain(DOMAIN) + .addUser(USER, PASSWORD); + } + + @Test + void restoreDeletedMessageShouldSucceed(GuiceJamesServer jamesServer) throws Exception { + // Create a message + int imapPort = jamesServer.getProbe(ImapGuiceProbe.class).getImapPort(); + smtpMessageSender.connect(JAMES_SERVER_HOST, jamesServer.getProbe(SmtpGuiceProbe.class).getSmtpPort()) + .authenticate(USER, PASSWORD) + .sendMessageWithHeaders(USER, USER, "Subject: thisIsASubject\r\n\r\nBody"); + testIMAPClient.connect(JAMES_SERVER_HOST, imapPort) + .login(USER, PASSWORD) + .select(TestIMAPClient.INBOX) + .awaitMessageCount(AWAIT, 1); + + // Delete the message + testIMAPClient.setFlagsForAllMessagesInMailbox("\\Deleted"); + testIMAPClient.expunge(); + testIMAPClient.awaitNoMessage(AWAIT); + + // Restore the message using the Deleted message vault webadmin endpoint + String restoreBySubjectQuery = "{" + + " \"combinator\": \"and\"," + + " \"limit\": 1," + + " \"criteria\": [" + + " {" + + " \"fieldName\": \"subject\"," + + " \"operator\": \"equals\"," + + " \"value\": \"thisIsASubject\"" + + " }" + + " ]" + + "}"; + DeletedMessagesVaultRequests.restoreMessagesForUserWithQuery(webAdminApi, USER, restoreBySubjectQuery); + + // await the message to be restored + testIMAPClient.select(DefaultMailboxes.RESTORED_MESSAGES) + .awaitMessageCount(AWAIT, 1); + } + +} diff --git a/server/protocols/webadmin-integration-test/postgres-webadmin-integration-test/src/test/resources/dnsservice.xml b/server/protocols/webadmin-integration-test/postgres-webadmin-integration-test/src/test/resources/dnsservice.xml new file mode 100644 index 00000000000..6e4fbd2efb5 --- /dev/null +++ b/server/protocols/webadmin-integration-test/postgres-webadmin-integration-test/src/test/resources/dnsservice.xml @@ -0,0 +1,25 @@ + + + + + true + false + 50000 + diff --git a/server/protocols/webadmin-integration-test/postgres-webadmin-integration-test/src/test/resources/domainlist.xml b/server/protocols/webadmin-integration-test/postgres-webadmin-integration-test/src/test/resources/domainlist.xml new file mode 100644 index 00000000000..fe17431a1ea --- /dev/null +++ b/server/protocols/webadmin-integration-test/postgres-webadmin-integration-test/src/test/resources/domainlist.xml @@ -0,0 +1,24 @@ + + + + + false + false + diff --git a/server/protocols/webadmin-integration-test/postgres-webadmin-integration-test/src/test/resources/eml/emailWithOnlyAttachment.eml b/server/protocols/webadmin-integration-test/postgres-webadmin-integration-test/src/test/resources/eml/emailWithOnlyAttachment.eml new file mode 100644 index 00000000000..452d4cc26d4 --- /dev/null +++ b/server/protocols/webadmin-integration-test/postgres-webadmin-integration-test/src/test/resources/eml/emailWithOnlyAttachment.eml @@ -0,0 +1,16 @@ +Return-Path: +Subject: 29989 btellier +From: +Content-Disposition: attachment +MIME-Version: 1.0 +Date: Sun, 02 Apr 2017 22:09:04 -0000 +Content-Type: application/zip; name="9559333830.zip" +To: +Message-ID: <149117094410.10639.6001033367375624@any.com> +Content-Transfer-Encoding: base64 + +UEsDBBQAAgAIAEQeg0oN2YT/EAsAAMsWAAAIABwAMjIwODUuanNVVAkAAxBy4VgQcuFYdXgLAAEE +AAAAAAQAAAAApZhbi1zHFYWfY/B/MP3i7kwj1/2CokAwBPIQ+sGPkgJ1tURkdeiMbYzQf8+3q8+M +ZmQllgn2aHrqnNq1L2uvtavnj2/b7evz26/Op5M6q/P+8OUX77784g8/lQtLisXTU/68vfzCv/Lg +D9vqs/3b8fNXf92273ey4XTCykk9w9LpfD7tX+zGzU83b8pPg39uBr/Kmxe7w9PLuP3xwpFKTJ32 +AAEEAAAAAAQAAAAAUEsFBgAAAAABAAEATgAAAFILAAAAAA== diff --git a/server/protocols/webadmin-integration-test/postgres-webadmin-integration-test/src/test/resources/imapserver.xml b/server/protocols/webadmin-integration-test/postgres-webadmin-integration-test/src/test/resources/imapserver.xml new file mode 100644 index 00000000000..f7429d1ac37 --- /dev/null +++ b/server/protocols/webadmin-integration-test/postgres-webadmin-integration-test/src/test/resources/imapserver.xml @@ -0,0 +1,41 @@ + + + + + + + + imapserver + 0.0.0.0:0 + 200 + + + file://conf/keystore + james72laBalle + org.bouncycastle.jce.provider.BouncyCastleProvider + + 0 + 0 + false + false + + diff --git a/server/protocols/webadmin-integration-test/postgres-webadmin-integration-test/src/test/resources/jwt_publickey b/server/protocols/webadmin-integration-test/postgres-webadmin-integration-test/src/test/resources/jwt_publickey new file mode 100644 index 00000000000..53914e0533a --- /dev/null +++ b/server/protocols/webadmin-integration-test/postgres-webadmin-integration-test/src/test/resources/jwt_publickey @@ -0,0 +1,9 @@ +-----BEGIN PUBLIC KEY----- +MIIBIjANBgkqhkiG9w0BAQEFAAOCAQ8AMIIBCgKCAQEAtlChO/nlVP27MpdkG0Bh +16XrMRf6M4NeyGa7j5+1UKm42IKUf3lM28oe82MqIIRyvskPc11NuzSor8HmvH8H +lhDs5DyJtx2qp35AT0zCqfwlaDnlDc/QDlZv1CoRZGpQk1Inyh6SbZwYpxxwh0fi ++d/4RpE3LBVo8wgOaXPylOlHxsDizfkL8QwXItyakBfMO6jWQRrj7/9WDhGf4Hi+ +GQur1tPGZDl9mvCoRHjFrD5M/yypIPlfMGWFVEvV5jClNMLAQ9bYFuOc7H1fEWw6 +U1LZUUbJW9/CH45YXz82CYqkrfbnQxqRb2iVbVjs/sHopHd1NTiCfUtwvcYJiBVj +kwIDAQAB +-----END PUBLIC KEY----- diff --git a/server/protocols/webadmin-integration-test/postgres-webadmin-integration-test/src/test/resources/keystore b/server/protocols/webadmin-integration-test/postgres-webadmin-integration-test/src/test/resources/keystore new file mode 100644 index 00000000000..536a6c792b0 Binary files /dev/null and b/server/protocols/webadmin-integration-test/postgres-webadmin-integration-test/src/test/resources/keystore differ diff --git a/server/protocols/webadmin-integration-test/postgres-webadmin-integration-test/src/test/resources/listeners.xml b/server/protocols/webadmin-integration-test/postgres-webadmin-integration-test/src/test/resources/listeners.xml new file mode 100644 index 00000000000..ff2e5172324 --- /dev/null +++ b/server/protocols/webadmin-integration-test/postgres-webadmin-integration-test/src/test/resources/listeners.xml @@ -0,0 +1,49 @@ + + + + + + org.apache.james.mailbox.cassandra.MailboxOperationLoggingListener + + + org.apache.james.mailbox.quota.mailing.listeners.QuotaThresholdCrossingListener + QuotaThresholdCrossingListener-lower-threshold + + + + 0.1 + + + first + + + + org.apache.james.mailbox.quota.mailing.listeners.QuotaThresholdCrossingListener + QuotaThresholdCrossingListener-upper-threshold + + + + 0.2 + + + second + + + \ No newline at end of file diff --git a/server/protocols/webadmin-integration-test/postgres-webadmin-integration-test/src/test/resources/lmtpserver.xml b/server/protocols/webadmin-integration-test/postgres-webadmin-integration-test/src/test/resources/lmtpserver.xml new file mode 100644 index 00000000000..f838adb5f01 --- /dev/null +++ b/server/protocols/webadmin-integration-test/postgres-webadmin-integration-test/src/test/resources/lmtpserver.xml @@ -0,0 +1,23 @@ + + + + + + diff --git a/server/protocols/webadmin-integration-test/postgres-webadmin-integration-test/src/test/resources/mailetcontainer.xml b/server/protocols/webadmin-integration-test/postgres-webadmin-integration-test/src/test/resources/mailetcontainer.xml new file mode 100644 index 00000000000..5b3de6b3255 --- /dev/null +++ b/server/protocols/webadmin-integration-test/postgres-webadmin-integration-test/src/test/resources/mailetcontainer.xml @@ -0,0 +1,128 @@ + + + + + + + + postmaster + + + + 20 + postgres://var/mail/error/ + + + + + + + + transport + + + + + + ignore + + + postgres://var/mail/error/ + ignore + + + + + + postgres://var/mail/rrt-error/ + + + + + + + + + + + + + bcc + + + + ignore + + + ignore + + + ignore + + + + local-address-error + 550 - Requested action not taken: no such user here + + + + outgoing + 5000, 100000, 500000 + 3 + 0 + 10 + true + bounces + + + relay-denied + + + + + + none + + + postgres://var/mail/address-error/ + + + + + + none + + + postgres://var/mail/relay-denied/ + Warning: You are sending an e-mail to a remote server. You must be authentified to perform such an operation + + + + + + false + + + + + + + + diff --git a/server/protocols/webadmin-integration-test/postgres-webadmin-integration-test/src/test/resources/mailrepositorystore.xml b/server/protocols/webadmin-integration-test/postgres-webadmin-integration-test/src/test/resources/mailrepositorystore.xml new file mode 100644 index 00000000000..689745af60f --- /dev/null +++ b/server/protocols/webadmin-integration-test/postgres-webadmin-integration-test/src/test/resources/mailrepositorystore.xml @@ -0,0 +1,31 @@ + + + + + + + + + postgres + + + + + diff --git a/server/protocols/webadmin-integration-test/postgres-webadmin-integration-test/src/test/resources/managesieveserver.xml b/server/protocols/webadmin-integration-test/postgres-webadmin-integration-test/src/test/resources/managesieveserver.xml new file mode 100644 index 00000000000..f136a432b8a --- /dev/null +++ b/server/protocols/webadmin-integration-test/postgres-webadmin-integration-test/src/test/resources/managesieveserver.xml @@ -0,0 +1,32 @@ + + + + + + + + + + + + diff --git a/server/protocols/webadmin-integration-test/postgres-webadmin-integration-test/src/test/resources/pop3server.xml b/server/protocols/webadmin-integration-test/postgres-webadmin-integration-test/src/test/resources/pop3server.xml new file mode 100644 index 00000000000..bec385ae306 --- /dev/null +++ b/server/protocols/webadmin-integration-test/postgres-webadmin-integration-test/src/test/resources/pop3server.xml @@ -0,0 +1,23 @@ + + + + + + diff --git a/server/protocols/webadmin-integration-test/postgres-webadmin-integration-test/src/test/resources/smtpserver.xml b/server/protocols/webadmin-integration-test/postgres-webadmin-integration-test/src/test/resources/smtpserver.xml new file mode 100644 index 00000000000..2fd612d961b --- /dev/null +++ b/server/protocols/webadmin-integration-test/postgres-webadmin-integration-test/src/test/resources/smtpserver.xml @@ -0,0 +1,54 @@ + + + + + + + smtpserver-global + 0.0.0.0:0 + 200 + + file://conf/keystore + james72laBalle + org.bouncycastle.jce.provider.BouncyCastleProvider + SunX509 + + 360 + 0 + 0 + + never + false + true + + 0.0.0.0/0 + false + 0 + true + Apache JAMES awesome SMTP Server + + + + + false + + + + diff --git a/server/protocols/webadmin-integration-test/postgres-webadmin-integration-test/src/test/resources/webadmin.properties b/server/protocols/webadmin-integration-test/postgres-webadmin-integration-test/src/test/resources/webadmin.properties new file mode 100644 index 00000000000..78a176aabda --- /dev/null +++ b/server/protocols/webadmin-integration-test/postgres-webadmin-integration-test/src/test/resources/webadmin.properties @@ -0,0 +1,27 @@ +# 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. + +# This template file can be used as example for James Server configuration +# DO NOT USE IT AS SUCH AND ADAPT IT TO YOUR NEEDS + +# Read https://james.apache.org/server/config-webadmin.html for further details + +enabled=true +port=0 +host=127.0.0.1 + +extensions.routes=org.apache.james.webadmin.dropwizard.MetricsRoutes \ No newline at end of file diff --git a/server/protocols/webadmin/webadmin-jmap/src/main/java/org/apache/james/webadmin/data/jmap/JmapUploadRoutes.java b/server/protocols/webadmin/webadmin-jmap/src/main/java/org/apache/james/webadmin/data/jmap/JmapUploadRoutes.java index 950c850c35c..49730d138f4 100644 --- a/server/protocols/webadmin/webadmin-jmap/src/main/java/org/apache/james/webadmin/data/jmap/JmapUploadRoutes.java +++ b/server/protocols/webadmin/webadmin-jmap/src/main/java/org/apache/james/webadmin/data/jmap/JmapUploadRoutes.java @@ -23,7 +23,7 @@ import jakarta.inject.Inject; -import org.apache.james.jmap.cassandra.upload.CassandraUploadRepository; +import org.apache.james.jmap.api.upload.UploadRepository; import org.apache.james.task.Task; import org.apache.james.task.TaskManager; import org.apache.james.webadmin.Routes; @@ -39,12 +39,12 @@ public class JmapUploadRoutes implements Routes { public static final String BASE_PATH = "/jmap/uploads"; - private final CassandraUploadRepository uploadRepository; + private final UploadRepository uploadRepository; private final TaskManager taskManager; private final JsonTransformer jsonTransformer; @Inject - public JmapUploadRoutes(CassandraUploadRepository uploadRepository, TaskManager taskManager, JsonTransformer jsonTransformer) { + public JmapUploadRoutes(UploadRepository uploadRepository, TaskManager taskManager, JsonTransformer jsonTransformer) { this.uploadRepository = uploadRepository; this.taskManager = taskManager; this.jsonTransformer = jsonTransformer; diff --git a/server/protocols/webadmin/webadmin-jmap/src/main/java/org/apache/james/webadmin/data/jmap/UploadCleanupTaskDTO.java b/server/protocols/webadmin/webadmin-jmap/src/main/java/org/apache/james/webadmin/data/jmap/UploadCleanupTaskDTO.java index 8a3aa2b8720..6ffaffee7f7 100644 --- a/server/protocols/webadmin/webadmin-jmap/src/main/java/org/apache/james/webadmin/data/jmap/UploadCleanupTaskDTO.java +++ b/server/protocols/webadmin/webadmin-jmap/src/main/java/org/apache/james/webadmin/data/jmap/UploadCleanupTaskDTO.java @@ -21,7 +21,7 @@ import java.util.Locale; -import org.apache.james.jmap.cassandra.upload.CassandraUploadRepository; +import org.apache.james.jmap.api.upload.UploadRepository; import org.apache.james.json.DTOModule; import org.apache.james.server.task.json.dto.TaskDTO; import org.apache.james.server.task.json.dto.TaskDTOModule; @@ -48,7 +48,7 @@ public String getScope() { return scope; } - public static TaskDTOModule module(CassandraUploadRepository uploadRepository) { + public static TaskDTOModule module(UploadRepository uploadRepository) { return DTOModule .forDomainObject(UploadRepositoryCleanupTask.class) .convertToDTO(UploadCleanupTaskDTO.class) diff --git a/server/protocols/webadmin/webadmin-jmap/src/main/java/org/apache/james/webadmin/data/jmap/UploadRepositoryCleanupTask.java b/server/protocols/webadmin/webadmin-jmap/src/main/java/org/apache/james/webadmin/data/jmap/UploadRepositoryCleanupTask.java index aeaee5ee1df..419c9cf707d 100644 --- a/server/protocols/webadmin/webadmin-jmap/src/main/java/org/apache/james/webadmin/data/jmap/UploadRepositoryCleanupTask.java +++ b/server/protocols/webadmin/webadmin-jmap/src/main/java/org/apache/james/webadmin/data/jmap/UploadRepositoryCleanupTask.java @@ -22,11 +22,12 @@ import static org.apache.james.webadmin.data.jmap.UploadRepositoryCleanupTask.CleanupScope.EXPIRED; import java.time.Clock; +import java.time.Duration; import java.time.Instant; import java.util.Arrays; import java.util.Optional; -import org.apache.james.jmap.cassandra.upload.CassandraUploadRepository; +import org.apache.james.jmap.api.upload.UploadRepository; import org.apache.james.task.Task; import org.apache.james.task.TaskExecutionDetails; import org.apache.james.task.TaskType; @@ -40,6 +41,7 @@ public class UploadRepositoryCleanupTask implements Task { private static final Logger LOGGER = LoggerFactory.getLogger(UploadRepositoryCleanupTask.class); public static final TaskType TASK_TYPE = TaskType.of("UploadRepositoryCleanupTask"); + public static final Duration EXPIRE_DURATION = Duration.ofDays(7); enum CleanupScope { EXPIRED; @@ -79,10 +81,10 @@ public CleanupScope getScope() { } } - private final CassandraUploadRepository uploadRepository; + private final UploadRepository uploadRepository; private final CleanupScope scope; - public UploadRepositoryCleanupTask(CassandraUploadRepository uploadRepository, CleanupScope scope) { + public UploadRepositoryCleanupTask(UploadRepository uploadRepository, CleanupScope scope) { this.uploadRepository = uploadRepository; this.scope = scope; } @@ -90,7 +92,7 @@ public UploadRepositoryCleanupTask(CassandraUploadRepository uploadRepository, C @Override public Result run() { if (EXPIRED.equals(scope)) { - return uploadRepository.purge() + return Mono.from(uploadRepository.deleteByUploadDateBefore(EXPIRE_DURATION)) .thenReturn(Result.COMPLETED) .onErrorResume(error -> { LOGGER.error("Error when cleaning upload repository", error); diff --git a/server/task/task-memory/src/test/java/org/apache/james/task/eventsourcing/TaskExecutionDetailsProjectionContract.java b/server/task/task-memory/src/test/java/org/apache/james/task/eventsourcing/TaskExecutionDetailsProjectionContract.java index 131812e497e..0c89c93aec2 100644 --- a/server/task/task-memory/src/test/java/org/apache/james/task/eventsourcing/TaskExecutionDetailsProjectionContract.java +++ b/server/task/task-memory/src/test/java/org/apache/james/task/eventsourcing/TaskExecutionDetailsProjectionContract.java @@ -45,7 +45,14 @@ default void loadShouldBeAbleToRetrieveASavedRecord() { testee.update(TASK_EXECUTION_DETAILS()); Optional taskExecutionDetails = OptionConverters.toJava(testee.load(TASK_ID())); - assertThat(taskExecutionDetails).contains(TASK_EXECUTION_DETAILS()); + + assertThat(taskExecutionDetails.get()) + .usingRecursiveComparison() + .ignoringFields("submittedDate") + .isEqualTo(TASK_EXECUTION_DETAILS()); + + assertThat(taskExecutionDetails.get().getSubmittedDate().isEqual(TASK_EXECUTION_DETAILS().getSubmittedDate())) + .isTrue(); } @Test @@ -54,7 +61,14 @@ default void readDetailsShouldBeAbleToRetrieveASavedRecordWithAdditionalInformat testee.update(TASK_EXECUTION_DETAILS_WITH_ADDITIONAL_INFORMATION()); Optional taskExecutionDetails = OptionConverters.toJava(testee.load(TASK_ID())); - assertThat(taskExecutionDetails).contains(TASK_EXECUTION_DETAILS_WITH_ADDITIONAL_INFORMATION()); + + assertThat(taskExecutionDetails.get()) + .usingRecursiveComparison() + .ignoringFields("submittedDate") + .isEqualTo(TASK_EXECUTION_DETAILS_WITH_ADDITIONAL_INFORMATION()); + + assertThat(taskExecutionDetails.get().getSubmittedDate().isEqual(TASK_EXECUTION_DETAILS_WITH_ADDITIONAL_INFORMATION().getSubmittedDate())) + .isTrue(); } @Test @@ -65,7 +79,14 @@ default void updateShouldUpdateRecords() { testee.update(TASK_EXECUTION_DETAILS_UPDATED()); Optional taskExecutionDetails = OptionConverters.toJava(testee.load(TASK_ID())); - assertThat(taskExecutionDetails).contains(TASK_EXECUTION_DETAILS_UPDATED()); + + assertThat(taskExecutionDetails.get()) + .usingRecursiveComparison() + .ignoringFields("submittedDate") + .isEqualTo(TASK_EXECUTION_DETAILS_UPDATED()); + + assertThat(taskExecutionDetails.get().getSubmittedDate().isEqual(TASK_EXECUTION_DETAILS_UPDATED().getSubmittedDate())) + .isTrue(); } @Test @@ -89,7 +110,10 @@ default void listShouldReturnAllRecords() { testee.update(TASK_EXECUTION_DETAILS_2()); List taskExecutionDetails = asJava(testee.list()); - assertThat(taskExecutionDetails).containsOnly(TASK_EXECUTION_DETAILS(), TASK_EXECUTION_DETAILS_2()); + + assertThat(taskExecutionDetails) + .usingRecursiveFieldByFieldElementComparatorIgnoringFields("submittedDate") + .containsOnly(TASK_EXECUTION_DETAILS(), TASK_EXECUTION_DETAILS_2()); } @Test @@ -99,6 +123,8 @@ default void listDetailsShouldReturnLastUpdatedRecords() { testee.update(TASK_EXECUTION_DETAILS_UPDATED()); List taskExecutionDetails = asJava(testee.list()); - assertThat(taskExecutionDetails).containsOnly(TASK_EXECUTION_DETAILS_UPDATED()); + assertThat(taskExecutionDetails) + .usingRecursiveFieldByFieldElementComparatorIgnoringFields("submittedDate") + .containsOnly(TASK_EXECUTION_DETAILS_UPDATED()); } } diff --git a/server/task/task-postgres/pom.xml b/server/task/task-postgres/pom.xml new file mode 100644 index 00000000000..35160283f7d --- /dev/null +++ b/server/task/task-postgres/pom.xml @@ -0,0 +1,134 @@ + + + 4.0.0 + + org.apache.james + james-server + 3.9.0-SNAPSHOT + ../../pom.xml + + + james-server-task-postgres + Apache James :: Server :: Task :: PostgreSQL + Distributed task manager leveraging PostgreSQL + + + + ${james.groupId} + apache-james-backends-postgres + + + ${james.groupId} + apache-james-backends-postgres + test-jar + test + + + ${james.groupId} + james-json + + + ${james.groupId} + james-json + test-jar + test + + + ${james.groupId} + james-server-guice-common + test-jar + test + + + ${james.groupId} + james-server-lifecycle-api + + + ${james.groupId} + james-server-task-api + test-jar + test + + + ${james.groupId} + james-server-task-json + + + ${james.groupId} + james-server-task-json + test-jar + test + + + ${james.groupId} + james-server-task-memory + + + ${james.groupId} + james-server-task-memory + test-jar + test + + + ${james.groupId} + james-server-testing + test + + + ${james.groupId} + metrics-tests + test + + + ${james.groupId} + testing-base + test + + + com.fasterxml.jackson.core + jackson-databind + + + commons-codec + commons-codec + test + + + net.javacrumbs.json-unit + json-unit-assertj + test + + + org.awaitility + awaitility + test + + + org.mockito + mockito-core + test + + + org.scala-lang + scala-library + + + org.scala-lang.modules + scala-java8-compat_${scala.base} + + + org.testcontainers + postgresql + test + + + + + + + net.alchim31.maven + scala-maven-plugin + + + + diff --git a/server/task/task-postgres/src/main/scala/org/apache/james/task/eventsourcing/postgres/PostgresTaskExecutionDetailsProjection.scala b/server/task/task-postgres/src/main/scala/org/apache/james/task/eventsourcing/postgres/PostgresTaskExecutionDetailsProjection.scala new file mode 100644 index 00000000000..57271eb7d8e --- /dev/null +++ b/server/task/task-postgres/src/main/scala/org/apache/james/task/eventsourcing/postgres/PostgresTaskExecutionDetailsProjection.scala @@ -0,0 +1,54 @@ + /*************************************************************** + * 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.task.eventsourcing.postgres + +import java.time.Instant + +import jakarta.inject.Inject +import org.apache.james.task.eventsourcing.TaskExecutionDetailsProjection +import org.apache.james.task.{TaskExecutionDetails, TaskId} +import org.reactivestreams.Publisher + +import scala.compat.java8.OptionConverters._ +import scala.jdk.CollectionConverters._ + +class PostgresTaskExecutionDetailsProjection @Inject()(taskExecutionDetailsProjectionDAO: PostgresTaskExecutionDetailsProjectionDAO) + extends TaskExecutionDetailsProjection { + + override def load(taskId: TaskId): Option[TaskExecutionDetails] = + taskExecutionDetailsProjectionDAO.readDetails(taskId).blockOptional().asScala + + override def list: List[TaskExecutionDetails] = + taskExecutionDetailsProjectionDAO.listDetails().collectList().block().asScala.toList + + override def update(details: TaskExecutionDetails): Unit = + taskExecutionDetailsProjectionDAO.saveDetails(details).block() + + override def loadReactive(taskId: TaskId): Publisher[TaskExecutionDetails] = + taskExecutionDetailsProjectionDAO.readDetails(taskId) + + override def listReactive(): Publisher[TaskExecutionDetails] = taskExecutionDetailsProjectionDAO.listDetails() + + override def updateReactive(details: TaskExecutionDetails): Publisher[Void] = taskExecutionDetailsProjectionDAO.saveDetails(details) + + override def listDetailsByBeforeDate(beforeDate: Instant): Publisher[TaskExecutionDetails] = taskExecutionDetailsProjectionDAO.listDetailsByBeforeDate(beforeDate) + + override def remove(taskExecutionDetails: TaskExecutionDetails): Publisher[Void] = taskExecutionDetailsProjectionDAO.remove(taskExecutionDetails) +} diff --git a/server/task/task-postgres/src/main/scala/org/apache/james/task/eventsourcing/postgres/PostgresTaskExecutionDetailsProjectionDAO.scala b/server/task/task-postgres/src/main/scala/org/apache/james/task/eventsourcing/postgres/PostgresTaskExecutionDetailsProjectionDAO.scala new file mode 100644 index 00000000000..a938485a721 --- /dev/null +++ b/server/task/task-postgres/src/main/scala/org/apache/james/task/eventsourcing/postgres/PostgresTaskExecutionDetailsProjectionDAO.scala @@ -0,0 +1,112 @@ +/**************************************************************** + * 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.task.eventsourcing.postgres + +import java.time.{Instant, LocalDateTime} +import java.util.Optional + +import com.google.common.collect.ImmutableMap +import jakarta.inject.Inject +import org.apache.james.backends.postgres.PostgresCommons.{LOCAL_DATE_TIME_ZONED_DATE_TIME_FUNCTION, ZONED_DATE_TIME_TO_LOCAL_DATE_TIME, INSTANT_TO_LOCAL_DATE_TIME} +import org.apache.james.backends.postgres.utils.PostgresExecutor +import org.apache.james.server.task.json.JsonTaskAdditionalInformationSerializer +import org.apache.james.task._ +import org.apache.james.task.eventsourcing.postgres.PostgresTaskExecutionDetailsProjectionModule._ +import org.apache.james.util.ReactorUtils +import org.jooq.JSONB.jsonb +import org.jooq.{InsertQuery, Record} +import reactor.core.publisher.{Flux, Mono} + +class PostgresTaskExecutionDetailsProjectionDAO @Inject()(postgresExecutor: PostgresExecutor, jsonTaskAdditionalInformationSerializer: JsonTaskAdditionalInformationSerializer) { + + def saveDetails(details: TaskExecutionDetails): Mono[Void] = + Mono.from(serializeAdditionalInformation(details) + .flatMap(serializedAdditionalInformation => postgresExecutor.executeVoid(dsl => { + val insertValues: ImmutableMap[Any, Any] = toInsertValues(details, serializedAdditionalInformation) + + val insertStatement: InsertQuery[Record] = dsl.insertQuery(TABLE_NAME) + insertStatement.addValue(TASK_ID, details.getTaskId.getValue) + insertStatement.addValues(insertValues) + insertStatement.onConflict(TASK_ID) + insertStatement.onDuplicateKeyUpdate(true) + insertStatement.addValuesForUpdate(insertValues) + + Mono.from(insertStatement) + }))) + + private def toInsertValues(details: TaskExecutionDetails, serializedAdditionalInformation: Optional[String]): ImmutableMap[Any, Any] = { + val builder: ImmutableMap.Builder[Any, Any] = ImmutableMap.builder() + builder.put(TYPE, details.getType.asString()) + builder.put(STATUS, details.getStatus.getValue) + builder.put(SUBMITTED_DATE, ZONED_DATE_TIME_TO_LOCAL_DATE_TIME.apply(details.getSubmittedDate)) + builder.put(SUBMITTED_NODE, details.getSubmittedNode.asString) + details.getStartedDate.ifPresent(startedDate => builder.put(STARTED_DATE, ZONED_DATE_TIME_TO_LOCAL_DATE_TIME.apply(startedDate))) + details.getRanNode.ifPresent(hostname => builder.put(RAN_NODE, hostname.asString)) + details.getCompletedDate.ifPresent(completedDate => builder.put(COMPLETED_DATE, ZONED_DATE_TIME_TO_LOCAL_DATE_TIME.apply(completedDate))) + details.getCanceledDate.ifPresent(canceledDate => builder.put(CANCELED_DATE, ZONED_DATE_TIME_TO_LOCAL_DATE_TIME.apply(canceledDate))) + details.getCancelRequestedNode.ifPresent(hostname => builder.put(CANCEL_REQUESTED_NODE, hostname.asString)) + details.getFailedDate.ifPresent(failedDate => builder.put(FAILED_DATE, ZONED_DATE_TIME_TO_LOCAL_DATE_TIME.apply(failedDate))) + serializedAdditionalInformation.ifPresent(info => builder.put(ADDITIONAL_INFORMATION, jsonb(info))) + builder.build() + } + + private def serializeAdditionalInformation(details: TaskExecutionDetails): Mono[Optional[String]] = Mono.fromCallable(() => details + .getAdditionalInformation + .map(jsonTaskAdditionalInformationSerializer.serialize(_))) + .cast(classOf[Optional[String]]) + .subscribeOn(ReactorUtils.BLOCKING_CALL_WRAPPER) + + def readDetails(taskId: TaskId): Mono[TaskExecutionDetails] = + postgresExecutor.executeRow(dsl => Mono.from(dsl.selectFrom(TABLE_NAME) + .where(TASK_ID.eq(taskId.getValue)))) + .map(toTaskExecutionDetails) + + def listDetails(): Flux[TaskExecutionDetails] = + postgresExecutor.executeRows(dsl => Flux.from(dsl.selectFrom(TABLE_NAME))) + .map(toTaskExecutionDetails) + + def listDetailsByBeforeDate(beforeDate: Instant): Flux[TaskExecutionDetails] = + postgresExecutor.executeRows(dsl => Flux.from(dsl.selectFrom(TABLE_NAME) + .where(SUBMITTED_DATE.lt(INSTANT_TO_LOCAL_DATE_TIME.apply(beforeDate))))) + .map(toTaskExecutionDetails) + + def remove(details: TaskExecutionDetails): Mono[Void] = + postgresExecutor.executeVoid(dsl => Mono.from(dsl.deleteFrom(TABLE_NAME) + .where(TASK_ID.eq(details.getTaskId.getValue)))) + + private def toTaskExecutionDetails(record: Record): TaskExecutionDetails = + new TaskExecutionDetails( + taskId = TaskId.fromUUID(record.get(TASK_ID)), + `type` = TaskType.of(record.get(TYPE)), + status = TaskManager.Status.fromString(record.get(STATUS)), + submittedDate = LOCAL_DATE_TIME_ZONED_DATE_TIME_FUNCTION.apply(record.get(SUBMITTED_DATE, classOf[LocalDateTime])), + submittedNode = Hostname(record.get(SUBMITTED_NODE)), + startedDate = Optional.ofNullable(LOCAL_DATE_TIME_ZONED_DATE_TIME_FUNCTION.apply(record.get(STARTED_DATE, classOf[LocalDateTime]))), + ranNode = Optional.ofNullable(record.get(RAN_NODE)).map(Hostname(_)), + completedDate = Optional.ofNullable(LOCAL_DATE_TIME_ZONED_DATE_TIME_FUNCTION.apply(record.get(COMPLETED_DATE, classOf[LocalDateTime]))), + canceledDate = Optional.ofNullable(LOCAL_DATE_TIME_ZONED_DATE_TIME_FUNCTION.apply(record.get(CANCELED_DATE, classOf[LocalDateTime]))), + cancelRequestedNode = Optional.ofNullable(record.get(CANCEL_REQUESTED_NODE)).map(Hostname(_)), + failedDate = Optional.ofNullable(LOCAL_DATE_TIME_ZONED_DATE_TIME_FUNCTION.apply(record.get(FAILED_DATE, classOf[LocalDateTime]))), + additionalInformation = () => deserializeAdditionalInformation(record)) + + private def deserializeAdditionalInformation(record: Record): Optional[TaskExecutionDetails.AdditionalInformation] = + Optional.ofNullable(record.get(ADDITIONAL_INFORMATION)) + .map(additionalInformation => jsonTaskAdditionalInformationSerializer.deserialize(additionalInformation.data())) +} diff --git a/server/task/task-postgres/src/main/scala/org/apache/james/task/eventsourcing/postgres/PostgresTaskExecutionDetailsProjectionModule.scala b/server/task/task-postgres/src/main/scala/org/apache/james/task/eventsourcing/postgres/PostgresTaskExecutionDetailsProjectionModule.scala new file mode 100644 index 00000000000..21918fd8042 --- /dev/null +++ b/server/task/task-postgres/src/main/scala/org/apache/james/task/eventsourcing/postgres/PostgresTaskExecutionDetailsProjectionModule.scala @@ -0,0 +1,72 @@ +/**************************************************************** + * 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.task.eventsourcing.postgres + +import java.time.LocalDateTime +import java.util.UUID + +import org.apache.james.backends.postgres.{PostgresCommons, PostgresIndex, PostgresModule, PostgresTable} +import org.jooq.impl.{DSL, SQLDataType} +import org.jooq.{Field, JSONB, Record, Table} + +object PostgresTaskExecutionDetailsProjectionModule { + val TABLE_NAME: Table[Record] = DSL.table("task_execution_details_projection") + + val TASK_ID: Field[UUID] = DSL.field("task_id", SQLDataType.UUID.notNull) + val ADDITIONAL_INFORMATION: Field[JSONB] = DSL.field("additional_information", SQLDataType.JSONB) + val TYPE: Field[String] = DSL.field("type", SQLDataType.VARCHAR) + val STATUS: Field[String] = DSL.field("status", SQLDataType.VARCHAR) + val SUBMITTED_DATE: Field[LocalDateTime] = DSL.field("submitted_date", PostgresCommons.DataTypes.TIMESTAMP) + val SUBMITTED_NODE: Field[String] = DSL.field("submitted_node", SQLDataType.VARCHAR) + val STARTED_DATE: Field[LocalDateTime] = DSL.field("started_date", PostgresCommons.DataTypes.TIMESTAMP) + val RAN_NODE: Field[String] = DSL.field("ran_node", SQLDataType.VARCHAR) + val COMPLETED_DATE: Field[LocalDateTime] = DSL.field("completed_date", PostgresCommons.DataTypes.TIMESTAMP) + val CANCELED_DATE: Field[LocalDateTime] = DSL.field("canceled_date", PostgresCommons.DataTypes.TIMESTAMP) + val CANCEL_REQUESTED_NODE: Field[String] = DSL.field("cancel_requested_node", SQLDataType.VARCHAR) + val FAILED_DATE: Field[LocalDateTime] = DSL.field("failed_date", PostgresCommons.DataTypes.TIMESTAMP) + + private val TABLE: PostgresTable = PostgresTable.name(TABLE_NAME.getName) + .createTableStep((dsl, tableName) => dsl.createTableIfNotExists(tableName) + .column(TASK_ID) + .column(ADDITIONAL_INFORMATION) + .column(TYPE) + .column(STATUS) + .column(SUBMITTED_DATE) + .column(SUBMITTED_NODE) + .column(STARTED_DATE) + .column(RAN_NODE) + .column(COMPLETED_DATE) + .column(CANCELED_DATE) + .column(CANCEL_REQUESTED_NODE) + .column(FAILED_DATE) + .constraint(DSL.primaryKey(TASK_ID))) + .disableRowLevelSecurity + .build + + private val SUBMITTED_DATE_INDEX: PostgresIndex = PostgresIndex.name("task_execution_details_projection_submittedDate_index") + .createIndexStep((dsl, indexName) => dsl.createIndexIfNotExists(indexName) + .on(TABLE_NAME, SUBMITTED_DATE)); + + val MODULE: PostgresModule = PostgresModule + .builder + .addTable(TABLE) + .addIndex(SUBMITTED_DATE_INDEX) + .build +} diff --git a/server/task/task-postgres/src/test/java/org/apache/james/task/eventsourcing/postgres/PostgresTaskExecutionDetailsProjectionDAOTest.java b/server/task/task-postgres/src/test/java/org/apache/james/task/eventsourcing/postgres/PostgresTaskExecutionDetailsProjectionDAOTest.java new file mode 100644 index 00000000000..85b8508444e --- /dev/null +++ b/server/task/task-postgres/src/test/java/org/apache/james/task/eventsourcing/postgres/PostgresTaskExecutionDetailsProjectionDAOTest.java @@ -0,0 +1,202 @@ +/**************************************************************** + * 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.task.eventsourcing.postgres; + +import static org.apache.james.task.TaskExecutionDetailsFixture.TASK_EXECUTION_DETAILS; +import static org.apache.james.task.TaskExecutionDetailsFixture.TASK_EXECUTION_DETAILS_2; +import static org.apache.james.task.TaskExecutionDetailsFixture.TASK_EXECUTION_DETAILS_UPDATED; +import static org.apache.james.task.TaskExecutionDetailsFixture.TASK_EXECUTION_DETAILS_WITH_ADDITIONAL_INFORMATION; +import static org.apache.james.task.TaskExecutionDetailsFixture.TASK_ID; +import static org.apache.james.task.TaskExecutionDetailsFixture.TASK_ID_2; +import static org.assertj.core.api.Assertions.assertThat; + +import java.time.Instant; +import java.time.ZoneId; +import java.time.ZonedDateTime; +import java.util.Optional; +import java.util.stream.Stream; + +import org.apache.james.backends.postgres.PostgresExtension; +import org.apache.james.server.task.json.JsonTaskAdditionalInformationSerializer; +import org.apache.james.server.task.json.dto.MemoryReferenceWithCounterTaskAdditionalInformationDTO; +import org.apache.james.task.TaskExecutionDetails; +import org.apache.james.task.TaskExecutionDetailsFixture; +import org.apache.james.task.TaskManager; +import org.apache.james.task.TaskType; +import org.junit.jupiter.api.BeforeEach; +import org.junit.jupiter.api.Test; +import org.junit.jupiter.api.extension.RegisterExtension; + +import reactor.core.publisher.Flux; + +class PostgresTaskExecutionDetailsProjectionDAOTest { + @RegisterExtension + static PostgresExtension postgresExtension = PostgresExtension.withoutRowLevelSecurity(PostgresTaskExecutionDetailsProjectionModule.MODULE()); + + private static final JsonTaskAdditionalInformationSerializer JSON_TASK_ADDITIONAL_INFORMATION_SERIALIZER = JsonTaskAdditionalInformationSerializer.of(MemoryReferenceWithCounterTaskAdditionalInformationDTO.SERIALIZATION_MODULE); + + private PostgresTaskExecutionDetailsProjectionDAO testee; + + @BeforeEach + void setUp() { + testee = new PostgresTaskExecutionDetailsProjectionDAO(postgresExtension.getDefaultPostgresExecutor(), JSON_TASK_ADDITIONAL_INFORMATION_SERIALIZER); + } + + @Test + void readDetailsShouldBeAbleToRetrieveASavedRecord() { + testee.saveDetails(TASK_EXECUTION_DETAILS()).block(); + + TaskExecutionDetails taskExecutionDetails = testee.readDetails(TASK_ID()).block(); + + assertThat(taskExecutionDetails) + .usingRecursiveComparison() + .ignoringFields("submittedDate") + .isEqualTo(TASK_EXECUTION_DETAILS()); + } + + @Test + void readDetailsShouldBeAbleToRetrieveASavedRecordWithAdditionalInformation() { + testee.saveDetails(TASK_EXECUTION_DETAILS_WITH_ADDITIONAL_INFORMATION()).block(); + + TaskExecutionDetails taskExecutionDetails = testee.readDetails(TASK_ID()).block(); + + assertThat(taskExecutionDetails) + .usingRecursiveComparison() + .ignoringFields("submittedDate") + .isEqualTo(TASK_EXECUTION_DETAILS_WITH_ADDITIONAL_INFORMATION()); + + assertThat(taskExecutionDetails.getSubmittedDate().isEqual(TASK_EXECUTION_DETAILS_WITH_ADDITIONAL_INFORMATION().getSubmittedDate())) + .isTrue(); + } + + @Test + void saveDetailsShouldUpdateRecords() { + testee.saveDetails(TASK_EXECUTION_DETAILS()).block(); + + testee.saveDetails(TASK_EXECUTION_DETAILS_UPDATED()).block(); + + TaskExecutionDetails taskExecutionDetails = testee.readDetails(TASK_ID()).block(); + + assertThat(taskExecutionDetails) + .usingRecursiveComparison() + .ignoringFields("submittedDate") + .isEqualTo(TASK_EXECUTION_DETAILS_UPDATED()); + + assertThat(taskExecutionDetails.getSubmittedDate().isEqual(TASK_EXECUTION_DETAILS_UPDATED().getSubmittedDate())) + .isTrue(); + } + + @Test + void readDetailsShouldReturnEmptyWhenNone() { + Optional taskExecutionDetails = testee.readDetails(TASK_ID()).blockOptional(); + assertThat(taskExecutionDetails).isEmpty(); + } + + @Test + void listDetailsShouldReturnEmptyWhenNone() { + Stream taskExecutionDetails = testee.listDetails().toStream(); + assertThat(taskExecutionDetails).isEmpty(); + } + + @Test + void listDetailsShouldReturnAllRecords() { + testee.saveDetails(TASK_EXECUTION_DETAILS()).block(); + testee.saveDetails(TASK_EXECUTION_DETAILS_2()).block(); + + Stream taskExecutionDetails = testee.listDetails().toStream(); + + assertThat(taskExecutionDetails) + .usingRecursiveFieldByFieldElementComparatorIgnoringFields("submittedDate") + .containsOnly(TASK_EXECUTION_DETAILS(), TASK_EXECUTION_DETAILS_2()); + } + + @Test + void listDetailsShouldReturnLastUpdatedRecords() { + testee.saveDetails(TASK_EXECUTION_DETAILS()).block(); + testee.saveDetails(TASK_EXECUTION_DETAILS_UPDATED()).block(); + + Stream taskExecutionDetails = testee.listDetails().toStream(); + assertThat(taskExecutionDetails) + .usingRecursiveFieldByFieldElementComparatorIgnoringFields("submittedDate") + .containsOnly(TASK_EXECUTION_DETAILS_UPDATED()); + } + + @Test + void listBeforeDateShouldReturnCorrectEntry() { + TaskExecutionDetails taskExecutionDetails1 = new TaskExecutionDetails(TASK_ID(), + TaskType.of("type"), + TaskManager.Status.COMPLETED, + ZonedDateTime.ofInstant(Instant.parse("2000-01-01T00:00:00Z"), ZoneId.systemDefault()), + TaskExecutionDetailsFixture.SUBMITTED_NODE(), + Optional::empty, + Optional.empty(), + Optional.empty(), + Optional.empty(), + Optional.empty(), + Optional.empty(), + Optional.empty()); + + TaskExecutionDetails taskExecutionDetails2 = new TaskExecutionDetails(TASK_ID_2(), + TaskType.of("type"), + TaskManager.Status.COMPLETED, + ZonedDateTime.ofInstant(Instant.parse("2000-01-20T00:00:00Z"), ZoneId.systemDefault()), + TaskExecutionDetailsFixture.SUBMITTED_NODE(), + Optional::empty, + Optional.empty(), + Optional.empty(), + Optional.empty(), + Optional.empty(), + Optional.empty(), + Optional.empty()); + + testee.saveDetails(taskExecutionDetails1).block(); + testee.saveDetails(taskExecutionDetails2).block(); + + assertThat(Flux.from(testee.listDetailsByBeforeDate(Instant.parse("2000-01-15T12:00:55Z"))).collectList().block()) + .usingRecursiveFieldByFieldElementComparatorIgnoringFields("submittedDate") + .containsOnly(taskExecutionDetails1); + } + + @Test + void removeShouldDeleteAssignEntry() { + TaskExecutionDetails taskExecutionDetails1 = new TaskExecutionDetails(TASK_ID(), + TaskType.of("type"), + TaskManager.Status.COMPLETED, + ZonedDateTime.ofInstant(Instant.parse("2000-01-01T00:00:00Z"), ZoneId.systemDefault()), + TaskExecutionDetailsFixture.SUBMITTED_NODE(), + Optional::empty, + Optional.empty(), + Optional.empty(), + Optional.empty(), + Optional.empty(), + Optional.empty(), + Optional.empty()); + + testee.saveDetails(taskExecutionDetails1).block(); + + assertThat(testee.listDetails().collectList().block()) + .hasSize(1); + + testee.remove(taskExecutionDetails1).block(); + + assertThat(testee.listDetails().collectList().block()) + .isEmpty(); + } +} \ No newline at end of file diff --git a/server/task/task-postgres/src/test/java/org/apache/james/task/eventsourcing/postgres/PostgresTaskExecutionDetailsProjectionTest.java b/server/task/task-postgres/src/test/java/org/apache/james/task/eventsourcing/postgres/PostgresTaskExecutionDetailsProjectionTest.java new file mode 100644 index 00000000000..287c6c3d262 --- /dev/null +++ b/server/task/task-postgres/src/test/java/org/apache/james/task/eventsourcing/postgres/PostgresTaskExecutionDetailsProjectionTest.java @@ -0,0 +1,52 @@ +/**************************************************************** + * 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.task.eventsourcing.postgres; + +import java.util.function.Supplier; + +import org.apache.james.backends.postgres.PostgresExtension; +import org.apache.james.server.task.json.JsonTaskAdditionalInformationSerializer; +import org.apache.james.server.task.json.dto.MemoryReferenceWithCounterTaskAdditionalInformationDTO; +import org.apache.james.task.eventsourcing.TaskExecutionDetailsProjection; +import org.apache.james.task.eventsourcing.TaskExecutionDetailsProjectionContract; +import org.junit.jupiter.api.BeforeEach; +import org.junit.jupiter.api.extension.RegisterExtension; + +class PostgresTaskExecutionDetailsProjectionTest implements TaskExecutionDetailsProjectionContract { + @RegisterExtension + static PostgresExtension postgresExtension = PostgresExtension.withoutRowLevelSecurity(PostgresTaskExecutionDetailsProjectionModule.MODULE()); + + private static final JsonTaskAdditionalInformationSerializer JSON_TASK_ADDITIONAL_INFORMATION_SERIALIZER = JsonTaskAdditionalInformationSerializer.of(MemoryReferenceWithCounterTaskAdditionalInformationDTO.SERIALIZATION_MODULE); + + private Supplier testeeSupplier; + + @BeforeEach + void setUp() { + PostgresTaskExecutionDetailsProjectionDAO dao = new PostgresTaskExecutionDetailsProjectionDAO(postgresExtension.getDefaultPostgresExecutor(), + JSON_TASK_ADDITIONAL_INFORMATION_SERIALIZER); + testeeSupplier = () -> new PostgresTaskExecutionDetailsProjection(dao); + } + + @Override + public TaskExecutionDetailsProjection testee() { + return testeeSupplier.get(); + } + +} diff --git a/src/adr/0070-postgresql-adoption.md b/src/adr/0070-postgresql-adoption.md index 115594daa03..5d1caf4f262 100644 --- a/src/adr/0070-postgresql-adoption.md +++ b/src/adr/0070-postgresql-adoption.md @@ -1,4 +1,4 @@ -# 68. Native PostgreSQL adoption +# 70. Native PostgreSQL adoption Date: 2023-10-31 diff --git a/src/adr/0071-postgresql-mailbox-tables-structure.md b/src/adr/0071-postgresql-mailbox-tables-structure.md new file mode 100644 index 00000000000..df859422d46 --- /dev/null +++ b/src/adr/0071-postgresql-mailbox-tables-structure.md @@ -0,0 +1,58 @@ +# 71. Postgresql Mailbox tables structure + +Date: 2023-12-14 + +## Status + +Implemented + +## Context + +As described in [ADR-70](link), we are willing to provide a Postgres implementation for Apache James. +The current document is willing to detail the inner working of the mailbox of the target implementation. + +## Decision + +![diagram for mailbox tables](img/adr-71-mailbox-tables-diagram.png) + +Table list: +- mailbox +- mailbox_annotations +- message +- message_mailbox +- subscription + +Indexes in table message_mailbox: +- message_mailbox_message_id_index (message_id) +- mailbox_id_mail_uid_index (mailbox_id, message_uid) +- mailbox_id_is_seen_mail_uid_index (mailbox_id, is_seen, message_uid) +- mailbox_id_is_recent_mail_uid_index (mailbox_id, is_recent, message_uid) +- mailbox_id_is_delete_mail_uid_index (mailbox_id, is_deleted, message_uid) + +Indexes are used to find records faster. + +The table structure is mostly normalized which mitigates storage costs and achieves consistency easily. + +Foreign key constraints (mailbox_id in mailbox_annotations, message_id in message_mailbox) help to ensure data consistency. For example, message_id 1 in table message_mailbox could not exist if message_id 1 in table message does not exist + +For some fields, hstore data type are used. Hstore is key-value hashmap data structure. Hstore allows us to model complex data types without the need for complex joins. + +Special postgres clauses such as RETURNING, ON CONFLICT are used to ensure consistency without the need of combining multiple queries in a single transaction. + +## Consequences + +Pros: +- Indexes could increase query performance significantly + +Cons: +- Too many indexes in a table could reduce the performance of updating data in the table + +## Alternatives + +## References + +- [JIRA](https://issues.apache.org/jira/browse/JAMES-2586) +- [PostgreSQL](https://www.postgresql.org/) + + + diff --git a/src/adr/0072-postgresql-flags-update-concurrency-control.md b/src/adr/0072-postgresql-flags-update-concurrency-control.md new file mode 100644 index 00000000000..060e0c60960 --- /dev/null +++ b/src/adr/0072-postgresql-flags-update-concurrency-control.md @@ -0,0 +1,57 @@ +# 72. Postgresql flags update concurrency control mechanism + +Date: 2023-12-19 + +## Status + +Not-Implemented + +## Context + +We are facing a concurrency issue when update flags concurrently. +The multiple queries from clients simultaneously access the `user_flags` column of the `message_mailbox` table in PostgreSQL. +Currently, the James fetches the current data, performs changes, and then updates to database. +However, this approach does not ensure thread safety and may lead to concurrency issues. + +CRDT (conflict-free replicated data types) principles semantic can lay the ground to solving concurrency issues in a lock-free manner, and could thus be used for the problem at hand. This explores a different paradigm for addressing concurrency challenges without resorting to traditional transactions. + +## Decision + +To address the concurrency issue when clients make changes to the user_flags column, +we decide to use PostgreSQL's built-in functions to perform direct operations on the `user_flags` array column +(without fetching the current data and recalculating on James application). + +Specifically, we will use PostgreSQL functions such as +`array_remove`, `array_cat`, or `array_append` to perform specific operations as requested by the client (e.g., add, remove, replace elements). + +Additionally, we will create a custom function, say `remove_elements_from_array`, +for removing elements from the array since PostgreSQL does not support `array_remove` with an array input. + +## Consequences + +Pros: +- This solution reduces the complexity of working with the evaluate new user flags on James. +- Eliminates the step of fetching the current data and recalculating the new value of user_flags before updating. +- Ensures thread safety and reduces the risk of concurrency issues. + +Cons: +- The performance will depend on the performance of the PostgreSQL functions. + +## Alternatives + +- Optimistic Concurrency Control (OCC): Using optimistic concurrency control to ensure that only one version of the data is updated at a time. +However, this may increase the complexity of the code and require careful management of data versions. +The chosen solution using PostgreSQL functions was preferred for its simplicity and direct support for array operations. + +- Read-Then-Write Logic into Transactions: Transactions come with associated costs, including extra locking, coordination overhead, +and dependency on connection pooling. By avoiding the use of transactions, we aim to reduce these potential drawbacks +and explore other mechanisms for ensuring data consistency. + +## References + +- [JIRA](https://issues.apache.org/jira/browse/JAMES-2586) +- [PostgreSQL Array Functions and Operators](https://www.postgresql.org/docs/current/functions-array.html) +- [CRDT](https://en.wikipedia.org/wiki/Conflict-free_replicated_data_type) + + + diff --git a/src/adr/img/adr-71-mailbox-tables-diagram.png b/src/adr/img/adr-71-mailbox-tables-diagram.png new file mode 100644 index 00000000000..c9b2d11b5f5 Binary files /dev/null and b/src/adr/img/adr-71-mailbox-tables-diagram.png differ diff --git a/src/site/xdoc/server/config-mailrepositorystore.xml b/src/site/xdoc/server/config-mailrepositorystore.xml index 365b559c0e9..d8fd9f285c2 100644 --- a/src/site/xdoc/server/config-mailrepositorystore.xml +++ b/src/site/xdoc/server/config-mailrepositorystore.xml @@ -90,6 +90,12 @@

Cassandra Guice wiring allows to use the cassandra:// protocol for your ToRepository mailets.

+ + +

Postgres Guice wiring allows to use the postgres:// protocol for your ToRepository mailets.

+ +

This repository stores mail metadata in the Postgres database while the headers and body to the blob store.

+