diff --git a/.github/workflows/verify-hibernate.yml b/.github/workflows/verify-hibernate.yml new file mode 100644 index 0000000000..ec0e913f05 --- /dev/null +++ b/.github/workflows/verify-hibernate.yml @@ -0,0 +1,46 @@ +name: Verify (Hibernate) + +on: + push: + branches: + - master + paths-ignore: + - '.3rd-party/**' + - 'site/**' + - '**.md' + pull_request: + paths-ignore: + - '.3rd-party/**' + - 'site/**' + - '**.md' + +jobs: + verify-hibernate: + runs-on: ubuntu-latest + + services: + rabbitmq: + image: rabbitmq:3-management-alpine + env: + RABBITMQ_DEFAULT_VHOST: / + RABBITMQ_DEFAULT_USER: guest + RABBITMQ_DEFAULT_PASS: guest + ports: + - 15672:15672 + - 5672:5672 + + steps: + - uses: actions/checkout@v4 + + - name: Set up JDK 17 + uses: actions/setup-java@v4 + with: + distribution: 'temurin' + java-version: 17 + cache: 'maven' + + - name: Check file license headers + run: mvn license:check --batch-mode + + - name: Run tests & javadoc + run: mvn verify javadoc:javadoc --batch-mode -Djpa.vendor=hibernate -Dlogging.level.org.hibernate.collection.spi.AbstractPersistentCollection=ERROR \ No newline at end of file diff --git a/hawkbit-mgmt/hawkbit-mgmt-resource/src/main/java/org/eclipse/hawkbit/mgmt/rest/resource/MgmtDistributionSetResource.java b/hawkbit-mgmt/hawkbit-mgmt-resource/src/main/java/org/eclipse/hawkbit/mgmt/rest/resource/MgmtDistributionSetResource.java index 9506948205..7ab72fb117 100644 --- a/hawkbit-mgmt/hawkbit-mgmt-resource/src/main/java/org/eclipse/hawkbit/mgmt/rest/resource/MgmtDistributionSetResource.java +++ b/hawkbit-mgmt/hawkbit-mgmt-resource/src/main/java/org/eclipse/hawkbit/mgmt/rest/resource/MgmtDistributionSetResource.java @@ -131,8 +131,7 @@ public ResponseEntity> getDistributionSets( } @Override - public ResponseEntity getDistributionSet( - final Long distributionSetId) { + public ResponseEntity getDistributionSet(final Long distributionSetId) { final DistributionSet foundDs = distributionSetManagement.getOrElseThrowException(distributionSetId); final MgmtDistributionSet response = MgmtDistributionSetMapper.toResponse(foundDs); @@ -142,8 +141,7 @@ public ResponseEntity getDistributionSet( } @Override - public ResponseEntity> createDistributionSets( - final List sets) { + public ResponseEntity> createDistributionSets(final List sets) { log.debug("creating {} distribution sets", sets.size()); // set default Ds type if ds type is null final String defaultDsKey = systemSecurityContext diff --git a/hawkbit-mgmt/hawkbit-mgmt-resource/src/test/java/org/eclipse/hawkbit/mgmt/rest/resource/MgmtDistributionSetTypeResourceTest.java b/hawkbit-mgmt/hawkbit-mgmt-resource/src/test/java/org/eclipse/hawkbit/mgmt/rest/resource/MgmtDistributionSetTypeResourceTest.java index 607a804dbd..c5328b3cde 100644 --- a/hawkbit-mgmt/hawkbit-mgmt-resource/src/test/java/org/eclipse/hawkbit/mgmt/rest/resource/MgmtDistributionSetTypeResourceTest.java +++ b/hawkbit-mgmt/hawkbit-mgmt-resource/src/test/java/org/eclipse/hawkbit/mgmt/rest/resource/MgmtDistributionSetTypeResourceTest.java @@ -746,14 +746,4 @@ private DistributionSetType generateTestType() { assertThat(testType.getMandatoryModuleTypes()).containsExactly(osType); return testType; } - - private void createSoftwareModulesAlphabetical(final int amount) { - char character = 'a'; - for (int index = 0; index < amount; index++) { - final String str = String.valueOf(character); - softwareModuleManagement.create( - entityFactory.softwareModule().create().name(str).description(str).vendor(str).version(str)); - character++; - } - } } \ No newline at end of file diff --git a/hawkbit-repository/hawkbit-repository-jpa-api/README.md b/hawkbit-repository/hawkbit-repository-jpa-api/README.md new file mode 100644 index 0000000000..dd3c4c1798 --- /dev/null +++ b/hawkbit-repository/hawkbit-repository-jpa-api/README.md @@ -0,0 +1,3 @@ +# hawkBit JPA EclipseLink Vendor integration + +Implementation of [EclipseLink](http://www.eclipse.org/eclipselink/) JPA vendor. \ No newline at end of file diff --git a/hawkbit-repository/hawkbit-repository-jpa-api/pom.xml b/hawkbit-repository/hawkbit-repository-jpa-api/pom.xml new file mode 100644 index 0000000000..246cdeccf6 --- /dev/null +++ b/hawkbit-repository/hawkbit-repository-jpa-api/pom.xml @@ -0,0 +1,48 @@ + + + 4.0.0 + + org.eclipse.hawkbit + ${revision} + hawkbit-repository + ../pom.xml + + + hawkbit-repository-jpa-api + hawkBit :: Repository :: JPA API + + + ${project.build.directory}/generated-sources/apt/ + + + + + org.eclipse.hawkbit + hawkbit-repository-core + ${project.version} + + + + org.springframework.boot + spring-boot-starter-data-jpa + + + + + org.hibernate.orm + hibernate-jpamodelgen + true + + + diff --git a/hawkbit-repository/hawkbit-repository-jpa/src/main/java/org/eclipse/hawkbit/repository/jpa/EntityInterceptor.java b/hawkbit-repository/hawkbit-repository-jpa-api/src/main/java/org/eclipse/hawkbit/repository/jpa/EntityInterceptor.java similarity index 100% rename from hawkbit-repository/hawkbit-repository-jpa/src/main/java/org/eclipse/hawkbit/repository/jpa/EntityInterceptor.java rename to hawkbit-repository/hawkbit-repository-jpa-api/src/main/java/org/eclipse/hawkbit/repository/jpa/EntityInterceptor.java diff --git a/hawkbit-repository/hawkbit-repository-jpa/src/main/java/org/eclipse/hawkbit/repository/jpa/executor/AfterTransactionCommitDefaultServiceExecutor.java b/hawkbit-repository/hawkbit-repository-jpa-api/src/main/java/org/eclipse/hawkbit/repository/jpa/executor/AfterTransactionCommitDefaultServiceExecutor.java similarity index 96% rename from hawkbit-repository/hawkbit-repository-jpa/src/main/java/org/eclipse/hawkbit/repository/jpa/executor/AfterTransactionCommitDefaultServiceExecutor.java rename to hawkbit-repository/hawkbit-repository-jpa-api/src/main/java/org/eclipse/hawkbit/repository/jpa/executor/AfterTransactionCommitDefaultServiceExecutor.java index 25d28b8fe8..69918b727a 100644 --- a/hawkbit-repository/hawkbit-repository-jpa/src/main/java/org/eclipse/hawkbit/repository/jpa/executor/AfterTransactionCommitDefaultServiceExecutor.java +++ b/hawkbit-repository/hawkbit-repository-jpa-api/src/main/java/org/eclipse/hawkbit/repository/jpa/executor/AfterTransactionCommitDefaultServiceExecutor.java @@ -45,7 +45,7 @@ public void afterCommit() { @Override @SuppressWarnings({ "squid:S1217" }) public void afterCompletion(final int status) { - log.debug("Transaction completed after commit with status {}", status == STATUS_COMMITTED ? "COMMITTED" : "ROLLEDBACK"); + log.debug("Transaction completed after commit with status {}", status == TransactionSynchronization.STATUS_COMMITTED ? "COMMITTED" : "ROLLEDBACK"); } private void afterCommit(final Runnable runnable) { diff --git a/hawkbit-repository/hawkbit-repository-jpa/src/main/java/org/eclipse/hawkbit/repository/jpa/executor/AfterTransactionCommitExecutor.java b/hawkbit-repository/hawkbit-repository-jpa-api/src/main/java/org/eclipse/hawkbit/repository/jpa/executor/AfterTransactionCommitExecutor.java similarity index 100% rename from hawkbit-repository/hawkbit-repository-jpa/src/main/java/org/eclipse/hawkbit/repository/jpa/executor/AfterTransactionCommitExecutor.java rename to hawkbit-repository/hawkbit-repository-jpa-api/src/main/java/org/eclipse/hawkbit/repository/jpa/executor/AfterTransactionCommitExecutor.java diff --git a/hawkbit-repository/hawkbit-repository-jpa-api/src/main/java/org/eclipse/hawkbit/repository/jpa/model/AbstractBaseEntity.java b/hawkbit-repository/hawkbit-repository-jpa-api/src/main/java/org/eclipse/hawkbit/repository/jpa/model/AbstractBaseEntity.java new file mode 100644 index 0000000000..4609548f61 --- /dev/null +++ b/hawkbit-repository/hawkbit-repository-jpa-api/src/main/java/org/eclipse/hawkbit/repository/jpa/model/AbstractBaseEntity.java @@ -0,0 +1,116 @@ +/** + * Copyright (c) 2015 Bosch Software Innovations GmbH and others + * + * This program and the accompanying materials are made + * available under the terms of the Eclipse Public License 2.0 + * which is available at https://www.eclipse.org/legal/epl-2.0/ + * + * SPDX-License-Identifier: EPL-2.0 + */ +package org.eclipse.hawkbit.repository.jpa.model; + +import java.io.Serializable; +import java.util.concurrent.TimeUnit; + +import jakarta.persistence.EntityListeners; +import jakarta.persistence.MappedSuperclass; +import jakarta.persistence.PostPersist; +import jakarta.persistence.PostRemove; +import jakarta.persistence.PostUpdate; + +import org.eclipse.hawkbit.repository.Identifiable; +import org.eclipse.hawkbit.repository.jpa.model.helper.AfterTransactionCommitExecutorHolder; +import org.eclipse.hawkbit.repository.model.BaseEntity; +import org.eclipse.hawkbit.tenancy.TenantAwareAuthenticationDetails; +import org.springframework.data.jpa.domain.support.AuditingEntityListener; +import org.springframework.security.core.context.SecurityContextHolder; + +/** + * Core information of all entities. + */ +@MappedSuperclass +@EntityListeners({ AuditingEntityListener.class, EntityInterceptorListener.class }) +public abstract class AbstractBaseEntity implements BaseEntity, Serializable { + + /** + * Defined equals/hashcode strategy for the repository in general is that an entity is equal if it has the same {@link #getId()} and + * {@link #getOptLockRevision()} and class. + */ + @Override + // Exception squid:S864 - generated code + @SuppressWarnings({ "squid:S864" }) + public int hashCode() { + final int prime = 31; + int result = 1; + result = prime * result + (getId() == null ? 0 : getId().hashCode()); + result = prime * result + getOptLockRevision(); + result = prime * result + getClass().getName().hashCode(); + return result; + } + + /** + * Defined equals/hashcode strategy for the repository in general is that an entity is equal if it has the same {@link #getId()} and + * {@link #getOptLockRevision()} and class. + */ + @Override + public boolean equals(final Object obj) { + if (this == obj) { + return true; + } + if (obj == null) { + return false; + } + if (!getClass().isInstance(obj)) { + return false; + } + final BaseEntity other = (BaseEntity) obj; + final Long id = getId(); + final Long otherId = other.getId(); + if (id == null) { + if (otherId != null) { + return false; + } + } else if (!id.equals(otherId)) { + return false; + } + return getOptLockRevision() == other.getOptLockRevision(); + } + + @Override + public String toString() { + return getClass().getSimpleName() + " [id=" + getId() + "]"; + } + + @PostPersist + public void postInsert() { + if (this instanceof EventAwareEntity eventAwareEntity) { + doNotify(eventAwareEntity::fireCreateEvent); + } + } + + @PostUpdate + public void postUpdate() { + if (this instanceof EventAwareEntity eventAwareEntity) { + doNotify(eventAwareEntity::fireUpdateEvent); + } + } + + @PostRemove + public void postDelete() { + if (this instanceof EventAwareEntity eventAwareEntity) { + doNotify(eventAwareEntity::fireDeleteEvent); + } + } + + protected static void doNotify(final Runnable runnable) { + // fire events onl AFTER transaction commit + AfterTransactionCommitExecutorHolder.getInstance().getAfterCommit().afterCommit(runnable); + } + + protected boolean isController() { + return SecurityContextHolder.getContext().getAuthentication() != null + && SecurityContextHolder.getContext().getAuthentication() + .getDetails() instanceof TenantAwareAuthenticationDetails tenantAwareDetails + && tenantAwareDetails.isController(); + } +} \ No newline at end of file diff --git a/hawkbit-repository/hawkbit-repository-jpa/src/main/java/org/eclipse/hawkbit/repository/jpa/model/EntityInterceptorListener.java b/hawkbit-repository/hawkbit-repository-jpa-api/src/main/java/org/eclipse/hawkbit/repository/jpa/model/EntityInterceptorListener.java similarity index 100% rename from hawkbit-repository/hawkbit-repository-jpa/src/main/java/org/eclipse/hawkbit/repository/jpa/model/EntityInterceptorListener.java rename to hawkbit-repository/hawkbit-repository-jpa-api/src/main/java/org/eclipse/hawkbit/repository/jpa/model/EntityInterceptorListener.java diff --git a/hawkbit-repository/hawkbit-repository-jpa/src/main/java/org/eclipse/hawkbit/repository/jpa/model/EventAwareEntity.java b/hawkbit-repository/hawkbit-repository-jpa-api/src/main/java/org/eclipse/hawkbit/repository/jpa/model/EventAwareEntity.java similarity index 100% rename from hawkbit-repository/hawkbit-repository-jpa/src/main/java/org/eclipse/hawkbit/repository/jpa/model/EventAwareEntity.java rename to hawkbit-repository/hawkbit-repository-jpa-api/src/main/java/org/eclipse/hawkbit/repository/jpa/model/EventAwareEntity.java diff --git a/hawkbit-repository/hawkbit-repository-jpa/src/main/java/org/eclipse/hawkbit/repository/jpa/model/helper/AfterTransactionCommitExecutorHolder.java b/hawkbit-repository/hawkbit-repository-jpa-api/src/main/java/org/eclipse/hawkbit/repository/jpa/model/helper/AfterTransactionCommitExecutorHolder.java similarity index 100% rename from hawkbit-repository/hawkbit-repository-jpa/src/main/java/org/eclipse/hawkbit/repository/jpa/model/helper/AfterTransactionCommitExecutorHolder.java rename to hawkbit-repository/hawkbit-repository-jpa-api/src/main/java/org/eclipse/hawkbit/repository/jpa/model/helper/AfterTransactionCommitExecutorHolder.java diff --git a/hawkbit-repository/hawkbit-repository-jpa/src/main/java/org/eclipse/hawkbit/repository/jpa/model/helper/EntityInterceptorHolder.java b/hawkbit-repository/hawkbit-repository-jpa-api/src/main/java/org/eclipse/hawkbit/repository/jpa/model/helper/EntityInterceptorHolder.java similarity index 100% rename from hawkbit-repository/hawkbit-repository-jpa/src/main/java/org/eclipse/hawkbit/repository/jpa/model/helper/EntityInterceptorHolder.java rename to hawkbit-repository/hawkbit-repository-jpa-api/src/main/java/org/eclipse/hawkbit/repository/jpa/model/helper/EntityInterceptorHolder.java diff --git a/hawkbit-repository/hawkbit-repository-jpa/src/main/java/org/eclipse/hawkbit/repository/jpa/model/helper/SecurityTokenGeneratorHolder.java b/hawkbit-repository/hawkbit-repository-jpa-api/src/main/java/org/eclipse/hawkbit/repository/jpa/model/helper/SecurityTokenGeneratorHolder.java similarity index 100% rename from hawkbit-repository/hawkbit-repository-jpa/src/main/java/org/eclipse/hawkbit/repository/jpa/model/helper/SecurityTokenGeneratorHolder.java rename to hawkbit-repository/hawkbit-repository-jpa-api/src/main/java/org/eclipse/hawkbit/repository/jpa/model/helper/SecurityTokenGeneratorHolder.java diff --git a/hawkbit-repository/hawkbit-repository-jpa/src/main/java/org/eclipse/hawkbit/repository/jpa/model/helper/TenantAwareHolder.java b/hawkbit-repository/hawkbit-repository-jpa-api/src/main/java/org/eclipse/hawkbit/repository/jpa/model/helper/TenantAwareHolder.java similarity index 100% rename from hawkbit-repository/hawkbit-repository-jpa/src/main/java/org/eclipse/hawkbit/repository/jpa/model/helper/TenantAwareHolder.java rename to hawkbit-repository/hawkbit-repository-jpa-api/src/main/java/org/eclipse/hawkbit/repository/jpa/model/helper/TenantAwareHolder.java diff --git a/hawkbit-repository/hawkbit-repository-jpa/src/main/java/org/eclipse/hawkbit/repository/jpa/utils/MapAttributeConverter.java b/hawkbit-repository/hawkbit-repository-jpa-api/src/main/java/org/eclipse/hawkbit/repository/jpa/utils/MapAttributeConverter.java similarity index 100% rename from hawkbit-repository/hawkbit-repository-jpa/src/main/java/org/eclipse/hawkbit/repository/jpa/utils/MapAttributeConverter.java rename to hawkbit-repository/hawkbit-repository-jpa-api/src/main/java/org/eclipse/hawkbit/repository/jpa/utils/MapAttributeConverter.java diff --git a/hawkbit-repository/hawkbit-repository-jpa-eclipselink/README.md b/hawkbit-repository/hawkbit-repository-jpa-eclipselink/README.md new file mode 100644 index 0000000000..dd3c4c1798 --- /dev/null +++ b/hawkbit-repository/hawkbit-repository-jpa-eclipselink/README.md @@ -0,0 +1,3 @@ +# hawkBit JPA EclipseLink Vendor integration + +Implementation of [EclipseLink](http://www.eclipse.org/eclipselink/) JPA vendor. \ No newline at end of file diff --git a/hawkbit-repository/hawkbit-repository-jpa-eclipselink/pom.xml b/hawkbit-repository/hawkbit-repository-jpa-eclipselink/pom.xml new file mode 100644 index 0000000000..06c16c85a1 --- /dev/null +++ b/hawkbit-repository/hawkbit-repository-jpa-eclipselink/pom.xml @@ -0,0 +1,49 @@ + + + 4.0.0 + + org.eclipse.hawkbit + ${revision} + hawkbit-repository + ../pom.xml + + + hawkbit-repository-jpa-eclipselink + hawkBit :: Repository :: JPA EclipseLink Vendor + + + ${project.build.directory}/generated-sources/apt/ + + + + + org.eclipse.hawkbit + hawkbit-repository-jpa-api + ${project.version} + + + + + org.hibernate.orm + hibernate-jpamodelgen + true + + + + org.eclipse.persistence + org.eclipse.persistence.jpa + ${eclipselink.version} + + + diff --git a/hawkbit-repository/hawkbit-repository-jpa/src/main/java/org/eclipse/hawkbit/repository/jpa/HawkbitEclipseLinkJpaDialect.java b/hawkbit-repository/hawkbit-repository-jpa-eclipselink/src/main/java/org/eclipse/hawkbit/repository/jpa/HawkbitEclipseLinkJpaDialect.java similarity index 97% rename from hawkbit-repository/hawkbit-repository-jpa/src/main/java/org/eclipse/hawkbit/repository/jpa/HawkbitEclipseLinkJpaDialect.java rename to hawkbit-repository/hawkbit-repository-jpa-eclipselink/src/main/java/org/eclipse/hawkbit/repository/jpa/HawkbitEclipseLinkJpaDialect.java index 1e0385bb09..0562a17816 100644 --- a/hawkbit-repository/hawkbit-repository-jpa/src/main/java/org/eclipse/hawkbit/repository/jpa/HawkbitEclipseLinkJpaDialect.java +++ b/hawkbit-repository/hawkbit-repository-jpa-eclipselink/src/main/java/org/eclipse/hawkbit/repository/jpa/HawkbitEclipseLinkJpaDialect.java @@ -49,7 +49,7 @@ * 3.b.) the cause is not an {@link SQLException} and as a result cannot be * mapped. */ -public class HawkbitEclipseLinkJpaDialect extends EclipseLinkJpaDialect { +class HawkbitEclipseLinkJpaDialect extends EclipseLinkJpaDialect { @Serial private static final long serialVersionUID = 1L; @@ -99,4 +99,4 @@ private static SQLException findSqlException(final RuntimeException jpaSystemExc return null; } -} +} \ No newline at end of file diff --git a/hawkbit-repository/hawkbit-repository-jpa/src/main/java/org/eclipse/hawkbit/repository/jpa/Jpa.java b/hawkbit-repository/hawkbit-repository-jpa-eclipselink/src/main/java/org/eclipse/hawkbit/repository/jpa/Jpa.java similarity index 59% rename from hawkbit-repository/hawkbit-repository-jpa/src/main/java/org/eclipse/hawkbit/repository/jpa/Jpa.java rename to hawkbit-repository/hawkbit-repository-jpa-eclipselink/src/main/java/org/eclipse/hawkbit/repository/jpa/Jpa.java index ab9c9f0fe3..ad743e032a 100644 --- a/hawkbit-repository/hawkbit-repository-jpa/src/main/java/org/eclipse/hawkbit/repository/jpa/Jpa.java +++ b/hawkbit-repository/hawkbit-repository-jpa-eclipselink/src/main/java/org/eclipse/hawkbit/repository/jpa/Jpa.java @@ -16,40 +16,35 @@ import jakarta.persistence.Query; import lombok.NoArgsConstructor; +import lombok.extern.slf4j.Slf4j; @NoArgsConstructor(access = lombok.AccessLevel.PRIVATE) +@Slf4j public class Jpa { public enum JpaVendor { ECLIPSELINK, - HIBERNATE // NOT SUPPORTED! + HIBERNATE } public static final JpaVendor JPA_VENDOR = JpaVendor.ECLIPSELINK; + static { + log.info("JPA vendor: {}", JPA_VENDOR); + } - public static char NATIVE_QUERY_PARAMETER_PREFIX = switch (JPA_VENDOR) { - case ECLIPSELINK -> '?'; - case HIBERNATE -> ':'; - }; + public static final char NATIVE_QUERY_PARAMETER_PREFIX = '?'; public static String formatNativeQueryInClause(final String name, final List list) { - return switch (Jpa.JPA_VENDOR) { - case ECLIPSELINK -> formatEclipseLinkNativeQueryInClause(IntStream.range(0, list.size()).mapToObj(i -> name + "_" + i).toList()); - case HIBERNATE -> ":" + name; - }; + return formatEclipseLinkNativeQueryInClause(IntStream.range(0, list.size()).mapToObj(i -> name + "_" + i).toList()); } public static void setNativeQueryInParameter(final Query deleteQuery, final String name, final List list) { - if (Jpa.JPA_VENDOR == Jpa.JpaVendor.ECLIPSELINK) { - for (int i = 0, len = list.size(); i < len; i++) { - deleteQuery.setParameter(name + "_" + i, list.get(i)); - } - } else if (Jpa.JPA_VENDOR == Jpa.JpaVendor.HIBERNATE) { - deleteQuery.setParameter(name, list); + for (int i = 0, len = list.size(); i < len; i++) { + deleteQuery.setParameter(name + "_" + i, list.get(i)); } } private static String formatEclipseLinkNativeQueryInClause(final Collection elements) { return "?" + String.join(",?", elements); } -} +} \ No newline at end of file diff --git a/hawkbit-repository/hawkbit-repository-jpa-eclipselink/src/main/java/org/eclipse/hawkbit/repository/jpa/JpaConfiguration.java b/hawkbit-repository/hawkbit-repository-jpa-eclipselink/src/main/java/org/eclipse/hawkbit/repository/jpa/JpaConfiguration.java new file mode 100644 index 0000000000..ecb91ff2f3 --- /dev/null +++ b/hawkbit-repository/hawkbit-repository-jpa-eclipselink/src/main/java/org/eclipse/hawkbit/repository/jpa/JpaConfiguration.java @@ -0,0 +1,86 @@ +/** + * Copyright (c) 2015 Bosch Software Innovations GmbH and others + * + * This program and the accompanying materials are made + * available under the terms of the Eclipse Public License 2.0 + * which is available at https://www.eclipse.org/legal/epl-2.0/ + * + * SPDX-License-Identifier: EPL-2.0 + */ +package org.eclipse.hawkbit.repository.jpa; + +import java.util.HashMap; +import java.util.Map; + +import javax.sql.DataSource; + +import org.eclipse.persistence.config.PersistenceUnitProperties; +import org.springframework.beans.factory.ObjectProvider; +import org.springframework.boot.autoconfigure.orm.jpa.JpaBaseConfiguration; +import org.springframework.boot.autoconfigure.orm.jpa.JpaProperties; +import org.springframework.boot.autoconfigure.transaction.TransactionManagerCustomizers; +import org.springframework.context.annotation.Bean; +import org.springframework.context.annotation.Configuration; +import org.springframework.orm.jpa.vendor.AbstractJpaVendorAdapter; +import org.springframework.orm.jpa.vendor.EclipseLinkJpaDialect; +import org.springframework.orm.jpa.vendor.EclipseLinkJpaVendorAdapter; +import org.springframework.transaction.PlatformTransactionManager; +import org.springframework.transaction.jta.JtaTransactionManager; + +/** + * General EclipseLink configuration for hawkBit's Repository. + */ +@Configuration +public class JpaConfiguration extends JpaBaseConfiguration { + + protected JpaConfiguration( + final DataSource dataSource, final JpaProperties properties, + final ObjectProvider jtaTransactionManagerProvider) { + super(dataSource, properties, jtaTransactionManagerProvider); + } + + /** + * {@link MultiTenantJpaTransactionManager} bean. + * + * @return a new {@link PlatformTransactionManager} + * @see org.springframework.boot.autoconfigure.orm.jpa.JpaBaseConfiguration#transactionManager(ObjectProvider) + */ + @Override + @Bean + public PlatformTransactionManager transactionManager(final ObjectProvider transactionManagerCustomizers) { + return new MultiTenantJpaTransactionManager(); + } + + @Override + protected AbstractJpaVendorAdapter createJpaVendorAdapter() { + return new EclipseLinkJpaVendorAdapter() { + + private final HawkbitEclipseLinkJpaDialect jpaDialect = new HawkbitEclipseLinkJpaDialect(); + + @Override + public EclipseLinkJpaDialect getJpaDialect() { + return jpaDialect; + } + }; + } + + @Override + protected Map getVendorProperties() { + final Map properties = new HashMap<>(7); + // Turn off dynamic weaving to disable LTW lookup in static weaving mode + properties.put(PersistenceUnitProperties.WEAVING, "false"); + // needed for reports + properties.put(PersistenceUnitProperties.ALLOW_NATIVE_SQL_QUERIES, "true"); + // flyway + properties.put(PersistenceUnitProperties.DDL_GENERATION, "none"); + // Embed into hawkBit logging + properties.put(PersistenceUnitProperties.LOGGING_LOGGER, "JavaLogger"); + // Ensure that we flush only at the end of the transaction + properties.put(PersistenceUnitProperties.PERSISTENCE_CONTEXT_FLUSH_MODE, "COMMIT"); + // Enable batch writing + properties.put(PersistenceUnitProperties.BATCH_WRITING, "JDBC"); + // Batch size + properties.put(PersistenceUnitProperties.BATCH_WRITING_SIZE, "500"); + return properties; + } +} \ No newline at end of file diff --git a/hawkbit-repository/hawkbit-repository-jpa-eclipselink/src/main/java/org/eclipse/hawkbit/repository/jpa/MultiTenantJpaTransactionManager.java b/hawkbit-repository/hawkbit-repository-jpa-eclipselink/src/main/java/org/eclipse/hawkbit/repository/jpa/MultiTenantJpaTransactionManager.java new file mode 100644 index 0000000000..0370b58ef6 --- /dev/null +++ b/hawkbit-repository/hawkbit-repository-jpa-eclipselink/src/main/java/org/eclipse/hawkbit/repository/jpa/MultiTenantJpaTransactionManager.java @@ -0,0 +1,91 @@ +/** + * Copyright (c) 2015 Bosch Software Innovations GmbH and others + * + * This program and the accompanying materials are made + * available under the terms of the Eclipse Public License 2.0 + * which is available at https://www.eclipse.org/legal/epl-2.0/ + * + * SPDX-License-Identifier: EPL-2.0 + */ +package org.eclipse.hawkbit.repository.jpa; + +import java.io.Serial; +import java.util.Objects; + +import jakarta.persistence.EntityManager; +import jakarta.transaction.Transaction; + +import org.eclipse.hawkbit.repository.jpa.model.EntityPropertyChangeListener; +import org.eclipse.hawkbit.tenancy.TenantAware; +import org.eclipse.persistence.config.PersistenceUnitProperties; +import org.eclipse.persistence.descriptors.ClassDescriptor; +import org.eclipse.persistence.descriptors.DescriptorEventManager; +import org.eclipse.persistence.sessions.Session; +import org.springframework.beans.factory.annotation.Autowired; +import org.springframework.orm.jpa.EntityManagerHolder; +import org.springframework.orm.jpa.JpaTransactionManager; +import org.springframework.transaction.TransactionDefinition; +import org.springframework.transaction.support.TransactionSynchronizationManager; + +/** + * {@link JpaTransactionManager} that sets the {@link TenantAware#getCurrentTenant()} in the eclipselink session. This has + * to be done in eclipselink after a {@link Transaction} has been started. + *

+ * The class also handles setting the {@link EntityPropertyChangeListener} to the {@link DescriptorEventManager} of the + */ +class MultiTenantJpaTransactionManager extends JpaTransactionManager { + + @Serial + private static final long serialVersionUID = 1L; + + @Autowired + private transient TenantAware tenantAware; + + private static final Class JPA_TARGET; + + static { + try { + JPA_TARGET = Class.forName("org.eclipse.hawkbit.repository.jpa.model.JpaTarget"); + } catch (final RuntimeException e) { + throw e; + } catch (final Exception e) { + throw new IllegalStateException(e); + } + } + + private static final EntityPropertyChangeListener ENTITY_PROPERTY_CHANGE_LISTENER = new EntityPropertyChangeListener(); + + @Override + protected void doBegin(final Object transaction, final TransactionDefinition definition) { + super.doBegin(transaction, definition); + + final EntityManager em = Objects.requireNonNull( + (EntityManagerHolder) TransactionSynchronizationManager.getResource( + Objects.requireNonNull( + getEntityManagerFactory(), + "No EntityManagerFactory provided by TransactionSynchronizationManager")), + "No EntityManagerHolder provided by TransactionSynchronizationManager") + .getEntityManager(); + + final ClassDescriptor classDescriptor = em.unwrap(Session.class).getClassDescriptor(JPA_TARGET); + if (classDescriptor != null) { + final DescriptorEventManager dem = classDescriptor.getEventManager(); + if (dem != null && !dem.getEventListeners().contains(ENTITY_PROPERTY_CHANGE_LISTENER)) { + dem.addListener(ENTITY_PROPERTY_CHANGE_LISTENER); + } + } + + final String currentTenant = tenantAware.getCurrentTenant(); + if (currentTenant == null) { + cleanupTenant(em); + } else { + em.setProperty(PersistenceUnitProperties.MULTITENANT_PROPERTY_DEFAULT, currentTenant.toUpperCase()); + } + } + + private void cleanupTenant(final EntityManager em) { + if (em.isOpen() && em.getProperties().containsKey(PersistenceUnitProperties.MULTITENANT_PROPERTY_DEFAULT)) { + em.setProperty(PersistenceUnitProperties.MULTITENANT_PROPERTY_DEFAULT, ""); + } + } +} \ No newline at end of file diff --git a/hawkbit-repository/hawkbit-repository-jpa-eclipselink/src/main/java/org/eclipse/hawkbit/repository/jpa/model/AbstractJpaBaseEntity.java b/hawkbit-repository/hawkbit-repository-jpa-eclipselink/src/main/java/org/eclipse/hawkbit/repository/jpa/model/AbstractJpaBaseEntity.java new file mode 100644 index 0000000000..0771489b04 --- /dev/null +++ b/hawkbit-repository/hawkbit-repository-jpa-eclipselink/src/main/java/org/eclipse/hawkbit/repository/jpa/model/AbstractJpaBaseEntity.java @@ -0,0 +1,119 @@ +/** + * Copyright (c) 2015 Bosch Software Innovations GmbH and others + * + * This program and the accompanying materials are made + * available under the terms of the Eclipse Public License 2.0 + * which is available at https://www.eclipse.org/legal/epl-2.0/ + * + * SPDX-License-Identifier: EPL-2.0 + */ +package org.eclipse.hawkbit.repository.jpa.model; + +import java.io.Serial; + +import jakarta.persistence.Access; +import jakarta.persistence.AccessType; +import jakarta.persistence.Column; +import jakarta.persistence.GeneratedValue; +import jakarta.persistence.GenerationType; +import jakarta.persistence.Id; +import jakarta.persistence.MappedSuperclass; +import jakarta.persistence.Version; + +import lombok.AccessLevel; +import lombok.Getter; +import lombok.NoArgsConstructor; +import lombok.Setter; +import org.springframework.data.annotation.CreatedBy; +import org.springframework.data.annotation.CreatedDate; +import org.springframework.data.annotation.LastModifiedBy; +import org.springframework.data.annotation.LastModifiedDate; + +/** + * Base hawkBit entity class containing the common attributes for EclipseLink. + */ +@NoArgsConstructor(access = AccessLevel.PROTECTED) // Default constructor needed for JPA entities. +@MappedSuperclass +// exception squid:S2160 - BaseEntity equals/hashcode is handling correctly for sub entities +@SuppressWarnings("squid:S2160") +public abstract class AbstractJpaBaseEntity extends AbstractBaseEntity { + + protected static final int USERNAME_FIELD_LENGTH = 64; + + @Serial + private static final long serialVersionUID = 1L; + + @Setter // should be used just for test purposes + @Getter + @Id + @GeneratedValue(strategy = GenerationType.IDENTITY) + @Column(name = "id") + private Long id; + + @Setter // should be used just for test purposes + @Getter + @Version + @Column(name = "optlock_revision") + private int optLockRevision; + + // Audit fields. use property access to ensure that setters will be called and checked for modification + // (touch implementation depends on setLastModifiedAt(1). + @Column(name = "created_by", updatable = false, nullable = false, length = USERNAME_FIELD_LENGTH) + private String createdBy; + @Column(name = "created_at", updatable = false, nullable = false) + private long createdAt; + @Column(name = "last_modified_by", nullable = false, length = USERNAME_FIELD_LENGTH) + private String lastModifiedBy; + @Column(name = "last_modified_at", nullable = false) + private long lastModifiedAt; + + @CreatedBy + public void setCreatedBy(final String createdBy) { + this.createdBy = createdBy; + } + + @Access(AccessType.PROPERTY) + public String getCreatedBy() { + return createdBy; + } + + @CreatedDate + public void setCreatedAt(final long createdAt) { + this.createdAt = createdAt; + } + + @Access(AccessType.PROPERTY) + public long getCreatedAt() { + return createdAt; + } + + @LastModifiedBy + public void setLastModifiedBy(final String lastModifiedBy) { + if (this.lastModifiedBy != null && isController()) { + // initialized and controller = doesn't update + return; + } + + this.lastModifiedBy = lastModifiedBy; + } + + @Access(AccessType.PROPERTY) + public String getLastModifiedBy() { + return lastModifiedBy == null ? createdBy : lastModifiedBy; + } + + @LastModifiedDate + public void setLastModifiedAt(final long lastModifiedAt) { + if (this.lastModifiedAt != 0 && isController()) { + // initialized and controller = doesn't update + return; + } + + this.lastModifiedAt = lastModifiedAt; + } + + @Access(AccessType.PROPERTY) + public long getLastModifiedAt() { + return lastModifiedAt == 0 ? createdAt : lastModifiedAt; + } +} \ No newline at end of file diff --git a/hawkbit-repository/hawkbit-repository-jpa/src/main/java/org/eclipse/hawkbit/repository/jpa/model/AbstractJpaTenantAwareBaseEntity.java b/hawkbit-repository/hawkbit-repository-jpa-eclipselink/src/main/java/org/eclipse/hawkbit/repository/jpa/model/AbstractJpaTenantAwareBaseEntity.java similarity index 90% rename from hawkbit-repository/hawkbit-repository-jpa/src/main/java/org/eclipse/hawkbit/repository/jpa/model/AbstractJpaTenantAwareBaseEntity.java rename to hawkbit-repository/hawkbit-repository-jpa-eclipselink/src/main/java/org/eclipse/hawkbit/repository/jpa/model/AbstractJpaTenantAwareBaseEntity.java index 2bee50b42f..e12ba9bb98 100644 --- a/hawkbit-repository/hawkbit-repository-jpa/src/main/java/org/eclipse/hawkbit/repository/jpa/model/AbstractJpaTenantAwareBaseEntity.java +++ b/hawkbit-repository/hawkbit-repository-jpa-eclipselink/src/main/java/org/eclipse/hawkbit/repository/jpa/model/AbstractJpaTenantAwareBaseEntity.java @@ -55,10 +55,7 @@ public abstract class AbstractJpaTenantAwareBaseEntity extends AbstractJpaBaseEn */ @Override public int hashCode() { - final int prime = 31; - int result = super.hashCode(); - result = prime * result + (tenant == null ? 0 : tenant.hashCode()); - return result; + return 31 * super.hashCode() + (tenant == null ? 0 : tenant.hashCode()); } /** @@ -73,13 +70,12 @@ public boolean equals(final Object obj) { if (!super.equals(obj)) { return false; } - final AbstractJpaTenantAwareBaseEntity other = (AbstractJpaTenantAwareBaseEntity) obj; - return Objects.equals(getTenant(), other.getTenant()); + return Objects.equals(getTenant(), ((AbstractJpaTenantAwareBaseEntity) obj).getTenant()); } @Override public String toString() { - return "BaseEntity [id=" + super.getId() + "]"; + return getClass().getSimpleName() + " [tenant=" + getTenant() + ", id=" + getId() + "]"; } /** diff --git a/hawkbit-repository/hawkbit-repository-jpa-eclipselink/src/main/java/org/eclipse/hawkbit/repository/jpa/model/EntityPropertyChangeListener.java b/hawkbit-repository/hawkbit-repository-jpa-eclipselink/src/main/java/org/eclipse/hawkbit/repository/jpa/model/EntityPropertyChangeListener.java new file mode 100644 index 0000000000..0025e575d9 --- /dev/null +++ b/hawkbit-repository/hawkbit-repository-jpa-eclipselink/src/main/java/org/eclipse/hawkbit/repository/jpa/model/EntityPropertyChangeListener.java @@ -0,0 +1,37 @@ +/** + * Copyright (c) 2015 Bosch Software Innovations GmbH and others + * + * This program and the accompanying materials are made + * available under the terms of the Eclipse Public License 2.0 + * which is available at https://www.eclipse.org/legal/epl-2.0/ + * + * SPDX-License-Identifier: EPL-2.0 + */ +package org.eclipse.hawkbit.repository.jpa.model; + +import java.util.List; + +import org.eclipse.persistence.descriptors.DescriptorEvent; +import org.eclipse.persistence.descriptors.DescriptorEventAdapter; +import org.eclipse.persistence.queries.UpdateObjectQuery; + +/** + * Listens to updates on JpaTarget entities, filtering out updates that only change the + * "lastTargetQuery" or "address" fields. + */ +public class EntityPropertyChangeListener extends DescriptorEventAdapter { + + private static final List TARGET_UPDATE_EVENT_IGNORE_FIELDS = List.of( + "lastTargetQuery", "address", // actual to be skipped + "optLockRevision", "lastModifiedAt", "lastModifiedBy" // system to be skipped + ); + + @Override + public void postUpdate(final DescriptorEvent event) { + final Object object = event.getObject(); + if (((UpdateObjectQuery) event.getQuery()).getObjectChangeSet().getChangedAttributeNames().stream() + .anyMatch(field -> !TARGET_UPDATE_EVENT_IGNORE_FIELDS.contains(field))) { + ((AbstractJpaBaseEntity)object).doNotify(() -> ((EventAwareEntity) object).fireUpdateEvent()); + } + } +} \ No newline at end of file diff --git a/hawkbit-repository/hawkbit-repository-jpa/src/test/java/org/eclipse/hawkbit/repository/jpa/HawkBitEclipseLinkJpaDialectTest.java b/hawkbit-repository/hawkbit-repository-jpa-eclipselink/src/test/java/org/eclipse/hawkbit/repository/jpa/HawkBitEclipseLinkJpaDialectTest.java similarity index 85% rename from hawkbit-repository/hawkbit-repository-jpa/src/test/java/org/eclipse/hawkbit/repository/jpa/HawkBitEclipseLinkJpaDialectTest.java rename to hawkbit-repository/hawkbit-repository-jpa-eclipselink/src/test/java/org/eclipse/hawkbit/repository/jpa/HawkBitEclipseLinkJpaDialectTest.java index c3cce1d754..524cfa20e7 100644 --- a/hawkbit-repository/hawkbit-repository-jpa/src/test/java/org/eclipse/hawkbit/repository/jpa/HawkBitEclipseLinkJpaDialectTest.java +++ b/hawkbit-repository/hawkbit-repository-jpa-eclipselink/src/test/java/org/eclipse/hawkbit/repository/jpa/HawkBitEclipseLinkJpaDialectTest.java @@ -30,22 +30,21 @@ */ @Feature("Unit Tests - Repository") @Story("Exception handling") -public class HawkBitEclipseLinkJpaDialectTest { +class HawkBitEclipseLinkJpaDialectTest { private final HawkbitEclipseLinkJpaDialect hawkBitEclipseLinkJpaDialectUnderTest = new HawkbitEclipseLinkJpaDialect(); @Test @Description("Use Case: PersistenceException that can be mapped by EclipseLinkJpaDialect into corresponding DataAccessException.") - public void jpaOptimisticLockExceptionIsConcurrencyFailureException() { - assertThat( - hawkBitEclipseLinkJpaDialectUnderTest.translateExceptionIfPossible(mock(OptimisticLockException.class))) + void jpaOptimisticLockExceptionIsConcurrencyFailureException() { + assertThat(hawkBitEclipseLinkJpaDialectUnderTest.translateExceptionIfPossible(mock(OptimisticLockException.class))) .isInstanceOf(ConcurrencyFailureException.class); } @Test @Description("Use Case: PersistenceException that could not be mapped by EclipseLinkJpaDialect directly but " + "instead is wrapped into JpaSystemException. Cause of PersistenceException is an SQLException.") - public void jpaSystemExceptionWithSqlDeadLockExceptionIsConcurrencyFailureException() { + void jpaSystemExceptionWithSqlDeadLockExceptionIsConcurrencyFailureException() { final PersistenceException persEception = mock(PersistenceException.class); when(persEception.getCause()).thenReturn(new SQLException("simulated transaction ER_LOCK_DEADLOCK", "40001")); @@ -56,7 +55,7 @@ public void jpaSystemExceptionWithSqlDeadLockExceptionIsConcurrencyFailureExcept @Test @Description("Use Case: PersistenceException that could not be mapped by EclipseLinkJpaDialect directly but instead is wrapped" + " into JpaSystemException. Cause of PersistenceException is not an SQLException.") - public void jpaSystemExceptionWithNumberFormatExceptionIsNull() { + void jpaSystemExceptionWithNumberFormatExceptionIsNull() { final PersistenceException persEception = mock(PersistenceException.class); when(persEception.getCause()).thenReturn(new NumberFormatException()); @@ -67,7 +66,7 @@ public void jpaSystemExceptionWithNumberFormatExceptionIsNull() { @Test @Description("Use Case: RuntimeException that could not be mapped by EclipseLinkJpaDialect directly. Cause of " + "RuntimeException is an SQLException.") - public void runtimeExceptionWithSqlDeadLockExceptionIsConcurrencyFailureException() { + void runtimeExceptionWithSqlDeadLockExceptionIsConcurrencyFailureException() { final RuntimeException persEception = mock(RuntimeException.class); when(persEception.getCause()).thenReturn(new SQLException("simulated transaction ER_LOCK_DEADLOCK", "40001")); @@ -78,11 +77,10 @@ public void runtimeExceptionWithSqlDeadLockExceptionIsConcurrencyFailureExceptio @Test @Description("Use Case: RuntimeException that could not be mapped by EclipseLinkJpaDialect directly. Cause of " + "RuntimeException is not an SQLException.") - public void runtimeExceptionWithNumberFormatExceptionIsNull() { + void runtimeExceptionWithNumberFormatExceptionIsNull() { final RuntimeException persEception = mock(RuntimeException.class); when(persEception.getCause()).thenReturn(new NumberFormatException()); assertThat(hawkBitEclipseLinkJpaDialectUnderTest.translateExceptionIfPossible(persEception)).isNull(); } - -} +} \ No newline at end of file diff --git a/hawkbit-repository/hawkbit-repository-jpa-hibernate/README.md b/hawkbit-repository/hawkbit-repository-jpa-hibernate/README.md new file mode 100644 index 0000000000..91e3a66a40 --- /dev/null +++ b/hawkbit-repository/hawkbit-repository-jpa-hibernate/README.md @@ -0,0 +1,25 @@ +# hawkBit JPA Hibernate Vendor integration + +Implementation of [Hibernate](https://hibernate.org/) JPA vendor. + +To use this vendor you could exclude the org.eclipse.hawkbit:hawkbit.repository-jpa-eclipselink and include this module. +For instance if you use org.eclipse.hawkbit:hawkbit-repository-jpa via org.eclipse.hawkbit:hawkbit-starter you could do it like this: + +```xml + + org.eclipse.hawkbit + hawkbit-starter + ${project.version} + + + org.eclipse.hawkbit + hawkbit-repository-jpa-eclipselink + + + + + org.eclipse.hawkbit + hawkbit-repository-jpa-hibernate + ${project.version} + +``` \ No newline at end of file diff --git a/hawkbit-repository/hawkbit-repository-jpa-hibernate/pom.xml b/hawkbit-repository/hawkbit-repository-jpa-hibernate/pom.xml new file mode 100644 index 0000000000..bef2c32a47 --- /dev/null +++ b/hawkbit-repository/hawkbit-repository-jpa-hibernate/pom.xml @@ -0,0 +1,42 @@ + + + 4.0.0 + + org.eclipse.hawkbit + ${revision} + hawkbit-repository + + + hawkbit-repository-jpa-hibernate + hawkBit :: Repository :: JPA Hibernate Vendor + + + ${project.build.directory}/generated-sources/apt/ + + + + + org.eclipse.hawkbit + hawkbit-repository-jpa-api + ${project.version} + + + + + org.hibernate.orm + hibernate-jpamodelgen + true + + + diff --git a/hawkbit-repository/hawkbit-repository-jpa-hibernate/src/main/java/org/eclipse/hawkbit/repository/jpa/Jpa.java b/hawkbit-repository/hawkbit-repository-jpa-hibernate/src/main/java/org/eclipse/hawkbit/repository/jpa/Jpa.java new file mode 100644 index 0000000000..34213c1b03 --- /dev/null +++ b/hawkbit-repository/hawkbit-repository-jpa-hibernate/src/main/java/org/eclipse/hawkbit/repository/jpa/Jpa.java @@ -0,0 +1,42 @@ +/** + * Copyright (c) 2024 Contributors to the Eclipse Foundation + * + * This program and the accompanying materials are made + * available under the terms of the Eclipse Public License 2.0 + * which is available at https://www.eclipse.org/legal/epl-2.0/ + * + * SPDX-License-Identifier: EPL-2.0 + */ +package org.eclipse.hawkbit.repository.jpa; + +import java.util.List; + +import jakarta.persistence.Query; + +import lombok.NoArgsConstructor; +import lombok.extern.slf4j.Slf4j; + +@NoArgsConstructor(access = lombok.AccessLevel.PRIVATE) +@Slf4j +public class Jpa { + + public enum JpaVendor { + ECLIPSELINK, + HIBERNATE + } + + public static final JpaVendor JPA_VENDOR = JpaVendor.HIBERNATE; + static { + log.info("JPA vendor: {}", JPA_VENDOR); + } + + public static final char NATIVE_QUERY_PARAMETER_PREFIX = ':'; + + public static String formatNativeQueryInClause(final String name, final List list) { + return ":" + name; + } + + public static void setNativeQueryInParameter(final Query deleteQuery, final String name, final List list) { + deleteQuery.setParameter(name, list); + } +} \ No newline at end of file diff --git a/hawkbit-repository/hawkbit-repository-jpa-hibernate/src/main/java/org/eclipse/hawkbit/repository/jpa/JpaConfiguration.java b/hawkbit-repository/hawkbit-repository-jpa-hibernate/src/main/java/org/eclipse/hawkbit/repository/jpa/JpaConfiguration.java new file mode 100644 index 0000000000..045b25a6c7 --- /dev/null +++ b/hawkbit-repository/hawkbit-repository-jpa-hibernate/src/main/java/org/eclipse/hawkbit/repository/jpa/JpaConfiguration.java @@ -0,0 +1,103 @@ +/** + * Copyright (c) 2015 Bosch Software Innovations GmbH and others + * + * This program and the accompanying materials are made + * available under the terms of the Eclipse Public License 2.0 + * which is available at https://www.eclipse.org/legal/epl-2.0/ + * + * SPDX-License-Identifier: EPL-2.0 + */ +package org.eclipse.hawkbit.repository.jpa; + +import java.util.Collections; +import java.util.HashMap; +import java.util.Map; + +import javax.sql.DataSource; + +import org.eclipse.hawkbit.repository.jpa.model.EntityPropertyChangeListener; + +import org.eclipse.hawkbit.tenancy.TenantAware; +import org.hibernate.boot.Metadata; +import org.hibernate.boot.spi.BootstrapContext; +import org.hibernate.cfg.AvailableSettings; +import org.hibernate.context.spi.CurrentTenantIdentifierResolver; +import org.hibernate.engine.spi.SessionFactoryImplementor; +import org.hibernate.event.service.spi.EventListenerRegistry; +import org.hibernate.event.spi.EventType; +import org.hibernate.integrator.spi.Integrator; +import org.hibernate.jpa.boot.spi.IntegratorProvider; +import org.hibernate.service.spi.SessionFactoryServiceRegistry; +import org.springframework.beans.factory.ObjectProvider; +import org.springframework.boot.autoconfigure.orm.jpa.JpaBaseConfiguration; +import org.springframework.boot.autoconfigure.orm.jpa.JpaProperties; +import org.springframework.context.annotation.Bean; +import org.springframework.context.annotation.Configuration; +import org.springframework.orm.jpa.vendor.AbstractJpaVendorAdapter; +import org.springframework.orm.jpa.vendor.HibernateJpaDialect; +import org.springframework.orm.jpa.vendor.HibernateJpaVendorAdapter; +import org.springframework.transaction.jta.JtaTransactionManager; + +/** + * General Hibernate configuration for hawkBit's Repository. + */ +@Configuration +public class JpaConfiguration extends JpaBaseConfiguration { + + private final TenantIdentifier tenantIdentifier; + + protected JpaConfiguration( + final DataSource dataSource, final JpaProperties properties, + final ObjectProvider jtaTransactionManagerProvider, + final TenantAware tenantAware) { + super(dataSource, properties, jtaTransactionManagerProvider); + tenantIdentifier = new TenantIdentifier(tenantAware); + } + + @Bean + CurrentTenantIdentifierResolver currentTenantIdentifierResolver() { + return tenantIdentifier; + } + + @Override + protected AbstractJpaVendorAdapter createJpaVendorAdapter() { + return new HibernateJpaVendorAdapter() { + + private final HibernateJpaDialect jpaDialect = new HibernateJpaDialect(); + + @Override + public HibernateJpaDialect getJpaDialect() { + return jpaDialect; + } + }; + } + + @Override + protected Map getVendorProperties() { + final Map properties = new HashMap<>(4); + + properties.put(AvailableSettings.MULTI_TENANT_IDENTIFIER_RESOLVER, tenantIdentifier); + properties.put("hibernate.multiTenancy", "DISCRIMINATOR"); + // LAZY_LOAD - Enable lazy loading of lazy fields when session is closed - N + 1 problem occur. + // So it would be good if in future hawkBit run without that + // Otherwise, if false, call for the lazy field will throw LazyInitializationException + properties.put("hibernate.enable_lazy_load_no_trans", "true"); + properties.put("hibernate.integrator_provider", (IntegratorProvider) () -> Collections.singletonList(new Integrator() { + + @Override + public void integrate( + final Metadata metadata, final BootstrapContext bootstrapContext, + final SessionFactoryImplementor sessionFactory) { + sessionFactory.getServiceRegistry() + .getService(EventListenerRegistry.class) + .appendListeners(EventType.POST_UPDATE, new EntityPropertyChangeListener()); + } + + @Override + public void disintegrate(final SessionFactoryImplementor sessionFactory, final SessionFactoryServiceRegistry serviceRegistry) { + // do nothing + } + })); + return properties; + } +} \ No newline at end of file diff --git a/hawkbit-repository/hawkbit-repository-jpa-hibernate/src/main/java/org/eclipse/hawkbit/repository/jpa/TenantIdentifier.java b/hawkbit-repository/hawkbit-repository-jpa-hibernate/src/main/java/org/eclipse/hawkbit/repository/jpa/TenantIdentifier.java new file mode 100644 index 0000000000..bcb09c7a79 --- /dev/null +++ b/hawkbit-repository/hawkbit-repository-jpa-hibernate/src/main/java/org/eclipse/hawkbit/repository/jpa/TenantIdentifier.java @@ -0,0 +1,40 @@ +/** + * Copyright (c) 2024 Contributors to the Eclipse Foundation + * + * This program and the accompanying materials are made + * available under the terms of the Eclipse Public License 2.0 + * which is available at https://www.eclipse.org/legal/epl-2.0/ + * + * SPDX-License-Identifier: EPL-2.0 + */ +package org.eclipse.hawkbit.repository.jpa; + +import java.util.Optional; + +import org.eclipse.hawkbit.tenancy.TenantAware; +import org.hibernate.context.spi.CurrentTenantIdentifierResolver; +import org.springframework.boot.autoconfigure.orm.jpa.HibernatePropertiesCustomizer; + +/** + * {@link CurrentTenantIdentifierResolver} and {@link HibernatePropertiesCustomizer} that resolves the + * {@link TenantAware#getCurrentTenant()} for hibernate. + */ +class TenantIdentifier implements CurrentTenantIdentifierResolver { + + private final TenantAware tenantAware; + + TenantIdentifier(final TenantAware tenantAware) { + this.tenantAware = tenantAware; + } + + @Override + public String resolveCurrentTenantIdentifier() { + // on bootstrapping hibernate requests tenant and want to be non-null + return Optional.ofNullable(tenantAware.getCurrentTenant()).map(String::toUpperCase).orElse(""); + } + + @Override + public boolean validateExistingCurrentSessions() { + return true; + } +} \ No newline at end of file diff --git a/hawkbit-repository/hawkbit-repository-jpa-hibernate/src/main/java/org/eclipse/hawkbit/repository/jpa/model/AbstractJpaBaseEntity.java b/hawkbit-repository/hawkbit-repository-jpa-hibernate/src/main/java/org/eclipse/hawkbit/repository/jpa/model/AbstractJpaBaseEntity.java new file mode 100644 index 0000000000..d358430678 --- /dev/null +++ b/hawkbit-repository/hawkbit-repository-jpa-hibernate/src/main/java/org/eclipse/hawkbit/repository/jpa/model/AbstractJpaBaseEntity.java @@ -0,0 +1,119 @@ +/** + * Copyright (c) 2015 Bosch Software Innovations GmbH and others + * + * This program and the accompanying materials are made + * available under the terms of the Eclipse Public License 2.0 + * which is available at https://www.eclipse.org/legal/epl-2.0/ + * + * SPDX-License-Identifier: EPL-2.0 + */ +package org.eclipse.hawkbit.repository.jpa.model; + +import java.io.Serial; + +import jakarta.persistence.Access; +import jakarta.persistence.AccessType; +import jakarta.persistence.Column; +import jakarta.persistence.GeneratedValue; +import jakarta.persistence.GenerationType; +import jakarta.persistence.Id; +import jakarta.persistence.MappedSuperclass; +import jakarta.persistence.Version; + +import lombok.AccessLevel; +import lombok.Getter; +import lombok.NoArgsConstructor; +import lombok.Setter; +import org.springframework.data.annotation.CreatedBy; +import org.springframework.data.annotation.CreatedDate; +import org.springframework.data.annotation.LastModifiedBy; +import org.springframework.data.annotation.LastModifiedDate; + +/** + * Base hawkBit entity class containing the common attributes for Hibernate. + */ +@NoArgsConstructor(access = AccessLevel.PROTECTED) // Default constructor needed for JPA entities. +@MappedSuperclass +// exception squid:S2160 - BaseEntity equals/hashcode is handling correctly for sub entities +@SuppressWarnings("squid:S2160") +public abstract class AbstractJpaBaseEntity extends AbstractBaseEntity { + + protected static final int USERNAME_FIELD_LENGTH = 64; + + @Serial + private static final long serialVersionUID = 1L; + + @Setter // should be used just for test purposes + @Getter + @Id + @GeneratedValue(strategy = GenerationType.IDENTITY) + @Column(name = "id") + private Long id; + + @Setter // should be used just for test purposes + @Getter + @Version + @Column(name = "optlock_revision") + private int optLockRevision; + + // Audit fields. use property access to ensure that setters will be called and checked for modification + // (touch implementation depends on setLastModifiedAt(1). + private String createdBy; + private long createdAt; + private String lastModifiedBy; + private long lastModifiedAt; + + @CreatedBy + public void setCreatedBy(final String createdBy) { + this.createdBy = createdBy; + } + + @Column(name = "created_by", updatable = false, nullable = false, length = USERNAME_FIELD_LENGTH) + @Access(AccessType.PROPERTY) + public String getCreatedBy() { + return createdBy; + } + + @CreatedDate + public void setCreatedAt(final long createdAt) { + this.createdAt = createdAt; + } + + @Column(name = "created_at", updatable = false, nullable = false) + @Access(AccessType.PROPERTY) + public long getCreatedAt() { + return createdAt; + } + + @LastModifiedBy + public void setLastModifiedBy(final String lastModifiedBy) { + if (this.lastModifiedBy != null && isController()) { + // initialized and controller = doesn't update + return; + } + + this.lastModifiedBy = lastModifiedBy; + } + + @Column(name = "last_modified_by", nullable = false, length = USERNAME_FIELD_LENGTH) + @Access(AccessType.PROPERTY) + public String getLastModifiedBy() { + return lastModifiedBy == null ? createdBy : lastModifiedBy; + } + + @LastModifiedDate + public void setLastModifiedAt(final long lastModifiedAt) { + if (this.lastModifiedAt != 0 && isController()) { + // initialized and controller = doesn't update + return; + } + + this.lastModifiedAt = lastModifiedAt; + } + + @Column(name = "last_modified_at", nullable = false) + @Access(AccessType.PROPERTY) + public long getLastModifiedAt() { + return lastModifiedAt == 0 ? createdAt : lastModifiedAt; + } +} \ No newline at end of file diff --git a/hawkbit-repository/hawkbit-repository-jpa-hibernate/src/main/java/org/eclipse/hawkbit/repository/jpa/model/AbstractJpaTenantAwareBaseEntity.java b/hawkbit-repository/hawkbit-repository-jpa-hibernate/src/main/java/org/eclipse/hawkbit/repository/jpa/model/AbstractJpaTenantAwareBaseEntity.java new file mode 100644 index 0000000000..8f0bbd0e23 --- /dev/null +++ b/hawkbit-repository/hawkbit-repository-jpa-hibernate/src/main/java/org/eclipse/hawkbit/repository/jpa/model/AbstractJpaTenantAwareBaseEntity.java @@ -0,0 +1,95 @@ +/** + * Copyright (c) 2015 Bosch Software Innovations GmbH and others + * + * This program and the accompanying materials are made + * available under the terms of the Eclipse Public License 2.0 + * which is available at https://www.eclipse.org/legal/epl-2.0/ + * + * SPDX-License-Identifier: EPL-2.0 + */ +package org.eclipse.hawkbit.repository.jpa.model; + +import java.io.Serial; +import java.util.Objects; + +import jakarta.persistence.Column; +import jakarta.persistence.MappedSuperclass; +import jakarta.persistence.PrePersist; +import jakarta.validation.constraints.NotNull; +import jakarta.validation.constraints.Size; + +import lombok.AccessLevel; +import lombok.Getter; +import lombok.NoArgsConstructor; +import lombok.Setter; +import org.eclipse.hawkbit.repository.exception.TenantNotExistException; +import org.eclipse.hawkbit.repository.jpa.model.helper.TenantAwareHolder; +import org.eclipse.hawkbit.repository.model.TenantAwareBaseEntity; +import org.hibernate.annotations.TenantId; + +/** + * Holder of the base attributes common to all tenant aware entities. + */ +@NoArgsConstructor(access = AccessLevel.PROTECTED) // Default constructor needed for JPA entities. +@Setter +@Getter +@MappedSuperclass +public abstract class AbstractJpaTenantAwareBaseEntity extends AbstractJpaBaseEntity implements TenantAwareBaseEntity { + + @Serial + private static final long serialVersionUID = 1L; + + @Column(name = "tenant", nullable = false, insertable = true, updatable = false, length = 40) + @Size(min = 1, max = 40) + @NotNull + @TenantId // Hibernate MultiTenant support + private String tenant; + + /** + * Tenant aware entities extend the equals/hashcode strategy with the tenant name. That would allow for instance in a + * multi-schema based data separation setup to have the same primary key for different entities of different tenants. + */ + @Override + public int hashCode() { + return 31 * super.hashCode() + (tenant == null ? 0 : tenant.hashCode()); + } + + /** + * Tenant aware entities extend the equals/hashcode strategy with the tenant name. That would allow for instance in a + * multi-schema based data separation setup to have the same primary key for different entities of + * different tenants. + */ + @Override + // exception squid:S2259 - obj is checked for null in super + @SuppressWarnings("squid:S2259") + public boolean equals(final Object obj) { + if (!super.equals(obj)) { + return false; + } + return Objects.equals(getTenant(), ((AbstractJpaTenantAwareBaseEntity) obj).getTenant()); + } + + @Override + public String toString() { + return getClass().getSimpleName() + " [tenant=" + getTenant() + ", id=" + getId() + "]"; + } + + /** + * PrePersist listener method for all {@link TenantAwareBaseEntity} entities. + * + * // TODO - check if the tenant support should set tenant from context + * // TODO - should we check if tenant exists in the system? Note: seems it's not good to work with db in the listener + */ + @PrePersist + void prePersist() { + // before persisting the entity check the current ID of the tenant by using the TenantAware service + final String currentTenant = TenantAwareHolder.getInstance().getTenantAware().getCurrentTenant(); + if (currentTenant == null) { + throw new TenantNotExistException( + String.format( + "Tenant %s does not exists, cannot create entity %s with id %d", + TenantAwareHolder.getInstance().getTenantAware().getCurrentTenant(), getClass(), getId())); + } + setTenant(currentTenant.toUpperCase()); + } +} \ No newline at end of file diff --git a/hawkbit-repository/hawkbit-repository-jpa-hibernate/src/main/java/org/eclipse/hawkbit/repository/jpa/model/EntityPropertyChangeListener.java b/hawkbit-repository/hawkbit-repository-jpa-hibernate/src/main/java/org/eclipse/hawkbit/repository/jpa/model/EntityPropertyChangeListener.java new file mode 100644 index 0000000000..9b0d9201e1 --- /dev/null +++ b/hawkbit-repository/hawkbit-repository-jpa-hibernate/src/main/java/org/eclipse/hawkbit/repository/jpa/model/EntityPropertyChangeListener.java @@ -0,0 +1,68 @@ +/** + * Copyright (c) 2015 Bosch Software Innovations GmbH and others + * + * This program and the accompanying materials are made + * available under the terms of the Eclipse Public License 2.0 + * which is available at https://www.eclipse.org/legal/epl-2.0/ + * + * SPDX-License-Identifier: EPL-2.0 + */ +package org.eclipse.hawkbit.repository.jpa.model; + +import java.util.List; + +import org.hibernate.event.spi.PostUpdateEvent; +import org.hibernate.event.spi.PostUpdateEventListener; +import org.hibernate.persister.entity.EntityPersister; + +/** + * Listens to updates on JpaTarget entities, filtering out updates that only change the + * "lastTargetQuery" or "address" fields. + */ +public class EntityPropertyChangeListener implements PostUpdateEventListener { + + private static final List TARGET_UPDATE_EVENT_IGNORE_FIELDS = List.of( + "lastTargetQuery", "address", // actual to be skipped + "optLockRevision", "lastModifiedAt", "lastModifiedBy" // system to be skipped + ); + + private static final Class JPA_TARGET; + static { + try { + JPA_TARGET = Class.forName("org.eclipse.hawkbit.repository.jpa.model.JpaTarget"); + } catch (final RuntimeException e) { + throw e; + } catch (final Exception e) { + throw new IllegalStateException(e); + } + } + + @Override + public void onPostUpdate(final PostUpdateEvent event) { + if (!JPA_TARGET.isAssignableFrom(event.getEntity().getClass())) { + // only target entity updates goes through here + return; + } + + boolean lastTargetQueryChanged = false; + boolean hasNonIgnoredChanges = false; + for (int i : event.getDirtyProperties()) { + final String attribute = event.getPersister().getAttributeMapping(i).getAttributeName(); + if ("lastTargetQuery".equals(attribute)) { + lastTargetQueryChanged = true; + } else if (!TARGET_UPDATE_EVENT_IGNORE_FIELDS.contains(attribute)) { + hasNonIgnoredChanges = true; + break; + } + } + + if (hasNonIgnoredChanges || !lastTargetQueryChanged) { + ((AbstractJpaBaseEntity)event.getEntity()).doNotify(() -> ((EventAwareEntity) event.getEntity()).fireUpdateEvent()); + } + } + + @Override + public boolean requiresPostCommitHandling(final EntityPersister persister) { + return false; + } +} \ No newline at end of file diff --git a/hawkbit-repository/hawkbit-repository-jpa/pom.xml b/hawkbit-repository/hawkbit-repository-jpa/pom.xml index 80aa20d55f..efc32dabc3 100644 --- a/hawkbit-repository/hawkbit-repository-jpa/pom.xml +++ b/hawkbit-repository/hawkbit-repository-jpa/pom.xml @@ -23,32 +23,65 @@ ${project.build.directory}/generated-sources/apt/ + eclipselink + + + eclipselink + + + + !jpa.vendor + + + + + + org.eclipse.hawkbit + hawkbit-repository-jpa-eclipselink + ${project.version} + + + + + hibernate + + + jpa.vendor + hibernate + + + + + + org.eclipse.hawkbit + hawkbit-repository-jpa-hibernate + ${project.version} + + + + + org.eclipse.hawkbit - hawkbit-repository-api + hawkbit-repository-jpa-api ${project.version} + + - org.eclipse.hawkbit - hawkbit-repository-core - ${project.version} + org.hibernate.orm + hibernate-jpamodelgen + true + org.springframework spring-core - - org.springframework.boot - spring-boot-starter-data-jpa - - - org.eclipse.persistence - org.eclipse.persistence.jpa - org.springframework.security spring-security-core @@ -70,13 +103,6 @@ commons-collections4 - - - org.hibernate.orm - hibernate-jpamodelgen - true - - org.eclipse.hawkbit @@ -90,30 +116,4 @@ test - - - - - - - org.codehaus.mojo - build-helper-maven-plugin - - - add-source - generate-sources - - add-source - - - - ${apt.source.dir} - - - - - - - - diff --git a/hawkbit-repository/hawkbit-repository-jpa/src/main/java/org/eclipse/hawkbit/repository/jpa/RepositoryApplicationConfiguration.java b/hawkbit-repository/hawkbit-repository-jpa/src/main/java/org/eclipse/hawkbit/repository/jpa/RepositoryApplicationConfiguration.java index daa3531e9b..3041a733e4 100644 --- a/hawkbit-repository/hawkbit-repository-jpa/src/main/java/org/eclipse/hawkbit/repository/jpa/RepositoryApplicationConfiguration.java +++ b/hawkbit-repository/hawkbit-repository-jpa/src/main/java/org/eclipse/hawkbit/repository/jpa/RepositoryApplicationConfiguration.java @@ -9,14 +9,10 @@ */ package org.eclipse.hawkbit.repository.jpa; -import java.util.HashMap; import java.util.List; -import java.util.Map; import java.util.Optional; import java.util.concurrent.ScheduledExecutorService; -import javax.sql.DataSource; - import jakarta.persistence.EntityManager; import jakarta.validation.Validation; @@ -81,7 +77,6 @@ import org.eclipse.hawkbit.repository.jpa.builder.JpaTargetBuilder; import org.eclipse.hawkbit.repository.jpa.builder.JpaTargetFilterQueryBuilder; import org.eclipse.hawkbit.repository.jpa.builder.JpaTargetTypeBuilder; -import org.eclipse.hawkbit.repository.jpa.configuration.MultiTenantJpaTransactionManager; import org.eclipse.hawkbit.repository.jpa.event.JpaEventEntityManager; import org.eclipse.hawkbit.repository.jpa.executor.AfterTransactionCommitDefaultServiceExecutor; import org.eclipse.hawkbit.repository.jpa.executor.AfterTransactionCommitExecutor; @@ -166,7 +161,7 @@ import org.eclipse.hawkbit.tenancy.TenantAware; import org.eclipse.hawkbit.tenancy.UserAuthoritiesResolver; import org.eclipse.hawkbit.utils.TenantConfigHelper; -import org.eclipse.persistence.config.PersistenceUnitProperties; +import org.hibernate.context.spi.CurrentTenantIdentifierResolver; import org.hibernate.validator.BaseHibernateValidatorConfiguration; import org.springframework.beans.BeansException; import org.springframework.beans.factory.ObjectProvider; @@ -177,7 +172,6 @@ import org.springframework.boot.autoconfigure.condition.ConditionalOnProperty; import org.springframework.boot.autoconfigure.domain.EntityScan; import org.springframework.boot.autoconfigure.jdbc.DataSourceAutoConfiguration; -import org.springframework.boot.autoconfigure.orm.jpa.JpaBaseConfiguration; import org.springframework.boot.autoconfigure.orm.jpa.JpaProperties; import org.springframework.boot.autoconfigure.transaction.TransactionManagerCustomizers; import org.springframework.context.annotation.Bean; @@ -191,9 +185,6 @@ import org.springframework.data.jpa.repository.config.EnableJpaRepositories; import org.springframework.integration.support.locks.LockRegistry; import org.springframework.lang.NonNull; -import org.springframework.orm.jpa.vendor.AbstractJpaVendorAdapter; -import org.springframework.orm.jpa.vendor.EclipseLinkJpaDialect; -import org.springframework.orm.jpa.vendor.EclipseLinkJpaVendorAdapter; import org.springframework.retry.annotation.EnableRetry; import org.springframework.scheduling.annotation.EnableScheduling; import org.springframework.transaction.PlatformTransactionManager; @@ -213,15 +204,9 @@ @EnableRetry @EntityScan("org.eclipse.hawkbit.repository.jpa.model") @PropertySource("classpath:/hawkbit-jpa-defaults.properties") -@Import({ RepositoryDefaultConfiguration.class, DataSourceAutoConfiguration.class, - SystemManagementCacheKeyGenerator.class }) +@Import({ JpaConfiguration.class, RepositoryDefaultConfiguration.class, DataSourceAutoConfiguration.class, SystemManagementCacheKeyGenerator.class }) @AutoConfigureAfter(DataSourceAutoConfiguration.class) -public class RepositoryApplicationConfiguration extends JpaBaseConfiguration { - - protected RepositoryApplicationConfiguration(final DataSource dataSource, final JpaProperties properties, - final ObjectProvider jtaTransactionManagerProvider) { - super(dataSource, properties, jtaTransactionManagerProvider); - } +public class RepositoryApplicationConfiguration { /** * Defines the validation processor bean. @@ -234,57 +219,11 @@ public MethodValidationPostProcessor methodValidationPostProcessor() { // ValidatorFactory shall NOT be closed because after closing the generated Validator // methods shall not be called - we need the validator in future processor.setValidator(Validation.byDefaultProvider().configure() - .addProperty(BaseHibernateValidatorConfiguration.ALLOW_PARALLEL_METHODS_DEFINE_PARAMETER_CONSTRAINTS,"true") + .addProperty(BaseHibernateValidatorConfiguration.ALLOW_PARALLEL_METHODS_DEFINE_PARAMETER_CONSTRAINTS, "true") .buildValidatorFactory().getValidator()); return processor; } - /** - * {@link MultiTenantJpaTransactionManager} bean. - * - * @return a new {@link PlatformTransactionManager} - * @see org.springframework.boot.autoconfigure.orm.jpa.JpaBaseConfiguration#transactionManager(ObjectProvider) - */ - @Override - @Bean - public PlatformTransactionManager transactionManager( - final ObjectProvider transactionManagerCustomizers) { - return new MultiTenantJpaTransactionManager(); - } - - @Override - protected AbstractJpaVendorAdapter createJpaVendorAdapter() { - return new EclipseLinkJpaVendorAdapter() { - - private final HawkbitEclipseLinkJpaDialect jpaDialect = new HawkbitEclipseLinkJpaDialect(); - - @Override - public EclipseLinkJpaDialect getJpaDialect() { - return jpaDialect; - } - }; - } - - @Override - protected Map getVendorProperties() { - final Map properties = new HashMap<>(7); - // Turn off dynamic weaving to disable LTW lookup in static weaving mode - properties.put(PersistenceUnitProperties.WEAVING, "false"); - // needed for reports - properties.put(PersistenceUnitProperties.ALLOW_NATIVE_SQL_QUERIES, "true"); - // flyway - properties.put(PersistenceUnitProperties.DDL_GENERATION, "none"); - // Embed into hawkBit logging - properties.put(PersistenceUnitProperties.LOGGING_LOGGER, "JavaLogger"); - // Ensure that we flush only at the end of the transaction - properties.put(PersistenceUnitProperties.PERSISTENCE_CONTEXT_FLUSH_MODE, "COMMIT"); - // Enable batch writing - properties.put(PersistenceUnitProperties.BATCH_WRITING, "JDBC"); - // Batch size - properties.put(PersistenceUnitProperties.BATCH_WRITING_SIZE, "500"); - return properties; - } - @Bean public BeanPostProcessor entityManagerBeanPostProcessor( @Autowired(required = false) final AccessController artifactAccessController, diff --git a/hawkbit-repository/hawkbit-repository-jpa/src/main/java/org/eclipse/hawkbit/repository/jpa/configuration/MultiTenantJpaTransactionManager.java b/hawkbit-repository/hawkbit-repository-jpa/src/main/java/org/eclipse/hawkbit/repository/jpa/configuration/MultiTenantJpaTransactionManager.java deleted file mode 100644 index 3096586674..0000000000 --- a/hawkbit-repository/hawkbit-repository-jpa/src/main/java/org/eclipse/hawkbit/repository/jpa/configuration/MultiTenantJpaTransactionManager.java +++ /dev/null @@ -1,53 +0,0 @@ -/** - * Copyright (c) 2015 Bosch Software Innovations GmbH and others - * - * This program and the accompanying materials are made - * available under the terms of the Eclipse Public License 2.0 - * which is available at https://www.eclipse.org/legal/epl-2.0/ - * - * SPDX-License-Identifier: EPL-2.0 - */ -package org.eclipse.hawkbit.repository.jpa.configuration; - -import java.io.Serial; -import java.util.Objects; - -import jakarta.persistence.EntityManager; -import jakarta.persistence.EntityManagerFactory; -import jakarta.transaction.Transaction; - -import org.eclipse.hawkbit.tenancy.TenantAware; -import org.eclipse.persistence.config.PersistenceUnitProperties; -import org.springframework.beans.factory.annotation.Autowired; -import org.springframework.orm.jpa.EntityManagerHolder; -import org.springframework.orm.jpa.JpaTransactionManager; -import org.springframework.transaction.TransactionDefinition; -import org.springframework.transaction.support.TransactionSynchronizationManager; - -/** - * {@link JpaTransactionManager} that sets the {@link TenantAware#getCurrentTenant()} in the eclipselink session. This has - * to be done in eclipselink after a {@link Transaction} has been started. - */ -public class MultiTenantJpaTransactionManager extends JpaTransactionManager { - - @Serial - private static final long serialVersionUID = 1L; - - @Autowired - private transient TenantAware tenantAware; - - @Override - protected void doBegin(final Object transaction, final TransactionDefinition definition) { - super.doBegin(transaction, definition); - - final String currentTenant = tenantAware.getCurrentTenant(); - if (currentTenant != null) { - final EntityManagerFactory emFactory = Objects.requireNonNull(getEntityManagerFactory()); - final EntityManagerHolder emHolder = Objects.requireNonNull( - (EntityManagerHolder) TransactionSynchronizationManager.getResource(emFactory), - "No EntityManagerHolder provided by TransactionSynchronizationManager"); - final EntityManager em = emHolder.getEntityManager(); - em.setProperty(PersistenceUnitProperties.MULTITENANT_PROPERTY_DEFAULT, currentTenant.toUpperCase()); - } - } -} \ No newline at end of file diff --git a/hawkbit-repository/hawkbit-repository-jpa/src/main/java/org/eclipse/hawkbit/repository/jpa/management/JpaArtifactManagement.java b/hawkbit-repository/hawkbit-repository-jpa/src/main/java/org/eclipse/hawkbit/repository/jpa/management/JpaArtifactManagement.java index 9645a85d26..c4c7f3653f 100644 --- a/hawkbit-repository/hawkbit-repository-jpa/src/main/java/org/eclipse/hawkbit/repository/jpa/management/JpaArtifactManagement.java +++ b/hawkbit-repository/hawkbit-repository-jpa/src/main/java/org/eclipse/hawkbit/repository/jpa/management/JpaArtifactManagement.java @@ -23,7 +23,6 @@ import org.eclipse.hawkbit.artifact.repository.model.AbstractDbArtifact; import org.eclipse.hawkbit.artifact.repository.model.DbArtifact; import org.eclipse.hawkbit.artifact.repository.model.DbArtifactHash; -import org.eclipse.hawkbit.im.authentication.SpPermission; import org.eclipse.hawkbit.repository.ArtifactEncryptionService; import org.eclipse.hawkbit.repository.ArtifactManagement; import org.eclipse.hawkbit.repository.QuotaManagement; @@ -36,7 +35,6 @@ import org.eclipse.hawkbit.repository.exception.InvalidSHA1HashException; import org.eclipse.hawkbit.repository.exception.InvalidSHA256HashException; import org.eclipse.hawkbit.repository.jpa.EncryptionAwareDbArtifact; -import org.eclipse.hawkbit.repository.jpa.Jpa; import org.eclipse.hawkbit.repository.jpa.JpaManagementHelper; import org.eclipse.hawkbit.repository.jpa.acm.AccessController; import org.eclipse.hawkbit.repository.jpa.configuration.Constants; @@ -53,17 +51,13 @@ import org.eclipse.hawkbit.repository.model.ArtifactUpload; import org.eclipse.hawkbit.repository.model.SoftwareModule; import org.eclipse.hawkbit.tenancy.TenantAware; -import org.springframework.beans.factory.annotation.Autowired; import org.springframework.dao.ConcurrencyFailureException; import org.springframework.data.domain.Page; import org.springframework.data.domain.Pageable; import org.springframework.retry.annotation.Backoff; import org.springframework.retry.annotation.Retryable; -import org.springframework.security.access.prepost.PreAuthorize; import org.springframework.transaction.PlatformTransactionManager; import org.springframework.transaction.annotation.Transactional; -import org.springframework.transaction.support.TransactionSynchronization; -import org.springframework.transaction.support.TransactionSynchronizationManager; import org.springframework.validation.annotation.Validated; /** diff --git a/hawkbit-repository/hawkbit-repository-jpa/src/main/java/org/eclipse/hawkbit/repository/jpa/management/JpaTargetManagement.java b/hawkbit-repository/hawkbit-repository-jpa/src/main/java/org/eclipse/hawkbit/repository/jpa/management/JpaTargetManagement.java index facb408063..a3847b5b8c 100644 --- a/hawkbit-repository/hawkbit-repository-jpa/src/main/java/org/eclipse/hawkbit/repository/jpa/management/JpaTargetManagement.java +++ b/hawkbit-repository/hawkbit-repository-jpa/src/main/java/org/eclipse/hawkbit/repository/jpa/management/JpaTargetManagement.java @@ -36,7 +36,6 @@ import org.apache.commons.collections4.ListUtils; import org.eclipse.hawkbit.repository.DistributionSetManagement; import org.eclipse.hawkbit.repository.FilterParams; -import org.eclipse.hawkbit.repository.OffsetBasedPageRequest; import org.eclipse.hawkbit.repository.QuotaManagement; import org.eclipse.hawkbit.repository.TargetFields; import org.eclipse.hawkbit.repository.TargetManagement; @@ -90,7 +89,6 @@ import org.springframework.data.domain.Page; import org.springframework.data.domain.Pageable; import org.springframework.data.domain.Slice; -import org.springframework.data.domain.Sort; import org.springframework.data.jpa.domain.Specification; import org.springframework.orm.jpa.vendor.Database; import org.springframework.retry.annotation.Backoff; diff --git a/hawkbit-repository/hawkbit-repository-jpa/src/main/java/org/eclipse/hawkbit/repository/jpa/model/AbstractJpaBaseEntity.java b/hawkbit-repository/hawkbit-repository-jpa/src/main/java/org/eclipse/hawkbit/repository/jpa/model/AbstractJpaBaseEntity.java deleted file mode 100644 index 07033f753a..0000000000 --- a/hawkbit-repository/hawkbit-repository-jpa/src/main/java/org/eclipse/hawkbit/repository/jpa/model/AbstractJpaBaseEntity.java +++ /dev/null @@ -1,213 +0,0 @@ -/** - * Copyright (c) 2015 Bosch Software Innovations GmbH and others - * - * This program and the accompanying materials are made - * available under the terms of the Eclipse Public License 2.0 - * which is available at https://www.eclipse.org/legal/epl-2.0/ - * - * SPDX-License-Identifier: EPL-2.0 - */ -package org.eclipse.hawkbit.repository.jpa.model; - -import java.io.Serial; - -import jakarta.persistence.Access; -import jakarta.persistence.AccessType; -import jakarta.persistence.Column; -import jakarta.persistence.EntityListeners; -import jakarta.persistence.GeneratedValue; -import jakarta.persistence.GenerationType; -import jakarta.persistence.Id; -import jakarta.persistence.MappedSuperclass; -import jakarta.persistence.PostPersist; -import jakarta.persistence.PostRemove; -import jakarta.persistence.PostUpdate; -import jakarta.persistence.Version; - -import lombok.AccessLevel; -import lombok.Getter; -import lombok.NoArgsConstructor; -import lombok.Setter; -import org.eclipse.hawkbit.repository.jpa.model.helper.AfterTransactionCommitExecutorHolder; -import org.eclipse.hawkbit.repository.model.BaseEntity; -import org.eclipse.hawkbit.tenancy.TenantAwareAuthenticationDetails; -import org.springframework.data.annotation.CreatedBy; -import org.springframework.data.annotation.CreatedDate; -import org.springframework.data.annotation.LastModifiedBy; -import org.springframework.data.annotation.LastModifiedDate; -import org.springframework.data.jpa.domain.support.AuditingEntityListener; -import org.springframework.security.core.context.SecurityContextHolder; - -/** - * Base hawkBit entity class containing the common attributes. - */ -@NoArgsConstructor(access = AccessLevel.PROTECTED) // Default constructor needed for JPA entities. -@MappedSuperclass -@EntityListeners({ AuditingEntityListener.class, EntityInterceptorListener.class }) -public abstract class AbstractJpaBaseEntity implements BaseEntity { - - protected static final int USERNAME_FIELD_LENGTH = 64; - - @Serial - private static final long serialVersionUID = 1L; - - @Setter // should be used just for test purposes - @Getter - @Id - @GeneratedValue(strategy = GenerationType.IDENTITY) - @Column(name = "id") - private Long id; - - @Setter // should be used just for test purposes - @Getter - @Version - @Column(name = "optlock_revision") - private int optLockRevision; - - // Audit fields. use property access to ensure that setters will be called and checked for modification - // (touch implementation depends on setLastModifiedAt(1). - @Column(name = "created_by", updatable = false, nullable = false, length = USERNAME_FIELD_LENGTH) - private String createdBy; - @Column(name = "created_at", updatable = false, nullable = false) - private long createdAt; - @Column(name = "last_modified_by", nullable = false, length = USERNAME_FIELD_LENGTH) - private String lastModifiedBy; - @Column(name = "last_modified_at", nullable = false) - private long lastModifiedAt; - - @CreatedBy - public void setCreatedBy(final String createdBy) { - this.createdBy = createdBy; - } - - // maybe needed to have correct createdBy value in the database - @Access(AccessType.PROPERTY) - public String getCreatedBy() { - return createdBy; - } - - @CreatedDate - public void setCreatedAt(final long createdAt) { - this.createdAt = createdAt; - } - - // property access to make entity manager to detect touch - @Access(AccessType.PROPERTY) - public long getCreatedAt() { - return createdAt; - } - - @LastModifiedBy - public void setLastModifiedBy(final String lastModifiedBy) { - if (this.lastModifiedBy != null && isController()) { - // initialized and controller = doesn't update - return; - } - - this.lastModifiedBy = lastModifiedBy; - } - - // property access to make entity manager to detect touch - @Access(AccessType.PROPERTY) - public String getLastModifiedBy() { - return lastModifiedBy == null ? createdBy : lastModifiedBy; - } - - @LastModifiedDate - public void setLastModifiedAt(final long lastModifiedAt) { - if (this.lastModifiedAt != 0 && isController()) { - // initialized and controller = doesn't update - return; - } - - this.lastModifiedAt = lastModifiedAt; - } - - // property access to make entity manager to detect touch - @Access(AccessType.PROPERTY) - public long getLastModifiedAt() { - return lastModifiedAt == 0 ? createdAt : lastModifiedAt; - } - - /** - * Defined equals/hashcode strategy for the repository in general is that an entity is equal if it has the same {@link #getId()} and - * {@link #getOptLockRevision()} and class. - */ - @Override - // Exception squid:S864 - generated code - @SuppressWarnings({ "squid:S864" }) - public int hashCode() { - final int prime = 31; - int result = 1; - result = prime * result + (id == null ? 0 : id.hashCode()); - result = prime * result + optLockRevision; - result = prime * result + this.getClass().getName().hashCode(); - return result; - } - - /** - * Defined equals/hashcode strategy for the repository in general is that an entity is equal if it has the same {@link #getId()} and - * {@link #getOptLockRevision()} and class. - */ - @Override - public boolean equals(final Object obj) { - if (this == obj) { - return true; - } - if (obj == null) { - return false; - } - if (!(this.getClass().isInstance(obj))) { - return false; - } - final AbstractJpaBaseEntity other = (AbstractJpaBaseEntity) obj; - final Long id = getId(); - final Long otherId = other.getId(); - if (id == null) { - if (otherId != null) { - return false; - } - } else if (!id.equals(otherId)) { - return false; - } - return getOptLockRevision() == other.getOptLockRevision(); - } - - @Override - public String toString() { - return this.getClass().getSimpleName() + " [id=" + id + "]"; - } - - @PostPersist - public void postInsert() { - if (this instanceof EventAwareEntity eventAwareEntity) { - doNotify(eventAwareEntity::fireCreateEvent); - } - } - - @PostUpdate - public void postUpdate() { - if (this instanceof EventAwareEntity eventAwareEntity) { - doNotify(eventAwareEntity::fireUpdateEvent); - } - } - - @PostRemove - public void postDelete() { - if (this instanceof EventAwareEntity eventAwareEntity) { - doNotify(eventAwareEntity::fireDeleteEvent); - } - } - - protected static void doNotify(final Runnable runnable) { - // fire events onl AFTER transaction commit - AfterTransactionCommitExecutorHolder.getInstance().getAfterCommit().afterCommit(runnable); - } - - private boolean isController() { - return SecurityContextHolder.getContext().getAuthentication() != null - && SecurityContextHolder.getContext().getAuthentication() - .getDetails() instanceof TenantAwareAuthenticationDetails tenantAwareDetails - && tenantAwareDetails.isController(); - } -} \ No newline at end of file diff --git a/hawkbit-repository/hawkbit-repository-jpa/src/main/java/org/eclipse/hawkbit/repository/jpa/model/JpaAction.java b/hawkbit-repository/hawkbit-repository-jpa/src/main/java/org/eclipse/hawkbit/repository/jpa/model/JpaAction.java index 5acadac92f..e1e75585d3 100644 --- a/hawkbit-repository/hawkbit-repository-jpa/src/main/java/org/eclipse/hawkbit/repository/jpa/model/JpaAction.java +++ b/hawkbit-repository/hawkbit-repository-jpa/src/main/java/org/eclipse/hawkbit/repository/jpa/model/JpaAction.java @@ -19,8 +19,6 @@ import java.util.Map; import java.util.Optional; -import jakarta.persistence.Access; -import jakarta.persistence.AccessType; import jakarta.persistence.CascadeType; import jakarta.persistence.Column; import jakarta.persistence.ConstraintMode; @@ -37,7 +35,6 @@ import jakarta.persistence.NamedEntityGraphs; import jakarta.persistence.NamedSubgraph; import jakarta.persistence.OneToMany; -import jakarta.persistence.PostUpdate; import jakarta.persistence.Table; import jakarta.validation.constraints.Max; import jakarta.validation.constraints.Min; @@ -56,7 +53,6 @@ import org.eclipse.hawkbit.repository.model.RolloutGroup; import org.eclipse.hawkbit.repository.model.Target; import org.eclipse.hawkbit.repository.model.helper.EventPublisherHolder; -import org.springframework.data.annotation.CreatedDate; /** * JPA implementation of {@link Action}. diff --git a/hawkbit-repository/hawkbit-repository-jpa/src/main/java/org/eclipse/hawkbit/repository/jpa/model/JpaTarget.java b/hawkbit-repository/hawkbit-repository-jpa/src/main/java/org/eclipse/hawkbit/repository/jpa/model/JpaTarget.java index f662392735..eb155fdd48 100644 --- a/hawkbit-repository/hawkbit-repository-jpa/src/main/java/org/eclipse/hawkbit/repository/jpa/model/JpaTarget.java +++ b/hawkbit-repository/hawkbit-repository-jpa/src/main/java/org/eclipse/hawkbit/repository/jpa/model/JpaTarget.java @@ -16,7 +16,6 @@ import java.util.HashSet; import java.util.List; import java.util.Map; -import java.util.Objects; import java.util.Set; import jakarta.persistence.CascadeType; @@ -27,7 +26,6 @@ import jakarta.persistence.Converter; import jakarta.persistence.ElementCollection; import jakarta.persistence.Entity; -import jakarta.persistence.EntityListeners; import jakarta.persistence.FetchType; import jakarta.persistence.ForeignKey; import jakarta.persistence.Index; @@ -67,11 +65,6 @@ import org.eclipse.hawkbit.repository.model.helper.SystemSecurityContextHolder; import org.eclipse.hawkbit.repository.model.helper.TenantConfigurationManagementHolder; import org.eclipse.hawkbit.security.SystemSecurityContext; -import org.eclipse.persistence.descriptors.DescriptorEvent; -import org.eclipse.persistence.descriptors.DescriptorEventAdapter; -import org.eclipse.persistence.queries.UpdateObjectQuery; -import org.hibernate.event.spi.PreUpdateEvent; -import org.hibernate.event.spi.PreUpdateEventListener; /** * JPA implementation of {@link Target}. @@ -88,7 +81,6 @@ uniqueConstraints = @UniqueConstraint(columnNames = { "controller_id", "tenant" }, name = "uk_tenant_controller_id")) // exception squid:S2160 - BaseEntity equals/hashcode is handling correctly for sub entities @SuppressWarnings("squid:S2160") -@EntityListeners({ JpaTarget.EntityPropertyChangeListener.class }) // add listener to the listeners declared into suppers @Slf4j public class JpaTarget extends AbstractJpaNamedEntity implements Target, EventAwareEntity { @@ -330,40 +322,4 @@ public TargetUpdateStatusConverter() { ), null); } } - - /** - * Listens to updates on {@link JpaTarget} entities, Filtering out updates that only change the "lastTargetQuery" or "address" fields. - */ - public static class EntityPropertyChangeListener extends DescriptorEventAdapter implements PreUpdateEventListener { - - private static final List TARGET_UPDATE_EVENT_IGNORE_FIELDS = List.of( - "lastTargetQuery", "address", // actual to be skipped - "optLockRevision", "lastModifiedAt", "lastModifiedBy" // system to be skipped - ); - - @Override - public void postUpdate(final DescriptorEvent event) { - final Object object = event.getObject(); - if (((UpdateObjectQuery) event.getQuery()).getObjectChangeSet().getChangedAttributeNames().stream() - .anyMatch(field -> !TARGET_UPDATE_EVENT_IGNORE_FIELDS.contains(field))) { - doNotify(() -> ((EventAwareEntity) object).fireUpdateEvent()); - } - } - - @Override - public boolean onPreUpdate(final PreUpdateEvent event) { - final Object[] oldState = event.getOldState(); - final Object[] newState = event.getState(); - for (int i = 0; i < newState.length; i++) { - if (!Objects.equals(oldState[i], newState[i])) { - final String attribute = event.getPersister().getAttributeMapping(i).getAttributeName(); - if (!TARGET_UPDATE_EVENT_IGNORE_FIELDS.contains(attribute)) { - doNotify(() -> ((EventAwareEntity) event.getEntity()).fireUpdateEvent()); - break; - } - } - } - return false; - } - } } \ No newline at end of file diff --git a/hawkbit-repository/hawkbit-repository-jpa/src/main/java/org/eclipse/hawkbit/repository/jpa/rsql/RsqlParserValidationOracle.java b/hawkbit-repository/hawkbit-repository-jpa/src/main/java/org/eclipse/hawkbit/repository/jpa/rsql/RsqlParserValidationOracle.java index 52966cd804..a2afe49085 100644 --- a/hawkbit-repository/hawkbit-repository-jpa/src/main/java/org/eclipse/hawkbit/repository/jpa/rsql/RsqlParserValidationOracle.java +++ b/hawkbit-repository/hawkbit-repository-jpa/src/main/java/org/eclipse/hawkbit/repository/jpa/rsql/RsqlParserValidationOracle.java @@ -34,7 +34,6 @@ import org.eclipse.hawkbit.repository.rsql.SuggestionContext; import org.eclipse.hawkbit.repository.rsql.SyntaxErrorContext; import org.eclipse.hawkbit.repository.rsql.ValidationOracleContext; -import org.eclipse.persistence.exceptions.ConversionException; import org.springframework.orm.jpa.JpaSystemException; import org.springframework.util.CollectionUtils; @@ -75,8 +74,14 @@ public ValidationOracleContext suggest(final String rsqlQuery, final int cursorP } catch (final RSQLParameterUnsupportedFieldException | IllegalArgumentException ex) { errorContext.setErrorMessage(getCustomMessage(ex.getMessage(), null)); log.trace("Illegal argument on parsing :", ex); - } catch (@SuppressWarnings("squid:S1166") final ConversionException | JpaSystemException e) { + } catch (final JpaSystemException e) { // noop + } catch (final RuntimeException e) { + if ("org.eclipse.persistence.exceptions.ConversionException".equals(e.getClass().getName())) { + // noop + } else { + throw e; + } } return context; } diff --git a/hawkbit-repository/hawkbit-repository-jpa/src/test/java/org/eclipse/hawkbit/repository/jpa/management/DistributionSetManagementTest.java b/hawkbit-repository/hawkbit-repository-jpa/src/test/java/org/eclipse/hawkbit/repository/jpa/management/DistributionSetManagementTest.java index cadaf9be63..d9f15f9677 100644 --- a/hawkbit-repository/hawkbit-repository-jpa/src/test/java/org/eclipse/hawkbit/repository/jpa/management/DistributionSetManagementTest.java +++ b/hawkbit-repository/hawkbit-repository-jpa/src/test/java/org/eclipse/hawkbit/repository/jpa/management/DistributionSetManagementTest.java @@ -607,17 +607,15 @@ void findDistributionSetsWithoutLazy() { void lockDistributionSet() { final DistributionSet distributionSet = testdataFactory.createDistributionSet("ds-1"); assertThat( - distributionSetManagement.get(distributionSet.getId()).map(DistributionSet::isLocked) - .orElse(true)) + distributionSetManagement.get(distributionSet.getId()).map(DistributionSet::isLocked).orElse(true)) .isFalse(); distributionSetManagement.lock(distributionSet.getId()); assertThat( - distributionSetManagement.get(distributionSet.getId()).map(DistributionSet::isLocked) - .orElse(false)) + distributionSetManagement.get(distributionSet.getId()).map(DistributionSet::isLocked).orElse(false)) .isTrue(); // assert software modules are locked assertThat(distributionSet.getModules().size()).isNotEqualTo(0); - distributionSetManagement.get(distributionSet.getId()).map(DistributionSet::getModules) + distributionSetManagement.getWithDetails(distributionSet.getId()).map(DistributionSet::getModules) .orElseThrow().forEach(module -> assertThat(module.isLocked()).isTrue()); } @@ -668,7 +666,7 @@ void unlockDistributionSet() { .isFalse(); // assert software modules are not unlocked assertThat(distributionSet.getModules().size()).isNotEqualTo(0); - distributionSetManagement.get(distributionSet.getId()).map(DistributionSet::getModules) + distributionSetManagement.getWithDetails(distributionSet.getId()).map(DistributionSet::getModules) .orElseThrow().forEach(module -> assertThat(module.isLocked()).isTrue()); } @@ -688,7 +686,7 @@ void lockDistributionSetApplied() { .as("Attempt to modify a locked DS software modules should throw an exception") .isThrownBy(() -> distributionSetManagement.assignSoftwareModules( distributionSet.getId(), List.of(testdataFactory.createSoftwareModule("sm-1").getId()))); - assertThat(distributionSetManagement.get(distributionSet.getId()).get().getModules().size()) + assertThat(distributionSetManagement.getWithDetails(distributionSet.getId()).get().getModules().size()) .as("Software module shall not be added to a locked DS.") .isEqualTo(softwareModuleCount); @@ -697,7 +695,7 @@ void lockDistributionSetApplied() { .as("Attempt to modify a locked DS software modules should throw an exception") .isThrownBy(() -> distributionSetManagement.unassignSoftwareModule( distributionSet.getId(), distributionSet.getModules().stream().findFirst().get().getId())); - assertThat(distributionSetManagement.get(distributionSet.getId()).get().getModules().size()) + assertThat(distributionSetManagement.getWithDetails(distributionSet.getId()).get().getModules().size()) .as("Software module shall not be removed from a locked DS.") .isEqualTo(softwareModuleCount); } @@ -722,8 +720,7 @@ void isImplicitLockApplicableForDistributionSet() { .toList()); // assert that implicit lock locks for every skip tag skipTags.forEach(skipTag -> { - DistributionSet distributionSetWithSkipTag = - testdataFactory.createDistributionSet("ds-skip-" + skipTag.getName()); + DistributionSet distributionSetWithSkipTag = testdataFactory.createDistributionSet("ds-skip-" + skipTag.getName()); distributionSetManagement.assignTag(List.of(distributionSetWithSkipTag.getId()), skipTag.getId()); distributionSetWithSkipTag = distributionSetManagement.get(distributionSetWithSkipTag.getId()).orElseThrow(); // assert that implicit lock isn't applicable for skip tags diff --git a/hawkbit-repository/hawkbit-repository-jpa/src/test/java/org/eclipse/hawkbit/repository/jpa/rsql/RSQLToSQL.java b/hawkbit-repository/hawkbit-repository-jpa/src/test/java/org/eclipse/hawkbit/repository/jpa/rsql/RSQLToSQL.java index 895ab02d73..da4d5758d0 100644 --- a/hawkbit-repository/hawkbit-repository-jpa/src/test/java/org/eclipse/hawkbit/repository/jpa/rsql/RSQLToSQL.java +++ b/hawkbit-repository/hawkbit-repository-jpa/src/test/java/org/eclipse/hawkbit/repository/jpa/rsql/RSQLToSQL.java @@ -9,6 +9,8 @@ */ package org.eclipse.hawkbit.repository.jpa.rsql; +import java.lang.reflect.InvocationTargetException; +import java.lang.reflect.Method; import java.util.List; import jakarta.persistence.EntityManager; @@ -23,11 +25,9 @@ import cz.jirutka.rsql.parser.ast.RSQLOperators; import cz.jirutka.rsql.parser.ast.RSQLVisitor; import org.eclipse.hawkbit.repository.RsqlQueryField; +import org.eclipse.hawkbit.repository.jpa.Jpa; import org.eclipse.hawkbit.repository.rsql.RsqlConfigHolder; import org.eclipse.hawkbit.repository.rsql.VirtualPropertyReplacer; -import org.eclipse.persistence.config.PersistenceUnitProperties; -import org.eclipse.persistence.jpa.JpaQuery; -import org.eclipse.persistence.queries.DatabaseQuery; import org.springframework.orm.jpa.vendor.Database; import org.springframework.util.CollectionUtils; @@ -40,23 +40,33 @@ public RSQLToSQL(final EntityManager entityManager) { this.entityManager = entityManager; } - public & RsqlQueryField> String toSQL(final Class domainClass, final Class fieldsClass, final String rsql, - final boolean legacyRsqlVisitor) { - return createDbQuery(domainClass, fieldsClass, rsql, legacyRsqlVisitor).getSQLString(); - } - - public & RsqlQueryField> DatabaseQuery createDbQuery(final Class domainClass, final Class fieldsClass, - final String rsql, final boolean legacyRsqlVisitor) { + public & RsqlQueryField> String toSQL( + final Class domainClass, final Class fieldsClass, final String rsql, final boolean legacyRsqlVisitor) { final CriteriaQuery query = createQuery(domainClass, fieldsClass, rsql, legacyRsqlVisitor); final TypedQuery typedQuery = entityManager.createQuery(query); // executes the query - otherwise the SQL string is not generated - typedQuery.setParameter(PersistenceUnitProperties.MULTITENANT_PROPERTY_DEFAULT, "DEFAULT"); - typedQuery.getResultList(); - return typedQuery.unwrap(JpaQuery.class).getDatabaseQuery(); + if (Jpa.JPA_VENDOR.equals(Jpa.JpaVendor.ECLIPSELINK)) { + typedQuery.setParameter("eclipselink.tenant-id", "DEFAULT"); + typedQuery.getResultList(); + try { + final Class jpaQueryClass = Class.forName("org.eclipse.persistence.jpa.JpaQuery"); + final Method getDatabaseQueryMethod = jpaQueryClass.getMethod("getDatabaseQuery"); + final Method getSQLString = getDatabaseQueryMethod.getReturnType().getMethod("getSQLString"); + return (String)getSQLString.invoke(getDatabaseQueryMethod.invoke(typedQuery.unwrap(jpaQueryClass))); + } catch (final RuntimeException e) { + throw e; + } catch (final ClassNotFoundException | NoSuchMethodException | IllegalAccessException e) { + throw new UnsupportedOperationException("EclipseLink is not supported", e); + } catch (final InvocationTargetException e) { + throw e.getCause() instanceof RuntimeException ? (RuntimeException)e.getCause() : new RuntimeException(e.getCause()); + } + } else { // hibernate + throw new UnsupportedOperationException("Hibernate is not supported"); + } } - private & RsqlQueryField> CriteriaQuery createQuery(final Class domainClass, final Class fieldsClass, - final String rsql, final boolean legacyRsqlVisitor) { + private & RsqlQueryField> CriteriaQuery createQuery( + final Class domainClass, final Class fieldsClass, final String rsql, final boolean legacyRsqlVisitor) { final CriteriaQuery query = entityManager.getCriteriaBuilder().createQuery(domainClass); final CriteriaBuilder cb = entityManager.getCriteriaBuilder(); return query.where( @@ -78,16 +88,15 @@ private & RsqlQueryField> Predicate toPredicate( query.distinct(true); final RSQLVisitor, String> jpqQueryRSQLVisitor = - legacyRsqlVisitor ? - new JpaQueryRsqlVisitor<>( + legacyRsqlVisitor + ? new JpaQueryRsqlVisitor<>( root, cb, fieldsClass, virtualPropertyReplacer, DATABASE, query, !RsqlConfigHolder.getInstance().isCaseInsensitiveDB() && RsqlConfigHolder.getInstance().isIgnoreCase()) - : - new JpaQueryRsqlVisitorG2<>( - fieldsClass, root, query, cb, - DATABASE, virtualPropertyReplacer, - !RsqlConfigHolder.getInstance().isCaseInsensitiveDB() && RsqlConfigHolder.getInstance().isIgnoreCase()); + : new JpaQueryRsqlVisitorG2<>( + fieldsClass, root, query, cb, + DATABASE, virtualPropertyReplacer, + !RsqlConfigHolder.getInstance().isCaseInsensitiveDB() && RsqlConfigHolder.getInstance().isIgnoreCase()); final List accept = rootNode.accept(jpqQueryRSQLVisitor); if (CollectionUtils.isEmpty(accept)) { diff --git a/hawkbit-repository/hawkbit-repository-jpa/src/test/resources/jpa-test.properties b/hawkbit-repository/hawkbit-repository-jpa/src/test/resources/jpa-test.properties index 8c0d0f6092..ab9117e809 100644 --- a/hawkbit-repository/hawkbit-repository-jpa/src/test/resources/jpa-test.properties +++ b/hawkbit-repository/hawkbit-repository-jpa/src/test/resources/jpa-test.properties @@ -15,7 +15,18 @@ logging.level.org.eclipse.persistence=ERROR spring.jpa.properties.eclipselink.logging.level=FINE spring.jpa.properties.eclipselink.logging.level.sql=FINE spring.jpa.properties.eclipselink.logging.parameters=true + +#hibernate.generate_statistics=true +#logging.level.org.hibernate.SQL=TRACE +#logging.level.org.hibernate.stat=TRACE # Debug utility functions - END +# Switch to mysql +#spring.jpa.database=MYSQL +#spring.datasource.url=jdbc:mariadb://localhost:3306/hawkbit_test +#spring.datasource.driverClassName=org.mariadb.jdbc.Driver +#spring.datasource.username=root +#spring.datasource.password= + # enable / disable case sensitiveness of the DB when playing around #hawkbit.rsql.caseInsensitiveDB=true diff --git a/hawkbit-repository/pom.xml b/hawkbit-repository/pom.xml index 76e2492643..c8ad17eeb3 100644 --- a/hawkbit-repository/pom.xml +++ b/hawkbit-repository/pom.xml @@ -17,6 +17,7 @@ hawkbit-parent ${revision} + hawkbit-repository hawkBit :: Repository :: Parent pom @@ -24,6 +25,9 @@ hawkbit-repository-api hawkbit-repository-core + hawkbit-repository-jpa-api + hawkbit-repository-jpa-eclipselink + hawkbit-repository-jpa-hibernate hawkbit-repository-jpa hawkbit-repository-jpa-flyway diff --git a/pom.xml b/pom.xml index 2eeec8f7b2..5ac40120fa 100644 --- a/pom.xml +++ b/pom.xml @@ -671,16 +671,6 @@ - - org.eclipse.persistence - org.eclipse.persistence.core - ${eclipselink.version} - - - org.eclipse.persistence - org.eclipse.persistence.jpa - ${eclipselink.version} - org.springframework.plugin spring-plugin-core