/****************************************************************************** * 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.process; import java.sql.Timestamp; import java.util.ArrayList; import java.util.HashMap; import java.util.List; import java.util.logging.Level; import org.adempiere.exceptions.NoVendorForProductException; import org.apache.commons.collections.keyvalue.MultiKey; import org.compiere.model.MBPartner; import org.compiere.model.MCharge; import org.compiere.model.MConversionType; import org.compiere.model.MOrder; import org.compiere.model.MOrderLine; import org.compiere.model.MProcessPara; import org.compiere.model.MProduct; import org.compiere.model.MProductPO; import org.compiere.model.MRequisition; import org.compiere.model.MRequisitionLine; import org.compiere.model.POResultSet; import org.compiere.model.Query; import org.compiere.util.AdempiereUserError; import org.compiere.util.DB; import org.compiere.util.Msg; /** * Create PO from Requisition * * * @author Jorg Janke * @version $Id: RequisitionPOCreate.java,v 1.2 2006/07/30 00:51:01 jjanke Exp $ * * @author Teo Sarca, www.arhipac.ro *
  • BF [ 2609760 ] RequisitionPOCreate not using DateRequired *
  • BF [ 2605888 ] CreatePOfromRequisition creates more PO than needed *
  • BF [ 2811718 ] Create PO from Requisition without any parameter terminate in NPE * https://sourceforge.net/p/adempiere/bugs/1954/ *
  • FR [ 2844074 ] Requisition PO Create - more selection fields * https://sourceforge.net/p/adempiere/feature-requests/796/ */ @org.adempiere.base.annotation.Process public class RequisitionPOCreate extends SvrProcess { /** Org */ private int p_AD_Org_ID = 0; /** Warehouse */ private int p_M_Warehouse_ID = 0; /** Doc Date From */ private Timestamp p_DateDoc_From; /** Doc Date To */ private Timestamp p_DateDoc_To; /** Doc Date From */ private Timestamp p_DateRequired_From; /** Doc Date To */ private Timestamp p_DateRequired_To; /** Priority */ private String p_PriorityRule = null; /** User */ private int p_AD_User_ID = 0; /** Product */ private int p_M_Product_ID = 0; /** Product Category */ private int p_M_Product_Category_ID = 0; /** BPartner Group */ private int p_C_BP_Group_ID = 0; /** Requisition */ private int p_M_Requisition_ID = 0; /** Consolidate */ private boolean p_ConsolidateDocument = false; /** Order */ private MOrder m_order = null; /** Order Line */ private MOrderLine m_orderLine = null; /** Orders Cache : (C_BPartner_ID, DateRequired, M_PriceList_ID) -> MOrder */ private HashMap m_cacheOrders = new HashMap(); /** * Prepare - e.g., get Parameters. */ protected void prepare() { ProcessInfoParameter[] para = getParameter(); for (int i = 0; i < para.length; i++) { String name = para[i].getParameterName(); if (para[i].getParameter() == null && para[i].getParameter_To() == null) ; else if (name.equals("AD_Org_ID")) p_AD_Org_ID = para[i].getParameterAsInt(); else if (name.equals("M_Warehouse_ID")) p_M_Warehouse_ID = para[i].getParameterAsInt(); else if (name.equals("DateDoc")) { p_DateDoc_From = (Timestamp)para[i].getParameter(); p_DateDoc_To = (Timestamp)para[i].getParameter_To(); } else if (name.equals("DateRequired")) { p_DateRequired_From = (Timestamp)para[i].getParameter(); p_DateRequired_To = (Timestamp)para[i].getParameter_To(); } else if (name.equals("PriorityRule")) p_PriorityRule = (String)para[i].getParameter(); else if (name.equals("AD_User_ID")) p_AD_User_ID = para[i].getParameterAsInt(); else if (name.equals("M_Product_ID")) p_M_Product_ID = para[i].getParameterAsInt(); else if (name.equals("M_Product_Category_ID")) p_M_Product_Category_ID = para[i].getParameterAsInt(); else if (name.equals("C_BP_Group_ID")) p_C_BP_Group_ID = para[i].getParameterAsInt(); else if (name.equals("M_Requisition_ID")) p_M_Requisition_ID = para[i].getParameterAsInt(); else if (name.equals("ConsolidateDocument")) p_ConsolidateDocument = "Y".equals(para[i].getParameter()); else MProcessPara.validateUnknownParameter(getProcessInfo().getAD_Process_ID(), para[i]); } } // prepare /** * Process * @return info * @throws Exception */ protected String doIt() throws Exception { // Specific if (p_M_Requisition_ID != 0) { if (log.isLoggable(Level.INFO)) log.info("M_Requisition_ID=" + p_M_Requisition_ID); MRequisition req = new MRequisition(getCtx(), p_M_Requisition_ID, get_TrxName()); if (!MRequisition.DOCSTATUS_Completed.equals(req.getDocStatus())) { throw new AdempiereUserError("@DocStatus@ = " + req.getDocStatus()); } MRequisitionLine[] lines = req.getLines(); for (int i = 0; i < lines.length; i++) { if (lines[i].getC_OrderLine_ID() == 0) { process (lines[i]); } } closeOrder(); return ""; } // single Requisition // if (log.isLoggable(Level.INFO)) log.info("AD_Org_ID=" + p_AD_Org_ID + ", M_Warehouse_ID=" + p_M_Warehouse_ID + ", DateDoc=" + p_DateDoc_From + "/" + p_DateDoc_To + ", DateRequired=" + p_DateRequired_From + "/" + p_DateRequired_To + ", PriorityRule=" + p_PriorityRule + ", AD_User_ID=" + p_AD_User_ID + ", M_Product_ID=" + p_M_Product_ID + ", ConsolidateDocument" + p_ConsolidateDocument); ArrayList params = new ArrayList(); StringBuilder whereClause = new StringBuilder("C_OrderLine_ID IS NULL"); if (p_AD_Org_ID > 0) { whereClause.append(" AND AD_Org_ID=?"); params.add(p_AD_Org_ID); } if (p_M_Product_ID > 0) { whereClause.append(" AND M_Product_ID=?"); params.add(p_M_Product_ID); } else if (p_M_Product_Category_ID > 0) { whereClause.append(" AND EXISTS (SELECT 1 FROM M_Product p WHERE M_RequisitionLine.M_Product_ID=p.M_Product_ID") .append(" AND p.M_Product_Category_ID=?)"); params.add(p_M_Product_Category_ID); } if (p_C_BP_Group_ID > 0) { whereClause.append(" AND (") .append("M_RequisitionLine.C_BPartner_ID IS NULL") .append(" OR EXISTS (SELECT 1 FROM C_BPartner bp WHERE M_RequisitionLine.C_BPartner_ID=bp.C_BPartner_ID AND bp.C_BP_Group_ID=?)") .append(")"); params.add(p_C_BP_Group_ID); } // // Requisition Header whereClause.append(" AND EXISTS (SELECT 1 FROM M_Requisition r WHERE M_RequisitionLine.M_Requisition_ID=r.M_Requisition_ID") .append(" AND r.DocStatus=?"); params.add(MRequisition.DOCSTATUS_Completed); if (p_M_Warehouse_ID > 0) { whereClause.append(" AND r.M_Warehouse_ID=?"); params.add(p_M_Warehouse_ID); } if (p_DateDoc_From != null) { whereClause.append(" AND r.DateDoc >= ?"); params.add(p_DateDoc_From); } if (p_DateDoc_To != null) { whereClause.append(" AND r.DateDoc <= ?"); params.add(p_DateDoc_To); } if (p_DateRequired_From != null) { whereClause.append(" AND r.DateRequired >= ?"); params.add(p_DateRequired_From); } if (p_DateRequired_To != null) { whereClause.append(" AND r.DateRequired <= ?"); params.add(p_DateRequired_To); } if (p_PriorityRule != null) { whereClause.append(" AND r.PriorityRule >= ?"); params.add(p_PriorityRule); } if (p_AD_User_ID > 0) { whereClause.append(" AND r.AD_User_ID=?"); params.add(p_AD_User_ID); } whereClause.append(")"); // End Requisition Header // // ORDER BY clause StringBuilder orderClause = new StringBuilder(); if (!p_ConsolidateDocument) { orderClause.append("M_Requisition_ID, "); } orderClause.append("(SELECT DateRequired FROM M_Requisition r WHERE M_RequisitionLine.M_Requisition_ID=r.M_Requisition_ID),"); orderClause.append("M_Product_ID, C_Charge_ID, M_AttributeSetInstance_ID"); POResultSet rs = new Query(getCtx(), MRequisitionLine.Table_Name, whereClause.toString(), get_TrxName()) .setParameters(params) .setOrderBy(orderClause.toString()) .setClient_ID() .scroll(); try { while (rs.hasNext()) { process(rs.next()); } } finally { DB.close(rs); rs = null; } closeOrder(); return ""; } // doit private int m_M_Requisition_ID = 0; private int m_M_Product_ID = 0; private int m_M_AttributeSetInstance_ID = 0; /** BPartner */ private MBPartner m_bpartner = null; /** * Process Line * @param rLine request line * @throws Exception */ private void process (MRequisitionLine rLine) throws Exception { if (rLine.getM_Product_ID() == 0 && rLine.getC_Charge_ID() == 0) { log.warning("Ignored Line" + rLine.getLine() + " " + rLine.getDescription() + " - " + rLine.getLineNetAmt()); return; } if (!p_ConsolidateDocument && rLine.getM_Requisition_ID() != m_M_Requisition_ID) { closeOrder(); } if (m_orderLine == null || rLine.getM_Product_ID() != m_M_Product_ID || rLine.getM_AttributeSetInstance_ID() != m_M_AttributeSetInstance_ID || rLine.getC_Charge_ID() != 0 // single line per charge || m_order == null || (rLine.getC_BPartner_ID() > 0 && m_order.getC_BPartner_ID() != rLine.getC_BPartner_ID()) || m_order.getDatePromised().compareTo(rLine.getDateRequired()) != 0 ) { newLine(rLine); // No Order Line was produced (vendor was not valid/allowed) => SKIP if (m_orderLine == null) return; } // Update Order Line m_orderLine.setQty(m_orderLine.getQtyOrdered().add(rLine.getQty())); // Update Requisition Line rLine.setC_OrderLine_ID(m_orderLine.getC_OrderLine_ID()); rLine.saveEx(); } // process /** * Create new Order * @param rLine request line * @param C_BPartner_ID b.partner * @throws Exception */ private void newOrder(MRequisitionLine rLine, int C_BPartner_ID) throws Exception { if (m_order != null) { closeOrder(); } // BPartner if (m_bpartner == null || C_BPartner_ID != m_bpartner.get_ID()) { m_bpartner = MBPartner.get(getCtx(), C_BPartner_ID); } // Order Timestamp DateRequired = rLine.getDateRequired(); int M_PriceList_ID = rLine.getParent().getM_PriceList_ID(); MultiKey key = new MultiKey(C_BPartner_ID, DateRequired, M_PriceList_ID); m_order = m_cacheOrders.get(key); if (m_order == null) { m_order = new MOrder(getCtx(), 0, get_TrxName()); m_order.setAD_Org_ID(rLine.getAD_Org_ID()); m_order.setM_Warehouse_ID(rLine.getParent().getM_Warehouse_ID()); m_order.setDatePromised(DateRequired); m_order.setIsSOTrx(false); m_order.setC_DocTypeTarget_ID(); m_order.setBPartner(m_bpartner); m_order.setM_PriceList_ID(M_PriceList_ID); if (MConversionType.getDefault(getAD_Client_ID()) > 0) m_order.setC_ConversionType_ID(MConversionType.getDefault(getAD_Client_ID())); // default po document type if (!p_ConsolidateDocument) { StringBuilder msgsd= new StringBuilder().append(Msg.getElement(getCtx(), "M_Requisition_ID")) .append(": ").append(rLine.getParent().getDocumentNo()); m_order.setDescription(msgsd.toString()); } // Prepare Save m_order.saveEx(); // Put to cache m_cacheOrders.put(key, m_order); } m_M_Requisition_ID = rLine.getM_Requisition_ID(); } // newOrder /** * Close Order * @throws Exception */ private void closeOrder() throws Exception { if (m_orderLine != null) { m_orderLine.saveEx(); } if (m_order != null) { m_order.load(get_TrxName()); String message = Msg.parseTranslation(getCtx(), "@GeneratedPO@ " + m_order.getDocumentNo()); addBufferLog(0, null, m_order.getGrandTotal(), message, m_order.get_Table_ID(), m_order.getC_Order_ID()); } m_order = null; m_orderLine = null; } // closeOrder /** * New Order Line (different Product) * @param rLine request line * @throws Exception */ private void newLine(MRequisitionLine rLine) throws Exception { if (m_orderLine != null) { m_orderLine.saveEx(); } m_orderLine = null; MProduct product = MProduct.get(getCtx(), rLine.getM_Product_ID()); // Get Business Partner int C_BPartner_ID = rLine.getC_BPartner_ID(); if (C_BPartner_ID != 0) { ; } else if (rLine.getC_Charge_ID() != 0) { MCharge charge = MCharge.get(getCtx(), rLine.getC_Charge_ID()); C_BPartner_ID = charge.getC_BPartner_ID(); if (C_BPartner_ID == 0) { throw new AdempiereUserError("No Vendor for Charge " + charge.getName()); } } else { // Find Strategic Vendor for Product // TODO: refactor MProductPO[] ppos = MProductPO.getOfProduct(getCtx(), product.getM_Product_ID(), null); for (int i = 0; i < ppos.length; i++) { if (ppos[i].isCurrentVendor() && ppos[i].getC_BPartner_ID() != 0) { C_BPartner_ID = ppos[i].getC_BPartner_ID(); break; } } if (C_BPartner_ID == 0 && ppos.length > 0) { C_BPartner_ID = ppos[0].getC_BPartner_ID(); } if (C_BPartner_ID == 0) { throw new NoVendorForProductException(product.getName()); } } if (!isGenerateForVendor(C_BPartner_ID)) { if (log.isLoggable(Level.INFO)) log.info("Skip for partner "+C_BPartner_ID); return; } // New Order - Different Vendor if (m_order == null || m_order.getC_BPartner_ID() != C_BPartner_ID || m_order.getDatePromised().compareTo(rLine.getDateRequired()) != 0 ) { newOrder(rLine, C_BPartner_ID); } // No Order Line m_orderLine = new MOrderLine(m_order); m_orderLine.setDatePromised(rLine.getDateRequired()); if (product != null) { m_orderLine.setProduct(product); m_orderLine.setM_AttributeSetInstance_ID(rLine.getM_AttributeSetInstance_ID()); } else { m_orderLine.setC_Charge_ID(rLine.getC_Charge_ID()); m_orderLine.setPriceActual(rLine.getPriceActual()); } m_orderLine.setAD_Org_ID(rLine.getAD_Org_ID()); // Prepare Save m_M_Product_ID = rLine.getM_Product_ID(); m_M_AttributeSetInstance_ID = rLine.getM_AttributeSetInstance_ID(); m_orderLine.saveEx(); } // newLine /** * Do we need to generate Purchase Orders for given Vendor * @param C_BPartner_ID * @return true if it's allowed */ private boolean isGenerateForVendor(int C_BPartner_ID) { // No filter group was set => generate for all vendors if (p_C_BP_Group_ID <= 0) return true; if (m_excludedVendors.contains(C_BPartner_ID)) return false; // boolean match = new Query(getCtx(), MBPartner.Table_Name, "C_BPartner_ID=? AND C_BP_Group_ID=?", get_TrxName()) .setParameters(new Object[]{C_BPartner_ID, p_C_BP_Group_ID}) .match(); if (!match) { m_excludedVendors.add(C_BPartner_ID); } return match; } private List m_excludedVendors = new ArrayList(); } // RequisitionPOCreate