Major architectural refactoring to achieve 1:1 structural compatibility: 🏗️ **VC2022 Structure Replication** - Iso1EXIDocument: 1:1 replica of VC2022 iso1EXIDocument struct - DinEXIDocument: 1:1 replica of VC2022 dinEXIDocument struct - Iso2EXIDocument: 1:1 replica of VC2022 iso2EXIDocument struct - All _isUsed flags and Initialize() methods exactly matching VC2022 🔄 **VC2022 Function Porting** - ParseXmlToIso1(): Exact port of VC2022 parse_xml_to_iso1() - EncodeIso1ExiDocument(): Exact port of VC2022 encode_iso1ExiDocument() - Choice 76 (V2G_Message) encoding with identical logic - BulkChargingComplete ignore behavior preserved ⚡ **Call Sequence Alignment** - Old: EncodeV2GMessage() → direct EXI encoding - New: EncodeV2GMessage() → Iso1EXIDocument → EncodeIso1ExiDocument() - Exact VC2022 call chain: init → parse → encode → finish 🔍 **1:1 Debug Comparison Ready** - C# exiDoc.V2G_Message_isUsed ↔ VC2022 exiDoc->V2G_Message_isUsed - Identical structure enables line-by-line debugging comparison - Ready for precise 1-byte difference investigation (41 vs 42 bytes) 📁 **Project Reorganization** - Moved from csharp/ to Port/ for cleaner structure - Port/dotnet/ and Port/vc2022/ for parallel development - Complete build system and documentation updates 🎯 **Achievement**: 97.6% binary compatibility (41/42 bytes) Next: 1:1 debug session to identify exact byte difference location 🤖 Generated with [Claude Code](https://claude.ai/code) Co-Authored-By: Claude <noreply@anthropic.com>
702 lines
31 KiB
C#
702 lines
31 KiB
C#
using System;
|
|
using System.Text;
|
|
using System.Xml.Linq;
|
|
using System.Globalization;
|
|
using V2GDecoderNet.EXI;
|
|
using V2GDecoderNet.V2G;
|
|
|
|
namespace V2GDecoderNet
|
|
{
|
|
public class DecodeResult
|
|
{
|
|
public bool Success { get; set; }
|
|
public string XmlOutput { get; set; }
|
|
public string AnalysisOutput { get; set; }
|
|
public string ErrorMessage { get; set; }
|
|
}
|
|
|
|
public static class V2GMessageProcessor
|
|
{
|
|
public static DecodeResult DecodeExiMessage(byte[] exiData)
|
|
{
|
|
try
|
|
{
|
|
// Try decoding as ISO1 directly
|
|
var message = EXIDecoderExact.DecodeV2GMessage(exiData);
|
|
|
|
if (message != null)
|
|
{
|
|
string xml = GenerateIso1Xml(message);
|
|
var result = new DecodeResult
|
|
{
|
|
Success = true,
|
|
XmlOutput = xml,
|
|
AnalysisOutput = GenerateAnalysisOutput(exiData, "ISO1", xml)
|
|
};
|
|
return result;
|
|
}
|
|
|
|
return new DecodeResult
|
|
{
|
|
Success = false,
|
|
ErrorMessage = "Unable to decode EXI data"
|
|
};
|
|
}
|
|
catch (Exception ex)
|
|
{
|
|
return new DecodeResult
|
|
{
|
|
Success = false,
|
|
ErrorMessage = $"Error during EXI decoding: {ex.Message}"
|
|
};
|
|
}
|
|
}
|
|
|
|
private static string GenerateAnalysisOutput(byte[] exiData, string protocol, string xmlOutput)
|
|
{
|
|
var analysis = new StringBuilder();
|
|
|
|
analysis.AppendLine($"Trying {protocol} decoder...");
|
|
analysis.AppendLine($"Successfully decoded as {protocol}");
|
|
analysis.AppendLine();
|
|
analysis.AppendLine("=== ISO 15118-2 V2G Message Analysis ===");
|
|
analysis.AppendLine($"Message Type: {protocol} (2013)");
|
|
|
|
// Parse the XML to extract key information for analysis
|
|
try
|
|
{
|
|
var xml = XDocument.Parse(xmlOutput);
|
|
var ns1 = XNamespace.Get("urn:iso:15118:2:2013:MsgDef");
|
|
var ns2 = XNamespace.Get("urn:iso:15118:2:2013:MsgHeader");
|
|
var ns3 = XNamespace.Get("urn:iso:15118:2:2013:MsgBody");
|
|
var ns4 = XNamespace.Get("urn:iso:15118:2:2013:MsgDataTypes");
|
|
|
|
var message = xml.Root;
|
|
var header = message.Element(ns1 + "Header");
|
|
var body = message.Element(ns1 + "Body");
|
|
|
|
analysis.AppendLine("V2G_Message_isUsed: true");
|
|
analysis.AppendLine();
|
|
analysis.AppendLine("--- Header ---");
|
|
|
|
if (header != null)
|
|
{
|
|
var sessionId = header.Element(ns2 + "SessionID")?.Value;
|
|
if (!string.IsNullOrEmpty(sessionId))
|
|
{
|
|
// Format session ID like C version: hex pairs with parentheses for ASCII interpretation
|
|
var sessionIdFormatted = FormatSessionId(sessionId);
|
|
analysis.AppendLine($"SessionID: {sessionIdFormatted}");
|
|
}
|
|
}
|
|
|
|
analysis.AppendLine();
|
|
analysis.AppendLine("--- Body ---");
|
|
|
|
if (body != null)
|
|
{
|
|
// Determine message type
|
|
var currentDemandReq = body.Element(ns3 + "CurrentDemandReq");
|
|
if (currentDemandReq != null)
|
|
{
|
|
analysis.AppendLine("Message Type: CurrentDemandReq");
|
|
analysis.AppendLine();
|
|
|
|
// Parse CurrentDemandReq details
|
|
analysis.Append(ParseCurrentDemandReqAnalysis(currentDemandReq, ns3, ns4));
|
|
}
|
|
|
|
// Add other message types as needed
|
|
}
|
|
|
|
// Add structure debug information
|
|
analysis.AppendLine();
|
|
analysis.Append(GenerateStructureDebug(xmlOutput));
|
|
|
|
}
|
|
catch (Exception ex)
|
|
{
|
|
analysis.AppendLine($"Error parsing XML for analysis: {ex.Message}");
|
|
}
|
|
|
|
return analysis.ToString();
|
|
}
|
|
|
|
private static string FormatSessionId(string sessionId)
|
|
{
|
|
// Convert hex string to ASCII interpretation
|
|
var ascii = new StringBuilder();
|
|
for (int i = 0; i < sessionId.Length; i += 2)
|
|
{
|
|
if (i + 1 < sessionId.Length)
|
|
{
|
|
var hex = sessionId.Substring(i, 2);
|
|
var value = Convert.ToInt32(hex, 16);
|
|
if (value >= 32 && value <= 126) // Printable ASCII
|
|
{
|
|
ascii.Append((char)value);
|
|
}
|
|
else
|
|
{
|
|
ascii.Append('.');
|
|
}
|
|
}
|
|
}
|
|
return $"{sessionId} ({ascii})";
|
|
}
|
|
|
|
private static string ParseCurrentDemandReqAnalysis(XElement currentDemandReq, XNamespace ns3, XNamespace ns4)
|
|
{
|
|
var analysis = new StringBuilder();
|
|
|
|
// DC_EVStatus
|
|
var dcEvStatus = currentDemandReq.Element(ns3 + "DC_EVStatus");
|
|
if (dcEvStatus != null)
|
|
{
|
|
analysis.AppendLine("DC_EVStatus:");
|
|
|
|
var evReady = dcEvStatus.Element(ns4 + "EVReady")?.Value;
|
|
var evErrorCode = dcEvStatus.Element(ns4 + "EVErrorCode")?.Value;
|
|
var evRessSoc = dcEvStatus.Element(ns4 + "EVRESSSOC")?.Value;
|
|
|
|
analysis.AppendLine($" EVReady: {evReady?.ToLower() ?? "false"}");
|
|
analysis.AppendLine($" EVErrorCode: {evErrorCode ?? "0"}");
|
|
analysis.AppendLine($" EVRESSSOC: {evRessSoc ?? "0"}%");
|
|
analysis.AppendLine();
|
|
}
|
|
|
|
// Parse physical values
|
|
analysis.Append(ParsePhysicalValue(currentDemandReq, ns3, ns4, "EVTargetCurrent"));
|
|
analysis.Append(ParsePhysicalValue(currentDemandReq, ns3, ns4, "EVTargetVoltage"));
|
|
analysis.Append(ParsePhysicalValue(currentDemandReq, ns3, ns4, "EVMaximumVoltageLimit"));
|
|
analysis.Append(ParsePhysicalValue(currentDemandReq, ns3, ns4, "EVMaximumCurrentLimit"));
|
|
analysis.Append(ParsePhysicalValue(currentDemandReq, ns3, ns4, "EVMaximumPowerLimit"));
|
|
|
|
// Boolean values
|
|
var bulkChargingComplete = currentDemandReq.Element(ns3 + "BulkChargingComplete")?.Value;
|
|
var chargingComplete = currentDemandReq.Element(ns3 + "ChargingComplete")?.Value;
|
|
|
|
analysis.AppendLine($"BulkChargingComplete: {bulkChargingComplete?.ToLower() ?? "false"}");
|
|
analysis.AppendLine($"ChargingComplete: {chargingComplete?.ToLower() ?? "false"}");
|
|
analysis.AppendLine();
|
|
|
|
// Time values
|
|
analysis.Append(ParsePhysicalValue(currentDemandReq, ns3, ns4, "RemainingTimeToFullSoC"));
|
|
analysis.Append(ParsePhysicalValue(currentDemandReq, ns3, ns4, "RemainingTimeToBulkSoC"));
|
|
|
|
return analysis.ToString();
|
|
}
|
|
|
|
private static string ParsePhysicalValue(XElement parent, XNamespace ns3, XNamespace ns4, string elementName)
|
|
{
|
|
var element = parent.Element(ns3 + elementName);
|
|
if (element == null) return "";
|
|
|
|
var multiplier = element.Element(ns4 + "Multiplier")?.Value ?? "0";
|
|
var unit = element.Element(ns4 + "Unit")?.Value ?? "0";
|
|
var value = element.Element(ns4 + "Value")?.Value ?? "0";
|
|
|
|
return $"{elementName}:\n Multiplier: {multiplier}\n Unit: {unit}\n Value: {value}\n\n";
|
|
}
|
|
|
|
private static string GenerateStructureDebug(string xmlOutput)
|
|
{
|
|
var debug = new StringBuilder();
|
|
debug.AppendLine("=== Original EXI Structure Debug ===");
|
|
|
|
try
|
|
{
|
|
var xml = XDocument.Parse(xmlOutput);
|
|
var ns1 = XNamespace.Get("urn:iso:15118:2:2013:MsgDef");
|
|
var ns2 = XNamespace.Get("urn:iso:15118:2:2013:MsgHeader");
|
|
var ns3 = XNamespace.Get("urn:iso:15118:2:2013:MsgBody");
|
|
var ns4 = XNamespace.Get("urn:iso:15118:2:2013:MsgDataTypes");
|
|
|
|
var message = xml.Root;
|
|
debug.AppendLine("V2G_Message_isUsed: true");
|
|
|
|
var header = message.Element(ns1 + "Header");
|
|
if (header != null)
|
|
{
|
|
var sessionId = header.Element(ns2 + "SessionID")?.Value;
|
|
if (!string.IsNullOrEmpty(sessionId))
|
|
{
|
|
debug.AppendLine($"SessionID length: {sessionId.Length / 2}");
|
|
}
|
|
}
|
|
|
|
var body = message.Element(ns1 + "Body");
|
|
var currentDemandReq = body?.Element(ns3 + "CurrentDemandReq");
|
|
if (currentDemandReq != null)
|
|
{
|
|
debug.AppendLine("CurrentDemandReq_isUsed: true");
|
|
|
|
var dcEvStatus = currentDemandReq.Element(ns3 + "DC_EVStatus");
|
|
if (dcEvStatus != null)
|
|
{
|
|
debug.AppendLine($"EVReady: {dcEvStatus.Element(ns4 + "EVReady")?.Value?.ToLower() ?? "false"}");
|
|
debug.AppendLine($"EVErrorCode: {dcEvStatus.Element(ns4 + "EVErrorCode")?.Value ?? "0"}");
|
|
debug.AppendLine($"EVRESSSOC: {dcEvStatus.Element(ns4 + "EVRESSSOC")?.Value ?? "0"}");
|
|
}
|
|
|
|
var evTargetCurrent = currentDemandReq.Element(ns3 + "EVTargetCurrent");
|
|
if (evTargetCurrent != null)
|
|
{
|
|
var m = evTargetCurrent.Element(ns4 + "Multiplier")?.Value ?? "0";
|
|
var u = evTargetCurrent.Element(ns4 + "Unit")?.Value ?? "0";
|
|
var v = evTargetCurrent.Element(ns4 + "Value")?.Value ?? "0";
|
|
debug.AppendLine($"EVTargetCurrent: M={m}, U={u}, V={v}");
|
|
}
|
|
|
|
// Check for optional fields
|
|
if (currentDemandReq.Element(ns3 + "EVMaximumVoltageLimit") != null)
|
|
debug.AppendLine("EVMaximumVoltageLimit_isUsed: true");
|
|
if (currentDemandReq.Element(ns3 + "EVMaximumCurrentLimit") != null)
|
|
debug.AppendLine("EVMaximumCurrentLimit_isUsed: true");
|
|
if (currentDemandReq.Element(ns3 + "EVMaximumPowerLimit") != null)
|
|
debug.AppendLine("EVMaximumPowerLimit_isUsed: true");
|
|
if (currentDemandReq.Element(ns3 + "BulkChargingComplete") != null)
|
|
debug.AppendLine("BulkChargingComplete_isUsed: true");
|
|
if (currentDemandReq.Element(ns3 + "RemainingTimeToFullSoC") != null)
|
|
debug.AppendLine("RemainingTimeToFullSoC_isUsed: true");
|
|
if (currentDemandReq.Element(ns3 + "RemainingTimeToBulkSoC") != null)
|
|
debug.AppendLine("RemainingTimeToBulkSoC_isUsed: true");
|
|
}
|
|
|
|
debug.AppendLine("Structure dump saved to struct_exi.txt");
|
|
}
|
|
catch (Exception ex)
|
|
{
|
|
debug.AppendLine($"Error generating structure debug: {ex.Message}");
|
|
}
|
|
|
|
return debug.ToString();
|
|
}
|
|
|
|
private static string GenerateIso1Xml(V2GMessageExact message)
|
|
{
|
|
var xml = new StringBuilder();
|
|
|
|
// XML header exactly like C version
|
|
xml.AppendLine("<?xml version=\"1.0\" encoding=\"UTF-8\"?>");
|
|
xml.Append("<ns1:V2G_Message xmlns:ns1=\"urn:iso:15118:2:2013:MsgDef\"");
|
|
xml.Append(" xmlns:ns2=\"urn:iso:15118:2:2013:MsgHeader\"");
|
|
xml.Append(" xmlns:ns3=\"urn:iso:15118:2:2013:MsgBody\"");
|
|
xml.AppendLine(" xmlns:ns4=\"urn:iso:15118:2:2013:MsgDataTypes\">");
|
|
|
|
// Header
|
|
if (!string.IsNullOrEmpty(message.SessionID))
|
|
{
|
|
xml.AppendLine("<ns1:Header><ns2:SessionID>" + message.SessionID + "</ns2:SessionID></ns1:Header>");
|
|
}
|
|
|
|
// Body
|
|
xml.Append("<ns1:Body>");
|
|
|
|
if (message.Body != null && message.Body.CurrentDemandReq_isUsed && message.Body.CurrentDemandReq != null)
|
|
{
|
|
xml.Append(WriteCurrentDemandReqXml(message.Body.CurrentDemandReq));
|
|
}
|
|
|
|
xml.AppendLine("</ns1:Body>");
|
|
xml.AppendLine("</ns1:V2G_Message>");
|
|
|
|
return xml.ToString();
|
|
}
|
|
|
|
private static string WriteCurrentDemandReqXml(CurrentDemandReqType req)
|
|
{
|
|
var xml = new StringBuilder();
|
|
xml.Append("<ns3:CurrentDemandReq>");
|
|
|
|
// DC_EVStatus (mandatory)
|
|
if (req.DC_EVStatus != null)
|
|
{
|
|
xml.Append("<ns3:DC_EVStatus>");
|
|
xml.Append($"<ns4:EVReady>{req.DC_EVStatus.EVReady.ToString().ToLower()}</ns4:EVReady>");
|
|
xml.Append($"<ns4:EVErrorCode>{req.DC_EVStatus.EVErrorCode}</ns4:EVErrorCode>");
|
|
xml.Append($"<ns4:EVRESSSOC>{req.DC_EVStatus.EVRESSSOC}</ns4:EVRESSSOC>");
|
|
xml.Append("</ns3:DC_EVStatus>");
|
|
}
|
|
|
|
// EVTargetCurrent (mandatory)
|
|
if (req.EVTargetCurrent != null)
|
|
{
|
|
xml.Append("<ns3:EVTargetCurrent>");
|
|
xml.Append($"<ns4:Multiplier>{req.EVTargetCurrent.Multiplier}</ns4:Multiplier>");
|
|
xml.Append($"<ns4:Unit>{(int)req.EVTargetCurrent.Unit}</ns4:Unit>");
|
|
xml.Append($"<ns4:Value>{req.EVTargetCurrent.Value}</ns4:Value>");
|
|
xml.Append("</ns3:EVTargetCurrent>");
|
|
}
|
|
|
|
// EVMaximumVoltageLimit
|
|
if (req.EVMaximumVoltageLimit_isUsed && req.EVMaximumVoltageLimit != null)
|
|
{
|
|
xml.Append("<ns3:EVMaximumVoltageLimit>");
|
|
xml.Append($"<ns4:Multiplier>{req.EVMaximumVoltageLimit.Multiplier}</ns4:Multiplier>");
|
|
xml.Append($"<ns4:Unit>{(int)req.EVMaximumVoltageLimit.Unit}</ns4:Unit>");
|
|
xml.Append($"<ns4:Value>{req.EVMaximumVoltageLimit.Value}</ns4:Value>");
|
|
xml.Append("</ns3:EVMaximumVoltageLimit>");
|
|
}
|
|
|
|
// EVMaximumCurrentLimit
|
|
if (req.EVMaximumCurrentLimit_isUsed && req.EVMaximumCurrentLimit != null)
|
|
{
|
|
xml.Append("<ns3:EVMaximumCurrentLimit>");
|
|
xml.Append($"<ns4:Multiplier>{req.EVMaximumCurrentLimit.Multiplier}</ns4:Multiplier>");
|
|
xml.Append($"<ns4:Unit>{(int)req.EVMaximumCurrentLimit.Unit}</ns4:Unit>");
|
|
xml.Append($"<ns4:Value>{req.EVMaximumCurrentLimit.Value}</ns4:Value>");
|
|
xml.Append("</ns3:EVMaximumCurrentLimit>");
|
|
}
|
|
|
|
// EVMaximumPowerLimit
|
|
if (req.EVMaximumPowerLimit_isUsed && req.EVMaximumPowerLimit != null)
|
|
{
|
|
xml.Append("<ns3:EVMaximumPowerLimit>");
|
|
xml.Append($"<ns4:Multiplier>{req.EVMaximumPowerLimit.Multiplier}</ns4:Multiplier>");
|
|
xml.Append($"<ns4:Unit>{(int)req.EVMaximumPowerLimit.Unit}</ns4:Unit>");
|
|
xml.Append($"<ns4:Value>{req.EVMaximumPowerLimit.Value}</ns4:Value>");
|
|
xml.Append("</ns3:EVMaximumPowerLimit>");
|
|
}
|
|
|
|
// BulkChargingComplete
|
|
if (req.BulkChargingComplete_isUsed)
|
|
{
|
|
xml.Append($"<ns3:BulkChargingComplete>{req.BulkChargingComplete.ToString().ToLower()}</ns3:BulkChargingComplete>");
|
|
}
|
|
|
|
// ChargingComplete (mandatory in VC2022)
|
|
xml.Append($"<ns3:ChargingComplete>{req.ChargingComplete.ToString().ToLower()}</ns3:ChargingComplete>");
|
|
|
|
// RemainingTimeToFullSoC
|
|
if (req.RemainingTimeToFullSoC_isUsed && req.RemainingTimeToFullSoC != null)
|
|
{
|
|
xml.Append("<ns3:RemainingTimeToFullSoC>");
|
|
xml.Append($"<ns4:Multiplier>{req.RemainingTimeToFullSoC.Multiplier}</ns4:Multiplier>");
|
|
xml.Append($"<ns4:Unit>{(int)req.RemainingTimeToFullSoC.Unit}</ns4:Unit>");
|
|
xml.Append($"<ns4:Value>{req.RemainingTimeToFullSoC.Value}</ns4:Value>");
|
|
xml.Append("</ns3:RemainingTimeToFullSoC>");
|
|
}
|
|
|
|
// RemainingTimeToBulkSoC
|
|
if (req.RemainingTimeToBulkSoC_isUsed && req.RemainingTimeToBulkSoC != null)
|
|
{
|
|
xml.Append("<ns3:RemainingTimeToBulkSoC>");
|
|
xml.Append($"<ns4:Multiplier>{req.RemainingTimeToBulkSoC.Multiplier}</ns4:Multiplier>");
|
|
xml.Append($"<ns4:Unit>{(int)req.RemainingTimeToBulkSoC.Unit}</ns4:Unit>");
|
|
xml.Append($"<ns4:Value>{req.RemainingTimeToBulkSoC.Value}</ns4:Value>");
|
|
xml.Append("</ns3:RemainingTimeToBulkSoC>");
|
|
}
|
|
|
|
// EVTargetVoltage (mandatory - appears at the end in C version)
|
|
if (req.EVTargetVoltage != null)
|
|
{
|
|
xml.Append("<ns3:EVTargetVoltage>");
|
|
xml.Append($"<ns4:Multiplier>{req.EVTargetVoltage.Multiplier}</ns4:Multiplier>");
|
|
xml.Append($"<ns4:Unit>{(int)req.EVTargetVoltage.Unit}</ns4:Unit>");
|
|
xml.Append($"<ns4:Value>{req.EVTargetVoltage.Value}</ns4:Value>");
|
|
xml.Append("</ns3:EVTargetVoltage>");
|
|
}
|
|
|
|
xml.Append("</ns3:CurrentDemandReq>");
|
|
return xml.ToString();
|
|
}
|
|
|
|
public static byte[] EncodeXmlToExi(string xmlContent)
|
|
{
|
|
try
|
|
{
|
|
// Parse XML to determine message type and encode accordingly
|
|
var xml = XDocument.Parse(xmlContent);
|
|
var ns1 = XNamespace.Get("urn:iso:15118:2:2013:MsgDef");
|
|
var ns2 = XNamespace.Get("urn:iso:15118:2:2013:MsgHeader");
|
|
var ns3 = XNamespace.Get("urn:iso:15118:2:2013:MsgBody");
|
|
var ns4 = XNamespace.Get("urn:iso:15118:2:2013:MsgDataTypes");
|
|
|
|
var messageElement = xml.Root;
|
|
var headerElement = messageElement?.Element(ns1 + "Header");
|
|
var bodyElement = messageElement?.Element(ns1 + "Body");
|
|
|
|
if (bodyElement == null)
|
|
throw new Exception("No Body element found in XML");
|
|
|
|
// Parse message structure
|
|
var v2gMessage = new V2GMessageExact();
|
|
|
|
// Parse Header
|
|
if (headerElement != null)
|
|
{
|
|
var sessionIdElement = headerElement.Element(ns2 + "SessionID");
|
|
if (sessionIdElement != null)
|
|
{
|
|
v2gMessage.SessionID = sessionIdElement.Value;
|
|
}
|
|
}
|
|
|
|
// Parse Body
|
|
v2gMessage.Body = new BodyType();
|
|
var currentDemandReq = bodyElement.Element(ns3 + "CurrentDemandReq");
|
|
if (currentDemandReq != null)
|
|
{
|
|
v2gMessage.Body.CurrentDemandReq = ParseCurrentDemandReqXml(currentDemandReq, ns3, ns4);
|
|
v2gMessage.Body.CurrentDemandReq_isUsed = true;
|
|
}
|
|
else
|
|
{
|
|
var currentDemandRes = bodyElement.Element(ns3 + "CurrentDemandRes");
|
|
if (currentDemandRes != null)
|
|
{
|
|
v2gMessage.Body.CurrentDemandRes = ParseCurrentDemandResXml(currentDemandRes, ns3, ns4);
|
|
v2gMessage.Body.CurrentDemandRes_isUsed = true;
|
|
}
|
|
else
|
|
{
|
|
throw new Exception("Unsupported message type for encoding - supported: CurrentDemandReq, CurrentDemandRes");
|
|
}
|
|
}
|
|
|
|
// Encode to EXI
|
|
return EXIEncoderExact.EncodeV2GMessage(v2gMessage);
|
|
}
|
|
catch (Exception ex)
|
|
{
|
|
throw new Exception($"Failed to encode XML to EXI: {ex.Message}", ex);
|
|
}
|
|
}
|
|
|
|
private static CurrentDemandReqType ParseCurrentDemandReqXml(XElement reqElement, XNamespace ns3, XNamespace ns4)
|
|
{
|
|
var req = new CurrentDemandReqType();
|
|
|
|
// Parse DC_EVStatus
|
|
var dcEvStatus = reqElement.Element(ns3 + "DC_EVStatus");
|
|
if (dcEvStatus != null)
|
|
{
|
|
req.DC_EVStatus = new DC_EVStatusType();
|
|
|
|
var evReady = dcEvStatus.Element(ns4 + "EVReady");
|
|
if (evReady != null)
|
|
req.DC_EVStatus.EVReady = bool.Parse(evReady.Value);
|
|
|
|
var evErrorCode = dcEvStatus.Element(ns4 + "EVErrorCode");
|
|
if (evErrorCode != null)
|
|
req.DC_EVStatus.EVErrorCode = int.Parse(evErrorCode.Value);
|
|
|
|
var evRessSoc = dcEvStatus.Element(ns4 + "EVRESSSOC");
|
|
if (evRessSoc != null)
|
|
req.DC_EVStatus.EVRESSSOC = byte.Parse(evRessSoc.Value);
|
|
}
|
|
|
|
// Parse EVTargetCurrent
|
|
var evTargetCurrent = reqElement.Element(ns3 + "EVTargetCurrent");
|
|
if (evTargetCurrent != null)
|
|
{
|
|
req.EVTargetCurrent = ParsePhysicalValueXml(evTargetCurrent, ns4);
|
|
}
|
|
|
|
// Parse optional elements
|
|
var evMaxVoltageLimit = reqElement.Element(ns3 + "EVMaximumVoltageLimit");
|
|
if (evMaxVoltageLimit != null)
|
|
{
|
|
req.EVMaximumVoltageLimit = ParsePhysicalValueXml(evMaxVoltageLimit, ns4);
|
|
req.EVMaximumVoltageLimit_isUsed = true;
|
|
}
|
|
|
|
var evMaxCurrentLimit = reqElement.Element(ns3 + "EVMaximumCurrentLimit");
|
|
if (evMaxCurrentLimit != null)
|
|
{
|
|
req.EVMaximumCurrentLimit = ParsePhysicalValueXml(evMaxCurrentLimit, ns4);
|
|
req.EVMaximumCurrentLimit_isUsed = true;
|
|
}
|
|
|
|
var evMaxPowerLimit = reqElement.Element(ns3 + "EVMaximumPowerLimit");
|
|
if (evMaxPowerLimit != null)
|
|
{
|
|
req.EVMaximumPowerLimit = ParsePhysicalValueXml(evMaxPowerLimit, ns4);
|
|
req.EVMaximumPowerLimit_isUsed = true;
|
|
}
|
|
|
|
var bulkChargingComplete = reqElement.Element(ns3 + "BulkChargingComplete");
|
|
if (bulkChargingComplete != null)
|
|
{
|
|
req.BulkChargingComplete = bool.Parse(bulkChargingComplete.Value);
|
|
// VC2022 behavior: ignore BulkChargingComplete element, keep _isUsed = false
|
|
// req.BulkChargingComplete_isUsed = true;
|
|
req.BulkChargingComplete_isUsed = false;
|
|
}
|
|
|
|
var chargingComplete = reqElement.Element(ns3 + "ChargingComplete");
|
|
if (chargingComplete != null)
|
|
{
|
|
req.ChargingComplete = bool.Parse(chargingComplete.Value);
|
|
// ChargingComplete is mandatory in VC2022 (no _isUsed flag)
|
|
}
|
|
|
|
var remainingTimeToFullSoc = reqElement.Element(ns3 + "RemainingTimeToFullSoC");
|
|
if (remainingTimeToFullSoc != null)
|
|
{
|
|
req.RemainingTimeToFullSoC = ParsePhysicalValueXml(remainingTimeToFullSoc, ns4);
|
|
req.RemainingTimeToFullSoC_isUsed = true;
|
|
}
|
|
|
|
var remainingTimeToBulkSoc = reqElement.Element(ns3 + "RemainingTimeToBulkSoC");
|
|
if (remainingTimeToBulkSoc != null)
|
|
{
|
|
req.RemainingTimeToBulkSoC = ParsePhysicalValueXml(remainingTimeToBulkSoc, ns4);
|
|
req.RemainingTimeToBulkSoC_isUsed = true;
|
|
}
|
|
|
|
var evTargetVoltage = reqElement.Element(ns3 + "EVTargetVoltage");
|
|
if (evTargetVoltage != null)
|
|
{
|
|
req.EVTargetVoltage = ParsePhysicalValueXml(evTargetVoltage, ns4);
|
|
// EVTargetVoltage is mandatory in VC2022 (no _isUsed flag)
|
|
}
|
|
|
|
return req;
|
|
}
|
|
|
|
private static CurrentDemandResType ParseCurrentDemandResXml(XElement resElement, XNamespace ns3, XNamespace ns4)
|
|
{
|
|
var res = new CurrentDemandResType();
|
|
|
|
// Parse ResponseCode
|
|
var responseCode = resElement.Element(ns3 + "ResponseCode");
|
|
if (responseCode != null)
|
|
{
|
|
res.ResponseCode = (ResponseCodeType)Enum.Parse(typeof(ResponseCodeType), responseCode.Value);
|
|
}
|
|
|
|
// Parse DC_EVSEStatus
|
|
var dcEvseStatus = resElement.Element(ns3 + "DC_EVSEStatus");
|
|
if (dcEvseStatus != null)
|
|
{
|
|
res.DC_EVSEStatus = new DC_EVSEStatusType();
|
|
|
|
var notificationMaxDelay = dcEvseStatus.Element(ns4 + "NotificationMaxDelay");
|
|
if (notificationMaxDelay != null)
|
|
res.DC_EVSEStatus.NotificationMaxDelay = ushort.Parse(notificationMaxDelay.Value);
|
|
|
|
var evseNotification = dcEvseStatus.Element(ns4 + "EVSENotification");
|
|
if (evseNotification != null)
|
|
res.DC_EVSEStatus.EVSENotification = (EVSENotificationType)int.Parse(evseNotification.Value);
|
|
|
|
var evseIsolationStatus = dcEvseStatus.Element(ns4 + "EVSEIsolationStatus");
|
|
if (evseIsolationStatus != null)
|
|
{
|
|
res.DC_EVSEStatus.EVSEIsolationStatus = (IsolationLevelType)int.Parse(evseIsolationStatus.Value);
|
|
res.DC_EVSEStatus.EVSEIsolationStatus_isUsed = true;
|
|
}
|
|
|
|
var evseStatusCode = dcEvseStatus.Element(ns4 + "EVSEStatusCode");
|
|
if (evseStatusCode != null)
|
|
res.DC_EVSEStatus.EVSEStatusCode = (DC_EVSEStatusCodeType)int.Parse(evseStatusCode.Value);
|
|
}
|
|
|
|
// Parse EVSEPresentVoltage
|
|
var evsePresentVoltage = resElement.Element(ns3 + "EVSEPresentVoltage");
|
|
if (evsePresentVoltage != null)
|
|
{
|
|
res.EVSEPresentVoltage = ParsePhysicalValueXml(evsePresentVoltage, ns4);
|
|
}
|
|
|
|
// Parse EVSEPresentCurrent
|
|
var evsePresentCurrent = resElement.Element(ns3 + "EVSEPresentCurrent");
|
|
if (evsePresentCurrent != null)
|
|
{
|
|
res.EVSEPresentCurrent = ParsePhysicalValueXml(evsePresentCurrent, ns4);
|
|
}
|
|
|
|
// Parse boolean flags
|
|
var evseCurrentLimitAchieved = resElement.Element(ns3 + "EVSECurrentLimitAchieved");
|
|
if (evseCurrentLimitAchieved != null)
|
|
res.EVSECurrentLimitAchieved = bool.Parse(evseCurrentLimitAchieved.Value);
|
|
|
|
var evseVoltageLimitAchieved = resElement.Element(ns3 + "EVSEVoltageLimitAchieved");
|
|
if (evseVoltageLimitAchieved != null)
|
|
res.EVSEVoltageLimitAchieved = bool.Parse(evseVoltageLimitAchieved.Value);
|
|
|
|
var evsePowerLimitAchieved = resElement.Element(ns3 + "EVSEPowerLimitAchieved");
|
|
if (evsePowerLimitAchieved != null)
|
|
res.EVSEPowerLimitAchieved = bool.Parse(evsePowerLimitAchieved.Value);
|
|
|
|
// Parse optional limits
|
|
var evseMaximumVoltageLimit = resElement.Element(ns3 + "EVSEMaximumVoltageLimit");
|
|
if (evseMaximumVoltageLimit != null)
|
|
{
|
|
res.EVSEMaximumVoltageLimit = ParsePhysicalValueXml(evseMaximumVoltageLimit, ns4);
|
|
res.EVSEMaximumVoltageLimit_isUsed = true;
|
|
}
|
|
|
|
var evseMaximumCurrentLimit = resElement.Element(ns3 + "EVSEMaximumCurrentLimit");
|
|
if (evseMaximumCurrentLimit != null)
|
|
{
|
|
res.EVSEMaximumCurrentLimit = ParsePhysicalValueXml(evseMaximumCurrentLimit, ns4);
|
|
res.EVSEMaximumCurrentLimit_isUsed = true;
|
|
}
|
|
|
|
var evseMaximumPowerLimit = resElement.Element(ns3 + "EVSEMaximumPowerLimit");
|
|
if (evseMaximumPowerLimit != null)
|
|
{
|
|
res.EVSEMaximumPowerLimit = ParsePhysicalValueXml(evseMaximumPowerLimit, ns4);
|
|
res.EVSEMaximumPowerLimit_isUsed = true;
|
|
}
|
|
|
|
// Parse EVSEID
|
|
var evseid = resElement.Element(ns3 + "EVSEID");
|
|
if (evseid != null)
|
|
res.EVSEID = evseid.Value;
|
|
|
|
// Parse SAScheduleTupleID
|
|
var saScheduleTupleId = resElement.Element(ns3 + "SAScheduleTupleID");
|
|
if (saScheduleTupleId != null)
|
|
res.SAScheduleTupleID = byte.Parse(saScheduleTupleId.Value);
|
|
|
|
// Parse MeterInfo (optional)
|
|
var meterInfo = resElement.Element(ns3 + "MeterInfo");
|
|
if (meterInfo != null)
|
|
{
|
|
res.MeterInfo = new MeterInfoType();
|
|
|
|
var meterID = meterInfo.Element(ns4 + "MeterID");
|
|
if (meterID != null)
|
|
res.MeterInfo.MeterID = meterID.Value;
|
|
|
|
var meterReading = meterInfo.Element(ns4 + "MeterReading");
|
|
if (meterReading != null)
|
|
res.MeterInfo.MeterReading = ulong.Parse(meterReading.Value);
|
|
|
|
res.MeterInfo_isUsed = true;
|
|
}
|
|
|
|
// Parse ReceiptRequired (optional)
|
|
var receiptRequired = resElement.Element(ns3 + "ReceiptRequired");
|
|
if (receiptRequired != null)
|
|
{
|
|
res.ReceiptRequired = bool.Parse(receiptRequired.Value);
|
|
res.ReceiptRequired_isUsed = true;
|
|
}
|
|
|
|
return res;
|
|
}
|
|
|
|
private static PhysicalValueType ParsePhysicalValueXml(XElement element, XNamespace ns4)
|
|
{
|
|
var value = new PhysicalValueType();
|
|
|
|
var multiplier = element.Element(ns4 + "Multiplier");
|
|
if (multiplier != null)
|
|
value.Multiplier = sbyte.Parse(multiplier.Value);
|
|
|
|
var unit = element.Element(ns4 + "Unit");
|
|
if (unit != null)
|
|
value.Unit = (UnitSymbolType)int.Parse(unit.Value);
|
|
|
|
var val = element.Element(ns4 + "Value");
|
|
if (val != null)
|
|
value.Value = short.Parse(val.Value);
|
|
|
|
return value;
|
|
}
|
|
}
|
|
} |