Skip to main content
Question

Create ARInvoice from Webhook


asommer
Jr Varsity III
Forum|alt.badge.img

Good day,

I’m working on a webhook for integration with a third party.  The webhook runs find until I try to create an ARInvoice and add an ARTran to the InvoiceEntry.   

The error I’m getting is in the SM_ARInvoiceEntry.cs file IsLineCreatedFromAppSO method.  The base.Base.Accessinfo.ScreenID is null.

We have a separate integration using a custom screen with a custom Action, but using webhooks will be much better for our purposes.  Is there a way I can set the ScreenID from a webhook in order to create the ARInvoice with ARTran items?

 

Here is the webhook code:

 

namespace EcrsWebhook
{
    public class EcrsWebhookHandler : PXGraph<ARInvoiceEntry>, IWebhookHandler
    {
        private static readonly Newtonsoft.Json.JsonSerializer Serializer = new JsonSerializer();
        decimal INFRA_DISCOUNT = (decimal)0.15;

        public async Task HandleAsync(WebhookContext context, CancellationToken cancellation)
        {

            ECRSConfigs MAXIO_KEY;

            using (JsonTextReader reader = new JsonTextReader(context.Request.CreateTextReader()))
            {
                // Create the PXGraph for CustomerMaint and ARInvoiceEntry.
                CustomerMaint customerMaint = PXGraph.CreateInstance<CustomerMaint>();
                ARInvoiceEntry invoiceEntry = PXGraph.CreateInstance<ARInvoiceEntry>();

                MAXIO_KEY = SelectFrom<ECRSConfigs>.
                    Where<ECRSConfigs.configKey.IsEqual<PX.Data.BQL.@P.AsString>>.View.
                    Select(customerMaint, "maxio_key");

                string hmac = EcrsCrypto.CalculateHMAC_SHA256(MAXIO_KEY.ConfigValue, context.Response.Body.ToString());
                string reqKey = context.Request.Query["signature_hmac_sha_25"];

                // Check the request signature.
                if (hmac != reqKey)
                {
                    context.Response.StatusCode = 401;
                    return;
                }

                // Parse the JSON into an object.
                PaymentSuccess paymentSuccess = Serializer.Deserialize<PaymentSuccess>(reader);

                // Find the Customer.
                Customer customer = SelectFrom<Customer>.
                    Where<Customer.acctCD.IsEqual<PX.Data.BQL.@P.AsString>>.View.
                    Select(customerMaint, $"RC{paymentSuccess.transaction.customer_id.ToString()}");

                int subscriptionId = paymentSuccess.subscription.id;
                int productId = paymentSuccess.subscription.product.id;

                // Find the InventoryItem for the Product.
                InventoryItem inventoryItem = SelectFrom<InventoryItem>.
                    Where<InventoryItem.inventoryCD.IsEqual<PX.Data.BQL.@P.AsString>>.View.
                    Select(customerMaint, $"REC{paymentSuccess.subscription.product.id.ToString()}");


                // Calculate the Financial Period.
                int finMonth = DateTime.Now.Month;
                string strMonth = "";
                if (finMonth < 10)
                {
                    strMonth = $"0{finMonth}";
                }
                else
                {
                    strMonth = finMonth.ToString();
                }
                string finPeriod = DateTime.Now.Year.ToString() + strMonth;

                // Get the last 4 of the CC to put in the description.
                string creditCardFinal4 = "";
                creditCardFinal4 = paymentSuccess.subscription.credit_card.masked_card_number;
                creditCardFinal4 = creditCardFinal4.Split('-').Last();

                string description = $"SubscriptionID: {paymentSuccess.subscription.id}, CC: {creditCardFinal4}, EventID: {paymentSuccess.event_id}";
                DateTime today = DateTime.Now;

                // Create new Invoice.
                ARInvoice invoice = new ARInvoice();
                invoice.CustomerID = customer.BAccountID;
                invoice.CustomerLocationID = customer.DefLocationID;
                invoice.DocDate = today;
                invoice.DueDate = today;
                invoice.DiscDate = today;
                invoice.FinPeriodID = finPeriod;
                invoice.ProjectID = 0;
                invoice.BranchID = 2;
                invoice.TermsID = "PREPAID";
                invoice.DocType = "INV";
                invoice.ARAccountID = 10;
                invoice.ARSubID = 2;
                invoice.Status = "B";
                invoice.TaxZoneID = "AVALARA";
                invoice.DocDesc = description;
                invoice.CreatedByScreenID = "AR301000";

                // Get the price after applying INFRA coupon if applicable.
                decimal productPrice = paymentSuccess.subscription.product.price_in_cents;
                foreach (String couponCode in paymentSuccess.subscription.coupon_codes)
                {
                    if (couponCode == "INFRA")
                    {
                        productPrice = productPrice - decimal.Multiply(productPrice, INFRA_DISCOUNT);
                    }
                }

                ARTran tran = new ARTran
                {
                    RefNbr = invoice.RefNbr,
                    InventoryID = inventoryItem.InventoryID,
                    Qty = 1,
                    CuryUnitPrice = productPrice,
                    CustomerID = customer.BAccountID,
                    AccountID = 89,
                    SubID = 150,
                    BranchID = 2,
                    TaxCategoryID = "SC070100",
                    CreatedByScreenID = "AR301000"
                };

                invoice = invoiceEntry.ARInvoice_CustomerID_DocType_RefNbr.Update(invoice);
                //invoiceEntry.Transactions.Cache.Update(tran);
                invoiceEntry.Transactions.Update(tran);


                invoiceEntry.Save.Press();
            }
        }
    }
}

 

Thanks All!

12 replies

Forum|alt.badge.img+6
  • Captain II
  • 563 replies
  • March 31, 2025

You shouldn’t need to set CreatedbyScreenID at all - that will happen behind the scenes from the PXGraph that you’re using.

It looks like you never make use of the ARInvoiceEntry in your logic. e.g.:

invoice = invoiceEntry.Document.Insert(invoice);

 

 


asommer
Jr Varsity III
Forum|alt.badge.img
  • Author
  • Jr Varsity III
  • 27 replies
  • March 31, 2025

Thanks for you reply Django. Maybe I’m not doing things in the correct order… I’m creating an ARInvoiceEntry object, then creating an ARInvoice object to add to the ARInvoiceEntry, and finally creating an ARTran object to add to the ARInvoiceEntry.  Is that correct?

The crash happens on the “invoiceEntry.Transactions.Update(tran);” line which goes into the IsLineCreatedFromAppSO method in SM_ARInvoiceEntry.cs and the exact error is:

 

Object reference not set to an instance of an object.

   at PX.Objects.FS.SM_ARInvoiceEntry.IsLineCreatedFromAppSO(PXGraph cleanerGraph, Object document, Object lineDoc, String fieldName) in C:\build\code_repo\WebSites\Pure\PX.Objects.FS\CustomBLC\SM_ARInvoiceEntry.cs:line 711

 

Thanks again!


Forum|alt.badge.img+6
  • Captain II
  • 563 replies
  • March 31, 2025

I’m creating an ARInvoiceEntry object, then creating an ARInvoice object to add to the ARInvoiceEntry, and finally creating an ARTran object to add to the ARInvoiceEntry.  Is that correct?

 

That’s the correct way - but you haven’t connected the two properly.

There are a few different ways to create the ARInvoice record. The way you’ve selected works just fine. Typically, you’ll want to populate the key fields and then call Insert.

ARInvoice invoice = new ARInvoice();
invoice.DocType = "INV";

invoice = invoiceEntry.Document.Insert(invoice);

invoice.CustomerID = customer.BAccountID;
invoice.CustomerLocationID = customer.DefLocationID;
invoice.DocDate = today;

...

invoice = invoiceEntry.Document.Update(invoice);

The same thing with the detail lines. You’ll create a new ARTran record and .Insert it into the cache:

ARTran tran = new ARTran
{
  //RefNbr = invoice.RefNbr, -- unnecessary as the DAC already manages this
  InventoryID = inventoryItem.InventoryID,
  Qty = 1,
  CuryUnitPrice = productPrice,
  //CustomerID = customer.BAccountID, -- unnecessary
  AccountID = 89,
  SubID = 150,
  //BranchID = 2, -- unnecessary unless you're doing inter-branch transactions
  TaxCategoryID = "SC070100" //probably unnecessary as the item should be set up properly
};

tran = invoiceEntry.Transactions.Insert(tran);

invoiceEntry, by itself, won’t do anything. You need to insert the records into the PXGraph object so that the business logic within ARInvoiceEntry can do all of the heavy lifting for you.


asommer
Jr Varsity III
Forum|alt.badge.img
  • Author
  • Jr Varsity III
  • 27 replies
  • March 31, 2025

Still getting this error from the “Exception Stack Trace” from the Request Details on the Webhooks screen: 

PX.Data.PXException: Error: An error occurred during processing of the field Qty: Object reference not set to an instance of an object.. ---> System.NullReferenceException: Object reference not set to an instance of an object.
   at PX.Objects.FS.SM_ARInvoiceEntry.IsLineCreatedFromAppSO(PXGraph cleanerGraph, Object document, Object lineDoc, String fieldName) in C:\build\code_repo\WebSites\Pure\PX.Objects.FS\CustomBLC\SM_ARInvoiceEntry.cs:line 711
   at PX.Objects.FS.SM_ARInvoiceEntry._(FieldUpdating`2 e) in C:\build\code_repo\WebSites\Pure\PX.Objects.FS\CustomBLC\SM_ARInvoiceEntry.cs:line 231
   at PX.Data.PXCache.OnFieldUpdating(String name, Object row, Object& newValue) in C:\build\code_repo\NetTools\PX.Data\Cache\ModelEventHandling.cs:line 854
   at PX.Data.PXCache`1.FillWithValues(TNode copy, TNode& item) in C:\build\code_repo\NetTools\PX.Data\Cache\Model.cs:line 1752
   --- End of inner exception stack trace ---
   at PX.Data.PXCache`1.FillWithValues(TNode copy, TNode& item) in C:\build\code_repo\NetTools\PX.Data\Cache\Model.cs:line 1831
   at PX.Data.PXCache`1.Insert(Object data, Boolean bypassinterceptor) in C:\build\code_repo\NetTools\PX.Data\Cache\Model.cs:line 5932
   at PX.Data.PXSelectBase`1.Insert(Table item) in C:\build\code_repo\NetTools\PX.Data\Graph\Indexer.cs:line 582
   at EcrsWebhook.EcrsWebhookHandler.<HandleAsync>d__2.MoveNext() in C:\Program Files\Acumatica ERP\AcumaticaERP\App_Data\Projects\Webhooks\Webhooks\Webhooks.cs:line 177
--- End of stack trace from previous location where exception was thrown ---
   at System.Runtime.ExceptionServices.ExceptionDispatchInfo.Throw()
   at System.Runtime.CompilerServices.TaskAwaiter.HandleNonSuccessAndDebuggerNotification(Task task)
   at PX.Api.Webhooks.Executors.CoreWebhookExecutor.<PX-Api-Webhooks-Executors-IWebhookExecutor-ExecuteAsync>d__5.MoveNext()

 

Did make the changes you outlined, but the same error is persisting. Is there some other field I’m missing trying to create an ARInvoice from a webhook?

 

Thanks again!


Forum|alt.badge.img+6
  • Captain II
  • 563 replies
  • March 31, 2025

It appears that Acumatica thinks your Qty field value is null.

Are you sure that your inventoryItem variable contains a value?

Are you set up to be able to use Visual Studio to debug your application? If not, I’d recommend that so you can look to see what your Qty field value looks like at the various stages of the process. You can use Postman to simulate the call to the webhook and then debug your code.


asommer
Jr Varsity III
Forum|alt.badge.img
  • Author
  • Jr Varsity III
  • 27 replies
  • April 1, 2025

Correct the Qty field is null when the object is used in the IsLineCreatedFromAppSO method, but in my method it has a value of 1:

 

 

Is there anything else I should try?

 

Thanks!


Forum|alt.badge.img+6
  • Captain II
  • 563 replies
  • April 1, 2025

I think there are some other errors within the code of your webhook.

Your declaration of the class, for example should be:

public class EcrsWebhookHandler : IWebhookHandler

Here is code from the example that Acumatica provides:

https://github.com/Acumatica/Help-and-Training-Examples/blob/2023R1/IntegrationDevelopment/Help/ConfiguringWebhooks/TogglWebhookHandler_Code/WebhookHandler/TogglWebhookHandler.cs

I could be wrong, but in your screenshot, it looks like you’ve hit the exception on line 483. What variable is null that the line is accessing? I suspect it is base but it isn’t clear. Your Qty value isn’t a problem, yet, until you get past line 483.

 


asommer
Jr Varsity III
Forum|alt.badge.img
  • Author
  • Jr Varsity III
  • 27 replies
  • April 1, 2025

The code in the screenshot is from the SM_ARInvoiceEntry.cs file found by clicking on the exception in the debug console.  My class is defined by the line: 

public class EcrsWebhookHandler : PXGraph<ARInvoiceEntry>, IWebhookHandler

 

I’ve tried it with and without the additional “PXGraph<ARInvoiceEntry>” and both ways receive the same exception.  

Thanks!


rimamanaseryan16
Freshman I

@asommer Is the problem already solved? I faced it as well today.


asommer
Jr Varsity III
Forum|alt.badge.img
  • Author
  • Jr Varsity III
  • 27 replies
  • April 3, 2025

It is not solved as of now.  I’ve been digging into the SM_ARInvoiceEntry.cs file in order to understand how it all works together. 

An approach I’ve been working on is catching the exception from inserting the ARTran, saving the ARInvoiceEntry then going back and adding the ARTran afterwards.  I’ll try to post an update later this afternoon when/if there’s progress.

Thanks ​@rimamanaseryan16 for the reply!


rimamanaseryan16
Freshman I

Thanks for letting me know. Will look forward to getting an update ​@asommer 


asommer
Jr Varsity III
Forum|alt.badge.img
  • Author
  • Jr Varsity III
  • 27 replies
  • April 3, 2025

Wasn’t able to get anything working.  Ran into other errors and exceptions while trying to create the ARInvoice then add ARTrans.  I’ll open a ticket with Acumatica support and see if it’s a bug with that if statement in IsLineCreatedFromAppSO method.

Thanks!


Reply


Cookie policy

We use cookies to enhance and personalize your experience. If you accept you agree to our full cookie policy. Learn more about our cookies.

 
Cookie settings