Chilkat HOME Android™ AutoIt C C# C++ Chilkat2-Python CkPython Classic ASP DataFlex Delphi DLL Go Java Node.js Objective-C PHP Extension Perl PowerBuilder PowerShell PureBasic Ruby SQL Server Swift Tcl Unicode C Unicode C++ VB.NET VBScript Visual Basic 6.0 Visual FoxPro Xojo Plugin
(C++) Hungary NAV Manage Annulment RequestDemonstrates the manageAnnulment request for the Hungarian NAV Online Invoicing System REST API v2.0. Note: This example requires Chilkat v9.5.0.83 or greater. For more information, see https://onlineszamla.nav.gov.hu/api/files/container/download/Online%20Szamla_Interfesz%20specifik%C3%A1ci%C3%B3_EN_v2.0.pdf
#include <CkCrypt2.h> #include <CkDateTime.h> #include <CkPrng.h> #include <CkStringBuilder.h> #include <CkXml.h> #include <CkBinData.h> #include <CkHttp.h> #include <CkHttpResponse.h> void ChilkatSample(void) { // This example assumes the Chilkat API to have been previously unlocked. // See Global Unlock Sample for sample code. // Build the following XML: // Use this online tool to generate code from sample XML: // Generate Code to Create XML // <?xml version="1.0" encoding="UTF-8"?> // <ManageAnnulmentRequest xmlns="http://schemas.nav.gov.hu/OSA/2.0/api"> // <header> // <requestId>RID338592103413</requestId> // <timestamp>2019-09-11T13:37:09.385Z</timestamp> // <requestVersion>2.0</requestVersion> // <headerVersion>1.0</headerVersion> // </header> // <user> // <login>lwilsmn0uqdxe6u</login> // <passwordHash>2F43840A882CFDB7DB0 ... F6CA57FE8CD74DC28E</passwordHash> // <taxNumber>11111111</taxNumber> // <requestSignature>A297E3BD9 ... 1959ADFAF5</requestSignature> // </user> // <software> // <softwareId>123456789123456789</softwareId> // <softwareName>string</softwareName> // <softwareOperation>LOCAL_SOFTWARE</softwareOperation> // <softwareMainVersion>string</softwareMainVersion> // <softwareDevName>string</softwareDevName> // <softwareDevContact>string</softwareDevContact> // <softwareDevCountryCode>HU</softwareDevCountryCode> // <softwareDevTaxNumber>string</softwareDevTaxNumber> // </software> // <exchangeToken>dbd03076-3a9b-4312-bbbb-0cee3a6472572P11CS49ASIL</exchangeToken> // <annulmentOperations> // <annulmentOperation> // <index>1</index> // <annulmentOperation>ANNUL</annulmentOperation> // <invoiceAnnulment>PD94bWwgdm ... ubnVsbWVudD4=</invoiceAnnulment> // </annulmentOperation> // </annulmentOperations> // </ManageAnnulmentRequest> bool success; CkCrypt2 crypt; CkDateTime dtNow; dtNow.SetFromCurrentSystemTime(); std::cout << dtNow.getAsTimestamp(false) << "\r\n"; // The hash algorithm for the password is SHA512 (not SHA3-512). crypt.put_HashAlgorithm("sha512"); crypt.put_EncodingMode("hex"); const char *myPassword = "my-password"; const char *passwordHash = crypt.hashStringENC(myPassword); // Generate a random request ID like "RID215118906689" CkPrng prng; CkStringBuilder sbRequestId; sbRequestId.Append("RID"); sbRequestId.Append(prng.randomString(12,true,false,false)); std::cout << "generated requestId = " << sbRequestId.getAsString() << "\r\n"; // Calculate the requestSignature crypt.put_HashAlgorithm("sha3-512"); const char *signatureKey = "ce-8f5e-215119fa7dd621DLMRHRLH2S"; CkStringBuilder sbFinalHashBase; // First append the timestamp because we are going to remove certain chars/parts. sbFinalHashBase.Append(dtNow.getAsTimestamp(false)); int numReplaced = sbFinalHashBase.Replace("Z",""); numReplaced = sbFinalHashBase.Replace("-",""); numReplaced = sbFinalHashBase.Replace(":",""); numReplaced = sbFinalHashBase.Replace("T",""); // Prepend the requestId and append the signatureKey sbFinalHashBase.Prepend(sbRequestId.getAsString()); sbFinalHashBase.Append(signatureKey); // Calculate first (and only) index hash. const char *invoiceOperation = "ANNUL"; const char *invoiceData = "PD94bWwgdmVyc2lvbj0iMS4wIiBlbmNvZGluZz0iVVRGLTgiIHN0YW5kYWxvbmU9InllcyI/Pg0KPEludm9pY2VBbm51bG1lbnQgeG1sbnM9Imh0dHA6Ly9zY2hlbWFzLm5hdi5nb3YuaHUvT1NBLzIuMC9hbm51bCI+DQoJCTxhbm51bG1lbnRSZWZlcmVuY2U+MjIyMjIyMjI8L2FubnVsbWVudFJlZmVyZW5jZT4NCgkJPGFubnVsbWVudFRpbWVzdGFtcD4yMDE4LTA2LTE4VDA5OjEwOjQ1LjMwOVo8L2FubnVsbWVudFRpbWVzdGFtcD4NCgkJPGFubnVsbWVudENvZGU+RVJSQVRJQ19EQVRBPC9hbm51bG1lbnRDb2RlPg0KCQk8YW5udWxtZW50UmVhc29uPmNyZWF0ZSBzemFtbGEgYW5udWw8L2FubnVsbWVudFJlYXNvbj4NCjwvSW52b2ljZUFubnVsbWVudD4="; CkStringBuilder sbHashBase; sbHashBase.Append(invoiceOperation); sbHashBase.Append(invoiceData); const char *firstHash = crypt.hashStringENC(sbHashBase.getAsString()); // Append the index hash to the final hash base. sbFinalHashBase.Append(firstHash); // Get our request signature (using sha3-512 because our HashAlgorithm was set to "sha3-512" up above...) const char *requestSignature = crypt.hashStringENC(sbFinalHashBase.getAsString()); // Load our recently obtained exchange token. // See Hungary NAV Invoicing Token Exchange Sample Code CkXml xmlExchangeToken; success = xmlExchangeToken.LoadXmlFile("qa_data/tokens/nav_exchange_token.xml"); if (success == false) { std::cout << xmlExchangeToken.lastErrorText() << "\r\n"; return; } // Get the base64 encoded/encrypted exchange token. // IMPORTANT: Make sure to use the exchange token before it expires. // If it expired, then get a new one.. const char *encodedEncryptedExchangeToken = xmlExchangeToken.getChildContent("encodedExchangeToken"); // Decode to binary. CkBinData bdExchangeToken; bdExchangeToken.AppendEncoded(encodedEncryptedExchangeToken,"base64"); // Decrypt using your 16-digit replacement key: crypt.put_CryptAlgorithm("aes"); crypt.put_CipherMode("ecb"); crypt.put_KeyLength(128); crypt.put_EncodingMode("base64"); // Pass your 16-digit replacement key here: crypt.SetEncodedKey("99952BBAAAAA8XYZ","ascii"); crypt.DecryptBd(bdExchangeToken); const char *exchangeToken = bdExchangeToken.getString("utf-8"); std::cout << "exchange token = " << exchangeToken << "\r\n"; // Now build the XML.. CkXml xml; xml.put_Tag("ManageAnnulmentRequest"); xml.AddAttribute("xmlns","http://schemas.nav.gov.hu/OSA/2.0/api"); xml.UpdateChildContent("header|requestId",sbRequestId.getAsString()); xml.UpdateChildContent("header|timestamp",dtNow.getAsTimestamp(false)); xml.UpdateChildContent("header|requestVersion","2.0"); xml.UpdateChildContent("header|headerVersion","1.0"); xml.UpdateChildContent("user|login","lwilsmn0uqdxe6u"); xml.UpdateChildContent("user|passwordHash",passwordHash); xml.UpdateChildContent("user|taxNumber","11111111"); xml.UpdateChildContent("user|requestSignature",requestSignature); xml.UpdateChildContent("software|softwareId","123456789123456789"); xml.UpdateChildContent("software|softwareName","string"); xml.UpdateChildContent("software|softwareOperation","LOCAL_SOFTWARE"); xml.UpdateChildContent("software|softwareMainVersion","string"); xml.UpdateChildContent("software|softwareDevName","string"); xml.UpdateChildContent("software|softwareDevContact","string"); xml.UpdateChildContent("software|softwareDevCountryCode","HU"); xml.UpdateChildContent("software|softwareDevTaxNumber","string"); xml.UpdateChildContent("exchangeToken",exchangeToken); xml.UpdateChildContent("annulmentOperations|annulmentOperation|index","1"); xml.UpdateChildContent("annulmentOperations|annulmentOperation|annulmentOperation",invoiceOperation); xml.UpdateChildContent("annulmentOperations|annulmentOperation|invoiceAnnulment",invoiceData); // POST the XML to https://api-test.onlineszamla.nav.gov.hu/invoiceService/v2/manageAnnulment CkHttp http; http.put_Accept("application/xml"); http.SetRequestHeader("Content-Type","application/xml"); const char *endpoint = "https://api-test.onlineszamla.nav.gov.hu/invoiceService/v2/manageAnnulment"; CkHttpResponse *resp = http.PostXml(endpoint,xml.getXml(),"utf-8"); if (http.get_LastMethodSuccess() == false) { std::cout << http.lastErrorText() << "\r\n"; return; } std::cout << "Response status code = " << resp->get_StatusCode() << "\r\n"; CkXml respXml; respXml.LoadXml(resp->bodyStr()); std::cout << "Response body:" << "\r\n"; std::cout << respXml.getXml() << "\r\n"; delete resp; // The result looks like this: // <?xml version="1.0" encoding="UTF-8" standalone="yes"?> // <ManageAnnulmentResponse xmlns="http://schemas.nav.gov.hu/OSA/2.0/api" xmlns:ns2="http://schemas.nav.gov.hu/OSA/2.0/data"> // <header> // <requestId>RID465317110495</requestId> // <timestamp>2020-03-25T15:08:40Z</timestamp> // <requestVersion>2.0</requestVersion> // <headerVersion>1.0</headerVersion> // </header> // <result> // <funcCode>OK</funcCode> // </result> // <software> // <softwareId>123456789123456789</softwareId> // <softwareName>string</softwareName> // <softwareOperation>LOCAL_SOFTWARE</softwareOperation> // <softwareMainVersion>string</softwareMainVersion> // <softwareDevName>string</softwareDevName> // <softwareDevContact>string</softwareDevContact> // <softwareDevCountryCode>HU</softwareDevCountryCode> // <softwareDevTaxNumber>string</softwareDevTaxNumber> // </software> // <transactionId>2WT6XG9LQ0U9VQNA</transactionId> // </ManageAnnulmentResponse> // Use this online tool to generate parsing code from sample XML: // Generate Parsing Code from XML // Chilkat functions returning "const char *" return a pointer to temporary internal memory owned and managed by Chilkat. // See this example explaining how this memory should be used: const char * functions. const char *ManageAnnulmentResponse_xmlns = respXml.getAttrValue("xmlns"); const char *ManageAnnulmentResponse_xmlns_ns2 = respXml.getAttrValue("xmlns:ns2"); const char *requestId = respXml.getChildContent("header|requestId"); const char *timestamp = respXml.getChildContent("header|timestamp"); const char *requestVersion = respXml.getChildContent("header|requestVersion"); const char *headerVersion = respXml.getChildContent("header|headerVersion"); const char *funcCode = respXml.getChildContent("result|funcCode"); const char *softwareId = respXml.getChildContent("software|softwareId"); const char *softwareName = respXml.getChildContent("software|softwareName"); const char *softwareOperation = respXml.getChildContent("software|softwareOperation"); const char *softwareMainVersion = respXml.getChildContent("software|softwareMainVersion"); const char *softwareDevName = respXml.getChildContent("software|softwareDevName"); const char *softwareDevContact = respXml.getChildContent("software|softwareDevContact"); const char *softwareDevCountryCode = respXml.getChildContent("software|softwareDevCountryCode"); const char *softwareDevTaxNumber = respXml.getChildContent("software|softwareDevTaxNumber"); const char *transactionId = respXml.getChildContent("transactionId"); } |
© 2000-2025 Chilkat Software, Inc. All Rights Reserved.