/****************************************************************************** * Product: Adempiere ERP & CRM Smart Business Solution * * Copyright (C) 1999-2006 ComPiere, Inc. All Rights Reserved. * * This program is free software; you can redistribute it and/or modify it * * under the terms version 2 of the GNU General Public License as published * * by the Free Software Foundation. This program is distributed in the hope * * that it will be useful, but WITHOUT ANY WARRANTY; without even the implied * * warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. * * See the GNU General Public License for more details. * * You should have received a copy of the GNU General Public License along * * with this program; if not, write to the Free Software Foundation, Inc., * * 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA. * * For the text or an alternative of this public license, you may reach us * * ComPiere, Inc., 2620 Augustine Dr. #245, Santa Clara, CA 95054, USA * * or via info@compiere.org or http://www.compiere.org/license.html * *****************************************************************************/ package org.compiere.wf; import java.math.BigDecimal; import java.sql.PreparedStatement; import java.sql.ResultSet; import java.sql.SQLException; import java.util.Properties; import java.util.logging.Level; import org.compiere.model.PO; import org.compiere.util.DB; import org.compiere.util.Env; import org.compiere.util.Util; import org.idempiere.cache.ImmutablePOSupport; import org.compiere.model.X_AD_WF_NextCondition; /** * Extended Workflow Transition Condition model for AD_WF_NextCondition * * @author Jorg Janke * @version $Id: MWFNextCondition.java,v 1.3 2006/07/30 00:51:05 jjanke Exp $ * * @author Teo Sarca, SC ARHIPAC SERVICE SRL *
  • BF [ 1943720 ] WF Next Condition: handling boolean values is poor */ public class MWFNextCondition extends X_AD_WF_NextCondition implements ImmutablePOSupport { /** * generated serial id */ private static final long serialVersionUID = 3119863973003103716L; /** * UUID based Constructor * @param ctx Context * @param AD_WF_NextCondition_UU UUID key * @param trxName Transaction */ public MWFNextCondition(Properties ctx, String AD_WF_NextCondition_UU, String trxName) { super(ctx, AD_WF_NextCondition_UU, trxName); } /** * Default Constructor * @param ctx context * @param id id * @param trxName transaction */ public MWFNextCondition (Properties ctx, int id, String trxName) { super (ctx, id, trxName); } // MWFNextCondition /** * Load Constructor * @param ctx context * @param rs result set * @param trxName transaction */ public MWFNextCondition (Properties ctx, ResultSet rs, String trxName) { super(ctx, rs, trxName); } // MWFNextCondition /** * Copy constructor * @param copy */ public MWFNextCondition(MWFNextCondition copy) { this(Env.getCtx(), copy); } /** * Copy constructor * @param ctx * @param copy */ public MWFNextCondition(Properties ctx, MWFNextCondition copy) { this(ctx, copy, (String) null); } /** * Copy constructor * @param ctx * @param copy * @param trxName */ public MWFNextCondition(Properties ctx, MWFNextCondition copy, String trxName) { this(ctx, 0, trxName); copyPO(copy); this.m_numeric = copy.m_numeric; } /** Numeric evaluation */ private boolean m_numeric = true; /** * Is Or Condition * @return true if OR */ public boolean isOr() { return ANDOR_Or.equals(getAndOr()); } // isOr /** * Evaluate Condition * @param activity activity * @return true if true */ public boolean evaluate (MWFActivity activity) { return evaluate(activity.getPO()); } // evaluate /** * Evaluate Condition * @param po PO * @return true if true */ protected boolean evaluate (PO po) { if (getAD_Column_ID() == 0 && Util.isEmpty(getSQLStatement(), true)) throw new IllegalStateException("No Column and SQL Statement defined - " + this); if (po == null || po.get_ID() == 0) throw new IllegalStateException("Could not evaluate " + po + " - " + this); // if (getOperation().equals(OPERATION_Sql)) { String sqlStatement = getSQLStatement(); if (Util.isEmpty(getSQLStatement(), true)) return false; if (sqlStatement.indexOf("@") >= 0) sqlStatement = Env.parseVariable(sqlStatement, po, po.get_TrxName(), false); String result = DB.getSQLValueStringEx(po.get_TrxName(), sqlStatement); return "true".equalsIgnoreCase(result) || "y".equalsIgnoreCase(result); } // Object valueObj = null; if (!Util.isEmpty(getSQLStatement(), true)) { try { valueObj = getColumnSQLValue(po); } catch (SQLException e) { throw new RuntimeException("Could not get result from column sql: " + getSQLStatement(), e); } } else { valueObj = po.get_ValueOfColumn(getAD_Column_ID()); } if (valueObj == null) valueObj = ""; String value1 = getDecodedValue(getValue(), po); // F3P: added value decoding if (value1 == null) value1 = ""; String value2 = getDecodedValue(getValue2(), po); // F3P: added value decoding if (value2 == null) value2 = ""; String resultStr = "PO:{" + valueObj + "} " + getOperation() + " Condition:{" + value1 + "}"; if (getOperation().equals(OPERATION_X)) resultStr += "{" + value2 + "}"; boolean result = false; if (valueObj instanceof Number) result = compareNumber ((Number)valueObj, value1, value2); else if (valueObj instanceof Boolean) result = compareBoolean((Boolean)valueObj, value1, value2); else result = compareString(valueObj, value1, value2); // if (log.isLoggable(Level.FINE)) log.fine(resultStr + " -> " + result + (m_numeric ? " (#)" : " ($)")); return result; } // evaluate /** * Decode value string.
    * If sValue start with @COL=, remaining text is interpreted as a column of the associated record.
    * Otherwise, parse sValue for context and PO variable (i.e resolve @tag@ variables). * * @param sValue value to be decoded * @param po PO model object bound to the activity * @return decoded value */ protected String getDecodedValue(String sValue, PO po) { String sRet = sValue; if (sValue == null) ; else if (sValue.startsWith("@COL=")) { Object o = po.get_Value(sValue.substring(5)); // if(o != null) sRet = o.toString(); } else if (sValue.startsWith("@") && sValue.endsWith("@") && sValue.length() > 1) { sRet = Env.parseVariable (sValue, po, null, false); } return sRet; } /** * Compare Number * @param valueObj source value for comparison * @param value1 first value * @param value2 second value for between comparison (OPERATION_X) * @return true if operation */ private boolean compareNumber (Number valueObj, String value1, String value2) { BigDecimal valueObjB = null; BigDecimal value1B = null; BigDecimal value2B = null; try { if (valueObj instanceof BigDecimal) valueObjB = (BigDecimal)valueObj; else if (valueObj instanceof Integer) valueObjB = new BigDecimal (((Integer)valueObj).intValue()); else valueObjB = new BigDecimal (String.valueOf(valueObj)); } catch (Exception e) { if (log.isLoggable(Level.FINE)) log.fine("compareNumber - valueObj=" + valueObj + " - " + e.toString()); return compareString(valueObj, value1, value2); } try { value1B = new BigDecimal (value1); } catch (Exception e) { if (log.isLoggable(Level.FINE)) log.fine("compareNumber - value1=" + value1 + " - " + e.toString()); return compareString(valueObj, value1, value2); } String op = getOperation(); if (OPERATION_Eq.equals(op)) return valueObjB.compareTo(value1B) == 0; else if (OPERATION_Gt.equals(op)) return valueObjB.compareTo(value1B) > 0; else if (OPERATION_GtEq.equals(op)) return valueObjB.compareTo(value1B) >= 0; else if (OPERATION_Le.equals(op)) return valueObjB.compareTo(value1B) < 0; else if (OPERATION_LeEq.equals(op)) return valueObjB.compareTo(value1B) <= 0; else if (OPERATION_Like.equals(op)) return valueObjB.compareTo(value1B) == 0; else if (OPERATION_NotEq.equals(op)) return valueObjB.compareTo(value1B) != 0; // else if (OPERATION_Sql.equals(op)) throw new IllegalArgumentException("SQL not Implemented"); // else if (OPERATION_X.equals(op)) { if (valueObjB.compareTo(value1B) < 0) return false; // To try { value2B = new BigDecimal (String.valueOf(value2)); return valueObjB.compareTo(value2B) <= 0; } catch (Exception e) { if (log.isLoggable(Level.FINE)) log.fine("compareNumber - value2=" + value2 + " - " + e.toString()); return false; } } // throw new IllegalArgumentException("Unknown Operation=" + op); } // compareNumber /** * Compare String * @param valueObj source value for comparison * @param value1S first value * @param value2S second value for between comparison (OPERATION_X) * @return true if operation */ private boolean compareString (Object valueObj, String value1S, String value2S) { m_numeric = false; String valueObjS = String.valueOf(valueObj); // String op = getOperation(); if (OPERATION_Eq.equals(op)) return valueObjS.compareTo(value1S) == 0; else if (OPERATION_Gt.equals(op)) return valueObjS.compareTo(value1S) > 0; else if (OPERATION_GtEq.equals(op)) return valueObjS.compareTo(value1S) >= 0; else if (OPERATION_Le.equals(op)) return valueObjS.compareTo(value1S) < 0; else if (OPERATION_LeEq.equals(op)) return valueObjS.compareTo(value1S) <= 0; else if (OPERATION_Like.equals(op)) return valueObjS.compareTo(value1S) == 0; else if (OPERATION_NotEq.equals(op)) return valueObjS.compareTo(value1S) != 0; // else if (OPERATION_Sql.equals(op)) throw new IllegalArgumentException("SQL not Implemented"); // else if (OPERATION_X.equals(op)) { if (valueObjS.compareTo(value1S) < 0) return false; // To return valueObjS.compareTo(value2S) <= 0; } // throw new IllegalArgumentException("Unknown Operation=" + op); } // compareString /** * Compare Boolean * @param valueObj source value for comparison * @param value1S first value * @param value2S ignore * @return true if operation */ private boolean compareBoolean (Boolean valueObj, String value1S, String value2S) { m_numeric = false; // Boolean value1B = Boolean.valueOf(value1S) || "Y".equalsIgnoreCase(value1S); // String op = getOperation(); if (OPERATION_Eq.equals(op)) return valueObj.equals(value1B); else if (OPERATION_NotEq.equals(op)) return !valueObj.equals(value1B); else throw new IllegalArgumentException("Not Supported =" + op); } // compareBoolean /** * String Representation * @return info */ @Override public String toString () { StringBuilder sb = new StringBuilder ("MWFNextCondition["); sb.append(get_ID()).append(",SeqNo=").append(getSeqNo()) .append ("]"); return sb.toString (); } // toString @Override public MWFNextCondition markImmutable() { if (is_Immutable()) return this; makeImmutable(); return this; } /** * Get value from Column SQL (SQLStatement) instead of from AD_Column_ID * @param po * @return Value from Column SQL * @throws SQLException */ private Object getColumnSQLValue(PO po) throws SQLException { String columnSQL = getSQLStatement(); if (columnSQL.indexOf("@") >= 0) { columnSQL = Env.parseVariable(columnSQL, po, po.get_TrxName(), false); } String tableName = po.get_TableName(); String pkName = po.get_KeyColumns() != null && po.get_KeyColumns().length==1 ? po.get_KeyColumns()[0] : po.getUUIDColumnName(); String resultSql = String.format("SELECT (%s) FROM %s WHERE %s = ?", columnSQL, tableName, pkName); try (PreparedStatement pstmt = DB.prepareStatement(resultSql, ResultSet.TYPE_FORWARD_ONLY, ResultSet.CONCUR_READ_ONLY, po.get_TrxName())) { if (pkName.endsWith("_UU")) pstmt.setString(1, po.get_ValueAsString(po.getUUIDColumnName())); else pstmt.setInt(1, po.get_ID()); ResultSet rs = pstmt.executeQuery(); if (rs.next()) return rs.getObject(1); } return null; } @Override protected boolean beforeSave(boolean newRecord) { if (!Util.isEmpty(getSQLStatement(), true)) { setAD_Column_ID(0); } return true; } } // MWFNextCondition