4da5287d01
Execute HCAL roadmap R1.2 (GetRuntimeParameterAsync) end-to-end, and in doing so
discover that the "string-handle wall" blocking R1.1/R1.4/R1.5/R1.6 was a handle
FORMAT bug, not a missing native session/filter registration.
R1.2 (shipped, live-verified):
- Captured native GetRuntimeParameter -> WCF op aa/Stat/GETRP (string-handle op,
GETHI's shape), via scripts/Capture-RuntimeParam.ps1 + instrument-wcf-{write,read}message.
- HistorianRuntimeParameterProtocol serializes pRequestBuff (54 67 01 00 + uint
nameCount + per-name uint charCount + UTF-16) and parses pResponseBuff (version +
uint resultCount + CRetVariant 0x43 VT_BSTR + uint16 len + uint16 charCount + UTF-16).
- IStatusServiceContract2.GetRuntimeParameter (GETRP) op; HistorianWcfStatusClient
passes the Open2 storage-session GUID as the string handle, UPPERCASE.
- Public HistorianClient.GetRuntimeParameterAsync(name) via the dialect.
- Golden WcfRuntimeParameterProtocolTests + gated live test; returns HistorianVersion.
String-handle wall RESOLVED (proven, public APIs deferred):
- The Open2 storage GUID works as the string handle when sent UPPERCASE
(ToString("D").ToUpperInvariant()); earlier "blocked" probes used lowercase.
- Live-probed GETHI (R1.4) -> returns data; ExeC (R1.1) -> Retr.GetV prime -> ExeC ->
GetR returns a BinaryFormatter-serialized .NET DataTable. Gated
StringHandleProbeDiagnosticTests + scripts/Capture-ExecSql.ps1 + exec-sql harness scenario.
- Docs flipped: wcf-string-handle-wall.md RESOLVED banner; roadmap R1.1/R1.4 reachable,
R1.5/R1.6 likely; wcf-status-localhost.md GETRP section.
- R1.1/R1.4 public APIs NOT shipped: ExeC needs a GetR paging loop + a BinaryFormatter-
stream parser (BinaryFormatter is removed from .NET 10); GETHI full-info struct needs
its own capture.
223 unit tests pass; gated live tests green against the local 2020 Historian.
Co-Authored-By: Claude Opus 4.8 (1M context) <noreply@anthropic.com>
Claude-Session: https://claude.ai/code/session_01B6mcaT2PjRFKcogzp9UkfC
3.8 KiB
3.8 KiB
WCF Status Evidence
Commands:
dotnet run --no-build --project tools\AVEVA.Historian.ReverseEngineering -- wcf-status localhost 32568
dotnet run --no-build --project tools\AVEVA.Historian.ReverseEngineering -- wcf-status localhost 32568 Version
Confirmed:
- The local status endpoint is
net.tcp://localhost:32568/Stat. IStatusServiceContract2is a static WCF contract namedStatin namespaceaa. The managed definitions now includeGetSystemParameter,GETHI,PNGS, andPNGP.GetInterfaceVersionreturns code0, version0on the local 2020 install.- The decompiled
CStatusConnectionWCF.GetServerTimeimplementation is a WCF-path stub that returns success without calling theStatservice. The managed direct call likewise returns code0with size0and no buffer.
Observed sanitized localhost results:
GetSystemTimeZoneName(handle: 0)returns code4and no value.IsDBCaseSensitive(handle: 0)returns code4.GetSystemParameter(handle: 0, "Version")returnsfalsewith no error buffer.
Re-tested 2026-06-20 with a real authenticated client handle (full Open2 auth
chain), not handle: 0:
GetSystemParameter(handle, "HistorianVersion")→ real version string (works; shipped asGetSystemParameterAsync).GetSystemTimeZoneName(handle)→ return code0x00000000(success) but an empty value string. Same channel/handle that makesGetSystemParameterreturn real data, so this is the op's own behavior, not an auth/marshalling gap.GetSystemTimeZoneNameis a member of theGetServerTimestub family: the 2020 WCF path returns success without producing a value (the native client computes the zone locally). It only becomes a real round-trip on the 2023 R2 gRPC front door (Status.GetSystemTimeZoneName), which is absent on this box.
Interpretation:
Statendpoint routing is confirmed, but status operations that require a real client handle are not usable until managed session open is solved.GetServerTimeshould not be promoted into the public SDK as a real server time call from this WCF path; native evidence shows it is a no-op stub here.GetServerTimeZoneAsync(roadmap R1.3) is NOT a trivial WCF op on 2020 — it is a stub returning empty. Do not ship it over the 2020 WCF transport. Deliver it only against a live 2023 R2 gRPC server. Reclassified indocs/plans/hcal-roadmap.md.
GETRP / GetRuntimeParameter (roadmap R1.2) — DONE, live-verified 2026-06-20
Captured the native HistorianAccess.GetRuntimeParameter(List<string>, out List<object>)
WCF traffic with scripts/Capture-RuntimeParam.ps1 (instrument-wcf-{write,read}message).
Findings:
- The WCF op is
aa/Stat/GETRP—bool GETRP(string handle, byte[] pRequestBuff, out byte[] pResponseBuff, out byte[] errorBuffer), i.e. the same string-handle + request/response-buffer shape as GETHI, not the simpleGetSystemParameter(uint, string)shape the roadmap originally assumed. - The
string handleis the Open2 storage-session GUID (the valueParseOpenConnectionResponsereads fromoutBuff[5..21]), sent UPPERCASE, dash-separated, no braces (ToString("D").ToUpperInvariant()). - Unlike GETHI (which the earlier probe found blocked), GETRP succeeds from the pure-managed
client with that handle:
GetRuntimeParameter("HistorianVersion")→20,0,000,000. pRequestBuff=54 67 01 00(sig+version) + uint nameCount + per name(uint charCount + UTF-16LE).pResponseBuff= version(1) + uint resultCount + CRetVariant(0x43VT_BSTR + uint16 payloadLen + uint16 charCount + UTF-16LE).
Shipped as HistorianClient.GetRuntimeParameterAsync(name). See
HistorianRuntimeParameterProtocol, golden WcfRuntimeParameterProtocolTests, and the
handle-format lead in wcf-string-handle-wall.md §Update (retry GETHI/ExeC uppercased).