Acumatica:能够覆盖DefaultEndpointImpl以将高级自定义逻辑添加到合同REST API中 [英] Acumatica: Ability to override DefaultEndpointImpl to add advanced custom logic to contract REST API

查看:62
本文介绍了Acumatica:能够覆盖DefaultEndpointImpl以将高级自定义逻辑添加到合同REST API中的处理方法,对大家解决问题具有一定的参考价值,需要的朋友们下面随着小编来一起学习吧!

问题描述

我想知道是否可以覆盖DefaultEndpointImpl.cs或在另一个文件中添加我自己的API逻辑?

I'm wondering if it possible to override DefaultEndpointImpl.cs or add my own API logic in another file?

我正在为几个API调用而苦苦挣扎要求此文件中的逻辑被覆盖或添加。例如,我能够通过api成功为采购订单创建购买收据,但是我无法以相同的方式添加转移收据购买收据。

I'm struggling with a few API calls which require the logic in this file to be overridden or added to. For example I am able to create a purchase receipt for a PO via the api successfully, however I'm not able to add a "Transfer Receipt" purchase receipt in the same way.

我在API端点中包含了许多非标准字段,这些字段引用了原始的转移,转移订单和发货,但是没有成功。 API调用成功,但未添加任何行。我已经能够获取生成的行以更新其数量,但是无法添加或删除当前行。

I've included various non-standard fields to the API endpoints that reference the original transfer, transfer order and shipment but have been unsuccessful. The API calls succeeds but no lines are added. I've been able to get lines to update their quantity once generated but can't add or delete current lines.

问题似乎源于此代码,这似乎添加可以正确处理添加图形的购买订单行逻辑但对其他收货类型不做任何特殊处理的功能。

The problem looks to stem from this code, which seems to add functionality that correctly handles adding Purchase order lines logic of the graph but doesn't do anything special for other receipt types.

[FieldsProcessed(new[] { "POLineNbr", "POOrderType", "POOrderNbr" })]
protected void PurchaseReceiptDetail_Insert(PXGraph graph, EntityImpl entity, EntityImpl targetEntity) {

    var receiptEntry = (POReceiptEntry)graph;

    var lineNbr = targetEntity.Fields.SingleOrDefault(f => f.Name == "POLineNbr") as EntityValueField;
    var orderType = targetEntity.Fields.SingleOrDefault(f => f.Name == "POOrderType") as EntityValueField;
    var orderNbr = targetEntity.Fields.SingleOrDefault(f => f.Name == "POOrderNbr") as EntityValueField;

    bool insertViaAddPO = lineNbr != null && orderNbr != null && orderType != null;

    if (!insertViaAddPO && (lineNbr != null || orderType != null || orderNbr != null)) {

        throw new PXException(PO.Messages.POTypeNbrLineNbrMustBeFilled);
    }

    var detailsCache = receiptEntry.transactions.Cache;

    if (insertViaAddPO) {

        receiptEntry.filter.Cache.Remove(receiptEntry.filter.Current);
        receiptEntry.filter.Cache.Insert(new POReceiptEntry.POOrderFilter());

        var filter = receiptEntry.filter.Current; var state = receiptEntry.filter.Cache.GetStateExt(filter, "OrderType") as PXStringState;

        if (state != null && state.AllowedLabels.Contains(orderType.Value)) {
            orderType.Value = state.ValueLabelDic.Single(p => p.Value == orderType.Value).Key;

        }

        receiptEntry.filter.Cache.SetValueExt(filter, "OrderType", orderType.Value);
        receiptEntry.filter.Cache.SetValueExt(filter, "OrderNbr", orderNbr.Value);
        receiptEntry.filter.Update(filter);

        var orders = receiptEntry.poLinesSelection.Select().Select(r => r.GetItem<POReceiptEntry.POLineS>());
        var order = orders.FirstOrDefault(o => o.LineNbr == int.Parse(lineNbr.Value));

        if (order == null) {
            throw new PXException(PO.Messages.PurchaseOrderLineNotFound);
        }

        order.Selected = true;

        receiptEntry.poLinesSelection.Update(order);
        receiptEntry.Actions["AddPOOrderLine2"].Press();

    } else {
        detailsCache.Current = detailsCache.Insert();
    }
}


推荐答案

之后经过反复试验,我设法使所有功能按需运行。但是,我担心的是,由于这不是文档化的信息,因此在不久的将来它可能会发生更改。

After a lot of trial and error I have managed to get everything working as I need. However I'm concerned that since this is not documented information that it may be subject to change in the near future.

DefaultEndpointImpl类中的方法是通过反射调用的。为了为我的默认终结点添加自定义功能,我扩展了默认类并为我的类和函数添加了必要的属性。

The methods in the class DefaultEndpointImpl are called via reflection. To add custom functionality for my default endpoint I have extended the default class and added the necessary attributes to my class and functions.

我的自定义类添加了以下内容:

My custom class adds the following:


  1. 覆盖默认的PurchaseReceiptDetail_Insert函数以添加转账收据功能。

  2. 添加ReceiptDetail_Insert方法以允许添加IN收据引用

第一个很简单,我对结果感到满意。第二个程序花了很多时间,调试起来很正确,很hacky,可能不适用于串行跟踪的项目。花费这么长时间的原因是因为有许多未知数,而我基本上是根据基类中的代码进行猜测的。例如,我什至不确定此功能在哪个阶段启动。看起来这些功能完全覆盖了其他一些不可见的默认逻辑。

The first one was pretty straight forward and I'm happy with the outcome. The second one took a lot of time and debugging to get right, is hacky and probably doesn't work with serial tracked items. The reason it's taken so long is because there are many unknowns and I've essentially been guessing based on the code in the base class. For example I'm not even sure at which stage this function fires. It looks the functions completely override some other default logic which is invisible.

因此,不管您选择还是保留它!也许我会联系Acumatica,看看是否可以获得更多信息。

So take it or leave it! Maybe I will reach out to Acumatica to see if I can get more information.

using PX.Api; 
using PX.Api.ContractBased; 
using PX.Api.ContractBased.Models; 
using PX.Data;
using PX.Objects.PO;
using PX.Objects.IN;
using System;
using System.Linq;
using PX.Objects.CM;
using PX.Objects.CS;

namespace AcuStock
{
  [PXVersion("5.30.001", "AcuStock")]  
  [PXVersion("6.00.001", "AcuStock")]  
  [PXVersion("17.200.001", "AcuStock")]  
  public class AcuStockEndpointImpl : PX.Objects.DefaultEndpointImpl
  {

    [FieldsProcessed(new[] { "POLineNbr", "POOrderType", "POOrderNbr", "OrigLineNbr", "OrigRefNbr" })]
    protected new void PurchaseReceiptDetail_Insert(PXGraph graph, EntityImpl entity, EntityImpl targetEntity){

        var lineNbr = targetEntity.Fields.SingleOrDefault(f => f.Name == "OrigLineNbr") as EntityValueField;
        var refNbr = targetEntity.Fields.SingleOrDefault(f => f.Name == "OrigRefNbr") as EntityValueField;
        var receiptQty = targetEntity.Fields.SingleOrDefault(f => f.Name == "ReceiptQty") as EntityValueField;
        var location = targetEntity.Fields.SingleOrDefault(f => f.Name == "Location") as EntityValueField;
        var inventoryID = targetEntity.Fields.SingleOrDefault(f => f.Name == "InventoryID") as EntityValueField;

        bool insertViaAddTR = lineNbr != null && refNbr != null;

        var receiptEntry = (POReceiptEntry) graph;

        if (insertViaAddTR){

            receiptEntry.addReceipt.Cache.Remove(receiptEntry.addReceipt.Current);
            receiptEntry.addReceipt.Cache.Insert(new POReceiptEntry.POReceiptLineS());

            var filter = receiptEntry.addReceipt.Current;

            receiptEntry.addReceipt.Cache.SetValueExt(filter, "OrigRefNbr", refNbr.Value);
            receiptEntry.addReceipt.Cache.SetValueExt(filter, "OrigLineNbr", lineNbr.Value);
            receiptEntry.addReceipt.Cache.SetValueExt(filter, "InventoryID", inventoryID.Value);
            receiptEntry.addReceipt.Cache.SetValueExt(filter, "ReceiptQty", receiptQty.Value);
            if (location != null)
                receiptEntry.addReceipt.Cache.SetValueExt(filter, "LocationID", location.Value);
            receiptEntry.addReceipt.Update(filter);

            var lines = receiptEntry.addReceipt.Select().Select(r => r.GetItem<POReceiptEntry.POReceiptLineS>());
            var line = lines.FirstOrDefault(o => o.OrigLineNbr == int.Parse(lineNbr.Value));

            if (line == null){
                throw new PXException(PX.Objects.PO.Messages.PurchaseOrderLineNotFound);
            }

            receiptEntry.addPOReceiptLine2.Press();

        } else {
            base.PurchaseReceiptDetail_Insert(graph, entity, targetEntity);
        }

        var allocations = (targetEntity.Fields.SingleOrDefault(f => string.Equals(f.Name, "Allocations")) as EntityListField).Value ?? new EntityImpl[0];

        if (allocations.Any(a => a.Fields != null && a.Fields.Length > 0)){
            // Remove automatically added allocation
            if (receiptEntry.splits.Current != null){
                receiptEntry.splits.Delete(receiptEntry.splits.Current);
            }
    }

    }

    [FieldsProcessed(new[] { "OrigLineNbr", "OrigRefNbr", "Quantity", "Location" })]
    protected void ReceiptDetails_Insert(PXGraph graph, EntityImpl entity, EntityImpl targetEntity) {

        var lineNbr = targetEntity.Fields.SingleOrDefault(f => f.Name == "OrigLineNbr") as EntityValueField;
        var receiptQty = targetEntity.Fields.SingleOrDefault(f => f.Name == "Quantity") as EntityValueField;
        var location = targetEntity.Fields.SingleOrDefault(f => f.Name == "Location") as EntityValueField;

        var allocations = (targetEntity.Fields.SingleOrDefault(f => string.Equals(f.Name, "Allocations")) as EntityListField).Value ?? new EntityImpl[0];
        var hasAllocations = allocations.Any(a => a.Fields != null && a.Fields.Length > 0);

        var receiptEntry = (INReceiptEntry) graph;

        string transferNbr = receiptEntry.receipt.Current.TransferNbr;

        var detailsCache = receiptEntry.transactions.Cache;

        if (lineNbr == null || transferNbr == null){

            detailsCache.Current = detailsCache.Insert();
            return;
        }

        INTran newtran = null;
        decimal newtranqty = Decimal.Parse(receiptQty.Value);
        decimal newtrancost = 0m;
        receiptEntry.ParseSubItemSegKeys();

        using (new PXReadBranchRestrictedScope())
        {
            foreach (PXResult<INTransitLine, INLocationStatus2, INTransitLineLotSerialStatus, INSite, InventoryItem, INTran> res in
                PXSelectJoin<INTransitLine,
                InnerJoin<INLocationStatus2, On<INLocationStatus2.locationID, Equal<INTransitLine.costSiteID>>,
                LeftJoin<INTransitLineLotSerialStatus,
                        On<INTransitLine.transferNbr, Equal<INTransitLineLotSerialStatus.transferNbr>,
                        And<INTransitLine.transferLineNbr, Equal<INTransitLineLotSerialStatus.transferLineNbr>>>,
                InnerJoin<INSite, On<INSite.siteID, Equal<INTransitLine.toSiteID>>,
                InnerJoin<InventoryItem, On<InventoryItem.inventoryID, Equal<INLocationStatus2.inventoryID>>,
                InnerJoin<INTran,
                    On<INTran.docType, Equal<INDocType.transfer>,
                    And<INTran.refNbr, Equal<INTransitLine.transferNbr>,
                    And<INTran.lineNbr, Equal<INTransitLine.transferLineNbr>,
                    And<INTran.invtMult, Equal<shortMinus1>>>>>>>>>>,
                Where<INTransitLine.transferNbr, Equal<Required<INTransitLine.transferNbr>>, And<INTransitLine.transferLineNbr, Equal<Required<INTransitLine.transferLineNbr>>>>,
                OrderBy<Asc<INTransitLine.transferNbr, Asc<INTransitLine.transferLineNbr>>>>
                .Select(receiptEntry, transferNbr, lineNbr.Value))
            {
                INTransitLine transitline = res;
                INLocationStatus2 stat = res;
                INTransitLineLotSerialStatus lotstat = res;
                INSite site = res;
                InventoryItem item = res;
                INTran tran = res; 

                if (stat.QtyOnHand == 0m || (lotstat != null && lotstat.QtyOnHand == 0m))
                    continue;

                if (newtran == null) { 

                    if (!object.Equals(receiptEntry.receipt.Current.BranchID, site.BranchID))
                    {
                        INRegister copy = PXCache<INRegister>.CreateCopy(receiptEntry.receipt.Current);
                        copy.BranchID = site.BranchID;
                        receiptEntry.receipt.Update(copy);
                    }

                    newtran = PXCache<INTran>.CreateCopy(tran);
                    newtran.OrigBranchID = newtran.BranchID;
                    newtran.OrigTranType = newtran.TranType;
                    newtran.OrigRefNbr = transitline.TransferNbr;
                    newtran.OrigLineNbr = transitline.TransferLineNbr;
                    newtran.BranchID = site.BranchID;
                    newtran.DocType = receiptEntry.receipt.Current.DocType;
                    newtran.RefNbr = receiptEntry.receipt.Current.RefNbr;
                    newtran.LineNbr = (int)PXLineNbrAttribute.NewLineNbr<INTran.lineNbr>(receiptEntry.transactions.Cache, receiptEntry.receipt.Current);
                    newtran.InvtMult = (short)1;
                    newtran.SiteID = transitline.ToSiteID;
                    newtran.LocationID = transitline.ToLocationID;
                    newtran.ToSiteID = null;
                    newtran.ToLocationID = null;
                    newtran.BaseQty = 0m;
                    newtran.Qty = 0m;
                    newtran.UnitCost = 0m;
                    newtran.Released = false;
                    newtran.InvtAcctID = null;
                    newtran.InvtSubID = null;
                    newtran.ReasonCode = null;
                    newtran.ARDocType = null;
                    newtran.ARRefNbr = null;
                    newtran.ARLineNbr = null;
                    newtran.ProjectID = null;
                    newtran.TaskID = null;
                    newtran.CostCodeID = null;
                    newtran.TranCost = 0m;

                    receiptEntry.splits.Current = null;

                    newtran = receiptEntry.transactions.Insert(newtran);

                    receiptEntry.transactions.Current = newtran;

                    if (receiptEntry.splits.Current != null)
                    {
                        receiptEntry.splits.Delete(receiptEntry.splits.Current);
                    }
                }

                if (hasAllocations){

                    newtranqty = 0m;

                    foreach (var allocation in allocations) { 

                        var newsplitqty = allocation.Fields.SingleOrDefault(f => f.Name == "Quantity") as EntityValueField;
                        var newsplitlocation = allocation.Fields.SingleOrDefault(f => f.Name == "Location") as EntityValueField;

                        INTranSplit newsplit = this.addReceiptSplitLine(receiptEntry, stat, lotstat, transitline, newtran, item, Decimal.Parse(newsplitqty.Value), newsplitlocation.Value);

                        newtrancost += newsplit.BaseQty.Value * newsplit.UnitCost.Value;
                        newtranqty += newsplit.BaseQty.Value;
                    }

                    break;

                } else {

                    INTranSplit newsplit = this.addReceiptSplitLine(receiptEntry, stat, lotstat, transitline, tran, item, newtranqty, null);

                    newtrancost += newsplit.BaseQty.Value * newsplit.UnitCost.Value;
                    newtranqty += newsplit.BaseQty.Value;

                }

            }

            receiptEntry.UpdateTranCostQty(newtran, newtranqty, newtrancost);

        }

    }

    [FieldsProcessed(new[] { "OrigLineNbr" })]
    protected void ReceiptAllocations_Insert(PXGraph graph, EntityImpl entity, EntityImpl targetEntity) {
        // no-op
    }

    private INTranSplit addReceiptSplitLine(INReceiptEntry receiptEntry, INLocationStatus2 stat, INTransitLineLotSerialStatus lotstat, INTransitLine transitline, INTran tran, InventoryItem item, Decimal qty, string location){

        INTranSplit newsplit;
        decimal newsplitqty;
        if (lotstat.QtyOnHand == null)
        {
            newsplit = new INTranSplit();
            newsplit.InventoryID = stat.InventoryID;
            newsplit.IsStockItem = true;
            newsplit.FromSiteID = transitline.SiteID;
            newsplit.SubItemID = stat.SubItemID;
            newsplit.LotSerialNbr = null;
            newsplitqty = qty;
        }
        else
        {
            newsplit = new INTranSplit();
            newsplit.InventoryID = lotstat.InventoryID;
            newsplit.IsStockItem = true;
            newsplit.FromSiteID = lotstat.FromSiteID;
            newsplit.SubItemID = lotstat.SubItemID;
            newsplit.LotSerialNbr = lotstat.LotSerialNbr;
            newsplitqty = qty;
        }

        newsplit.DocType = receiptEntry.receipt.Current.DocType;
        newsplit.RefNbr = receiptEntry.receipt.Current.RefNbr;
        newsplit.LineNbr = tran.LineNbr;
        newsplit.SplitLineNbr = (int)PXLineNbrAttribute.NewLineNbr<INTranSplit.splitLineNbr>(receiptEntry.splits.Cache, receiptEntry.receipt.Current);

        newsplit.UnitCost = 0m;
        newsplit.InvtMult = (short)1;
        newsplit.SiteID = transitline.ToSiteID;
        newsplit.PlanID = null;
        newsplit.Released = false;
        newsplit.ProjectID = null;
        newsplit.TaskID = null;

        if (location == null) { 
            newsplit.LocationID = lotstat.ToLocationID ?? transitline.ToLocationID;
        } else { 
            receiptEntry.splits.SetValueExt<INTranSplit.locationID>(newsplit, location);
        }

        newsplit = receiptEntry.splits.Insert(newsplit);

        newsplit.MaxTransferBaseQty = newsplitqty;
        newsplit.BaseQty = newsplitqty;
        newsplit.Qty = newsplit.BaseQty.Value;

        receiptEntry.UpdateCostSubItemID(newsplit, item);

        receiptEntry.SetCostAttributes(tran, newsplit, item, tran.OrigRefNbr);
        newsplit.UnitCost = PXCurrencyAttribute.BaseRound(receiptEntry, newsplit.UnitCost);
        receiptEntry.splits.Update(newsplit);

        return newsplit;
    }

  }
}

这篇关于Acumatica:能够覆盖DefaultEndpointImpl以将高级自定义逻辑添加到合同REST API中的文章就介绍到这了,希望我们推荐的答案对大家有所帮助,也希望大家多多支持IT屋!

查看全文
登录 关闭
扫码关注1秒登录
发送“验证码”获取 | 15天全站免登陆