org.dbflute.FunCustodial Maven / Gradle / Ivy
/*
* Copyright 2014-2017 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;
import java.util.Date;
import java.util.List;
import java.util.Locale;
import java.util.TimeZone;
import org.dbflute.dbmeta.accessory.EntityModifiedProperties;
import org.dbflute.exception.CharParameterShortSizeException;
import org.dbflute.exception.NonSpecifiedColumnAccessException;
import org.dbflute.exception.UndefinedClassificationCodeException;
import org.dbflute.helper.message.ExceptionMessageBuilder;
import org.dbflute.jdbc.Classification;
import org.dbflute.jdbc.ClassificationMeta;
import org.dbflute.jdbc.ClassificationUndefinedHandlingType;
import org.dbflute.jdbc.ShortCharHandlingMode;
import org.dbflute.system.DBFluteSystem;
import org.dbflute.util.DfTypeUtil;
import org.dbflute.util.Srl;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
/**
* The FunCustodial provides small services to your entities.
* @author jflute
* @since 1.1.0 (2014/10/18 Saturday)
*/
public class FunCustodial {
// ===================================================================================
// Definition
// ==========
/** The instance of logging object for classification meta. (NotNull) */
private static final Logger _clsMetaLog = LoggerFactory.getLogger(ClassificationMeta.class);
// ===================================================================================
// Value Conversion
// ================
public static String toClassTitle(Object entity) {
return DfTypeUtil.toClassTitle(entity);
}
@SuppressWarnings("unchecked")
public static NUMBER toNumber(Object obj, Class type) {
return (NUMBER) DfTypeUtil.toNumber(obj, type);
}
public static Boolean toBoolean(Object obj) {
return DfTypeUtil.toBoolean(obj);
}
public static String toStringDate(Date date, TimeZone timeZone, String pattern) {
if (date == null) {
return null;
}
final TimeZone realZone = timeZone != null ? timeZone : DBFluteSystem.getFinalTimeZone();
final Locale realLocale = DBFluteSystem.getFinalLocale(); // no specified because of basically debug string
final String str = DfTypeUtil.toStringDate(date, realZone, pattern, realLocale);
return (DfTypeUtil.isDateBC(date) ? "BC" : "") + str;
}
public static String toStringBytes(byte[] bytes) {
return "byte[" + (bytes != null ? String.valueOf(bytes.length) : "null") + "]";
}
public static String convertEmptyToNull(String value) {
return (value != null && value.length() == 0) ? null : value;
}
// -----------------------------------------------------
// Short Character
// ---------------
/**
* @param parameterName The name of the parameter. (NotNull)
* @param value The value of the parameter. (NullAllowed)
* @param size The size of parameter type. (NullAllowed)
* @param mode The handling mode. (NotNull)
* @return The filtered value. (NullAllowed)
*/
public static String handleShortChar(String parameterName, String value, Integer size, ShortCharHandlingMode mode) {
if (parameterName == null || parameterName.trim().length() == 0) {
String msg = "The argument 'parameterName' should not be null or empty:";
msg = msg + " value=" + value + " size=" + size + " mode=" + mode;
throw new IllegalArgumentException(msg);
}
if (mode == null) {
String msg = "The argument 'mode' should not be null:";
msg = msg + " parameterName=" + parameterName + " value=" + value + " size=" + size;
throw new IllegalArgumentException(msg);
}
if (value == null) {
return null;
}
if (size == null) {
return value;
}
if (value.length() >= size) {
return value;
}
if (mode.equals(ShortCharHandlingMode.RFILL)) {
return Srl.rfill(value, size);
} else if (mode.equals(ShortCharHandlingMode.LFILL)) {
return Srl.lfill(value, size);
} else if (mode.equals(ShortCharHandlingMode.EXCEPTION)) {
String msg = "The size of the parameter '" + parameterName + "' should be " + size + ":";
msg = msg + " value=[" + value + "] size=" + value.length();
throw new CharParameterShortSizeException(msg);
} else {
return value;
}
}
// ===================================================================================
// Specified Properties
// ====================
public static void checkSpecifiedProperty(Entity entity, String propertyName, EntityModifiedProperties specifiedProperties) {
if (specifiedProperties == null) { // means no need to check (e.g. all columns are selected)
return;
}
if (!entity.createdBySelect()) { // basically no way, selected when the properties exists but just in case
return;
}
// SpecifyColumn is used for the entity
if (specifiedProperties.isModifiedProperty(propertyName)) { // yes, setter is called when select mapping
return; // OK, OK
}
// no, you get the non-specified column, throws exception
throwNonSpecifiedColumnAccessException(entity, propertyName, specifiedProperties);
}
protected static void throwNonSpecifiedColumnAccessException(Entity entity, String propertyName,
EntityModifiedProperties specifiedProperties) {
final ExceptionMessageBuilder br = new ExceptionMessageBuilder();
br.addNotice("Non-specified column was accessed.");
br.addItem("Advice");
br.addElement("To get non-specified column from entity is not allowd.");
br.addElement("Non-specified means using SpecifyColumn but the column is non-specified.");
br.addElement("Mistake? Or specify the column by your condition-bean if you need it.");
br.addElement("For example:");
br.addElement(" (x):");
br.addElement(" memberBhv.selectEntity(cb -> {");
br.addElement(" cb.specify().columnMemberName();");
br.addElement(" cb.query().set...");
br.addElement(" }).alwaysPresent(member -> {");
br.addElement(" ... = member.getMemberName();");
br.addElement(" ... = member.getBirthdate(); // *NG: exception");
br.addElement(" });");
br.addElement(" (o):");
br.addElement(" memberBhv.selectEntity(cb -> {");
br.addElement(" cb.specify().columnMemberName();");
br.addElement(" cb.specify().columnBirthdate(); // *Point");
br.addElement(" cb.query().set...");
br.addElement(" }).alwaysPresent(member -> {");
br.addElement(" ... = member.getMemberName();");
br.addElement(" ... = member.getBirthdate(); // OK");
br.addElement(" });");
br.addElement("");
br.addElement("While, reluctantly you need to get the column without change conditions,");
br.addElement("you can enable non-specified column access by the condition-bean option.");
br.addElement("The method is cb.enable...()");
buildExceptionTableInfo(br, entity);
br.addItem("Non-Specified and Accessed");
br.addElement(propertyName);
br.addItem("Specified Column in the Table");
br.addElement(specifiedProperties);
final String msg = br.buildExceptionMessage();
throw new NonSpecifiedColumnAccessException(msg);
}
protected static void buildExceptionTableInfo(ExceptionMessageBuilder br, Entity entity) {
br.addItem("Table");
br.addElement(entity.asTableDbName());
try {
br.addElement(entity.asDBMeta().extractPrimaryKeyMap(entity));
} catch (RuntimeException continued) { // just in case
br.addElement("*Failed to get PK info:");
br.addElement(continued.getMessage());
}
}
// ===================================================================================
// Classification
// ==============
public static void checkClassificationCode(Entity entity, String columnDbName, ClassificationMeta meta, Object code) {
if (code == null) {
return;
}
final ClassificationUndefinedHandlingType undefinedHandlingType = meta.undefinedHandlingType();
if (!undefinedHandlingType.isChecked()) { // basically no way (not called if no check)
return;
}
if (meta.codeOf(code) != null) {
return;
}
final boolean allowedByOption = entity.myundefinedClassificationAccessAllowed();
handleUndefinedClassificationCode(entity.asTableDbName(), columnDbName, meta, code, allowedByOption);
}
public static void handleUndefinedClassificationCode(String tableDbName, String columnDbName, ClassificationMeta meta, Object code,
boolean allowedByOption) {
final ClassificationUndefinedHandlingType undefinedHandlingType = meta.undefinedHandlingType();
if (ClassificationUndefinedHandlingType.EXCEPTION.equals(undefinedHandlingType)) {
if (allowedByOption) { // e.g. ConditionBean's option
showUndefinedClassificationCodeMessage(tableDbName, columnDbName, meta, code); // logging at least
} else { // normally here
throwUndefinedClassificationCodeException(tableDbName, columnDbName, meta, code);
}
} else if (ClassificationUndefinedHandlingType.LOGGING.equals(undefinedHandlingType)) {
showUndefinedClassificationCodeMessage(tableDbName, columnDbName, meta, code);
}
// else means ALLOWED
}
protected static void throwUndefinedClassificationCodeException(String tableDbName, String columnDbName, ClassificationMeta meta,
Object code) {
final ExceptionMessageBuilder br = new ExceptionMessageBuilder();
br.addNotice("Undefined classification code was set to the entity.");
br.addItem("Advice");
br.addElement("Confirm the value of the classication column on your database,");
br.addElement("or setting value to your entity.");
br.addElement("The code is NOT one of classification code defined on DBFlute.");
br.addElement("");
br.addElement("_/_/_/_/_/_/_/_/_/_/_/_/_/_/_/_/_/_/_/");
br.addElement(" Use formal code!");
br.addElement(" Or add the code to classification definition.");
br.addElement("_/_/_/_/_/_/_/_/_/_/");
br.addElement("");
br.addElement("Or if you (reluctantly) need to allow it, change the option like this:");
br.addElement("but *Deprecated");
br.addElement("(classificationDefinitionMap.dfprop)");
br.addElement(" ; [classification-name] = list:{");
br.addElement(" ; map:{");
br.addElement(" ; topComment=...; codeType=...");
br.addElement(" ; undefinedHandlingType=ALLOWED");
br.addElement(" }");
br.addElement(" map:{...}");
br.addElement(" }");
br.addElement("*for your information, the default of undefinedHandlingType is LOGGING");
br.addItem("Table");
br.addElement(tableDbName);
br.addItem("Column");
br.addElement(columnDbName);
br.addItem("Classification");
br.addElement(meta.classificationName());
final List listAll = meta.listAll();
final StringBuilder sb = new StringBuilder();
for (Classification cls : listAll) {
if (sb.length() > 0) {
sb.append(", ");
}
sb.append(cls.name()).append("(").append(cls.code()).append(")");
}
br.addElement(sb.toString());
br.addItem("Undefined Code");
br.addElement(code);
final String msg = br.buildExceptionMessage();
throw new UndefinedClassificationCodeException(msg);
}
public static void showUndefinedClassificationCodeMessage(String tableDbName, String columnDbName, ClassificationMeta meta,
Object code) {
if (_clsMetaLog.isInfoEnabled()) {
final String classificationName = meta.classificationName();
final String exp = classificationName + "." + code + " of " + tableDbName + "." + columnDbName;
_clsMetaLog.info("*Undefined classification code was set: " + exp); // one line because of many called
}
}
// ===================================================================================
// Small Helper
// ============
// ------------
public static boolean isSameValue(Object value1, Object value2) {
if (value1 == null && value2 == null) {
return true;
}
if (value1 == null || value2 == null) {
return false;
}
if (value1 instanceof byte[] && value2 instanceof byte[]) {
return isSameValueBytes((byte[]) value1, (byte[]) value2);
}
return value1.equals(value2);
}
public static boolean isSameValueBytes(byte[] bytes1, byte[] bytes2) {
if (bytes1 == null && bytes2 == null) {
return true;
}
if (bytes1 == null || bytes2 == null) {
return false;
}
if (bytes1.length != bytes2.length) {
return false;
}
for (int i = 0; i < bytes1.length; i++) {
if (bytes1[i] != bytes2[i]) {
return false;
}
}
return true;
}
public static int calculateHashcode(int result, Object value) { // calculateHashcode()
if (value == null) {
return result;
}
return (31 * result) + (value instanceof byte[] ? ((byte[]) value).length : value.hashCode());
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy