Certo, è possibile in modo molto semplice se gli schemi sono gli stessi. E poiché hai creato entrambi i database con la stessa mappatura Hibernate, dovrebbero essere uguali in Entity
senso.
Hai solo bisogno di due unità di persistenza Hibernate (sorgenti dati). Se entrambi sono configurati correttamente e hai il particolare EntityManager
istanze a portata di mano, vai alla Session
di ibernazione livello - per quanto ne so JPA non lo supporta in questo modo (correggimi se sbaglio) - e replica la tua entità di origine nel tuo database di destinazione.
Poiché mi piace lavorare con Spring, userò Spring Boot per il seguente esempio. Fatta eccezione per la configurazione, il passaggio di replica verrebbe implementato allo stesso modo con qualsiasi applicazione Hibernate.
Sto anche usando due database PostgreSQL invece di un HSQLB solo per mantenerlo semplice. Estendi semplicemente la parte di configurazione se le tue configurazioni si discostano, l'unica differenza tra le mie unità di persistenza è l'URL dell'origine dati.
Quindi prima abbiamo bisogno di un'entità per testare la replica:
import javax.persistence.Entity;
import javax.persistence.GeneratedValue;
import javax.persistence.Id;
@Entity
public class StorageEntry {
@Id
@GeneratedValue
private Long id;
private String someValue;
// imagine getters and setter here
}
Questa è (la versione YAML di) la configurazione delle due origini dati (vedi il secondo URL di origine dati chiamato targetDatabaseUrl
), Tutte le altre parti della configurazione verranno utilizzate per entrambe le unità di persistenza:
spring:
datasource:
url: jdbc:postgresql://localhost/postgres
targetDatabaseUrl: jdbc:postgresql://localhost/postgres2
username: <username>
password: <password>
driver-class-name: org.postgresql.Driver
jpa:
database-platform: org.hibernate.dialect.PostgreSQLDialect
hibernate:
ddl-auto: create-drop
La parte successiva è la classe di configurazione per le origini dati:
import java.util.Properties;
import javax.persistence.EntityManager;
import javax.persistence.EntityManagerFactory;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.beans.factory.annotation.Value;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
import org.springframework.jdbc.datasource.DriverManagerDataSource;
import org.springframework.orm.jpa.JpaTransactionManager;
import org.springframework.orm.jpa.JpaVendorAdapter;
import org.springframework.orm.jpa.LocalContainerEntityManagerFactoryBean;
import org.springframework.transaction.PlatformTransactionManager;
@Configuration
public class PersistenceConfig {
@Autowired
private JpaVendorAdapter jpaVendorAdapter;
@Value("${spring.datasource.url}")
private String databaseUrl;
@Value("${spring.datasource.targetDatabaseUrl}")
private String targetDatabaseUrl;
@Value("${spring.datasource.username}")
private String username;
@Value("${spring.datasource.password}")
private String password;
@Value("${spring.datasource.driver-class-name}")
private String driverClassName;
@Value("${spring.jpa.database-platform}")
private String dialect;
@Value("${spring.jpa.hibernate.ddl-auto}")
private String ddlAuto;
@Bean
public EntityManager sourceEntityManager() {
return sourceEntityManagerFactory().createEntityManager();
}
@Bean
public EntityManager targetEntityManager() {
return targetEntityManagerFactory().createEntityManager();
}
@Bean
public EntityManagerFactory sourceEntityManagerFactory() {
return createEntityManagerFactory("source", databaseUrl);
}
@Bean
public EntityManagerFactory targetEntityManagerFactory() {
return createEntityManagerFactory("target", targetDatabaseUrl);
}
@Bean
public PlatformTransactionManager sourceTransactionManager() {
return new JpaTransactionManager(sourceEntityManagerFactory());
}
@Bean
public PlatformTransactionManager targetTransactionManager() {
return new JpaTransactionManager(targetEntityManagerFactory());
}
private EntityManagerFactory createEntityManagerFactory(final String persistenceUnitName,
final String databaseUrl) {
final LocalContainerEntityManagerFactoryBean entityManagerFactory = new LocalContainerEntityManagerFactoryBean();
final DriverManagerDataSource dataSource = new DriverManagerDataSource(databaseUrl, username, password);
dataSource.setDriverClassName(driverClassName);
entityManagerFactory.setDataSource(dataSource);
entityManagerFactory.setJpaVendorAdapter(jpaVendorAdapter);
entityManagerFactory.setPackagesToScan("com.example.model");
entityManagerFactory.setPersistenceUnitName(persistenceUnitName);
final Properties properties = new Properties();
properties.setProperty("hibernate.dialect", dialect);
properties.setProperty("hibernate.hbm2ddl.auto", ddlAuto);
entityManagerFactory.setJpaProperties(properties);
entityManagerFactory.afterPropertiesSet();
return entityManagerFactory.getObject();
}
}
Ora puoi utilizzare i diversi gestori di entità per leggere e scrivere semplicemente i tuoi dati da un'origine dati all'altra. Per dimostrarlo, ecco un piccolo test case:
import static org.hamcrest.CoreMatchers.notNullValue;
import static org.hamcrest.CoreMatchers.nullValue;
import static org.hamcrest.MatcherAssert.assertThat;
import javax.persistence.EntityManager;
import javax.persistence.PersistenceContext;
import org.hibernate.ReplicationMode;
import org.hibernate.Session;
import org.junit.Test;
import org.junit.runner.RunWith;
import org.springframework.boot.test.context.SpringBootTest;
import org.springframework.test.context.junit4.SpringRunner;
import org.springframework.transaction.annotation.Transactional;
import com.example.model.StorageEntry;
@SpringBootTest
@RunWith(SpringRunner.class)
@Transactional(transactionManager = "targetTransactionManager")
public class ReplicationTests {
@PersistenceContext(unitName = "source")
private EntityManager sourceEntityManager;
@PersistenceContext(unitName = "target")
private EntityManager targetEntityManager;
@Test
public void copyEntityBetweenPersistenceUnits() {
final StorageEntry entityToCopy = new StorageEntry();
entityToCopy.setSomeValue("copyMe!");
sourceEntityManager.persist(entityToCopy);
final Long id = entityToCopy.getId();
final StorageEntry sourceEntity = sourceEntityManager.find(StorageEntry.class, id);
assertThat("Entity should exist in default schema!", sourceEntity, notNullValue());
StorageEntry targetEntity = targetEntityManager.find(StorageEntry.class, id);
assertThat("Target schema should not contain the entity, yet!", targetEntity, nullValue());
final Session hibernateSession = targetEntityManager.unwrap(Session.class);
hibernateSession.replicate(sourceEntity, ReplicationMode.OVERWRITE);
targetEntityManager.flush();
targetEntityManager.clear();
targetEntity = targetEntityManager.find(StorageEntry.class, id);
assertThat("Entity should be copied now!", targetEntity, notNullValue());
}
}
Infine, scegli uno dei possibili modalità di replica che si adatta alle tue esigenze.
È tutto. Puoi persino utilizzare una transazione, decidere per una di entrambe le unità di persistenza e sfruttare il suo gestore delle transazioni come fa il test con @Transactional(transactionManager = "targetTransactionManager")
.