MergingPersistenceUnitManager.java
/*
 * Copyright 2011-2025 the original author or authors.
 *
 * Licensed under the Apache License, Version 2.0 (the "License");
 * you may not use this file except in compliance with the License.
 * You may obtain a copy of the License at
 *
 *      https://www.apache.org/licenses/LICENSE-2.0
 *
 * Unless required by applicable law or agreed to in writing, software
 * distributed under the License is distributed on an "AS IS" BASIS,
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 * See the License for the specific language governing permissions and
 * limitations under the License.
 */
package org.springframework.data.jpa.support;
import java.net.URISyntaxException;
import java.net.URL;
import org.apache.commons.logging.Log;
import org.apache.commons.logging.LogFactory;
import org.springframework.orm.jpa.persistenceunit.DefaultPersistenceUnitManager;
import org.springframework.orm.jpa.persistenceunit.MutablePersistenceUnitInfo;
import org.springframework.util.StringUtils;
/**
 * Extends {@link DefaultPersistenceUnitManager} to merge configurations of one persistence unit residing in multiple
 * {@code persistence.xml} files into one. This is necessary to allow the declaration of entities in separate modules.
 *
 * @author Oliver Gierke
 * @author Christoph Strobl
 * @link https://github.com/spring-projects/spring-framework/issues/7287
 */
public class MergingPersistenceUnitManager extends DefaultPersistenceUnitManager {
	private static final Log LOG = LogFactory.getLog(MergingPersistenceUnitManager.class);
	@Override
	protected void postProcessPersistenceUnitInfo(MutablePersistenceUnitInfo pui) {
		// Invoke normal post processing
		super.postProcessPersistenceUnitInfo(pui);
		if (StringUtils.hasText(pui.getPersistenceUnitName())) {
			MutablePersistenceUnitInfo oldPui = getPersistenceUnitInfo(pui.getPersistenceUnitName());
			if (oldPui != null) {
				postProcessPersistenceUnitInfo(pui, oldPui);
			}
		}
	}
	@Override
	protected boolean isPersistenceUnitOverrideAllowed() {
		return true;
	}
	void postProcessPersistenceUnitInfo(MutablePersistenceUnitInfo pui, MutablePersistenceUnitInfo oldPui) {
		String persistenceUnitName = pui.getPersistenceUnitName();
		for (URL url : oldPui.getJarFileUrls()) {
			if (!pui.getJarFileUrls().contains(url)) {
				if (LOG.isDebugEnabled()) {
					LOG.debug(String.format("Adding JAR file URL %s to persistence unit %s", url, persistenceUnitName));
				}
				pui.addJarFileUrl(url);
			}
		}
		for (String className : oldPui.getManagedClassNames()) {
			if (!pui.getManagedClassNames().contains(className)) {
				if (LOG.isDebugEnabled()) {
					LOG.debug(String.format("Adding class %s to PersistenceUnit %s", className, persistenceUnitName));
				}
				pui.addManagedClassName(className);
			}
		}
		for (String mappingFileName : oldPui.getMappingFileNames()) {
			if (!pui.getMappingFileNames().contains(mappingFileName)) {
				if (LOG.isDebugEnabled()) {
					LOG.debug(
							String.format("Adding mapping file %s to persistence unit %s", mappingFileName, persistenceUnitName));
				}
				pui.addMappingFileName(mappingFileName);
			}
		}
		URL newUrl = pui.getPersistenceUnitRootUrl();
		URL oldUrl = oldPui.getPersistenceUnitRootUrl();
		if (oldUrl == null || newUrl == null) {
			return;
		}
		try {
			boolean rootUrlsDiffer = !newUrl.toURI().equals(oldUrl.toURI());
			boolean urlNotInJarUrls = !pui.getJarFileUrls().contains(oldUrl);
			if (rootUrlsDiffer && urlNotInJarUrls) {
				pui.addJarFileUrl(oldUrl);
			}
		} catch (URISyntaxException ex) {
			throw new RuntimeException(ex);
		}
	}
}