From 9b4db9862e3f64bdb650a7de97bf833526d3ba67 Mon Sep 17 00:00:00 2001 From: Paulo Gomes da Cruz Junior Date: Mon, 28 Oct 2024 11:18:04 -0700 Subject: [PATCH 1/2] chore(db): simplifying db configuration (#422) --- backend/pom.xml | 5 ++ .../DataSourceConfiguration.java | 19 ---- .../configuration/OracleJpaConfiguration.java | 62 +++++++++++++ .../PostgresJpaConfiguration.java | 66 ++++++++++++++ .../OracleHikariConfiguration.java | 25 ------ .../configuration/OracleJpaConfiguration.java | 68 -------------- .../OraclePersistenceConfiguration.java | 50 ----------- .../PostgresHikariConfiguration.java | 27 ------ .../PostgresJpaConfiguration.java | 71 --------------- .../PostgresPersistenceConfiguration.java | 43 --------- .../endpoint/DashboardExtractionEndpoint.java | 1 - backend/src/main/resources/application.yml | 55 ++++++------ .../results/config/MultiFlywayConfig.java | 90 ------------------- .../AbstractTestContainerIntegrationTest.java | 55 ++++++------ .../UserOpeningRepositoryIntegrationTest.java | 5 -- 15 files changed, 190 insertions(+), 452 deletions(-) delete mode 100644 backend/src/main/java/ca/bc/gov/restapi/results/common/configuration/DataSourceConfiguration.java create mode 100644 backend/src/main/java/ca/bc/gov/restapi/results/common/configuration/OracleJpaConfiguration.java create mode 100644 backend/src/main/java/ca/bc/gov/restapi/results/common/configuration/PostgresJpaConfiguration.java delete mode 100644 backend/src/main/java/ca/bc/gov/restapi/results/oracle/configuration/OracleHikariConfiguration.java delete mode 100644 backend/src/main/java/ca/bc/gov/restapi/results/oracle/configuration/OracleJpaConfiguration.java delete mode 100644 backend/src/main/java/ca/bc/gov/restapi/results/oracle/configuration/OraclePersistenceConfiguration.java delete mode 100644 backend/src/main/java/ca/bc/gov/restapi/results/postgres/configuration/PostgresHikariConfiguration.java delete mode 100644 backend/src/main/java/ca/bc/gov/restapi/results/postgres/configuration/PostgresJpaConfiguration.java delete mode 100644 backend/src/main/java/ca/bc/gov/restapi/results/postgres/configuration/PostgresPersistenceConfiguration.java delete mode 100644 backend/src/test/java/ca/bc/gov/restapi/results/config/MultiFlywayConfig.java delete mode 100644 backend/src/test/java/ca/bc/gov/restapi/results/postgres/repository/UserOpeningRepositoryIntegrationTest.java diff --git a/backend/pom.xml b/backend/pom.xml index c7c0d940..6d3f6fcd 100644 --- a/backend/pom.xml +++ b/backend/pom.xml @@ -192,6 +192,11 @@ oracle-free test + + com.zaxxer + HikariCP + + diff --git a/backend/src/main/java/ca/bc/gov/restapi/results/common/configuration/DataSourceConfiguration.java b/backend/src/main/java/ca/bc/gov/restapi/results/common/configuration/DataSourceConfiguration.java deleted file mode 100644 index 5d62b9dd..00000000 --- a/backend/src/main/java/ca/bc/gov/restapi/results/common/configuration/DataSourceConfiguration.java +++ /dev/null @@ -1,19 +0,0 @@ -package ca.bc.gov.restapi.results.common.configuration; - -import lombok.Getter; -import lombok.Setter; -import org.springframework.boot.context.properties.ConfigurationProperties; -import org.springframework.context.annotation.Configuration; - -/** This class gets configuration from the application.yml file for both databases. */ -@Getter -@Setter -@Configuration -@ConfigurationProperties(prefix = "spring.datasource") -public class DataSourceConfiguration { - - private String driverClassName; - private String url; - private String username; - private String password; -} diff --git a/backend/src/main/java/ca/bc/gov/restapi/results/common/configuration/OracleJpaConfiguration.java b/backend/src/main/java/ca/bc/gov/restapi/results/common/configuration/OracleJpaConfiguration.java new file mode 100644 index 00000000..23376281 --- /dev/null +++ b/backend/src/main/java/ca/bc/gov/restapi/results/common/configuration/OracleJpaConfiguration.java @@ -0,0 +1,62 @@ +package ca.bc.gov.restapi.results.common.configuration; + +import com.zaxxer.hikari.HikariDataSource; +import jakarta.persistence.EntityManagerFactory; +import java.util.Map; +import javax.sql.DataSource; +import org.springframework.beans.factory.annotation.Qualifier; +import org.springframework.boot.context.properties.ConfigurationProperties; +import org.springframework.boot.jdbc.DataSourceBuilder; +import org.springframework.boot.orm.jpa.EntityManagerFactoryBuilder; +import org.springframework.context.annotation.Bean; +import org.springframework.context.annotation.Configuration; +import org.springframework.data.jpa.repository.config.EnableJpaRepositories; +import org.springframework.orm.jpa.JpaTransactionManager; +import org.springframework.orm.jpa.LocalContainerEntityManagerFactoryBean; +import org.springframework.transaction.PlatformTransactionManager; +import org.springframework.transaction.annotation.EnableTransactionManagement; + +/** + * This class holds JPA configurations for the Oracle database. + */ +@Configuration +@EnableJpaRepositories( + basePackages = {"ca.bc.gov.restapi.results.oracle"}, + entityManagerFactoryRef = "oracleEntityManagerFactory", + transactionManagerRef = "oracleTransactionManager" +) +@EnableTransactionManagement +public class OracleJpaConfiguration { + + @Bean(name = "oracleEntityManagerFactory") + public LocalContainerEntityManagerFactoryBean oracleEntityManagerFactory( + @Qualifier("oracleDataSource") HikariDataSource dataSource, + EntityManagerFactoryBuilder builder + ) { + return builder + .dataSource(dataSource) + .properties(Map.of( + "hibernate.dialect", "org.hibernate.dialect.OracleDialect", + "hibernate.boot.allow_jdbc_metadata_access","false", + "hibernate.hikari.connection.provider_class", + "org.hibernate.hikaricp.internal.HikariCPConnectionProvider", + "hibernate.connection.datasource", dataSource + )) + .packages("ca.bc.gov.restapi.results.oracle") + .persistenceUnit("oracle") + .build(); + } + + @Bean(name = "oracleDataSource") + @ConfigurationProperties(prefix = "spring.oracle.hikari") + public HikariDataSource oracleDataSource() { + return DataSourceBuilder.create().type(HikariDataSource.class).build(); + } + + @Bean(name = "oracleTransactionManager") + public PlatformTransactionManager oracleTransactionManager( + @Qualifier("oracleEntityManagerFactory") final EntityManagerFactory entityManagerFactory + ) { + return new JpaTransactionManager(entityManagerFactory); + } +} diff --git a/backend/src/main/java/ca/bc/gov/restapi/results/common/configuration/PostgresJpaConfiguration.java b/backend/src/main/java/ca/bc/gov/restapi/results/common/configuration/PostgresJpaConfiguration.java new file mode 100644 index 00000000..818cc524 --- /dev/null +++ b/backend/src/main/java/ca/bc/gov/restapi/results/common/configuration/PostgresJpaConfiguration.java @@ -0,0 +1,66 @@ +package ca.bc.gov.restapi.results.common.configuration; + +import com.zaxxer.hikari.HikariDataSource; +import jakarta.persistence.EntityManagerFactory; +import java.util.Map; +import javax.sql.DataSource; +import org.springframework.beans.factory.annotation.Qualifier; +import org.springframework.boot.context.properties.ConfigurationProperties; +import org.springframework.boot.jdbc.DataSourceBuilder; +import org.springframework.boot.orm.jpa.EntityManagerFactoryBuilder; +import org.springframework.context.annotation.Bean; +import org.springframework.context.annotation.Configuration; +import org.springframework.context.annotation.Primary; +import org.springframework.data.jpa.repository.config.EnableJpaRepositories; +import org.springframework.orm.jpa.JpaTransactionManager; +import org.springframework.orm.jpa.LocalContainerEntityManagerFactoryBean; +import org.springframework.transaction.PlatformTransactionManager; +import org.springframework.transaction.annotation.EnableTransactionManagement; + +/** + * This class holds JPA configurations for the Postgres database. + */ +@Configuration +@EnableTransactionManagement +@EnableJpaRepositories( + basePackages = {"ca.bc.gov.restapi.results.postgres"}, + entityManagerFactoryRef = "postgresEntityManagerFactory", + transactionManagerRef = "postgresTransactionManager") +public class PostgresJpaConfiguration { + + @Primary + @Bean(name = "postgresEntityManagerFactory") + public LocalContainerEntityManagerFactoryBean postgresEntityManagerFactory( + @Qualifier("postgresHikariDataSource") HikariDataSource dataSource, + EntityManagerFactoryBuilder builder + ) { + return builder + .dataSource(dataSource) + .properties(Map.of( + "hibernate.dialect", "org.hibernate.dialect.PostgreSQLDialect", + "hibernate.boot.allow_jdbc_metadata_access", "false", + "hibernate.hikari.connection.provider_class", + "org.hibernate.hikaricp.internal.HikariCPConnectionProvider", + "hibernate.connection.datasource", dataSource + )) + .packages("ca.bc.gov.restapi.results.postgres") + .persistenceUnit("postgres") + .build(); + } + + @Bean(name = "postgresHikariDataSource") + @ConfigurationProperties(prefix = "spring.datasource.hikari") + @Primary + public HikariDataSource postgresHikariDataSource() { + return DataSourceBuilder.create().type(HikariDataSource.class).build(); + } + + @Bean(name = "postgresTransactionManager") + @Primary + public PlatformTransactionManager postgresTransactionManager( + @Qualifier("postgresEntityManagerFactory") final EntityManagerFactory entityManagerFactory + ) { + return new JpaTransactionManager(entityManagerFactory); + } + +} diff --git a/backend/src/main/java/ca/bc/gov/restapi/results/oracle/configuration/OracleHikariConfiguration.java b/backend/src/main/java/ca/bc/gov/restapi/results/oracle/configuration/OracleHikariConfiguration.java deleted file mode 100644 index 436535a1..00000000 --- a/backend/src/main/java/ca/bc/gov/restapi/results/oracle/configuration/OracleHikariConfiguration.java +++ /dev/null @@ -1,25 +0,0 @@ -package ca.bc.gov.restapi.results.oracle.configuration; - -import lombok.Data; -import org.springframework.boot.context.properties.ConfigurationProperties; -import org.springframework.context.annotation.Configuration; - -/** This class holds configurations for the Oracle Hikari connection pool. */ -@Data -@Configuration -@ConfigurationProperties("spring.datasource.oracle") -public class OracleHikariConfiguration { - - private String driverClassName; - private String url; - private String username; - private String password; - private long connectionTimeout; - private long idleTimeout; - private long maxLifetime; - private long keepaliveTime; - private String poolName; - private int minimumIdle; - private int maximumPoolSize; - private long leakDetectionThreshold; -} diff --git a/backend/src/main/java/ca/bc/gov/restapi/results/oracle/configuration/OracleJpaConfiguration.java b/backend/src/main/java/ca/bc/gov/restapi/results/oracle/configuration/OracleJpaConfiguration.java deleted file mode 100644 index ded15cf5..00000000 --- a/backend/src/main/java/ca/bc/gov/restapi/results/oracle/configuration/OracleJpaConfiguration.java +++ /dev/null @@ -1,68 +0,0 @@ -package ca.bc.gov.restapi.results.oracle.configuration; - -import ca.bc.gov.restapi.results.oracle.entity.CutBlockOpenAdminEntity; -import ca.bc.gov.restapi.results.oracle.entity.OpeningEntity; -import ca.bc.gov.restapi.results.oracle.repository.CutBlockOpenAdminRepository; -import ca.bc.gov.restapi.results.oracle.repository.OpeningRepository; -import ca.bc.gov.restapi.results.oracle.service.CutBlockOpenAdminService; -import ca.bc.gov.restapi.results.oracle.service.OpeningService; -import java.util.Objects; -import javax.sql.DataSource; -import org.springframework.beans.factory.annotation.Qualifier; -import org.springframework.boot.orm.jpa.EntityManagerFactoryBuilder; -import org.springframework.context.annotation.Bean; -import org.springframework.context.annotation.Configuration; -import org.springframework.context.annotation.Primary; -import org.springframework.data.jpa.repository.config.EnableJpaRepositories; -import org.springframework.orm.jpa.JpaTransactionManager; -import org.springframework.orm.jpa.LocalContainerEntityManagerFactoryBean; -import org.springframework.transaction.PlatformTransactionManager; -import org.springframework.transaction.annotation.EnableTransactionManagement; - -/** This class holds JPA configurations for the Oracle database. */ -@Configuration -@EnableTransactionManagement -@EnableJpaRepositories( - basePackageClasses = { - // endpoints - CutBlockOpenAdminEntity.class, - // services - CutBlockOpenAdminService.class, - OpeningService.class, - // repositories - CutBlockOpenAdminRepository.class, - OpeningRepository.class, - // entities - OpeningEntity.class, - OpeningRepository.class, - }, - entityManagerFactoryRef = "oracleEntityManagerFactory", - transactionManagerRef = "oracleTransactionManager") -public class OracleJpaConfiguration { - - /** - * Creates the EntityManagerFactory for the oracle database. - * - * @param dataSource The Oracle DataSource. - * @param builder The EntityManagerFactory builder - * @return LocalContainerEntityManagerFactoryBean - */ - @Bean - @Primary - public LocalContainerEntityManagerFactoryBean oracleEntityManagerFactory( - @Qualifier("oracleDataSource") DataSource dataSource, EntityManagerFactoryBuilder builder) { - return builder.dataSource(dataSource).packages(getOracleEntities()).build(); - } - - private Class[] getOracleEntities() { - return new Class[] {CutBlockOpenAdminEntity.class, OpeningEntity.class}; - } - - @Bean - public PlatformTransactionManager oracleTransactionManager( - @Qualifier("oracleEntityManagerFactory") - LocalContainerEntityManagerFactoryBean oracleEntityManagerFactory) { - return new JpaTransactionManager( - Objects.requireNonNull(oracleEntityManagerFactory.getObject())); - } -} diff --git a/backend/src/main/java/ca/bc/gov/restapi/results/oracle/configuration/OraclePersistenceConfiguration.java b/backend/src/main/java/ca/bc/gov/restapi/results/oracle/configuration/OraclePersistenceConfiguration.java deleted file mode 100644 index d8fb6651..00000000 --- a/backend/src/main/java/ca/bc/gov/restapi/results/oracle/configuration/OraclePersistenceConfiguration.java +++ /dev/null @@ -1,50 +0,0 @@ -package ca.bc.gov.restapi.results.oracle.configuration; - -import com.zaxxer.hikari.HikariConfig; -import com.zaxxer.hikari.HikariDataSource; -import javax.sql.DataSource; -import org.springframework.beans.factory.annotation.Autowired; -import org.springframework.boot.autoconfigure.jdbc.DataSourceProperties; -import org.springframework.boot.context.properties.ConfigurationProperties; -import org.springframework.context.annotation.Bean; -import org.springframework.context.annotation.Configuration; -import org.springframework.context.annotation.Primary; - -/** - * This class holds persistence configurations for the Oracle database. - */ -@Configuration -public class OraclePersistenceConfiguration { - - @Autowired - private OracleHikariConfiguration oracleHikariConfiguration; - - @Bean - @ConfigurationProperties("spring.datasource.oracle") - public DataSourceProperties oracleDataSourceProperties() { - return new DataSourceProperties(); - } - - /** - * Creates a Postgres Datasource with all Hikari connection pool configuration. - */ - @Bean - @Primary - public DataSource oracleDataSource() { - HikariConfig config = new HikariConfig(); - config.setJdbcUrl(oracleHikariConfiguration.getUrl()); - config.setUsername(oracleHikariConfiguration.getUsername()); - config.setPassword(oracleHikariConfiguration.getPassword()); - config.setDriverClassName(oracleHikariConfiguration.getDriverClassName()); - config.setConnectionTimeout(oracleHikariConfiguration.getConnectionTimeout()); - config.setIdleTimeout(oracleHikariConfiguration.getIdleTimeout()); - config.setMaxLifetime(oracleHikariConfiguration.getMaxLifetime()); - config.setKeepaliveTime(oracleHikariConfiguration.getKeepaliveTime()); - config.setPoolName(oracleHikariConfiguration.getPoolName()); - config.setMinimumIdle(oracleHikariConfiguration.getMinimumIdle()); - config.setMaximumPoolSize(oracleHikariConfiguration.getMaximumPoolSize()); - config.setLeakDetectionThreshold(oracleHikariConfiguration.getLeakDetectionThreshold()); - - return new HikariDataSource(config); - } -} diff --git a/backend/src/main/java/ca/bc/gov/restapi/results/postgres/configuration/PostgresHikariConfiguration.java b/backend/src/main/java/ca/bc/gov/restapi/results/postgres/configuration/PostgresHikariConfiguration.java deleted file mode 100644 index d04729ff..00000000 --- a/backend/src/main/java/ca/bc/gov/restapi/results/postgres/configuration/PostgresHikariConfiguration.java +++ /dev/null @@ -1,27 +0,0 @@ -package ca.bc.gov.restapi.results.postgres.configuration; - -import lombok.Getter; -import lombok.Setter; -import org.springframework.boot.context.properties.ConfigurationProperties; -import org.springframework.context.annotation.Configuration; - -/** This class holds configurations for the Postgres Hikari connection pool. */ -@Getter -@Setter -@Configuration -@ConfigurationProperties("spring.datasource.postgres") -public class PostgresHikariConfiguration { - - private String driverClassName; - private String url; - private String username; - private String password; - private long connectionTimeout; - private long idleTimeout; - private long maxLifetime; - private long keepaliveTime; - private String poolName; - private int minimumIdle; - private int maximumPoolSize; - private long leakDetectionThreshold; -} diff --git a/backend/src/main/java/ca/bc/gov/restapi/results/postgres/configuration/PostgresJpaConfiguration.java b/backend/src/main/java/ca/bc/gov/restapi/results/postgres/configuration/PostgresJpaConfiguration.java deleted file mode 100644 index a69e974f..00000000 --- a/backend/src/main/java/ca/bc/gov/restapi/results/postgres/configuration/PostgresJpaConfiguration.java +++ /dev/null @@ -1,71 +0,0 @@ -package ca.bc.gov.restapi.results.postgres.configuration; - -import ca.bc.gov.restapi.results.postgres.endpoint.UserOpeningEndpoint; -import ca.bc.gov.restapi.results.postgres.entity.UserOpeningEntity; -import ca.bc.gov.restapi.results.postgres.repository.UserOpeningRepository; -import java.util.Objects; -import java.util.Properties; -import javax.sql.DataSource; -import org.springframework.beans.factory.annotation.Qualifier; -import org.springframework.boot.orm.jpa.EntityManagerFactoryBuilder; -import org.springframework.context.annotation.Bean; -import org.springframework.context.annotation.Configuration; -import org.springframework.data.jpa.repository.config.EnableJpaRepositories; -import org.springframework.orm.jpa.JpaTransactionManager; -import org.springframework.orm.jpa.LocalContainerEntityManagerFactoryBean; -import org.springframework.transaction.PlatformTransactionManager; -import org.springframework.transaction.annotation.EnableTransactionManagement; - -/** This class holds JPA configurations for the Postgres database. */ -@Configuration -@EnableTransactionManagement -@EnableJpaRepositories( - basePackageClasses = { - // endpoints - UserOpeningEndpoint.class, - // services - // none yet - // repositories - UserOpeningRepository.class, - // entities - UserOpeningEntity.class, - UserOpeningEndpoint.class - }, - entityManagerFactoryRef = "postgresEntityManagerFactory", - transactionManagerRef = "postgresTransactionManager") -public class PostgresJpaConfiguration { - - /** - * Creates the EntityManagerFactory for the postgres database. - * - * @param dataSource The Postgres DataSource. - * @param builder The EntityManagerFactory builder - * @return LocalContainerEntityManagerFactoryBean - */ - @Bean - public LocalContainerEntityManagerFactoryBean postgresEntityManagerFactory( - @Qualifier("postgresDataSource") DataSource dataSource, EntityManagerFactoryBuilder builder) { - Properties jpaProps = new Properties(); - jpaProps.setProperty("hibernate.default_schema", "silva"); - jpaProps.setProperty("hibernate.ddl-auto", "update"); - jpaProps.setProperty("jpa.defer-datasource-initialization", "true"); - jpaProps.setProperty("sql.init.mode", "always"); - - LocalContainerEntityManagerFactoryBean build = - builder.dataSource(dataSource).packages(getPostgresEntities()).build(); - build.setJpaProperties(jpaProps); - return build; - } - - private Class[] getPostgresEntities() { - return new Class[] {UserOpeningEntity.class}; - } - - @Bean - public PlatformTransactionManager postgresTransactionManager( - @Qualifier("postgresEntityManagerFactory") - LocalContainerEntityManagerFactoryBean postgresEntityManagerFactory) { - return new JpaTransactionManager( - Objects.requireNonNull(postgresEntityManagerFactory.getObject())); - } -} diff --git a/backend/src/main/java/ca/bc/gov/restapi/results/postgres/configuration/PostgresPersistenceConfiguration.java b/backend/src/main/java/ca/bc/gov/restapi/results/postgres/configuration/PostgresPersistenceConfiguration.java deleted file mode 100644 index cde31f82..00000000 --- a/backend/src/main/java/ca/bc/gov/restapi/results/postgres/configuration/PostgresPersistenceConfiguration.java +++ /dev/null @@ -1,43 +0,0 @@ -package ca.bc.gov.restapi.results.postgres.configuration; - -import com.zaxxer.hikari.HikariConfig; -import com.zaxxer.hikari.HikariDataSource; -import javax.sql.DataSource; -import org.springframework.beans.factory.annotation.Autowired; -import org.springframework.boot.autoconfigure.jdbc.DataSourceProperties; -import org.springframework.boot.context.properties.ConfigurationProperties; -import org.springframework.context.annotation.Bean; -import org.springframework.context.annotation.Configuration; - -/** This class holds persistence configurations for the Postgres database. */ -@Configuration -public class PostgresPersistenceConfiguration { - - @Autowired private PostgresHikariConfiguration postgresHikariConfiguration; - - @Bean - @ConfigurationProperties("spring.datasource.postgres") - public DataSourceProperties postgresDataSourceProperties() { - return new DataSourceProperties(); - } - - /** Creates a Postgres Datasource with all Hikari connection pool configuration. */ - @Bean - public DataSource postgresDataSource() { - HikariConfig config = new HikariConfig(); - config.setJdbcUrl(postgresHikariConfiguration.getUrl()); - config.setUsername(postgresHikariConfiguration.getUsername()); - config.setPassword(postgresHikariConfiguration.getPassword()); - config.setDriverClassName(postgresHikariConfiguration.getDriverClassName()); - config.setConnectionTimeout(postgresHikariConfiguration.getConnectionTimeout()); - config.setIdleTimeout(postgresHikariConfiguration.getIdleTimeout()); - config.setMaxLifetime(postgresHikariConfiguration.getMaxLifetime()); - config.setKeepaliveTime(postgresHikariConfiguration.getKeepaliveTime()); - config.setPoolName(postgresHikariConfiguration.getPoolName()); - config.setMinimumIdle(postgresHikariConfiguration.getMinimumIdle()); - config.setMaximumPoolSize(postgresHikariConfiguration.getMaximumPoolSize()); - config.setLeakDetectionThreshold(postgresHikariConfiguration.getLeakDetectionThreshold()); - - return new HikariDataSource(config); - } -} diff --git a/backend/src/main/java/ca/bc/gov/restapi/results/postgres/endpoint/DashboardExtractionEndpoint.java b/backend/src/main/java/ca/bc/gov/restapi/results/postgres/endpoint/DashboardExtractionEndpoint.java index c7c00aa8..729c4164 100644 --- a/backend/src/main/java/ca/bc/gov/restapi/results/postgres/endpoint/DashboardExtractionEndpoint.java +++ b/backend/src/main/java/ca/bc/gov/restapi/results/postgres/endpoint/DashboardExtractionEndpoint.java @@ -45,7 +45,6 @@ public ResponseEntity startExtractionProcessManually( Integer months, @RequestParam(value = "debug", required = false) Boolean debug) { - if (dashboardUserManagerConfiguration.getUserList().isEmpty()) { return ResponseEntity.status(HttpStatus.UNAUTHORIZED).build(); } diff --git a/backend/src/main/resources/application.yml b/backend/src/main/resources/application.yml index 33703131..185edb77 100644 --- a/backend/src/main/resources/application.yml +++ b/backend/src/main/resources/application.yml @@ -11,36 +11,42 @@ server: spring: application: name: results-api - -# Database, and JPA - Oracle - datasource: - oracle: + oracle: + jdbcUrl: jdbc:oracle:thin:@tcps://${DATABASE_HOST:nrcdb03.bcgov}:${DATABASE_PORT:1543}/${SERVICE_NAME:dbq01.nrs.bcgov}?javax.net.ssl.trustStore=${ca.bc.gov.nrs.oracle.keystore}&javax.net.ssl.trustStorePassword=${ca.bc.gov.nrs.oracle.secret}&javax.net.ssl.keyStore=${ca.bc.gov.nrs.oracle.keystore}&javax.net.ssl.keyStorePassword=${ca.bc.gov.nrs.oracle.secret}&oracle.net.ssl_certificate_alias=${ca.bc.gov.nrs.oracle.host}&oracle.net.ssl_server_dn_match=false + url: ${spring.datasource.oracle.jdbcUrl} + username: ${DATABASE_USER} + password: ${DATABASE_PASSWORD} + hikari: driver-class-name: oracle.jdbc.OracleDriver - url: jdbc:oracle:thin:@tcps://${DATABASE_HOST:nrcdb03.bcgov}:${DATABASE_PORT:1543}/${SERVICE_NAME:dbq01.nrs.bcgov}?javax.net.ssl.trustStore=${ca.bc.gov.nrs.oracle.keystore}&javax.net.ssl.trustStorePassword=${ca.bc.gov.nrs.oracle.secret}&javax.net.ssl.keyStore=${ca.bc.gov.nrs.oracle.keystore}&javax.net.ssl.keyStorePassword=${ca.bc.gov.nrs.oracle.secret}&oracle.net.ssl_certificate_alias=${ca.bc.gov.nrs.oracle.host}&oracle.net.ssl_server_dn_match=false - username: ${DATABASE_USER} - password: ${DATABASE_PASSWORD} - connectionTimeout: ${DB_POOL_CONN_TIMEOUT:120000} - idleTimeout: ${DB_POOL_IDLE_TIMEOUT:60000} - maxLifetime: ${DB_POOL_MAX_LIFETIME:150000} + jdbcUrl: ${spring.oracle.jdbcUrl} + url: ${spring.oracle.jdbcUrl} + username: ${spring.oracle.username} + password: ${spring.oracle.password} + connectionTimeout: 30000 + idleTimeout: 600000 + maxLifetime: 1800000 keepaliveTime: 30000 poolName: SilvaOracleConnPool - minimumIdle: ${DB_POOL_MIN_IDLE:1} - maximumPoolSize: ${DB_POOL_MAX_SIZE:1} + minimumIdle: 1 + maximumPoolSize: 3 leakDetectionThreshold: 60000 - -# Database, and JPA - Postgres - postgres: + datasource: + jdbcUrl: jdbc:postgresql://${POSTGRES_HOST:localhost}:5432/${POSTGRES_DB:nr-silva} + url: ${spring.datasource.jdbcUrl} + username: ${POSTGRES_USER:nr-silva} + password: ${POSTGRES_PASSWORD:default} + hikari: driver-class-name: org.postgresql.Driver - url: jdbc:postgresql://${POSTGRES_HOST:localhost}:5432/${POSTGRES_DB:nr-silva} - username: ${POSTGRES_USER:nr-silva} - password: ${POSTGRES_PASSWORD:default} - connectionTimeout: ${DB_POOL_CONN_TIMEOUT:120000} - idleTimeout: ${DB_POOL_IDLE_TIMEOUT:60000} - maxLifetime: ${DB_POOL_MAX_LIFETIME:150000} + jdbcUrl: ${spring.datasource.jdbcUrl} + username: ${spring.datasource.username} + password: ${spring.datasource.password} + connectionTimeout: 30000 + idleTimeout: 600000 + maxLifetime: 1800000 keepaliveTime: 30000 poolName: SilvaPostgresConnPool - minimumIdle: ${DB_POOL_MIN_IDLE:1} - maximumPoolSize: ${DB_POOL_MAX_SIZE:1} + minimumIdle: 1 + maximumPoolSize: 3 leakDetectionThreshold: 60000 # Common database settings @@ -60,9 +66,6 @@ spring: enabled: true baseline-on-migrate: true locations: classpath:db/migration - url: ${spring.datasource.postgres.url} - user: ${spring.datasource.postgres.username} - password: ${spring.datasource.postgres.password} # OpenSearch settings nr-results-ecs-version: 8.9 diff --git a/backend/src/test/java/ca/bc/gov/restapi/results/config/MultiFlywayConfig.java b/backend/src/test/java/ca/bc/gov/restapi/results/config/MultiFlywayConfig.java deleted file mode 100644 index 07b65fe5..00000000 --- a/backend/src/test/java/ca/bc/gov/restapi/results/config/MultiFlywayConfig.java +++ /dev/null @@ -1,90 +0,0 @@ -package ca.bc.gov.restapi.results.config; - -import ca.bc.gov.restapi.results.oracle.configuration.OracleHikariConfiguration; -import ca.bc.gov.restapi.results.oracle.configuration.OraclePersistenceConfiguration; -import ca.bc.gov.restapi.results.postgres.configuration.PostgresHikariConfiguration; -import ca.bc.gov.restapi.results.postgres.configuration.PostgresPersistenceConfiguration; -import javax.sql.DataSource; -import org.flywaydb.core.Flyway; -import org.springframework.boot.jdbc.DataSourceBuilder; -import org.springframework.context.annotation.Bean; -import org.springframework.context.annotation.Configuration; - -/** - * Configuration class for setting up multiple Flyway instances for different databases. This - * resides inside the test package to avoid conflicts with the main application. This class is used - * to configure Flyway for PostgreSQL and Oracle databases, so we can make use of the testcontainers - * to recreate the expected database structure. - */ -@Configuration -public class MultiFlywayConfig { - - /** - * Configures Flyway for PostgreSQL database. This will overwrite the original Flyway - * configuration for PostgreSQL. - * - * @param postgresHikariConfiguration the PostgreSQL Hikari configuration - * @return the configured Flyway instance for PostgreSQL - */ - @Bean - public Flyway flywayPostgres(PostgresHikariConfiguration postgresHikariConfiguration) { - return Flyway.configure() - //We build the datasource as the original Bean uses Hikari, and it failed due to Timeout - .dataSource(toDataSource(postgresHikariConfiguration)) - .locations("classpath:db/migration", "classpath:migration/postgres") - .baselineOnMigrate(true) - .load(); - } - - /** - * Configures Flyway for Oracle database. This will allow the test to be able to recreate the - * expected oracle database format. - * - * @param oracleHikariConfiguration the Oracle Hikari configuration - * @return the configured Flyway instance for Oracle - */ - @Bean - public Flyway flywayOracle(OracleHikariConfiguration oracleHikariConfiguration) { - return Flyway.configure() - //We build the datasource as the original Bean uses Hikari, and it failed due to Timeout - .dataSource(toDataSource(oracleHikariConfiguration)) - .locations("classpath:migration/oracle") - .schemas("THE") - .load(); - } - - /** - * Converts PostgreSQL Hikari configuration to a DataSource. The original Bean found at - * {@link PostgresPersistenceConfiguration} was failing due to - * a timeout caused by HikariCP - * - * @param postgresHikariConfiguration the PostgreSQL Hikari configuration - * @return the DataSource for PostgreSQL - */ - private DataSource toDataSource(PostgresHikariConfiguration postgresHikariConfiguration) { - return DataSourceBuilder.create() - .url(postgresHikariConfiguration.getUrl()) - .username(postgresHikariConfiguration.getUsername()) - .password(postgresHikariConfiguration.getPassword()) - .driverClassName(postgresHikariConfiguration.getDriverClassName()) - .build(); - } - - /** - * Converts Oracle Hikari configuration to a DataSource. The original Bean found at - * {@link OraclePersistenceConfiguration} was failing due to a - * timeout caused by HikariCP. - * - * @param oracleHikariConfiguration the Oracle Hikari configuration - * @return the DataSource for Oracle - */ - private DataSource toDataSource(OracleHikariConfiguration oracleHikariConfiguration) { - return DataSourceBuilder.create() - .url(oracleHikariConfiguration.getUrl()) - .username(oracleHikariConfiguration.getUsername()) - .password(oracleHikariConfiguration.getPassword()) - .driverClassName(oracleHikariConfiguration.getDriverClassName()) - .build(); - } - -} \ No newline at end of file diff --git a/backend/src/test/java/ca/bc/gov/restapi/results/extensions/AbstractTestContainerIntegrationTest.java b/backend/src/test/java/ca/bc/gov/restapi/results/extensions/AbstractTestContainerIntegrationTest.java index 969847e2..2c474935 100644 --- a/backend/src/test/java/ca/bc/gov/restapi/results/extensions/AbstractTestContainerIntegrationTest.java +++ b/backend/src/test/java/ca/bc/gov/restapi/results/extensions/AbstractTestContainerIntegrationTest.java @@ -28,14 +28,10 @@ @ContextConfiguration public abstract class AbstractTestContainerIntegrationTest { - /** - * PostgreSQL container instance. - */ static final PostgreSQLContainer postgres; - /** - * Oracle container instance. - */ static final OracleContainer oracle; + static final Flyway flywayPostgres; + static final Flyway flywayOracle; // Static fields declared like this are instantiated first by the JVM static { @@ -47,13 +43,24 @@ public abstract class AbstractTestContainerIntegrationTest { postgres.start(); oracle.start(); - } - @Autowired - private Flyway flywayPostgres; + flywayPostgres = + Flyway + .configure() + .dataSource(postgres.getJdbcUrl(), postgres.getUsername(), postgres.getPassword()) + .locations("classpath:db/migration", "classpath:migration/postgres") + .baselineOnMigrate(true) + .load(); - @Autowired - private Flyway flywayOracle; + flywayOracle = + Flyway + .configure() + .dataSource(oracle.getJdbcUrl(), oracle.getUsername(), oracle.getPassword()) + .locations("classpath:migration/oracle") + .schemas("THE") + .baselineOnMigrate(true) + .load(); + } /** * Migrate the databases using Flyway before each test. As we're using flyway, there's no need to @@ -75,22 +82,16 @@ public void setUp() { */ @DynamicPropertySource static void registerDynamicProperties(DynamicPropertyRegistry registry) { - // Overwrite the Postgres datasource with the testcontainer configuration - registry.add("spring.datasource.postgres.url", postgres::getJdbcUrl); - registry.add("spring.datasource.postgres.username", postgres::getUsername); - registry.add("spring.datasource.postgres.password", postgres::getPassword); - // Overwrite the Flyway for Postgres with the testcontainer configuration - registry.add("spring.flyway.postgres.url", postgres::getJdbcUrl); - registry.add("spring.flyway.postgres.user", postgres::getUsername); - registry.add("spring.flyway.postgres.password", postgres::getPassword); - // Overwrite the Oracle datasource with the testcontainer configuration - registry.add("spring.datasource.oracle.url", oracle::getJdbcUrl); - registry.add("spring.datasource.oracle.username", oracle::getUsername); - registry.add("spring.datasource.oracle.password", oracle::getPassword); - // Overwrite the Flyway for Oracle with the testcontainer configuration - registry.add("spring.flyway.oracle.url", oracle::getJdbcUrl); - registry.add("spring.flyway.oracle.user", oracle::getUsername); - registry.add("spring.flyway.oracle.password", oracle::getPassword); + registry.add("spring.datasource.jdbcUrl", postgres::getJdbcUrl); + registry.add("spring.datasource.username", postgres::getUsername); + registry.add("spring.datasource.password", postgres::getPassword); + registry.add("spring.datasource.hikari.username", postgres::getUsername); + registry.add("spring.datasource.hikari.password", postgres::getPassword); + + registry.add("spring.oracle.url", oracle::getJdbcUrl); + registry.add("spring.oracle.jdbcUrl", oracle::getJdbcUrl); + registry.add("spring.oracle.username", oracle::getUsername); + registry.add("spring.oracle.password", oracle::getPassword); } } diff --git a/backend/src/test/java/ca/bc/gov/restapi/results/postgres/repository/UserOpeningRepositoryIntegrationTest.java b/backend/src/test/java/ca/bc/gov/restapi/results/postgres/repository/UserOpeningRepositoryIntegrationTest.java deleted file mode 100644 index e644b25d..00000000 --- a/backend/src/test/java/ca/bc/gov/restapi/results/postgres/repository/UserOpeningRepositoryIntegrationTest.java +++ /dev/null @@ -1,5 +0,0 @@ -package ca.bc.gov.restapi.results.postgres.repository; - -class UserOpeningRepositoryIntegrationTest { - -} From 2cded86651a5899456737742ff624474f224ab35 Mon Sep 17 00:00:00 2001 From: Paulo Gomes da Cruz Junior Date: Tue, 29 Oct 2024 07:25:29 -0700 Subject: [PATCH 2/2] fix(configuration): fixed configuration issues (#425) --- .../ExternalApiConfiguration.java | 12 +- .../configuration/ProvidersConfiguration.java | 17 - .../configuration/SecurityConfiguration.java | 2 +- .../configuration/SilvaConfiguration.java | 46 ++ .../endpoint/SecretsServiceEndpoint.java | 14 +- .../endpoint/OpeningSearchEndpoint.java | 3 +- .../repository/OpeningSearchRepository.java | 8 +- .../oracle/repository/OrgUnitRepository.java | 2 +- .../oracle/service/OrgUnitService.java | 25 +- .../DashboardUserManagerConfiguration.java | 6 +- backend/src/main/resources/application.yml | 29 +- .../endpoint/SecretsServiceEndpointTest.java | 16 +- .../endpoint/OpeningSearchEndpointTest.java | 4 +- .../OpeningSearchRepositoryTest.java | 7 +- .../OrgUnitRepositoryIntegrationTest.java | 2 +- .../oracle/service/OpeningServiceTest.java | 501 ++++++------------ .../oracle/service/OrgUnitServiceTest.java | 14 +- .../test/resources/application-default.yml | 8 +- 18 files changed, 295 insertions(+), 421 deletions(-) delete mode 100644 backend/src/main/java/ca/bc/gov/restapi/results/common/configuration/ProvidersConfiguration.java create mode 100644 backend/src/main/java/ca/bc/gov/restapi/results/common/configuration/SilvaConfiguration.java diff --git a/backend/src/main/java/ca/bc/gov/restapi/results/common/configuration/ExternalApiConfiguration.java b/backend/src/main/java/ca/bc/gov/restapi/results/common/configuration/ExternalApiConfiguration.java index ef39be43..124a621b 100644 --- a/backend/src/main/java/ca/bc/gov/restapi/results/common/configuration/ExternalApiConfiguration.java +++ b/backend/src/main/java/ca/bc/gov/restapi/results/common/configuration/ExternalApiConfiguration.java @@ -15,15 +15,15 @@ public class ExternalApiConfiguration { /** * Creates a RestClient bean for the Forest Client API. * - * @param providersConfiguration the configuration properties for providers + * @param silvaConfiguration the configuration properties for providers * @return the configured RestClient instance for the Forest Client API */ @Bean - public RestClient forestClientApi(ProvidersConfiguration providersConfiguration) { + public RestClient forestClientApi(SilvaConfiguration silvaConfiguration) { return RestClient .builder() - .baseUrl(providersConfiguration.getForestClientBaseUri()) - .defaultHeader("X-API-KEY", providersConfiguration.getForestClientApiKey()) + .baseUrl(silvaConfiguration.getForestClientApi().getAddress()) + .defaultHeader("X-API-KEY", silvaConfiguration.getForestClientApi().getKey()) .defaultHeader("Content-Type", MediaType.APPLICATION_JSON_VALUE) .build(); } @@ -34,10 +34,10 @@ public RestClient forestClientApi(ProvidersConfiguration providersConfiguration) * @return the configured RestClient instance for the Open Maps API */ @Bean - public RestClient openMapsApi() { + public RestClient openMapsApi(SilvaConfiguration silvaConfiguration) { return RestClient .builder() - .baseUrl("https://openmaps.gov.bc.ca/geo/ows") + .baseUrl(silvaConfiguration.getOpenMaps().getAddress()) .defaultHeader("Content-Type", MediaType.APPLICATION_JSON_VALUE) .build(); } diff --git a/backend/src/main/java/ca/bc/gov/restapi/results/common/configuration/ProvidersConfiguration.java b/backend/src/main/java/ca/bc/gov/restapi/results/common/configuration/ProvidersConfiguration.java deleted file mode 100644 index 94764295..00000000 --- a/backend/src/main/java/ca/bc/gov/restapi/results/common/configuration/ProvidersConfiguration.java +++ /dev/null @@ -1,17 +0,0 @@ -package ca.bc.gov.restapi.results.common.configuration; - -import lombok.Getter; -import org.springframework.beans.factory.annotation.Value; -import org.springframework.context.annotation.Configuration; - -/** This class contains configurations for all external APIs like address and keys. */ -@Getter -@Configuration -public class ProvidersConfiguration { - - @Value("${forest-client-api.address}") - private String forestClientBaseUri; - - @Value("${forest-client-api.key}") - private String forestClientApiKey; -} diff --git a/backend/src/main/java/ca/bc/gov/restapi/results/common/configuration/SecurityConfiguration.java b/backend/src/main/java/ca/bc/gov/restapi/results/common/configuration/SecurityConfiguration.java index 41373c72..9410eef8 100644 --- a/backend/src/main/java/ca/bc/gov/restapi/results/common/configuration/SecurityConfiguration.java +++ b/backend/src/main/java/ca/bc/gov/restapi/results/common/configuration/SecurityConfiguration.java @@ -42,7 +42,7 @@ public SecurityFilterChain filterChain(HttpSecurity http) throws Exception { http.cors(Customizer.withDefaults()) .csrf( customize -> - customize.csrfTokenRepository(CookieCsrfTokenRepository.withHttpOnlyFalse())) + customize.csrfTokenRepository(new CookieCsrfTokenRepository())) .authorizeHttpRequests( customize -> customize diff --git a/backend/src/main/java/ca/bc/gov/restapi/results/common/configuration/SilvaConfiguration.java b/backend/src/main/java/ca/bc/gov/restapi/results/common/configuration/SilvaConfiguration.java new file mode 100644 index 00000000..8af1dc84 --- /dev/null +++ b/backend/src/main/java/ca/bc/gov/restapi/results/common/configuration/SilvaConfiguration.java @@ -0,0 +1,46 @@ +package ca.bc.gov.restapi.results.common.configuration; + +import java.util.List; +import lombok.AllArgsConstructor; +import lombok.Builder; +import lombok.Data; +import lombok.NoArgsConstructor; +import lombok.With; +import org.springframework.boot.context.properties.ConfigurationProperties; +import org.springframework.boot.context.properties.NestedConfigurationProperty; +import org.springframework.context.annotation.Configuration; +import org.springframework.stereotype.Component; + +/** + * This class contains configurations for all external APIs like address and keys. + */ +@Data +@NoArgsConstructor +@AllArgsConstructor +@With +@Builder +@Configuration +@Component +@ConfigurationProperties("ca.bc.gov.nrs") +public class SilvaConfiguration { + + private List dashboardJobUsers; + private List wmsWhitelist; + private List orgUnits; + + @NestedConfigurationProperty + private ExternalApiAddress forestClientApi; + @NestedConfigurationProperty + private ExternalApiAddress openMaps; + + @Data + @Builder + @NoArgsConstructor + @AllArgsConstructor + public static class ExternalApiAddress { + + private String address; + private String key; + } + +} diff --git a/backend/src/main/java/ca/bc/gov/restapi/results/common/endpoint/SecretsServiceEndpoint.java b/backend/src/main/java/ca/bc/gov/restapi/results/common/endpoint/SecretsServiceEndpoint.java index 9cf98a00..31761ef5 100644 --- a/backend/src/main/java/ca/bc/gov/restapi/results/common/endpoint/SecretsServiceEndpoint.java +++ b/backend/src/main/java/ca/bc/gov/restapi/results/common/endpoint/SecretsServiceEndpoint.java @@ -1,9 +1,9 @@ package ca.bc.gov.restapi.results.common.endpoint; +import ca.bc.gov.restapi.results.common.configuration.SilvaConfiguration; import ca.bc.gov.restapi.results.common.dto.WmsLayersWhitelistUserDto; import java.util.List; -import java.util.stream.Stream; -import org.springframework.beans.factory.annotation.Value; +import lombok.RequiredArgsConstructor; import org.springframework.web.bind.annotation.GetMapping; import org.springframework.web.bind.annotation.RequestMapping; import org.springframework.web.bind.annotation.RestController; @@ -13,10 +13,10 @@ */ @RestController @RequestMapping("/api/secrets") +@RequiredArgsConstructor public class SecretsServiceEndpoint { - @Value("${nr.results.config.wms-layers.whitelist}") - private String[] wmsLayersWhitelistUsers; + private final SilvaConfiguration silvaConfiguration; /** * Get all users allowed to see the WMS layers information. @@ -25,6 +25,10 @@ public class SecretsServiceEndpoint { */ @GetMapping("/wms-layers-whitelist") public List getWmsLayersWhitelistUsers() { - return Stream.of(wmsLayersWhitelistUsers).map(WmsLayersWhitelistUserDto::new).toList(); + return silvaConfiguration + .getWmsWhitelist() + .stream() + .map(WmsLayersWhitelistUserDto::new) + .toList(); } } diff --git a/backend/src/main/java/ca/bc/gov/restapi/results/oracle/endpoint/OpeningSearchEndpoint.java b/backend/src/main/java/ca/bc/gov/restapi/results/oracle/endpoint/OpeningSearchEndpoint.java index 536413d5..8a90bdbb 100644 --- a/backend/src/main/java/ca/bc/gov/restapi/results/oracle/endpoint/OpeningSearchEndpoint.java +++ b/backend/src/main/java/ca/bc/gov/restapi/results/oracle/endpoint/OpeningSearchEndpoint.java @@ -11,6 +11,7 @@ import ca.bc.gov.restapi.results.oracle.service.OpeningService; import ca.bc.gov.restapi.results.oracle.service.OrgUnitService; import jakarta.validation.Valid; +import java.util.Arrays; import java.util.List; import lombok.RequiredArgsConstructor; import org.springframework.web.bind.annotation.GetMapping; @@ -150,6 +151,6 @@ public List getOpeningOrgUnits() { public List getOpeningOrgUnitsByCode( @RequestParam(value = "orgUnitCodes", required = true) String[] codes) { - return orgUnitService.findAllOrgUnitsByCode(codes); + return orgUnitService.findAllOrgUnitsByCode(Arrays.asList(codes)); } } diff --git a/backend/src/main/java/ca/bc/gov/restapi/results/oracle/repository/OpeningSearchRepository.java b/backend/src/main/java/ca/bc/gov/restapi/results/oracle/repository/OpeningSearchRepository.java index b56f5c32..56fb66c0 100644 --- a/backend/src/main/java/ca/bc/gov/restapi/results/oracle/repository/OpeningSearchRepository.java +++ b/backend/src/main/java/ca/bc/gov/restapi/results/oracle/repository/OpeningSearchRepository.java @@ -9,6 +9,7 @@ import ca.bc.gov.restapi.results.oracle.enums.OpeningStatusEnum; import ca.bc.gov.restapi.results.oracle.util.PaginationUtil; import jakarta.persistence.EntityManager; +import jakarta.persistence.EntityManagerFactory; import jakarta.persistence.Query; import java.math.BigDecimal; import java.sql.Timestamp; @@ -16,18 +17,21 @@ import java.util.ArrayList; import java.util.List; import java.util.Objects; -import lombok.AllArgsConstructor; import lombok.extern.slf4j.Slf4j; +import org.springframework.beans.factory.annotation.Qualifier; import org.springframework.stereotype.Component; /** This class represents the Openings Search repository database access. */ @Slf4j @Component -@AllArgsConstructor public class OpeningSearchRepository { private final EntityManager em; + public OpeningSearchRepository(@Qualifier("oracleEntityManagerFactory") EntityManagerFactory emf) { + this.em = emf.createEntityManager(); + } + /** * Search Opening with filters. * diff --git a/backend/src/main/java/ca/bc/gov/restapi/results/oracle/repository/OrgUnitRepository.java b/backend/src/main/java/ca/bc/gov/restapi/results/oracle/repository/OrgUnitRepository.java index 2f48ba89..43861125 100644 --- a/backend/src/main/java/ca/bc/gov/restapi/results/oracle/repository/OrgUnitRepository.java +++ b/backend/src/main/java/ca/bc/gov/restapi/results/oracle/repository/OrgUnitRepository.java @@ -7,5 +7,5 @@ /** This interface provides methods to get, save, and manage org unit data in the database. */ public interface OrgUnitRepository extends JpaRepository { - List findAllByOrgUnitCodeIn(String[] orgUnitCodes); + List findAllByOrgUnitCodeIn(List orgUnitCodes); } diff --git a/backend/src/main/java/ca/bc/gov/restapi/results/oracle/service/OrgUnitService.java b/backend/src/main/java/ca/bc/gov/restapi/results/oracle/service/OrgUnitService.java index 507b84f9..4df7972c 100644 --- a/backend/src/main/java/ca/bc/gov/restapi/results/oracle/service/OrgUnitService.java +++ b/backend/src/main/java/ca/bc/gov/restapi/results/oracle/service/OrgUnitService.java @@ -1,25 +1,25 @@ package ca.bc.gov.restapi.results.oracle.service; +import ca.bc.gov.restapi.results.common.configuration.SilvaConfiguration; import ca.bc.gov.restapi.results.oracle.entity.OrgUnitEntity; import ca.bc.gov.restapi.results.oracle.repository.OrgUnitRepository; import java.util.Arrays; import java.util.List; import java.util.Objects; -import lombok.AllArgsConstructor; +import lombok.RequiredArgsConstructor; import lombok.extern.slf4j.Slf4j; -import org.springframework.beans.factory.annotation.Value; import org.springframework.stereotype.Service; -/** This class contains methods to handle Org Units. */ +/** + * This class contains methods to handle Org Units. + */ @Slf4j @Service -@AllArgsConstructor +@RequiredArgsConstructor public class OrgUnitService { private final OrgUnitRepository orgUnitRepository; - - @Value("${nr.results.config.opening-search.org-units}") - private String[] orgUnitsFromProps; + private final SilvaConfiguration silvaConfiguration; /** * Find all Org Units for the Openings Search. @@ -29,12 +29,15 @@ public class OrgUnitService { public List findAllOrgUnits() { log.info("Getting all org units for the search openings"); - if (Objects.isNull(orgUnitsFromProps) || orgUnitsFromProps.length == 0) { + if (Objects.isNull(silvaConfiguration.getOrgUnits()) + || silvaConfiguration.getOrgUnits().isEmpty() + ) { log.warn("No Org Units from the properties file."); return List.of(); } - List orgUnits = orgUnitRepository.findAllByOrgUnitCodeIn(orgUnitsFromProps); + List orgUnits = orgUnitRepository.findAllByOrgUnitCodeIn( + silvaConfiguration.getOrgUnits()); log.info("Found {} org units by codes", orgUnits.size()); return orgUnits; @@ -46,8 +49,8 @@ public List findAllOrgUnits() { * @param orgUnitCodes Org Unit codes to search for. * @return List of {@link OrgUnitEntity} with found categories. */ - public List findAllOrgUnitsByCode(String[] orgUnitCodes) { - log.info("Getting all org units by codes: {}", Arrays.toString(orgUnitCodes)); + public List findAllOrgUnitsByCode(List orgUnitCodes) { + log.info("Getting all org units by codes: {}", orgUnitCodes); List orgUnits = orgUnitRepository.findAllByOrgUnitCodeIn(orgUnitCodes); log.info("Found {} org units by codes", orgUnits.size()); diff --git a/backend/src/main/java/ca/bc/gov/restapi/results/postgres/configuration/DashboardUserManagerConfiguration.java b/backend/src/main/java/ca/bc/gov/restapi/results/postgres/configuration/DashboardUserManagerConfiguration.java index 2dc9151a..23e4a3ae 100644 --- a/backend/src/main/java/ca/bc/gov/restapi/results/postgres/configuration/DashboardUserManagerConfiguration.java +++ b/backend/src/main/java/ca/bc/gov/restapi/results/postgres/configuration/DashboardUserManagerConfiguration.java @@ -13,10 +13,10 @@ @Getter @Setter @Configuration -@ConfigurationProperties("nr.results") +@ConfigurationProperties("ca.bc.gov.nrs") public class DashboardUserManagerConfiguration { - private String[] dashboardJobUsers; + private List dashboardJobUsers; /** * Gets the users in a list format. @@ -27,6 +27,6 @@ public List getUserList() { if (Objects.isNull(dashboardJobUsers)) { return List.of(); } - return List.of(dashboardJobUsers); + return dashboardJobUsers; } } diff --git a/backend/src/main/resources/application.yml b/backend/src/main/resources/application.yml index 185edb77..91e89b55 100644 --- a/backend/src/main/resources/application.yml +++ b/backend/src/main/resources/application.yml @@ -79,10 +79,6 @@ management: health: show-details: always -# Native Cloud -springdoc: - enable-native-support: true - # https://github.com/quarkusio/quarkus/wiki/Migration-Guide-3.2 quarkus: native: @@ -93,28 +89,19 @@ ca: bc: gov: nrs: + dashboard-job-users: ${DASHBOARD_JOB_IDIR_USERS:NONE} + wms-whitelist: ${WMS_LAYERS_WHITELIST_USERS:NONE} + org-units: ${OPENING_SEARCH_ORG_UNITS:DCK,DSQ,DVA,DKM,DSC,DFN,DSI,DCR,DMK,DQC,DKA,DCS,DOS,DSE,DCC,DMH,DQU,DNI,DND,DRM,DPG,DSS,DPC} + forest-client-api: + address: ${FORESTCLIENTAPI_ADDRESS:https://nr-forest-client-api-prod.api.gov.bc.ca/api} + key: ${FORESTCLIENTAPI_KEY:placeholder-api-key} + open-maps: + address: ${OPENMAPS_URL:https://openmaps.gov.bc.ca/geo/ows} oracle: keystore: ${ORACLEDB_KEYSTORE:jssecacerts-path} secret: ${ORACLEDB_SECRET:changeit} host: ${DATABASE_HOST:nrcdb03.bcgov} -# Users allowed to manually trigger the Dashboard extraction job -nr: - results: - dashboard-job-users: ${DASHBOARD_JOB_IDIR_USERS} - -# Users allowed to see and download WMS layers information - config: - wms-layers: - whitelist: ${WMS_LAYERS_WHITELIST_USERS:NONE} - opening-search: - org-units: ${OPENING_SEARCH_ORG_UNITS:-"DCK,DSQ,DVA,DKM,DSC,DFN,DSI,DCR,DMK,DQC,DKA,DCS,DOS,DSE,DCC,DMH,DQU,DNI,DND,DRM,DPG,DSS,DPC"} - -# Forest Client API -forest-client-api: - address: ${FORESTCLIENTAPI_ADDRESS:https://nr-forest-client-api-prod.api.gov.bc.ca/api} - key: ${FORESTCLIENTAPI_KEY:placeholder-api-key} - # Logging logging: level: diff --git a/backend/src/test/java/ca/bc/gov/restapi/results/common/endpoint/SecretsServiceEndpointTest.java b/backend/src/test/java/ca/bc/gov/restapi/results/common/endpoint/SecretsServiceEndpointTest.java index a76da088..6930d54d 100644 --- a/backend/src/test/java/ca/bc/gov/restapi/results/common/endpoint/SecretsServiceEndpointTest.java +++ b/backend/src/test/java/ca/bc/gov/restapi/results/common/endpoint/SecretsServiceEndpointTest.java @@ -5,19 +5,21 @@ import static org.springframework.test.web.servlet.result.MockMvcResultMatchers.jsonPath; import static org.springframework.test.web.servlet.result.MockMvcResultMatchers.status; +import ca.bc.gov.restapi.results.extensions.AbstractTestContainerIntegrationTest; +import ca.bc.gov.restapi.results.extensions.WithMockJwt; import org.junit.jupiter.api.DisplayName; import org.junit.jupiter.api.Test; import org.springframework.beans.factory.annotation.Autowired; -import org.springframework.boot.test.autoconfigure.web.servlet.WebMvcTest; +import org.springframework.boot.test.autoconfigure.web.servlet.AutoConfigureMockMvc; import org.springframework.http.MediaType; -import org.springframework.security.test.context.support.WithMockUser; import org.springframework.test.web.servlet.MockMvc; -@WebMvcTest(SecretsServiceEndpoint.class) -@WithMockUser(roles = "user_read") -class SecretsServiceEndpointTest { - - @Autowired private MockMvc mockMvc; +@WithMockJwt +@AutoConfigureMockMvc +class SecretsServiceEndpointTest extends AbstractTestContainerIntegrationTest { + + @Autowired + private MockMvc mockMvc; @Test @DisplayName("Get WMS layers whitelist users happy path should succeed") diff --git a/backend/src/test/java/ca/bc/gov/restapi/results/oracle/endpoint/OpeningSearchEndpointTest.java b/backend/src/test/java/ca/bc/gov/restapi/results/oracle/endpoint/OpeningSearchEndpointTest.java index d616cc82..517e8969 100644 --- a/backend/src/test/java/ca/bc/gov/restapi/results/oracle/endpoint/OpeningSearchEndpointTest.java +++ b/backend/src/test/java/ca/bc/gov/restapi/results/oracle/endpoint/OpeningSearchEndpointTest.java @@ -257,7 +257,7 @@ void getOpeningOrgUnitsByCode_happyPath_shouldSucceed() throws Exception { List orgUnitEntityList = List.of(orgUnit); - when(orgUnitService.findAllOrgUnitsByCode(new String[]{"DAS"})).thenReturn(orgUnitEntityList); + when(orgUnitService.findAllOrgUnitsByCode(List.of("DAS"))).thenReturn(orgUnitEntityList); mockMvc .perform( @@ -287,7 +287,7 @@ void getOpeningOrgUnitsByCode_happyPath_shouldSucceed() throws Exception { @Test @DisplayName("Get Opening Org Units By Code not Found should Succeed") void getOpeningOrgUnitsByCode_notFound_shouldSucceed() throws Exception { - when(orgUnitService.findAllOrgUnitsByCode(new String[]{"DAS"})).thenReturn(List.of()); + when(orgUnitService.findAllOrgUnitsByCode(List.of("DAS"))).thenReturn(List.of()); mockMvc .perform( diff --git a/backend/src/test/java/ca/bc/gov/restapi/results/oracle/repository/OpeningSearchRepositoryTest.java b/backend/src/test/java/ca/bc/gov/restapi/results/oracle/repository/OpeningSearchRepositoryTest.java index e5cce349..2e1c6a0f 100644 --- a/backend/src/test/java/ca/bc/gov/restapi/results/oracle/repository/OpeningSearchRepositoryTest.java +++ b/backend/src/test/java/ca/bc/gov/restapi/results/oracle/repository/OpeningSearchRepositoryTest.java @@ -10,6 +10,7 @@ import ca.bc.gov.restapi.results.oracle.enums.OpeningCategoryEnum; import ca.bc.gov.restapi.results.oracle.enums.OpeningStatusEnum; import jakarta.persistence.EntityManager; +import jakarta.persistence.EntityManagerFactory; import jakarta.persistence.FlushModeType; import jakarta.persistence.LockModeType; import jakarta.persistence.Parameter; @@ -36,6 +37,7 @@ class OpeningSearchRepositoryTest { @Mock EntityManager entityManager; + @Mock EntityManagerFactory entityManagerFactory; private OpeningSearchRepository openingSearchRepository; @@ -295,7 +297,10 @@ public T unwrap(Class cls) { @BeforeEach void setup() { - openingSearchRepository = new OpeningSearchRepository(entityManager); + + when(entityManagerFactory.createEntityManager()).thenReturn(entityManager); + + openingSearchRepository = new OpeningSearchRepository(entityManagerFactory); } @Test diff --git a/backend/src/test/java/ca/bc/gov/restapi/results/oracle/repository/OrgUnitRepositoryIntegrationTest.java b/backend/src/test/java/ca/bc/gov/restapi/results/oracle/repository/OrgUnitRepositoryIntegrationTest.java index 8c49ba44..324b32bf 100644 --- a/backend/src/test/java/ca/bc/gov/restapi/results/oracle/repository/OrgUnitRepositoryIntegrationTest.java +++ b/backend/src/test/java/ca/bc/gov/restapi/results/oracle/repository/OrgUnitRepositoryIntegrationTest.java @@ -17,7 +17,7 @@ class OrgUnitRepositoryIntegrationTest extends AbstractTestContainerIntegrationT @Test @DisplayName("Find all by org unit code unit in happy path should succeed") void findAllByOrgUnitCodeIn_happyPath_shouldSucceed() { - List list = orgUnitRepository.findAllByOrgUnitCodeIn(new String[]{"DAS"}); + List list = orgUnitRepository.findAllByOrgUnitCodeIn(List.of("DAS")); Assertions.assertNotNull(list); Assertions.assertEquals(1, list.size()); diff --git a/backend/src/test/java/ca/bc/gov/restapi/results/oracle/service/OpeningServiceTest.java b/backend/src/test/java/ca/bc/gov/restapi/results/oracle/service/OpeningServiceTest.java index ff83205a..33dbe06f 100644 --- a/backend/src/test/java/ca/bc/gov/restapi/results/oracle/service/OpeningServiceTest.java +++ b/backend/src/test/java/ca/bc/gov/restapi/results/oracle/service/OpeningServiceTest.java @@ -1,188 +1,52 @@ package ca.bc.gov.restapi.results.oracle.service; -import static org.mockito.ArgumentMatchers.any; -import static org.mockito.Mockito.when; +import static com.github.tomakehurst.wiremock.core.WireMockConfiguration.wireMockConfig; import ca.bc.gov.restapi.results.common.exception.MaxPageSizeException; import ca.bc.gov.restapi.results.common.pagination.PaginatedResult; import ca.bc.gov.restapi.results.common.pagination.PaginationParameters; -import ca.bc.gov.restapi.results.common.provider.ForestClientApiProvider; -import ca.bc.gov.restapi.results.common.security.LoggedUserService; +import ca.bc.gov.restapi.results.extensions.AbstractTestContainerIntegrationTest; +import ca.bc.gov.restapi.results.extensions.WiremockLogNotifier; +import ca.bc.gov.restapi.results.extensions.WithMockJwt; import ca.bc.gov.restapi.results.oracle.dto.OpeningSearchFiltersDto; import ca.bc.gov.restapi.results.oracle.dto.OpeningSearchResponseDto; import ca.bc.gov.restapi.results.oracle.dto.RecentOpeningDto; -import ca.bc.gov.restapi.results.oracle.entity.CutBlockOpenAdminEntity; -import ca.bc.gov.restapi.results.oracle.entity.OpeningEntity; import ca.bc.gov.restapi.results.oracle.enums.OpeningCategoryEnum; import ca.bc.gov.restapi.results.oracle.enums.OpeningStatusEnum; -import ca.bc.gov.restapi.results.oracle.repository.OpeningRepository; -import ca.bc.gov.restapi.results.oracle.repository.OpeningSearchRepository; +import com.github.tomakehurst.wiremock.junit5.WireMockExtension; import java.math.BigDecimal; import java.time.LocalDateTime; -import java.util.ArrayList; -import java.util.List; import org.junit.jupiter.api.Assertions; -import org.junit.jupiter.api.BeforeEach; import org.junit.jupiter.api.DisplayName; import org.junit.jupiter.api.Test; -import org.junit.jupiter.api.extension.ExtendWith; -import org.mockito.Mock; -import org.springframework.data.domain.Page; -import org.springframework.data.domain.PageImpl; -import org.springframework.data.domain.Pageable; -import org.springframework.test.context.junit.jupiter.SpringExtension; - -@ExtendWith(SpringExtension.class) -class OpeningServiceTest { - - @Mock OpeningRepository openingRepository; - - @Mock CutBlockOpenAdminService cutBlockOpenAdminService; - - @Mock LoggedUserService loggedUserService; - - @Mock OpeningSearchRepository openingSearchRepository; - - @Mock ForestClientApiProvider forestClientApiProvider; - +import org.junit.jupiter.api.extension.RegisterExtension; +import org.springframework.beans.factory.annotation.Autowired; + +@DisplayName("Integrated Test | Opening Service") +@WithMockJwt(value = "ttester") +class OpeningServiceTest extends AbstractTestContainerIntegrationTest { + + @RegisterExtension + static WireMockExtension clientApiStub = WireMockExtension + .newInstance() + .options( + wireMockConfig() + .port(10000) + .notifier(new WiremockLogNotifier()) + .asynchronousResponseEnabled(true) + .stubRequestLoggingDisabled(false) + ) + .configureStaticDsl(true) + .build(); + + @Autowired private OpeningService openingService; - private OpeningSearchFiltersDto mockFilter( - String orgUnit, - String category, - List statusList, - Boolean myOpenings, - Boolean submittedToFrpa, - String disturbanceDateStart, - String disturbanceDateEnd, - String regenDelayDateStart, - String regenDelayDateEnd, - String freeGrowingDateStart, - String freeGrowingDateEnd, - String updateDateStart, - String updateDateEnd, - String cuttingPermitId, - String cutBlockId, - String timberMark, - String mainSearchTerm) { - return new OpeningSearchFiltersDto( - orgUnit, - category, - statusList, - myOpenings, - submittedToFrpa, - disturbanceDateStart, - disturbanceDateEnd, - regenDelayDateStart, - regenDelayDateEnd, - freeGrowingDateStart, - freeGrowingDateEnd, - updateDateStart, - updateDateEnd, - cuttingPermitId, - cutBlockId, - timberMark, - mainSearchTerm); - } - - private OpeningSearchFiltersDto mockOrgUnit(String orgUnit) { - return mockFilter( - orgUnit, null, null, null, null, null, null, null, null, null, null, null, null, null, null, - null, null); - } - - private OpeningSearchFiltersDto mockMainFilter(String mainSearchTerm) { - return mockFilter( - null, - null, - null, - null, - null, - null, - null, - null, - null, - null, - null, - null, - null, - null, - null, - null, - mainSearchTerm); - } - - @BeforeEach - void setup() { - openingService = - new OpeningService( - openingRepository, - cutBlockOpenAdminService, - loggedUserService, - openingSearchRepository, - forestClientApiProvider); - } - - @Test - @DisplayName("Get a list of recent openings without Cut Block for the home screen") - void getRecentOpenings_openingWithoutCutBlockAdmin_shouldSucceed() { - OpeningEntity openingEntity = new OpeningEntity(); - openingEntity.setId(123L); - - List openingList = new ArrayList<>(1); - openingList.add(openingEntity); - - int pages = 3; - int pageSize = 5; - int totalResults = pageSize * pages - 1; - Pageable openingsPageable = Pageable.ofSize(pageSize); - - Page openingsPage = new PageImpl<>(openingList, openingsPageable, totalResults); - when(openingRepository.findAllByEntryUserId(any(), any())).thenReturn(openingsPage); - - when(cutBlockOpenAdminService.findAllByOpeningIdIn(any())).thenReturn(List.of()); - - int currentPage = 1; - - PaginationParameters pagination = new PaginationParameters(currentPage, pages); - PaginatedResult paginatedResult = - openingService.getRecentOpeningsCurrentUser(pagination); - - Assertions.assertNotNull(paginatedResult); - Assertions.assertEquals(currentPage, paginatedResult.getPageIndex()); - Assertions.assertEquals(pages, paginatedResult.getTotalPages()); - Assertions.assertFalse(paginatedResult.getData().isEmpty()); - Assertions.assertEquals(1, paginatedResult.getData().size()); - } - @Test @DisplayName("Get a list of recent openings for logged user") void getRecentOpenings_fetchPaginated_shouldSucceed() { - OpeningEntity openingEntity = new OpeningEntity(); - openingEntity.setId(123L); - - List openingList = new ArrayList<>(1); - openingList.add(openingEntity); - int pages = 3; - int pageSize = 5; - int totalResults = pageSize * pages - 1; - Pageable openingsPageable = Pageable.ofSize(pageSize); - - Page openingsPage = new PageImpl<>(openingList, openingsPageable, totalResults); - when(openingRepository.findAllByEntryUserId(any(), any())).thenReturn(openingsPage); - - CutBlockOpenAdminEntity cutBlock = new CutBlockOpenAdminEntity(); - cutBlock.setId(222L); - cutBlock.setForestFileId("TFL47"); - cutBlock.setCuttingPermitId("12T"); - cutBlock.setTimberMark("47/12S"); - cutBlock.setCutBlockId("12-69"); - cutBlock.setOpeningGrossArea(new BigDecimal("12.9")); - cutBlock.setOpeningId(openingEntity.getId()); - when(cutBlockOpenAdminService.findAllByOpeningIdIn(any())).thenReturn(List.of(cutBlock)); - - int currentPage = 1; + int currentPage = 0; PaginationParameters pagination = new PaginationParameters(currentPage, pages); PaginatedResult paginatedResult = @@ -190,7 +54,7 @@ void getRecentOpenings_fetchPaginated_shouldSucceed() { Assertions.assertNotNull(paginatedResult); Assertions.assertEquals(currentPage, paginatedResult.getPageIndex()); - Assertions.assertEquals(pages, paginatedResult.getTotalPages()); + Assertions.assertEquals(1, paginatedResult.getTotalPages()); Assertions.assertFalse(paginatedResult.getData().isEmpty()); Assertions.assertEquals(1, paginatedResult.getData().size()); } @@ -198,10 +62,6 @@ void getRecentOpenings_fetchPaginated_shouldSucceed() { @Test @DisplayName("Get an empty list of recent openings for the home screen") void getRecentOpenings_emptyPages_shouldSucceed() { - Page openingsPage = new PageImpl<>(new ArrayList<>()); - when(openingRepository.findAllByEntryUserId(any(), any())).thenReturn(openingsPage); - - when(cutBlockOpenAdminService.findAllByOpeningIdIn(any())).thenReturn(List.of()); int currentPage = 0; int pages = 1; @@ -211,46 +71,20 @@ void getRecentOpenings_emptyPages_shouldSucceed() { Assertions.assertNotNull(paginatedResult); Assertions.assertEquals(currentPage, paginatedResult.getPageIndex()); - Assertions.assertEquals(0, paginatedResult.getTotalPages()); - Assertions.assertTrue(paginatedResult.getData().isEmpty()); + Assertions.assertEquals(1, paginatedResult.getTotalPages()); + Assertions.assertFalse(paginatedResult.getData().isEmpty()); } @Test @DisplayName("Get a list of recent openings without user") void getRecentOpenings_fetchNoUserPaginated_shouldSucceed() { - OpeningEntity openingEntity = new OpeningEntity(); - openingEntity.setId(123L); - - List openingList = new ArrayList<>(1); - openingList.add(openingEntity); - - int pages = 3; - int pageSize = 5; - int totalResults = pageSize * pages - 1; - Pageable openingsPageable = Pageable.ofSize(pageSize); - - Page openingsPage = new PageImpl<>(openingList, openingsPageable, totalResults); - when(openingRepository.findAll(any(Pageable.class))).thenReturn(openingsPage); - CutBlockOpenAdminEntity cutBlock = new CutBlockOpenAdminEntity(); - cutBlock.setId(222L); - cutBlock.setForestFileId("TFL47"); - cutBlock.setCuttingPermitId("12T"); - cutBlock.setTimberMark("47/12S"); - cutBlock.setCutBlockId("12-69"); - cutBlock.setOpeningGrossArea(new BigDecimal("12.9")); - cutBlock.setOpeningId(openingEntity.getId()); - when(cutBlockOpenAdminService.findAllByOpeningIdIn(any())).thenReturn(List.of(cutBlock)); - - int currentPage = 1; - - PaginationParameters pagination = new PaginationParameters(currentPage, pages); PaginatedResult paginatedResult = - openingService.getRecentOpenings(pagination); + openingService.getRecentOpenings(new PaginationParameters(1, 1)); Assertions.assertNotNull(paginatedResult); - Assertions.assertEquals(currentPage, paginatedResult.getPageIndex()); - Assertions.assertEquals(pages, paginatedResult.getTotalPages()); + Assertions.assertEquals(1, paginatedResult.getPageIndex()); + Assertions.assertEquals(3, paginatedResult.getTotalPages()); Assertions.assertFalse(paginatedResult.getData().isEmpty()); Assertions.assertEquals(1, paginatedResult.getData().size()); } @@ -258,163 +92,150 @@ void getRecentOpenings_fetchNoUserPaginated_shouldSucceed() { @Test @DisplayName("Opening search file id happy path should succeed") void openingSearch_fileId_shouldSucceed() { - PaginationParameters pagination = new PaginationParameters(0, 10); - PaginatedResult paginated = new PaginatedResult<>(); - paginated.setPageIndex(pagination.page()); - paginated.setPerPage(pagination.perPage()); - paginated.setTotalPages(1); - paginated.setHasNextPage(false); - - OpeningSearchResponseDto dto = new OpeningSearchResponseDto(); - dto.setOpeningId(123456); - dto.setOpeningNumber("123"); - dto.setCategory(OpeningCategoryEnum.FTML); - dto.setStatus(OpeningStatusEnum.RET); - dto.setCuttingPermitId(""); - dto.setTimberMark(""); - dto.setCutBlockId(""); - dto.setOpeningGrossAreaHa(BigDecimal.ZERO); - dto.setDisturbanceStartDate(LocalDateTime.now()); - dto.setOrgUnitCode(""); - dto.setOrgUnitName(""); - dto.setClientNumber(""); - dto.setClientAcronym(""); - dto.setRegenDelayDate(LocalDateTime.now()); - dto.setEarlyFreeGrowingDate(LocalDateTime.now()); - dto.setLateFreeGrowingDate(LocalDateTime.now()); - dto.setUpdateTimestamp(LocalDateTime.now()); - dto.setEntryUserId(""); - dto.setSubmittedToFrpa(false); - dto.setForestFileId("TFL47"); - paginated.setData(List.of(dto)); - - OpeningSearchFiltersDto filters = mockMainFilter("407"); - when(openingSearchRepository.searchOpeningQuery(filters, pagination)).thenReturn(paginated); PaginatedResult result = - openingService.openingSearch(filters, pagination); + openingService.openingSearch(new OpeningSearchFiltersDto( + null, + null, + null, + null, + null, + null, + null, + null, + null, + null, + null, + null, + null, + null, + null, + null, + "103" + ), + new PaginationParameters(0, 10) + ); Assertions.assertNotNull(result); Assertions.assertEquals(0, result.getPageIndex()); - Assertions.assertEquals(10, result.getPerPage()); + Assertions.assertEquals(1, result.getPerPage()); Assertions.assertEquals(1, result.getTotalPages()); Assertions.assertEquals(1, result.getData().size()); - Assertions.assertEquals(dto.getOpeningId(), result.getData().get(0).getOpeningId()); - Assertions.assertEquals(dto.getOpeningNumber(), result.getData().get(0).getOpeningNumber()); - Assertions.assertEquals(dto.getCategory(), result.getData().get(0).getCategory()); - Assertions.assertEquals(dto.getStatus(), result.getData().get(0).getStatus()); - Assertions.assertEquals(dto.getCuttingPermitId(), result.getData().get(0).getCuttingPermitId()); - Assertions.assertEquals(dto.getTimberMark(), result.getData().get(0).getTimberMark()); - Assertions.assertEquals(dto.getCutBlockId(), result.getData().get(0).getCutBlockId()); - Assertions.assertEquals( - dto.getOpeningGrossAreaHa(), result.getData().get(0).getOpeningGrossAreaHa()); - Assertions.assertEquals( - dto.getDisturbanceStartDate(), result.getData().get(0).getDisturbanceStartDate()); - Assertions.assertEquals(dto.getForestFileId(), result.getData().get(0).getForestFileId()); - Assertions.assertEquals(dto.getOrgUnitCode(), result.getData().get(0).getOrgUnitCode()); - Assertions.assertEquals(dto.getOrgUnitName(), result.getData().get(0).getOrgUnitName()); - Assertions.assertEquals(dto.getClientNumber(), result.getData().get(0).getClientNumber()); - Assertions.assertEquals(dto.getRegenDelayDate(), result.getData().get(0).getRegenDelayDate()); - Assertions.assertEquals( - dto.getEarlyFreeGrowingDate(), result.getData().get(0).getEarlyFreeGrowingDate()); - Assertions.assertEquals( - dto.getLateFreeGrowingDate(), result.getData().get(0).getLateFreeGrowingDate()); - Assertions.assertEquals(dto.getEntryUserId(), result.getData().get(0).getEntryUserId()); - Assertions.assertEquals(dto.getSubmittedToFrpa(), result.getData().get(0).getSubmittedToFrpa()); + Assertions.assertEquals(103, result.getData().get(0).getOpeningId()); + Assertions.assertNull(result.getData().get(0).getOpeningNumber()); + Assertions.assertEquals(OpeningCategoryEnum.FTML, result.getData().get(0).getCategory()); + Assertions.assertEquals(OpeningStatusEnum.APP, result.getData().get(0).getStatus()); + Assertions.assertEquals("14T", result.getData().get(0).getCuttingPermitId()); + Assertions.assertEquals("49/14S", result.getData().get(0).getTimberMark()); + Assertions.assertEquals("14-71", result.getData().get(0).getCutBlockId()); + Assertions.assertEquals(new BigDecimal("14.9"), + result.getData().get(0).getOpeningGrossAreaHa()); + Assertions.assertEquals(LocalDateTime.of(2024, 1, 24, 0, 0, 0), + result.getData().get(0).getDisturbanceStartDate()); + Assertions.assertEquals("TFL49", result.getData().get(0).getForestFileId()); + Assertions.assertEquals("TWO", result.getData().get(0).getOrgUnitCode()); + Assertions.assertEquals("Org two", result.getData().get(0).getOrgUnitName()); + Assertions.assertNull(result.getData().get(0).getClientNumber()); + Assertions.assertEquals(LocalDateTime.of(2024, 1, 24, 0, 0, 0), + result.getData().get(0).getRegenDelayDate()); + Assertions.assertEquals(LocalDateTime.of(2024, 1, 24, 0, 0, 0), + result.getData().get(0).getEarlyFreeGrowingDate()); + Assertions.assertEquals(LocalDateTime.of(2024, 1, 24, 0, 0, 0), + result.getData().get(0).getLateFreeGrowingDate()); + Assertions.assertEquals("IDIR@TTESTER", result.getData().get(0).getEntryUserId()); + Assertions.assertFalse(result.getData().get(0).getSubmittedToFrpa()); Assertions.assertFalse(result.isHasNextPage()); } @Test @DisplayName("Opening search org unit happy path should succeed") void openingSearch_orgUnit_shouldSucceed() { - PaginationParameters pagination = new PaginationParameters(0, 10); - PaginatedResult paginated = new PaginatedResult<>(); - paginated.setPageIndex(pagination.page()); - paginated.setPerPage(pagination.perPage()); - paginated.setTotalPages(1); - paginated.setHasNextPage(false); - - OpeningSearchResponseDto dto = new OpeningSearchResponseDto(); - dto.setOpeningId(123456); - dto.setOpeningNumber("123"); - dto.setCategory(OpeningCategoryEnum.FTML); - dto.setStatus(OpeningStatusEnum.RET); - dto.setCuttingPermitId(""); - dto.setTimberMark(""); - dto.setCutBlockId(""); - dto.setOpeningGrossAreaHa(BigDecimal.ZERO); - dto.setDisturbanceStartDate(LocalDateTime.now()); - dto.setOrgUnitCode(""); - dto.setOrgUnitName(""); - dto.setClientNumber(""); - dto.setClientAcronym(""); - dto.setRegenDelayDate(LocalDateTime.now()); - dto.setEarlyFreeGrowingDate(LocalDateTime.now()); - dto.setLateFreeGrowingDate(LocalDateTime.now()); - dto.setUpdateTimestamp(LocalDateTime.now()); - dto.setEntryUserId(""); - dto.setSubmittedToFrpa(false); - dto.setForestFileId("TFL47"); - paginated.setData(List.of(dto)); - - OpeningSearchFiltersDto filters = mockOrgUnit("DCR"); - when(openingSearchRepository.searchOpeningQuery(filters, pagination)).thenReturn(paginated); PaginatedResult result = - openingService.openingSearch(filters, pagination); + openingService.openingSearch(new OpeningSearchFiltersDto( + "TWO", + null, + null, + null, + null, + null, + null, + null, + null, + null, + null, + null, + null, + null, + null, + null, + null + ), + new PaginationParameters(0, 10) + ); Assertions.assertNotNull(result); Assertions.assertEquals(0, result.getPageIndex()); - Assertions.assertEquals(10, result.getPerPage()); + Assertions.assertEquals(1, result.getPerPage()); Assertions.assertEquals(1, result.getTotalPages()); Assertions.assertEquals(1, result.getData().size()); - Assertions.assertEquals(dto.getOpeningId(), result.getData().get(0).getOpeningId()); - Assertions.assertEquals(dto.getOpeningNumber(), result.getData().get(0).getOpeningNumber()); - Assertions.assertEquals(dto.getCategory(), result.getData().get(0).getCategory()); - Assertions.assertEquals(dto.getStatus(), result.getData().get(0).getStatus()); - Assertions.assertEquals(dto.getCuttingPermitId(), result.getData().get(0).getCuttingPermitId()); - Assertions.assertEquals(dto.getTimberMark(), result.getData().get(0).getTimberMark()); - Assertions.assertEquals(dto.getCutBlockId(), result.getData().get(0).getCutBlockId()); - Assertions.assertEquals( - dto.getOpeningGrossAreaHa(), result.getData().get(0).getOpeningGrossAreaHa()); - Assertions.assertEquals( - dto.getDisturbanceStartDate(), result.getData().get(0).getDisturbanceStartDate()); - Assertions.assertEquals(dto.getForestFileId(), result.getData().get(0).getForestFileId()); - Assertions.assertEquals(dto.getOrgUnitCode(), result.getData().get(0).getOrgUnitCode()); - Assertions.assertEquals(dto.getOrgUnitName(), result.getData().get(0).getOrgUnitName()); - Assertions.assertEquals(dto.getClientNumber(), result.getData().get(0).getClientNumber()); - Assertions.assertEquals(dto.getRegenDelayDate(), result.getData().get(0).getRegenDelayDate()); - Assertions.assertEquals( - dto.getEarlyFreeGrowingDate(), result.getData().get(0).getEarlyFreeGrowingDate()); - Assertions.assertEquals( - dto.getLateFreeGrowingDate(), result.getData().get(0).getLateFreeGrowingDate()); - Assertions.assertEquals(dto.getEntryUserId(), result.getData().get(0).getEntryUserId()); - Assertions.assertEquals(dto.getSubmittedToFrpa(), result.getData().get(0).getSubmittedToFrpa()); + Assertions.assertEquals(103, result.getData().get(0).getOpeningId()); + Assertions.assertNull(result.getData().get(0).getOpeningNumber()); + Assertions.assertEquals(OpeningCategoryEnum.FTML, result.getData().get(0).getCategory()); + Assertions.assertEquals(OpeningStatusEnum.APP, result.getData().get(0).getStatus()); + Assertions.assertEquals("14T", result.getData().get(0).getCuttingPermitId()); + Assertions.assertEquals("49/14S", result.getData().get(0).getTimberMark()); + Assertions.assertEquals("14-71", result.getData().get(0).getCutBlockId()); + Assertions.assertEquals(new BigDecimal("14.9"), + result.getData().get(0).getOpeningGrossAreaHa()); + Assertions.assertEquals(LocalDateTime.of(2024, 1, 24, 0, 0, 0), + result.getData().get(0).getDisturbanceStartDate()); + Assertions.assertEquals("TFL49", result.getData().get(0).getForestFileId()); + Assertions.assertEquals("TWO", result.getData().get(0).getOrgUnitCode()); + Assertions.assertEquals("Org two", result.getData().get(0).getOrgUnitName()); + Assertions.assertNull(result.getData().get(0).getClientNumber()); + Assertions.assertEquals(LocalDateTime.of(2024, 1, 24, 0, 0, 0), + result.getData().get(0).getRegenDelayDate()); + Assertions.assertEquals(LocalDateTime.of(2024, 1, 24, 0, 0, 0), + result.getData().get(0).getEarlyFreeGrowingDate()); + Assertions.assertEquals(LocalDateTime.of(2024, 1, 24, 0, 0, 0), + result.getData().get(0).getLateFreeGrowingDate()); + Assertions.assertEquals("IDIR@TTESTER", result.getData().get(0).getEntryUserId()); + Assertions.assertFalse(result.getData().get(0).getSubmittedToFrpa()); Assertions.assertFalse(result.isHasNextPage()); } @Test @DisplayName("Opening search no records found should succeed") void openingSearch_noRecordsFound_shouldSucceed() { - PaginationParameters pagination = new PaginationParameters(0, 10); - PaginatedResult paginated = new PaginatedResult<>(); - paginated.setPageIndex(pagination.page()); - paginated.setPerPage(pagination.perPage()); - paginated.setTotalPages(1); - paginated.setHasNextPage(false); - paginated.setData(List.of()); - - OpeningSearchFiltersDto filters = mockOrgUnit("AAA"); - - when(openingSearchRepository.searchOpeningQuery(filters, pagination)).thenReturn(paginated); - PaginatedResult result = - openingService.openingSearch(filters, pagination); + openingService.openingSearch( + new OpeningSearchFiltersDto( + null, + null, + null, + null, + null, + null, + null, + null, + null, + null, + null, + null, + null, + null, + null, + null, + "ABCD" + ), + new PaginationParameters(0, 10) + ); Assertions.assertNotNull(result); Assertions.assertEquals(0, result.getPageIndex()); Assertions.assertEquals(10, result.getPerPage()); - Assertions.assertEquals(1, result.getTotalPages()); + Assertions.assertEquals(0, result.getTotalPages()); Assertions.assertTrue(result.getData().isEmpty()); Assertions.assertFalse(result.isHasNextPage()); } @@ -422,13 +243,31 @@ void openingSearch_noRecordsFound_shouldSucceed() { @Test @DisplayName("Opening search max page exception should fail") void openingSearch_maxPageException_shouldFail() { - OpeningSearchFiltersDto filters = mockMainFilter("407"); - PaginationParameters pagination = new PaginationParameters(0, 2999); - Assertions.assertThrows( MaxPageSizeException.class, - () -> { - openingService.openingSearch(filters, pagination); - }); + () -> + openingService.openingSearch( + new OpeningSearchFiltersDto( + null, + null, + null, + null, + null, + null, + null, + null, + null, + null, + null, + null, + null, + null, + null, + null, + "FTML" + ), + new PaginationParameters(0, 2999) + ) + ); } } diff --git a/backend/src/test/java/ca/bc/gov/restapi/results/oracle/service/OrgUnitServiceTest.java b/backend/src/test/java/ca/bc/gov/restapi/results/oracle/service/OrgUnitServiceTest.java index e3e2a618..4919f4df 100644 --- a/backend/src/test/java/ca/bc/gov/restapi/results/oracle/service/OrgUnitServiceTest.java +++ b/backend/src/test/java/ca/bc/gov/restapi/results/oracle/service/OrgUnitServiceTest.java @@ -4,6 +4,7 @@ import static org.mockito.Mockito.verify; import static org.mockito.Mockito.when; +import ca.bc.gov.restapi.results.common.configuration.SilvaConfiguration; import ca.bc.gov.restapi.results.oracle.entity.OrgUnitEntity; import ca.bc.gov.restapi.results.oracle.repository.OrgUnitRepository; import java.time.LocalDate; @@ -22,10 +23,11 @@ class OrgUnitServiceTest { @Mock OrgUnitRepository orgUnitRepository; private OrgUnitService orgUnitService; + private final SilvaConfiguration silvaConfiguration = new SilvaConfiguration().withOrgUnits(List.of("DAS")); @BeforeEach void setup() { - orgUnitService = new OrgUnitService(orgUnitRepository, new String[] {"DAS"}); + orgUnitService = new OrgUnitService(orgUnitRepository, silvaConfiguration); } @Test @@ -48,7 +50,7 @@ void findAllOrgUnits_happyPath_shouldSucceed() { orgUnit.setExpiryDate(LocalDate.now().plusYears(3L)); orgUnit.setUpdateTimestamp(LocalDate.now()); - when(orgUnitRepository.findAllByOrgUnitCodeIn(new String[] {"DAS"})) + when(orgUnitRepository.findAllByOrgUnitCodeIn(List.of("DAS"))) .thenReturn(List.of(orgUnit)); List entities = orgUnitService.findAllOrgUnits(); @@ -76,9 +78,9 @@ void findAllOrgUnits_happyPath_shouldSucceed() { @Test @DisplayName("Find all org units empty response should succeed") void findAllOrgUnits_emptyResponse_shouldSucceed() { - orgUnitService = new OrgUnitService(orgUnitRepository, new String[] {}); + orgUnitService = new OrgUnitService(orgUnitRepository, silvaConfiguration); - when(orgUnitRepository.findAllByOrgUnitCodeIn(new String[] {"DAS"})).thenReturn(List.of()); + when(orgUnitRepository.findAllByOrgUnitCodeIn(List.of("DAS"))).thenReturn(List.of()); List entities = orgUnitService.findAllOrgUnits(); Assertions.assertNotNull(entities); @@ -107,7 +109,7 @@ void findAllOrgUnitsByCode_happyPath_shouldSucceed() { orgUnit.setExpiryDate(LocalDate.now().minusYears(1L)); orgUnit.setUpdateTimestamp(LocalDate.now()); - String[] units = new String[] {"DAS"}; + List units = List.of("DAS"); when(orgUnitRepository.findAllByOrgUnitCodeIn(units)).thenReturn(List.of(orgUnit)); List entities = orgUnitService.findAllOrgUnitsByCode(units); @@ -134,7 +136,7 @@ void findAllOrgUnitsByCode_happyPath_shouldSucceed() { @Test @DisplayName("Find all org units by code not found should succeed") void findAllOrgUnitsByCode_notFound_shouldSucceed() { - String[] units = new String[] {"DAS"}; + List units = List.of("DAS"); when(orgUnitRepository.findAllByOrgUnitCodeIn(units)).thenReturn(List.of()); List entities = orgUnitService.findAllOrgUnitsByCode(units); diff --git a/backend/src/test/resources/application-default.yml b/backend/src/test/resources/application-default.yml index 67b39a04..400798cf 100644 --- a/backend/src/test/resources/application-default.yml +++ b/backend/src/test/resources/application-default.yml @@ -14,11 +14,9 @@ ca: bc: gov: nrs: - results: - wms-layers: - whitelist: ${WMS_LAYERS_WHITELIST_USERS:NONE} - opening-search: - org-units: ${OPENING_SEARCH_ORG_UNITS:DAS} + dashboard-job-users: ${DASHBOARD_JOB_IDIR_USERS:NONE} + wms-whitelist: ${WMS_LAYERS_WHITELIST_USERS:NONE} + org-units: ${OPENING_SEARCH_ORG_UNITS:DAS} forest-client-api: address: http://localhost:10000 key: 123456789abcdef