org.dbflute.dbmeta.info.ForeignInfo Maven / Gradle / Ivy
/*
* Copyright 2014-2020 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
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND,
* either express or implied. See the License for the specific language
* governing permissions and limitations under the License.
*/
package org.dbflute.dbmeta.info;
import java.lang.reflect.Method;
import java.util.Collections;
import java.util.LinkedHashMap;
import java.util.List;
import java.util.Map;
import java.util.Map.Entry;
import org.dbflute.Entity;
import org.dbflute.dbmeta.DBMeta;
import org.dbflute.dbmeta.property.PropertyGateway;
import org.dbflute.dbmeta.property.PropertyMethodFinder;
import org.dbflute.util.DfReflectionUtil;
import org.dbflute.util.Srl;
/**
* The information of foreign relation.
* @author jflute
*/
public class ForeignInfo implements RelationInfo {
// ===================================================================================
// Attribute
// =========
protected final String _constraintName;
protected final String _foreignPropertyName;
protected final DBMeta _localDBMeta;
protected final DBMeta _foreignDBMeta;
protected final Map _localForeignColumnInfoMap;
protected final Map _foreignLocalColumnInfoMap;
protected final int _relationNo;
protected final Class> _objectNativeType; // always relation entity type, provided by DB meta
protected final Class> _propertyAccessType; // same as entity type or might be optional
protected final boolean _oneToOne;
protected final boolean _bizOneToOne;
protected final boolean _referrerAsOne;
protected final boolean _additionalFK;
protected final String _fixedCondition;
protected final List _dynamicParameterList;
protected final boolean _fixedInline;
protected final String _reversePropertyName;
protected final boolean _canBeNullObject;
protected final PropertyGateway _propertyGateway;
protected final PropertyMethodFinder _propertyMethodFinder;
protected final Method _readMethod;
protected final Method _writeMethod;
// ===================================================================================
// Constructor
// ===========
public ForeignInfo(String constraintName, String foreignPropertyName // name
, DBMeta localDBMeta, DBMeta foreignDBMeta // DB meta
, Map localForeignColumnInfoMap, int relationNo // relation attribute
, Class> propertyAccessType // property info (object native type is provided by DB meta)
, boolean oneToOne, boolean bizOneToOne, boolean referrerAsOne, boolean additionalFK // relation type
, String fixedCondition, List dynamicParameterList, boolean fixedInline // fixed condition
, String reversePropertyName, boolean canBeNullObject, PropertyMethodFinder propertyMethodFinder // various info
) { // big constructor
assertObjectNotNull("constraintName", constraintName);
assertObjectNotNull("foreignPropertyName", foreignPropertyName);
assertObjectNotNull("localDBMeta", localDBMeta);
assertObjectNotNull("foreignDBMeta", foreignDBMeta);
assertObjectNotNull("localForeignColumnInfoMap", localForeignColumnInfoMap);
assertObjectNotNull("propertyAccessType", propertyAccessType);
assertObjectNotNull("propertyMethodFinder", propertyMethodFinder);
_constraintName = constraintName;
_foreignPropertyName = foreignPropertyName;
_localDBMeta = localDBMeta;
_foreignDBMeta = foreignDBMeta;
_localForeignColumnInfoMap = Collections.unmodifiableMap(localForeignColumnInfoMap);
final Map foreignLocalColumnInfoMap = new LinkedHashMap(4);
for (Entry entry : localForeignColumnInfoMap.entrySet()) {
foreignLocalColumnInfoMap.put(entry.getValue(), entry.getKey());
}
_foreignLocalColumnInfoMap = Collections.unmodifiableMap(foreignLocalColumnInfoMap);
_relationNo = relationNo;
_objectNativeType = foreignDBMeta.getEntityType();
_propertyAccessType = propertyAccessType;
_oneToOne = oneToOne;
_bizOneToOne = bizOneToOne;
_referrerAsOne = referrerAsOne;
_additionalFK = additionalFK;
_fixedCondition = fixedCondition;
_fixedInline = fixedInline;
if (dynamicParameterList != null) {
_dynamicParameterList = Collections.unmodifiableList(dynamicParameterList);
} else {
_dynamicParameterList = Collections.emptyList();
}
_reversePropertyName = reversePropertyName;
_canBeNullObject = canBeNullObject;
_propertyGateway = findPropertyGateway();
_propertyMethodFinder = propertyMethodFinder;
_readMethod = findReadMethod();
_writeMethod = findWriteMethod();
}
// ===================================================================================
// Column Existence
// ================
public boolean containsLocalColumn(ColumnInfo localColumn) {
return doContainsLocalColumn(localColumn.getColumnDbName());
}
protected boolean doContainsLocalColumn(String columnName) {
for (ColumnInfo columnInfo : _localForeignColumnInfoMap.keySet()) {
if (columnInfo.getColumnDbName().equals(columnName)) {
return true;
}
}
return false;
}
public boolean containsForeignColumn(ColumnInfo foreignColumn) {
return doContainsForeignColumn(foreignColumn.getColumnDbName());
}
protected boolean doContainsForeignColumn(String columnName) {
for (ColumnInfo columnInfo : _foreignLocalColumnInfoMap.keySet()) {
if (columnInfo.getColumnDbName().equals(columnName)) {
return true;
}
}
return false;
}
// ===================================================================================
// Column Mapping
// ==============
public ColumnInfo findLocalByForeign(String foreignColumnDbName) {
final ColumnInfo keyColumnInfo = _foreignDBMeta.findColumnInfo(foreignColumnDbName);
final ColumnInfo resultColumnInfo = (ColumnInfo) _foreignLocalColumnInfoMap.get(keyColumnInfo);
if (resultColumnInfo == null) {
String msg = "Not found by foreignColumnDbName in foreignLocalColumnInfoMap:";
msg = msg + " foreignColumnDbName=" + foreignColumnDbName;
msg = msg + " foreignLocalColumnInfoMap=" + _foreignLocalColumnInfoMap;
throw new IllegalArgumentException(msg);
}
return resultColumnInfo;
}
public ColumnInfo findForeignByLocal(String localColumnDbName) {
final ColumnInfo keyColumnInfo = _localDBMeta.findColumnInfo(localColumnDbName);
final ColumnInfo resultColumnInfo = (ColumnInfo) _localForeignColumnInfoMap.get(keyColumnInfo);
if (resultColumnInfo == null) {
String msg = "Not found by localColumnDbName in localForeignColumnInfoMap:";
msg = msg + " localColumnDbName=" + localColumnDbName;
msg = msg + " localForeignColumnInfoMap=" + _localForeignColumnInfoMap;
throw new IllegalArgumentException(msg);
}
return resultColumnInfo;
}
// ===================================================================================
// Reflection
// ==========
// -----------------------------------------------------
// Read
// ----
/**
* Read the value to the entity by its gateway (means no reflection).
* It returns plain value in entity as property access type.
* @param The type of property, might be optional.
* @param localEntity The local entity of this column to read. (NotNull)
* @return The read instance of foreign entity, might be optional. (NotNull: when optional, NullAllowed: when native type)
*/
@SuppressWarnings("unchecked")
public PROPERTY read(Entity localEntity) {
return (PROPERTY) _propertyGateway.read(localEntity);
}
/**
* Get the read method for entity reflection.
* @return The read method, cached in this instance. (NotNull)
*/
public Method getReadMethod() { // basically unused in DBFlute, use gateway instead
return _readMethod;
}
// -----------------------------------------------------
// Write
// -----
/**
* Write the value to the entity by its gateway (means no reflection).
* No converting to optional so check the property access type.
* @param localEntity The local entity of this column to write. (NotNull)
* @param foreignEntity The written instance of foreign entity, might be optional. (NullAllowed: if null, null written)
*/
public void write(Entity localEntity, Object foreignEntity) {
_propertyGateway.write(localEntity, foreignEntity);
}
/**
* Get the write method for entity reflection.
* @return The writer method, cached in this instance. (NotNull)
*/
public Method getWriteMethod() { // basically unused in DBFlute, use gateway instead
return _writeMethod;
}
// -----------------------------------------------------
// Finder
// ------
protected PropertyGateway findPropertyGateway() {
final PropertyGateway gateway = _localDBMeta.findForeignPropertyGateway(_foreignPropertyName);
if (gateway == null) { // no way
String msg = "Not found the foreign property gateway by the name: " + _foreignPropertyName;
throw new IllegalStateException(msg);
}
return gateway;
}
protected Method findReadMethod() {
final Class extends Entity> localType = _localDBMeta.getEntityType();
return _propertyMethodFinder.findReadMethod(localType, _foreignPropertyName, _propertyAccessType);
}
protected Method findWriteMethod() {
final Class extends Entity> localType = _localDBMeta.getEntityType();
return _propertyMethodFinder.findWriteMethod(localType, _foreignPropertyName, _propertyAccessType);
}
// -----------------------------------------------------
// Invoker
// -------
protected Object invokeMethod(Method method, Object target, Object[] args) {
return DfReflectionUtil.invoke(method, target, args);
}
// ===================================================================================
// Relation Implementation
// =======================
public String getRelationPropertyName() {
return getForeignPropertyName();
}
public DBMeta getTargetDBMeta() {
return getForeignDBMeta();
}
public Map getLocalTargetColumnInfoMap() {
return getLocalForeignColumnInfoMap();
}
public boolean isReferrer() {
return false;
}
// ===================================================================================
// General Helper
// ==============
protected String initCap(final String name) {
return Srl.initCap(name);
}
protected void assertObjectNotNull(String variableName, Object value) {
if (variableName == null) {
String msg = "The value should not be null: variableName=null value=" + value;
throw new IllegalArgumentException(msg);
}
if (value == null) {
String msg = "The value should not be null: variableName=" + variableName;
throw new IllegalArgumentException(msg);
}
}
// ===================================================================================
// Basic Override
// ==============
@Override
public int hashCode() {
return _foreignPropertyName.hashCode() + _localDBMeta.hashCode() + _foreignDBMeta.hashCode();
}
@Override
public boolean equals(Object obj) {
if (obj == null || !(obj instanceof ForeignInfo)) {
return false;
}
final ForeignInfo target = (ForeignInfo) obj;
if (!this._foreignPropertyName.equals(target.getForeignPropertyName())) {
return false;
}
if (!this._localDBMeta.equals(target.getLocalDBMeta())) {
return false;
}
if (!this._foreignDBMeta.equals(target.getForeignDBMeta())) {
return false;
}
return true;
}
@Override
public String toString() {
return "{" + _localDBMeta.getTableDispName() + "." + _foreignPropertyName + "->" + _foreignDBMeta.getTableDbName() + "}";
}
// ===================================================================================
// Accessor
// ========
/** {@inheritDoc} */
public String getConstraintName() {
return _constraintName;
}
/**
* Get the property name of the foreign relation.
* This is unique name in the table.
* For example, if the member entity has getMemberStatus(), this returns 'memberStatus'.
* @return The string for property name. (NotNull)
*/
public String getForeignPropertyName() {
return _foreignPropertyName;
}
/** {@inheritDoc} */
public DBMeta getLocalDBMeta() {
return _localDBMeta;
}
/**
* Get the DB meta of the foreign table.
* For example, if the relation MEMBER and MEMBER_STATUS, this returns MEMBER_STATUS's one.
* @return The DB meta singleton instance. (NotNull)
*/
public DBMeta getForeignDBMeta() {
return _foreignDBMeta;
}
/**
* Get the read-only map, key is a local column info, value is a foreign column info.
* @return The read-only map. (NotNull, EmptyAllowed: only when fixedOnlyJoin)
*/
public Map getLocalForeignColumnInfoMap() {
return _localForeignColumnInfoMap;
}
/**
* Get the read-only map, key is a foreign column info, value is a local column info.
* @return The read-only map. (NotNull, EmptyAllowed: only when fixedOnlyJoin)
*/
public Map getForeignLocalColumnInfoMap() {
return _foreignLocalColumnInfoMap;
}
/**
* Get the number of a relation. (internal property)
* @return The number of a relation. (NotNull, NotMinus)
*/
public int getRelationNo() {
return _relationNo;
}
/** {@inheritDoc} */
public Class> getObjectNativeType() {
return _objectNativeType;
}
/** {@inheritDoc} */
public Class> getPropertyAccessType() {
return _propertyAccessType;
}
/** {@inheritDoc} */
public boolean isOneToOne() {
return _oneToOne;
}
/**
* Does the relation is biz-one-to-one?
* @return The determination, true or false.
*/
public boolean isBizOneToOne() {
return _bizOneToOne;
}
/**
* Does the relation is referrer-as-one?
* @return The determination, true or false.
*/
public boolean isReferrerAsOne() {
return _referrerAsOne;
}
/**
* Does the relation is from additional foreign key?
* @return The determination, true or false.
*/
public boolean isAdditionalFK() {
return _additionalFK;
}
/**
* Get the fixed-condition if it's additional foreign key.
* @return The string of fixed-condition. (NullAllowed)
*/
public String getFixedCondition() {
return _fixedCondition;
}
/**
* Get the read-only list of dynamic parameter name for fixed-condition if it's additional foreign key.
* @return The read-only list. (NotNull: if no fixed-condition, returns empty list)
*/
public List getDynamicParameterList() {
return _dynamicParameterList;
}
/**
* Does the fixed condition is for in-line view?
* @return The determination, true or false.
*/
public boolean isFixedInline() {
return _fixedInline;
}
/** {@inheritDoc} */
public RelationInfo getReverseRelation() {
return _reversePropertyName != null ? _foreignDBMeta.findRelationInfo(_reversePropertyName) : null;
}
/**
* Can the relation be null object?
* @return The determination, true or false.
*/
public boolean canBeNullObject() { // for future, nobody calls when first
return _canBeNullObject;
}
// -----------------------------------------------------
// Derived
// -------
/** {@inheritDoc} */
public boolean isCompoundKey() {
return getLocalForeignColumnInfoMap().size() > 1;
}
/**
* Does the relation is from pure foreign key?
* @return The determination, true or false.
*/
public boolean isPureFK() { // derived property
return !_additionalFK && !_referrerAsOne;
}
/**
* Do the FK columns have not null constraint?
* @return The determination, true or false.
*/
public boolean isNotNullFKColumn() {
for (Entry entry : getLocalForeignColumnInfoMap().entrySet()) {
final ColumnInfo localColumnInfo = entry.getKey();
if (!localColumnInfo.isNotNull()) {
return false;
}
}
return true;
}
/**
* Does it have fixed-condition for this relation?
* @return The determination, true or false.
*/
public boolean hasFixedCondition() {
return _fixedCondition != null && _fixedCondition.trim().length() > 0;
}
/**
* Does it have dynamic parameter of fixed-condition?
* @return The determination, true or false.
*/
public boolean hasFixedConditionDynamicParameter() {
return hasFixedCondition() && !_dynamicParameterList.isEmpty();
}
/**
* Is the dynamic parameters of fixed condition required?
* @return The determination, true or false. (true if NOT contains IF comment)
*/
public boolean isFixedConditionDynamicParameterRequired() {
return hasFixedConditionDynamicParameter() && !getFixedCondition().contains("/*IF ");
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy