我正在用Delphi 7修改一个旧的项目,添加使用证书和签名。在做了一些搜索后,我找到了XML Canonicalization Functions,但我不能让这些函数中的任何一个工作:
- WsStartReader规范化/WsEndReader规范化
- WsStartWriter规范化/WsEndWriter规范化
这些函数包含在Microsoft WebServices.dll中,我在Github上找到了它的头文件翻译,但我仍然没有运气。下面是我测试的代码:
`
{$APPTYPE CONSOLE}
program XmlBufferExample;
//This example shows some use of the xml buffer APIs.
//Original C++ code from Microsoft :
//https://msdn.microsoft.com/en-us/library/windows/desktop/dd819131(v=vs.85).aspx
uses
Windows, Sysutils, Classes,
webservices in 'webservices.pas';
Procedure PrintError(errorCode:HRESULT; error:PWS_ERROR);
var
hr:HRESULT;
errorCount,i:ULONG;
str:WS_STRING;
s:string;
begin
writeln(Format('Failure: errorCode=0x8%.x',[errorCode]));
if (errorCode=E_INVALIDARG) or (errorCode=WS_E_INVALID_OPERATION) then
begin
// Correct use of the APIs should never generate these errors
writeln('The error was due to an invalid use of an API. This is likely due to a bug in the program.');
exit;
end;
hr:=NOERROR;
if (error<>nil) then
begin
hr:=WsGetErrorProperty(error, WS_ERROR_PROPERTY_STRING_COUNT, @errorCount, sizeof(errorCount));
if (hr=NOERROR) and (errorCount>0) then
for i:=0 to errorCount-1 do
begin
hr:=WsGetErrorString(error, i, @str);
if (hr=NOERROR) then
begin
s:=copy(str.chars,1,str.length);
writeln(s);
end else
errorCount:=i; //exit for
end;
end;
if (hr<>NOERROR) then
writeln(Format('Could not get error string (errorCode=0x8%.x)',[hr]));
end;
var
MyCallbackStatus : Cardinal;
Found : BOOL;
Mybuffer:PWS_XML_BUFFER = nil;
const
HeapSize = 24 * 1024; //24 kb
function MyCallback(callbackState : pointer; buffer : PWS_BYTES; count : ULONG;
asyncContext : PWS_ASYNC_CONTEXT; error : PWS_ERROR):HRESULT; stdcall;
var
S : AnsiString;
begin
//debuging start
Writeln('Inside MyCallback :');
Writeln('MyCallbackStatus :', IntToStr(MyCallbackStatus));
if Assigned(error) then PrintError(0, error);
//debuging end
if Assigned(buffer) then
SetString(S, PAnsiChar(buffer.bytes), buffer.length);
Writeln(s);
end;
var
hr:HRESULT;
error:PWS_ERROR;
heap:PWS_HEAP;
buffer:PWS_XML_BUFFER;
writer:PWS_XML_WRITER;
reader:PWS_XML_READER;
newXml:pointer;
newXmlLength:ULONG;
xml:ansistring;
ExitCode:integer;
Stream : TMemoryStream;
begin
error:=nil;
heap:=nil;
buffer:=nil;
writer:=nil;
reader:=nil;
newXml:=nil;
newXmlLength:=0;
// Create an error object for storing rich error information
hr := WsCreateError(nil,
0,
@error);
if (hr=NOERROR) then
begin
// Create a heap to store deserialized data
hr := WsCreateHeap(2048, //maxSize
512, //trimSize
nil,
0,
@heap,
error);
end;
if hr=NOERROR then
begin
// Create an XML writer
hr := WsCreateWriter(nil,
0,
@writer,
error);
end;
if hr=NOERROR then
begin
// Create an XML reader
hr := WsCreateReader(nil,
0,
@reader,
error);
end;
// Some xml to read and write
xml:='<a><b>1</b><c>2</c></a>';
if hr=NOERROR then
begin
hr:=WsReadXmlBufferFromBytes(reader,
nil,
nil,
0,
PAnsiChar(xml), //@xml[1],
length(xml),
heap,
@buffer,
error);
end;
if hr=NOERROR then
begin
hr:=WsWriteXmlBufferToBytes(writer,
buffer,
nil,
nil,
0,
heap,
@newXml,
@newXmlLength,
error);
end;
if hr=NOERROR then
begin
writeln('new xml :');
writeln(copy(PAnsiChar(newXml),1,newXmlLength));
writeln;
ExitCode:=0;
end;
//----------------------------------------
//My test start
//----------------------------------------
if hr=NOERROR then
begin
if Assigned(reader) then
begin
WsFreeReader(reader);
reader := nil;
end;
hr := WsCreateReader(nil, 0, @reader, error);
if Assigned(heap) then
begin
WsFreeHeap(heap);
heap := nil;
end;
hr := WsCreateHeap(3 * HeapSize, 512, nil, 0, @heap, error);
if hr=NOERROR then
begin
//load xml file
Stream := TMemoryStream.Create();
try
Stream.LoadFromFile('Z:\zatca-einvoicing-sdk\test\100030.xml');
SetString(xml, PChar(Stream.Memory), Stream.Size);
Writeln;
Writeln('-------------------------------');
Writeln('XML File size is ' + IntToStr(Stream.Size) + ' Bytes');
finally
Stream.Free();
end;
hr:=WsReadXmlBufferFromBytes(reader, nil, nil, 0, PAnsiChar(xml), length(xml), heap, @buffer, error);
end;
(*
according to https://learn.microsoft.com/en-us/windows/win32/api/webservices/nf-webservices-wsstartreadercanonicalization
The usage pattern for canonicalization is:
1) Move the Reader to the element where canonicalization begins.
2) Call WsStartReaderCanonicalization.
3) Move the Reader forward to the end position.
4) Call WsEndReaderCanonicalization.
*)
// Step1: Move the Reader to the element where canonicalization begins. [this gives an error]
if hr=NOERROR then
hr := WsMoveReader(reader, WS_MOVE_TO_PARENT_ELEMENT, @Found, error);
// Step2 : Call WsStartReaderCanonicalization.
MyCallbackStatus := 0;
if hr=NOERROR then
hr:=WsStartReaderCanonicalization(reader, MyCallback, @MyCallbackStatus, nil, 0, error);
// Step3: Move the Reader forward to the end position. [this gives an error]
if hr=NOERROR then
hr := WsMoveReader(reader, WS_MOVE_TO_EOF, @Found, error);
// Step4 : Call WsEndReaderCanonicalization [this will call MyCallback]
if hr=NOERROR then
hr := WsEndReaderCanonicalization(reader, error);
end;
//----------------------------------------
//My test End
//----------------------------------------
if hr <> NOERROR then
begin
PrintError(hr,error);
ExitCode:=-1;
end;
if writer<>nil then WsFreeWriter(writer);
if reader<>nil then WsFreeReader(reader);
if heap<>nil then WsFreeHeap(heap);
if error<>nil then WsFreeError(error);
Readln;
halt(exitcode);
end.
`
正如你在代码中看到的,函数WsMoveReader()给出了一个错误。当我删除对WsMoveReader()的调用时,代码完成,没有错误,但是当调用MyCallback时,buffer参数为nil。任何帮助都将不胜感激。
1条答案
按热度按时间q9rjltbz1#
经过多次尝试,我终于找到了我问题的答案。这个解决方案有些原始,但它执行了所需的功能。下面是我的代码。
下面是一个伪函数,但是没有它代码就不能运行。我尝试用
nil
代替它,但是失败了。规范函数。
备注:
在某些情况下,此函数会在实际结果之前和之后返回“#10”,因此必须修改结果,如下所示: