Adds user-defined extended properties to an existing tag via the 2020 WCF
AddTEx (AddTagExtendedProperties) op. Write-enabled connection + uppercase
storage-session GUID handle; reuses the write orchestrator open/priming chain.
The AddTEx inBuff is the exact inverse of the R1.5 GetTepByNm read-response
framing, so the serializer mirrors the read parser:
uint32 groupCount + 0x01(group) + [0x09+u16+ASCII tag] + uint32 propCount
+ per prop{ 0x02 + [0x09+u16+ASCII name] + 0x43 VT_BSTR + u16 payloadLen
+ u16 charCount + UTF-16 value } + 0x01(group trailer) + 0x00(terminator).
The trailing 0x00 is required — without it inBuff is one byte short and the
server throws SErrorException in CHistStorage::AddTagExtendedProperties. The
golden fixture pins the clean inBuff the live server accepted (dumped via
AVEVA_HISTORIAN_TEP_DUMP); read-back verified via R1.5. String (0x43) values only.
Delete (DelTep) is deferred: the native DeleteTagExtendedPropertiesByName does a
client-side sync check and returns err 229 for a just-added property, so the
DelTep request never reaches the wire and its inBuff can't be captured yet.
Shipped: HistorianClient.AddTagExtendedPropertiesAsync/AddTagExtendedPropertyAsync;
HistorianTagExtendedPropertyProtocol.SerializeAddRequest; orchestrator path;
golden WcfTagExtendedPropertyWriteProtocolTests (4); gated live write/read-back test;
native-harness `add-tep` scenario + Capture-AddTagExtendedProperties.ps1 +
decode-add-tep-capture.py. Doc: wcf-add-tag-extended-properties.md. 233 tests green.
Co-Authored-By: Claude Opus 4.8 (1M context) <noreply@anthropic.com>
Claude-Session: https://claude.ai/code/session_01B6mcaT2PjRFKcogzp9UkfC
4.0 KiB
Extended-property write over 2020 WCF — AddTEx (HCAL R1.11)
Status: ✅ Add DONE + live-verified (2026-06-21). Delete (DelTep) deferred — see below.
HistorianClient.AddTagExtendedPropertiesAsync / AddTagExtendedPropertyAsync writes user-defined
extended properties onto an existing tag via the 2020 WCF AddTEx (AddTagExtendedProperties) op,
and they read back via the R1.5 GetTagExtendedPropertiesAsync path. Verified end-to-end from the
pure-managed .NET 10 client against the local 2020 Historian (create tag → add property → read back →
delete tag).
The op
bool AddTagExtendedProperties(string handle, byte[] inBuff, out byte[] errorBuffer) // AddTEx
On IHistoryServiceContract2 (History service). Requires a write-enabled connection (Open2 mode
0x401) and the uppercase storage-session GUID handle — the SDK reuses the write orchestrator's
open + priming chain (the same one used by EnsT2/DelT). The tag is referenced by name inside inBuff;
no extra per-connection tag registration was needed (the server resolves it).
The inBuff — the exact inverse of the R1.5 read response
The native AddTagExtendedProperties(TagExtendedPropertyGroupList, out err) packs its groups into the
AddTEx inBuff with the same framing the R1.5 GetTepByNm response uses, so the write serializer
is the inverse of HistorianTagExtendedPropertyProtocol.ParseResponse:
uint32 groupCount (= 1)
byte 0x01 (group marker)
0x09 + uint16 byteLen + ASCII tagName (compact-ASCII string)
uint32 propertyCount
repeated propertyCount times:
byte 0x02 (property marker)
0x09 + uint16 byteLen + ASCII propertyName
0x43 + uint16 payloadLen + uint16 charCount + UTF-16LE value (VT_BSTR variant; payloadLen = 2 + charCount*2)
byte 0x01 (group trailer)
byte 0x00 (buffer terminator)
⚠️ The trailing 0x01 0x00 matters: the group trailer is 0x01 (as in the read parser) plus a
final 0x00 buffer terminator. Omitting the 0x00 makes inBuff one byte short and the server throws
SErrorException in aahClientAccessPoint::CHistStorage::AddTagExtendedProperties (AddTEx returns
false). The read parser tolerates the extra byte because it only consumes one trailing byte per group.
Only the string (0x43 VT_BSTR) value variant is evidence-backed (matching the read path). The raw
instrument capture mangles the final byte with MDAS chunk markers, so the golden fixture pins the
clean byte[] the SDK handed the channel (dumped via AVEVA_HISTORIAN_TEP_DUMP) — the exact buffer
the live server accepted.
Delete (DelTep) — deferred
DeleteTagExtendedProperties (DelTep) is not shipped yet. The native
DeleteTagExtendedPropertiesByName(tag, propertyNames, deleteFromServer, out err) performs a
client-side sync check and returns error 229 ("Tag extended property not synchronized with
server") when deleting a just-added property — so the DelTep request never reached the wire and its
inBuff could not be captured. Capturing it needs a property that is already server-synchronized (add it
in one session, then delete in a later one). Left for a follow-up rather than shipping a guessed buffer.
Shipped surface
HistorianClient.AddTagExtendedPropertiesAsync(tag, IReadOnlyList<HistorianTagExtendedProperty>)andAddTagExtendedPropertyAsync(tag, name, value).HistorianTagExtendedPropertyProtocol.SerializeAddRequest(the inBuff serializer; lives beside the R1.5 read parser); orchestrator path inHistorianWcfTagWriteOrchestrator.- Golden
WcfTagExtendedPropertyWriteProtocolTests(pins the server-accepted buffer + layout); gated live testAddTagExtendedPropertiesAsync_AgainstLocalHistorian_WritesAndReadsBack.
Capture / decode tooling
scripts/Capture-AddTagExtendedProperties.ps1 (native-harness add-tep scenario +
instrument-wcf-{write,read}message; sandbox-guarded create→add→[optional delete]) and
scripts/decode-add-tep-capture.py.