{ +-------------------------------------------------------------+ } { | | } { | GM-Software | } { | =========== | } { | | } { | Project: All Projects | } { | | } { | Description: Simple SOAP implementations. | } { | | } { | | } { | Copyright (C) - 2010 - Gerrit Moeller. | } { | | } { | Source code distributed under MIT license. | } { | | } { | See: https://www.gm-software.de | } { | | } { +-------------------------------------------------------------+ } {$INCLUDE GMCompilerSettings.inc} unit GMSoap; interface uses {$IFDEF JEDIAPI}jwaWinType,{$ELSE}Windows,{$ENDIF} GMActiveX, GMStrDef, GMCollections, GMIntf, GMCommon, GMXml, GMHttp, TypInfo; const cDfltSoapPropTypeKinds = [tkInteger, tkChar, tkFloat, tkString, tkWChar, tkLString, tkWString, tkVariant, tkInt64]; // tkEnumeration, tkSet type // // Include RTTI to access properties at runtime // {$IFOPT M-} {$DEFINE RTTI_OFF} {$M+} {$ENDIF} TGMSoapValueCarrierObj = class(TGMRefCountedObj) public constructor Create(const ARefLifeTime: Boolean = False); reintroduce; virtual; end; {$IFDEF RTTI_OFF} {$M-} {$UNDEF RTTI_OFF} {$ENDIF} TGMSoapDataObjClass = class of TGMSoapValueCarrierObj; IGMLoadProperties = interface(IUNknown) ['{590F0CE0-A6C1-4796-B870-4BCC0A68B3C2}'] procedure LoadProperties(Node: IGMXmlNode; const ForceExist: Boolean = True); stdcall; end; IGMStoreProperties = interface(IUnknown) ['{BDBCD58A-8B7E-4a29-8F2D-5AA7B5C3C0B7}'] procedure StoreProperties(Node: IGMXmlNode); stdcall; end; EGMSoapException = class(EGMException); EGMSoapExceptionClass = class of EGMSoapException; TGMSoapMethodData = record NameSpace: TGMString; MethodName: TGMString; end; TGMSoapCallBase = class(TGMRefCountedObj) protected FTransportLayer: ISequentialStream; FXmlParseAttributes: TGMXmlParseAttributes; FOperationNS: TGMXmlNamedValueData; function InsertSoapEnvelope(const AParentNode: IGMXmlNode; const AAddOperationNS: Boolean = True): IGMXmlNode; virtual; function InsertSoapBody(const AParentNode: IGMXmlNode): IGMXmlNode; virtual; procedure AddEnvelopeAttributes(const AEnvelopeNode: IGMXmlNode; const AAddOperationNS: Boolean = True); virtual; function XmlTreeCreateClass: TGMXmlTreeClass; virtual; public constructor Create(const ATransportLayer: ISequentialStream; const ASoapPortTypeURL: TGMString; const AXmlParseAttributes: TGMXmlParseAttributes = cDfltXmlParseAttributes; const ARefLifeTime: Boolean = True); reintroduce; overload; end; TGMSoapClientCall = class(TGMSoapCallBase) protected FTypeNameSpace: TGMString; function WrongContentExceptionCreateClass: EGMExceptionClass; virtual; function ValueAsSoapString(const AValue: OleVariant): TGMString; virtual; function AddTypeAttributes: Boolean; virtual; procedure PrepareHttpRequest(const ARequest: IUnknown); virtual; function CreateSoapFaultExceptObj(const ASoapFaultNode: IGMXmlNode; const ACallingName: TGMString): EGMSoapException; virtual; abstract; procedure RaiseSoapFault(const ASoapFaultNode: IGMXmlNode; const ACallingName: TGMString); virtual; procedure CheckResponseXml(const AResponseXml: IGMXmlTree; const ACallingName: TGMString); virtual; procedure CheckResponseContentType(const AResponseContent: ISequentialStream; const AContentType, ACallingName: TGMString); virtual; function CheckResponseContent(const AResponseContent: ISequentialStream; const AContentType: TGMString; const ACallingName: TGMString): IGMXmlTree; virtual; //function ExtractSoapFaultMessage(const AResponseContent: IStream; var AXml: IGMXmlTree): TGMString; procedure AddEnvelopeAttributes(const AEnvelopeNode: IGMXmlNode; const AAddOperationNS: Boolean = True); override; public // constructor Create(const ATransportLayer: ISequentialStream; const ASoapPortTypeURL: TGMString; const AXmlParseAttributes: TGMXmlParseAttributes = cDfltXmlParseAttributes; const ARefLifeTime: Boolean = True); reintroduce; constructor Create(const ARefLifeTime: Boolean = True); override; function CreateSOAPCallXml(const ASoapMethodName: TGMString; var ANode: IGMXmlNode): IGMXmlTree; function AddSoapValue(const AParentNode: IGMXmlNode; const AName: TGMString; const AValue: OleVariant): IGMXmlNode; procedure StoreObjProps(const Obj: TObject; Node: IGMXmlNode; const ClassNodeName: TGMString; const TypeKinds: TTypeKinds); procedure LoadObjProps(const Obj: TObject; Node: IGMXmlNode; const ForceExist: Boolean; const TypeKinds: TTypeKinds); function ExecSoapCall(const ASession: IGMHttpClientSession; const AXML: IGMXmlTree; const ACallingName: TGMString = ''): IGMXmlTree; virtual; end; TGMSoapServerPort = class; IGMSoapServerPort = interface(IUnknown) ['{083F2619-5846-461F-B27A-5F78FD34D6B7}'] function Obj: TGMSoapServerPort; end; TGMSoapMethod = procedure (const ARequestMethodNode, AResponseBodyNode: IGMXmlNode) of object; TGMSoapMethodEntry = class(TGMRefCountedObj, IGMGetName) public SoapMethod: TGMSoapMethod; SoapMethodName: TGMString; constructor Create(const AMethodName: TGMString; const ASoapMethod: TGMSoapMethod; const ARefLifeTime: Boolean = False); reintroduce; function GetName: TGMString; stdcall; end; TGMSoapRequestResult = record HttpStatusCode: DWORD; LogMessage: AnsiString; ContentType: AnsiString; SOAPMethodName: TGMString; end; TGMSoapServerPort = class(TGMSoapCallBase, IGMSoapServerPort) protected FRegisteredSoapMethods: IGMObjArrayCollection; FSoapPortName: TGMString; function DoTracing: Boolean; virtual; function TracePrefix: TGMString; virtual; function InsertFaultDetailNode(const AFaultNode: IGMXmlNode; const AExceptObject: TObject): TGMString; virtual; abstract; function InsertFaultNode(const ABodyNode: IGMXmlNode; const AExceptObject: TObject): TGMString; procedure RegisterSOAPMethod(const AMethodName: AnsiString; const AMethod: TGMSoapMethod); function FindSoapMethod(const AMethodName: TGMString): TGMSoapMethod; virtual; public constructor Create(const ARefLifeTime: Boolean = True); overload; override; constructor Create(const ATransportLayer: ISequentialStream; const ASoapPortName, ASoapPortTypeURL: TGMString; const AXmlParseAttributes: TGMXmlParseAttributes = cDfltXmlParseAttributes; const ARefLifeTime: Boolean = True); reintroduce; overload; function Obj: TGMSoapServerPort; // function ProcessRequest(const AReadStrm, AWriteStrm: ISequentialStream): TGMSoapRequestResult; function ProcessRequest: TGMSoapRequestResult; virtual; // procedure SendResponseContent(const ATransportLayer: ISequentialStream); virtual; end; TGMSoapServerCallClass = class of TGMSoapServerPort; // TSMSUnkResultStateType(GMCheckGetEnumValFromName(TypeInfo(TSMSUnkResultStateType), GMCheckFindXmlSubValue(Node, cSoapResultState))); // GetEnumName(TypeInfo(SMSHelpTopicType), Ord(ATopic)) function GMSoapStrToInt(const AValue: TGMString; const ADefaultValue: LongInt = 0): LongInt; // ; ACaller: TObject = nil; const ACallingName: TGMString = '' function GMSoapBoolToStr(const AValue: Boolean): TGMString; //function GMIso8601DateTimeToStr(const ALocalTime: TDateTime): TGMString; //function GMAddSoapValue(const AParent: IGMXmlNode; const AName, ANameSpace: TGMString; const AValue: OleVariant): IGMXmlNode; function GMSoapTypeName(const AVType: LongInt): TGMString; procedure GMStoreObjProps(const Obj: TObject; Node: IGMXmlNode; const ClassNodeName: TGMString = ''; const TypeKinds: TTypeKinds = cDfltSoapPropTypeKinds); procedure GMLoadObjProps(const AObj: TObject; ANode: IGMXmlNode; const AForceExist: Boolean = True; const ATypeKinds: TTypeKinds = cDfltSoapPropTypeKinds); const cStrEnvNS = 'SOAP-ENV'; //cStrEncNS = 'SOAP-ENC'; cStrSoapEnv = 'Envelope'; cStrSoapHeader = 'Header'; cStrSoapBody = 'Body'; cStrSoapFault = 'Fault'; cStrSoapFaultString = 'faultstring'; // <- should be lowercase! cStrSoapFaultCode = 'faultcode'; // <- should be lowercase! cStrSoapFaultDetail = 'detail'; // <- should be lowercase! cStrExec = 'exec'; cStrXsd = 'xsd'; cStrXsi = 'xsi'; cStrType = 'type'; //cStrHtmlBody = cStrBody; //cStrArray = 'Array'; cStrSoap = 'SOAP'; implementation uses Classes, SysUtils, GMWinInetAPI, GMINetBase {$IFDEF DELPHI6},Variants{$ENDIF} ; const cStrHttpContentHtml = 'text/html'; resourcestring //RStrNoHttpStatusCode = 'The server did not transfer a http status code'; RStrNoSoapFaultMsg = 'No SOAP fault message transmitted by the server'; //RStrResponseContentStream = 'The response content stream'; RStrResonseContentNotXml = 'The response content is not XML/HTML'; RStrNoSOAPMethodNode = 'No SOAP method XML node found'; RStrEmptySoapMethodName = 'Empty Soap method name'; RStrSoapMethodNotImpl = 'SOAP method "%s" not implemented'; //RStrSOAPCallLogMsg = 'WFM SOAP call'; RStrException = 'Exception'; RStrUsingSoapPort = 'Using SOAP port'; //RStrInvalidIntFmt = 'Invalid integer value: %s'; { ------------------------- } { ---- Global Routines ---- } { ------------------------- } function GMSoapStrToInt(const AValue: TGMString; const ADefaultValue: LongInt): LongInt; // ; ACaller: TObject; const ACallingName: TGMString begin if Length(GMStrip(AValue, cWhiteSpace)) <= 0 then Result := ADefaultValue else Result := GMStrToInt(AValue); //Result := GMStrToInt(GMMakeDezInt(AValue)); // try // Result := GMStrToInt(AValue); //except // raise EGMException.ObjError(GMFormat(RStrInvalidIntFmt, [AValue]), ACaller, ACallingName); //end; end; //function GMSoapStrToInt(const AValue: TGMString; ACaller: TObject): LongInt; //begin //end; function GMSoapTypeName(const AVType: LongInt): TGMString; begin case VarTYpeMask and AVType of varInteger: Result := 'int'; varDate: Result := 'dateTime'; varBoolean: Result := 'boolean'; else Result := 'TGMString'; end; end; function GMSoapBoolToStr(const AValue: Boolean): TGMString; begin Result := GMBoolToStr(AValue, 'false', 'true'); end; function GMSoapVarToStr(const AValue: OleVariant): TGMString; begin case VarType(AValue) of varBoolean: Result := GMSoapBoolToStr(AValue); varDate: Result := GMIso8601DateTimeToStr(AValue); else Result := AValue; end; end; function GMAddSoapValue(const AParent: IGMXmlNode; const AName, ANameSpace: TGMString; const AValue: OleVariant): IGMXmlNode; begin if AParent <> nil then Result := AParent.Obj.Owner.CreateNewNode(AParent, AName, AValue, ANameSpace); end; procedure GMStoreObjProps(const Obj: TObject; Node: IGMXmlNode; const ClassNodeName: TGMString; const TypeKinds: TTypeKinds); var Count, i: Integer; PropList: PPropList; PIStore: IGMStoreProperties; begin if (Obj = nil) or (Node = nil) then Exit; if Obj.ClassInfo = nil then raise EGMSoapException.ObjError(GMFormat(RStrNeedRTTI, [Obj.ClassName]), Obj, 'GMStoreObjProps'); {ToDo: Klassentyp f?r XML Knoten?} if ClassNodeName <> '' then Node := GMCreateXmlNode(Node, ClassNodeName); Count := GetTypeData(PTypeInfo(Obj.ClassInfo))^.PropCount; if Count > 0 then begin GetMem(PropList, Count * SizeOf(Pointer)); try GetPropInfos(PTypeInfo(Obj.ClassInfo), PropList); for i:=0 to Count-1 do if PropList^[i].PropType^{$IFNDEF FPC}^{$ENDIF}.Kind in TypeKinds then GMAddSoapValue(Node, PropList^[i].Name, '', GetPropValue(Obj, PropList^[i].Name, False)); finally FreeMem(PropList); end; end; if Obj.GetInterface(IGMStoreProperties, PIStore) then PIStore.StoreProperties(Node); end; procedure GMLoadObjProps(const AObj: TObject; ANode: IGMXmlNode; const AForceExist: Boolean; const ATypeKinds: TTypeKinds); var Count, i: Integer; PropList: PPropList; ValNode: IGMXmlNode; PILoad: IGMLoadProperties; begin if (AObj = nil) or (ANode = nil) then Exit; if AObj.ClassInfo = nil then raise EGMSoapException.ObjError(GMFormat(RStrNeedRTTI, [AObj.ClassName]), AObj, 'GMLoadObjProps'); Count := GetTypeData(PTypeInfo(AObj.ClassInfo))^.PropCount; if Count > 0 then begin GetMem(PropList, Count * SizeOf(Pointer)); try GetPropInfos(PTypeInfo(AObj.ClassInfo), PropList); for i:=0 to Count-1 do if PropList^[i].PropType^{$IFNDEF FPC}^{$ENDIF}.Kind in ATypeKinds then begin if AForceExist then ValNode := ANode.Obj.CheckFindSubNode(PropList^[i].Name) else ValNode := ANode.Obj.FindSubNode(PropList^[i].Name); if (ValNode <> nil) and (ValNode.Obj.StrValue <> '') then SetPropValue(AObj, PropList^[i].Name, ValNode.Obj.StrValue) else case PropList^[i].PropType^.Kind of tkChar, tkString, tkWChar, tkLString, tkWString: SetPropValue(AObj, PropList^[i].Name, ''); tkInteger, tkFloat, tkInt64: SetPropValue(AObj, PropList^[i].Name, 0); tkVariant: SetPropValue(AObj, PropList^[i].Name, Null); //tkClass, tkInterface: SetPropValue(AObj, PropList^[i].Name, nil); end; end; finally FreeMem(PropList); end; end; if AObj.GetInterface(IGMLoadProperties, PILoad) then PILoad.LoadProperties(ANode, AForceExist); end; { -------------------------------- } { ---- TGMSoapValueCarrierObj ---- } { -------------------------------- } constructor TGMSoapValueCarrierObj.Create(const ARefLifeTime: Boolean); begin inherited Create(ARefLifeTime); end; { ------------------------- } { ---- TGMSoapCallBase ---- } { ------------------------- } constructor TGMSoapCallBase.Create(const ATransportLayer: ISequentialStream; const ASoapPortTypeURL: TGMString; const AXmlParseAttributes: TGMXmlParseAttributes; const ARefLifeTime: Boolean); begin Create(ARefLifeTime); FTransportLayer := ATransportLayer; FXmlParseAttributes := AXmlParseAttributes; FOperationNS := GMXmlNamedValueData(cStrExec, ASoapPortTypeURL); //FOperationNS.Value := URLExctractResourcePath(ASoapPortTypeURL); end; function TGMSoapCallBase.XmlTreeCreateClass: TGMXmlTreeClass; begin Result := TGMXmlTree; end; function TGMSoapCallBase.InsertSoapEnvelope(const AParentNode: IGMXmlNode; const AAddOperationNS: Boolean): IGMXmlNode; begin if AParentNode = nil then Exit; Result := GMCreateXmlNode(AParentNode, cStrSoapEnv, '', cStrEnvNS); AddEnvelopeAttributes(Result, AAddOperationNS); end; function TGMSoapCallBase.InsertSoapBody(const AParentNode: IGMXmlNode): IGMXmlNode; begin if AParentNode = nil then Exit; Result := GMCreateXmlNode(AParentNode, cStrSoapBody, '', cStrEnvNS); end; procedure TGMSoapCallBase.AddEnvelopeAttributes(const AEnvelopeNode: IGMXmlNode; const AAddOperationNS: Boolean); begin if AEnvelopeNode <> nil then with AEnvelopeNode.Obj do begin Attributes.Add(AttributeCreateClass.Create(GMXmlQualifiedName(cStrXmlns, cStrEnvNS), '"http://schemas.xmlsoap.org/soap/envelope/"')); if AAddOperationNS and (Length(FOperationNS.Name) > 0) and (Length(FOperationNS.Value) > 0) then Attributes.Add(AttributeCreateClass.Create(GMXmlQualifiedName(cStrXmlns, FOperationNS.Name), GMXmlAttrQuote(FOperationNS.Value))); end; end; { --------------------------- } { ---- TGMSoapClientCall ---- } { --------------------------- } constructor TGMSoapClientCall.Create(const ARefLifeTime: Boolean); begin inherited Create(ARefLifeTime); FTypeNameSpace := cStrXsi; end; function TGMSoapClientCall.AddTypeAttributes: Boolean; begin Result := Length(FTypeNameSpace) > 0; end; function TGMSoapClientCall.WrongContentExceptionCreateClass: EGMExceptionClass; begin Result := EGMSoapException; end; function TGMSoapClientCall.ValueAsSoapString(const AValue: OleVariant): TGMString; begin Result := GMSoapVarToStr(AValue); end; function TGMSoapClientCall.AddSoapValue(const AParentNode: IGMXmlNode; const AName: TGMString; const AValue: OleVariant): IGMXmlNode; begin if AParentNode = nil then Exit; Result := GMCreateXmlNode(AParentNode, AName, ValueAsSoapString(AValue)); if AddTypeAttributes then Result.Obj.Attributes.Add(Result.Obj.AttributeCreateClass.Create(GMXmlQualifiedName(FTypeNameSpace, cStrType), GMXmlAttrQuote(GMXmlQualifiedName(cStrXsd, GMSoapTypeName(VarType(AValue)))))); end; procedure TGMSoapClientCall.AddEnvelopeAttributes(const AEnvelopeNode: IGMXmlNode; const AAddOperationNS: Boolean); begin inherited; if AEnvelopeNode <> nil then with AEnvelopeNode.Obj do if AddTypeAttributes then Attributes.Add(AttributeCreateClass.Create(GMXmlQualifiedName(cStrXmlns, FTypeNameSpace), '"http://www.w3.org/2001/XMLSchema-instance"')); // with AEnvelopeNode.Obj do // begin // //Attributes.Add(AttributeCreateClass.Create(GMXmlQualifiedName(cStrXmlns, cStrEnvNS), '"http://schemas.xmlsoap.org/soap/envelope/"')); // if AddTypeAttributes then Attributes.Add(AttributeCreateClass.Create(GMXmlQualifiedName(cStrXmlns, FTypeNameSpace), '"http://www.w3.org/2001/XMLSchema-instance"')); //// Attributes.Add(AttributeCreateClass.Create(GMXmlQualifiedName(cStrXmlns, cStrXsd), '"http://www.w3.org/2001/XMLSchema"')); //// Attributes.Add(AttributeCreateClass.Create(GMXmlQualifiedName(cStrXmlns, cStrXsi), '"http://www.w3.org/2001/XMLSchema-instance"')); //// Attributes.Add(AttributeCreateClass.Create(GMXmlQualifiedName(cStrXmlns, cStrEncNS), '"http://schemas.xmlsoap.org/soap/encoding/"')); // end; end; procedure TGMSoapClientCall.StoreObjProps(const Obj: TObject; Node: IGMXmlNode; const ClassNodeName: TGMString; const TypeKinds: TTypeKinds); var Count, i: Integer; PropList: PPropList; PIStore: IGMStoreProperties; begin if (Obj = nil) or (Node = nil) then Exit; if Obj.ClassInfo = nil then raise EGMSoapException.ObjError(GMFormat(RStrNeedRTTI, [Obj.ClassName]), Obj, 'StoreObjProps'); {ToDo: Klassentyp f?r XML Knoten?} if ClassNodeName <> '' then Node := GMCreateXmlNode(Node, ClassNodeName, ''); Count := GetTypeData(PTypeInfo(Obj.ClassInfo))^.PropCount; if Count > 0 then begin GetMem(PropList, Count * SizeOf(Pointer)); try GetPropInfos(PTypeInfo(Obj.ClassInfo), PropList); for i:=0 to Count-1 do if PropList^[i].PropType^{$IFNDEF FPC}^{$ENDIF}.Kind in TypeKinds then AddSoapValue(Node, PropList^[i].Name, GetPropValue(Obj, PropList^[i].Name, False)); finally FreeMem(PropList); end; end; if Obj.GetInterface(IGMStoreProperties, PIStore) then PIStore.StoreProperties(Node); end; procedure TGMSoapClientCall.LoadObjProps(const Obj: TObject; Node: IGMXmlNode; const ForceExist: Boolean; const TypeKinds: TTypeKinds); var Count, i: Integer; PropList: PPropList; ValNode: IGMXmlNode; PILoad: IGMLoadProperties; begin if (Obj = nil) or (Node = nil) then Exit; if Obj.ClassInfo = nil then raise EGMSoapException.ObjError(GMFormat(RStrNeedRTTI, [Obj.ClassName]), Obj, 'GMLoadObjProps'); Count := GetTypeData(PTypeInfo(Obj.ClassInfo))^.PropCount; if Count > 0 then begin GetMem(PropList, Count * SizeOf(Pointer)); try GetPropInfos(PTypeInfo(Obj.ClassInfo), PropList); for i:=0 to Count-1 do if PropList^[i].PropType^{$IFNDEF FPC}^{$ENDIF}.Kind in TypeKinds then begin if ForceExist then ValNode := Node.Obj.CheckFindSubNode(PropList^[i].Name) else ValNode := Node.Obj.FindSubNode(PropList^[i].Name); if (ValNode <> nil) and (ValNode.Obj.StrValue <> '') then SetPropValue(Obj, PropList^[i].Name, ValNode.Obj.StrValue) else case PropList^[i].PropType^.Kind of tkChar, tkString, tkWChar, tkLString, tkWString: SetPropValue(Obj, PropList^[i].Name, ''); tkInteger, tkFloat, tkInt64: SetPropValue(Obj, PropList^[i].Name, 0); tkVariant: SetPropValue(Obj, PropList^[i].Name, Null); //tkClass, tkInterface: SetPropValue(Obj, PropList^[i].Name, nil); end; end; finally FreeMem(PropList); end; end; if Obj.GetInterface(IGMLoadProperties, PILoad) then PILoad.LoadProperties(Node, ForceExist); end; function TGMSoapClientCall.CreateSOAPCallXml(const ASoapMethodName: TGMString; var ANode: IGMXmlNode): IGMXmlTree; begin Result := XmlTreeCreateClass.CreateWrite; ANode := GMCreateXmlNode(InsertSoapBody(InsertSoapEnvelope(Result.Obj.RootNode)), GMXmlQualifiedName(FOperationNS.Name, ASoapMethodName)); end; procedure TGMSoapClientCall.RaiseSoapFault(const ASoapFaultNode: IGMXmlNode; const ACallingName: TGMString); function BuildFaultMsg: TGMString; var Node: IGMXmlNode; begin if (ASoapFaultNode <> nil) and ASoapFaultNode.Obj.FindSubNodeIntoVar(cStrSoapFaultString, nil, Node) then Result := GMStrip(Node.Obj.StrValue); if Length(Result) <= 0 then Result := RStrNoSoapFaultMsg; end; var SoapExceptObj: EGMSoapException; begin SoapExceptObj := nil; try SoapExceptObj := CreateSoapFaultExceptObj(ASoapFaultNode, ACallingName); if SoapExceptObj = nil then SoapExceptObj := EGMSoapException.ObjError(BuildFaultMsg, Self, BuildCallingName(ACallingName, {$I %CurrentRoutine%})); except GMFreeAndNil(SoapExceptObj); end; if SoapExceptObj <> nil then begin if Length(SoapExceptObj.Message) <= 0 then SoapExceptObj.Message := BuildFaultMsg; raise SoapExceptObj; end; end; procedure TGMSoapClientCall.CheckResponseXml(const AResponseXml: IGMXmlTree; const ACallingName: TGMString); var node: IGMXmlNode; begin if AResponseXml = nil then Exit; if GMGetXmlNodeByPath(AResponseXml.Obj.RootNode, [cStrSoapEnv, cStrSoapBody, cStrSoapFault], node) then RaiseSoapFault(node, BuildCallingName(ACallingName, {$I %CurrentRoutine%})); //if AResponseXml.Obj.RootNode.Obj.FindSubNode(cStrSoapEnv, node, 1) then // if node.Obj.FindSubNode(cStrSoapBody, node, 1) then // if node.Obj.FindSubNode(cStrSoapFault, node, 1) then RaiseSoapFault(node, BuildCallingName(ACallingName, {$I %CurrentRoutine%})); end; procedure TGMSoapClientCall.CheckResponseContentType(const AResponseContent: ISequentialStream; const AContentType, ACallingName: TGMString); begin if CompareText(cStrHttpContentXml, AContentType) <> 0 then raise WrongContentExceptionCreateClass.ObjError(GMStringJoin(GMTerminateStr(GMFormat(RStrWrongContentType, [AContentType, cStrHttpContentXml])), c2NewLine, GMExtractAnyTextResponse(AResponseContent, AContentType)), Self, BuildCallingName(ACallingName, {$I %CurrentRoutine%})); end; function TGMSoapClientCall.CheckResponseContent(const AResponseContent: ISequentialStream; const AContentType: TGMString; const ACallingName: TGMString): IGMXmlTree; //var StreamPosKeeper: IUnknown; begin if AResponseContent = nil then Exit; // begin Result := XmlTreeCreateClass.CreateRead(AResponseContent); Exit; end; //raise EGMException.ObjError(MsgPointerIsNil(RStrResponseContentStream), Self, {$I %CurrentRoutine%}); //StreamPosKeeper := TGMIStreamPosKeeper.Create(AResponseContent); //if not GMIsXmlContent(AResponseContent) then // raise WrongContentExceptionCreateClass.ObjError(RStrResonseContentNotXml, Self, BuildCallingName(ACallingName, {$I %CurrentRoutine%})); if CompareText(cStrHttpContentXml, AContentType) = 0 then Result := XmlTreeCreateClass.CreateRead(AResponseContent, FXmlParseAttributes) else //if CompareText(cStrHttpContentHtml, AContentType) <> 0 then Result := TGMHtmlTree.CreateRead(AResponseContent, FXmlParseAttributes - [paCheckHasXmlToken]); if CompareText(cStrHttpContentHtml, AContentType) = 0 then Result := TGMHtmlTree.CreateRead(AResponseContent, cRelaxedHtmlParseAttributes); //else // raise WrongContentExceptionCreateClass.ObjError(GMStringJoin(GMTerminateStr(GMFormat(RStrWrongContentType, [AContentType, cStrHttpContentXml])), c2NewLine, // GMExtractAnyTextResponse(AResponseContent, AContentType)), Self, BuildCallingName(ACallingName, {$I %CurrentRoutine%})); // // When Result is nil a later check of the content type will fail! // if Result <> nil then CheckResponseXml(Result, ACallingName); end; procedure TGMSoapClientCall.PrepareHttpRequest(const ARequest: IUnknown); begin {ToDo: Add "SOAPAction" header here!} // soapaction = "SOAPAction" ":" [ <"> URI-reference <"> ] // URI-reference = <as defined in RFC 2396 [4]> // Nothing! May be used by derived classes to add request headers. end; function TGMSoapClientCall.ExecSoapCall(const ASession: IGMHttpClientSession; const AXML: IGMXmlTree; const ACallingName: TGMString): IGMXmlTree; var responseStrm: ISequentialStream; xmlStrm: IStream; request: IGMHttpClientRequest; uriParts: RGMUriComponents; contentType: TGMString; begin if (ASession = nil) or (AXML = nil) then Exit; xmlStrm := TGMMemoryIStream.Create; //SrcStrm := TGMStreamAdapter.Create(xmlStrm); AXML.Obj.SaveToStream(xmlStrm); uriParts := GMParseUri(FOperationNS.Value); //request := FConnection.CreateHttpRequest(cStrHttpPost, FSoapPortURL); request := TGMHttpClientRequest.Create(cGMHttpAgent); PrepareHttpRequest(request); //request.AddHeaders(GMFormat('%s: %s', [cStrHdrUserAgent, cStrUserAgent])); //request.Obj.SendRequest('', xmlStrm.Obj.Memory, xmlStrm.Obj.Size, cGMTracePrefixes[tpXml]); responseStrm := ASession.ExecuteRequest(request, GMBuildUri('', '', '', '', '', uriParts.Path, uriParts.Query, uriParts.Fragment), cHttpMethodPOST, xmlStrm, cStrHttpContentXml); //responseStrm := TGMMemoryIStream.Create; //request.Obj.ReadResponseContent(responseStrm); // <- read responseStrm before HttpCheck to show it in the trace //contentType := GMStrip(GMFirstWord(request.Obj.GetStrHeader(HTTP_QUERY_CONTENT_TYPE), ';,'), cWhiteSpace); contentType := GMStrip(GMFirstWord(GMGetINetHeaderStrValue(request.Obj.ReceivedHeaders, cHttpContentType), ';,'), cWhiteSpace); //httpCode := request.Obj.GetIntHeader(HTTP_QUERY_STATUS_CODE, -1); // // Check the content before checking the content-type, because SOAP faults may be delivered as text/html by the server! // Result := CheckResponseContent(responseStrm, contentType, {$I %CurrentRoutine%}); CheckResponseContentType(responseStrm, contentType, {$I %CurrentRoutine%}); //if httpCode = -1 then raise EGMException.ObjError(RStrNoHttpStatusCode, Self, {$I %CurrentRoutine%}); // // SOAP faults come with HTTP code 500 - server error, so check for SOAP fault before raising a normal HTTP server error! // //if httpCode <> HTTP_STATUS_OK then // HttpCheck(httpCode, [HTTP_STATUS_OK], GMExtractAnyTextResponse(responseStrm, contentType), Self, {$I %CurrentRoutine%}); //GMCheckHTTPResponseContentType(request, cStrHttpContentXml, Self, {$I %CurrentRoutine%}, SoapFaultMsg); if Result = nil then Result := XmlTreeCreateClass.CreateRead(responseStrm, FXmlParseAttributes); end; { ---------------------------- } { ---- TGMSoapMethodEntry ---- } { ---------------------------- } constructor TGMSoapMethodEntry.Create(const AMethodName: TGMString; const ASoapMethod: TGMSoapMethod; const ARefLifeTime: Boolean); begin inherited Create(ARefLifeTime); SoapMethodName := AMethodName; SoapMethod := ASoapMethod; end; function TGMSoapMethodEntry.GetName: TGMString; begin Result := SoapMethodName; end; { --------------------------- } { ---- TGMSoapServerPort ---- } { --------------------------- } constructor TGMSoapServerPort.Create(const ARefLifeTime: Boolean); begin inherited Create(ARefLifeTime); FRegisteredSoapMethods := TGMObjArrayCollection.Create(True, False, True, GMCompareByName, True); end; constructor TGMSoapServerPort.Create(const ATransportLayer: ISequentialStream; const ASoapPortName, ASoapPortTypeURL: TGMString; const AXmlParseAttributes: TGMXmlParseAttributes; const ARefLifeTime: Boolean); begin inherited Create(ATransportLayer, ASoapPortTypeURL, AXmlParseAttributes, ARefLifeTime); FSoapPortName := ASoapPortName; end; function TGMSoapServerPort.Obj: TGMSoapServerPort; begin Result := Self; end; function TGMSoapServerPort.DoTracing: Boolean; begin Result := {$IFDEF DEBUG}True{$ELSE}False{$ENDIF}; end; function TGMSoapServerPort.TracePrefix: TGMString; begin Result := cStrSoap; end; procedure TGMSoapServerPort.RegisterSOAPMethod(const AMethodName: AnsiString; const AMethod: TGMSoapMethod); begin FRegisteredSoapMethods.Add(TGMSoapMethodEntry.Create(AMethodName, AMethod)); end; function TGMSoapServerPort.FindSoapMethod(const AMethodName: TGMString): TGMSoapMethod; var searchName: RGMNameRec; methodEntry: TGMSoapMethodEntry; begin searchName.Name := AMethodName; if FRegisteredSoapMethods.Find(searchName, methodEntry) then Result := methodEntry.SoapMethod else Result := nil; end; function TGMSoapServerPort.InsertFaultNode(const ABodyNode: IGMXmlNode; const AExceptObject: TObject): TGMString; var faultNode: IGMXmlNode; begin if (ABodyNode = nil) or (AExceptObject = nil) then Exit; faultNode := GMCreateXmlNode(ABodyNode, cStrSoapFault, '', cStrEnvNS); GMCreateXmlNode(faultNode, cStrSoapFaultCode, AExceptObject.ClassName); GMCreateXmlNode(faultNode, cStrSoapFaultString, RStrException); // GMMsgFromExceptObj(AExceptObject, False) Result := InsertFaultDetailNode(faultNode, AExceptObject); end; //function TGMSoapServerPort.ProcessRequest(const AReadStrm, AWriteStrm: ISequentialStream): TGMSoapRequestResult; function TGMSoapServerPort.ProcessRequest: TGMSoapRequestResult; var requestXml, responseXml: IGMXmlTree; responseBodyNode, requestBodyNode, requestMethodNode: IGMXmlNode; // responseEnvNode errMsg: TGMString; soapMethod: TGMSoapMethod; begin if DoTracing then vfGMTrace(RStrUsingSoapPort + ': "' + FSoapPortName+'"', TracePrefix); responseXml := XmlTreeCreateClass.CreateWrite; //responseEnvNode := InsertSoapEnvelope(responseXml.Obj.RootNode); //GMCreateXmlNode(responseEnvNode, cStrSoapHeader, '', cStrEnvNS); responseBodyNode := InsertSoapBody(InsertSoapEnvelope(responseXml.Obj.RootNode)); try // Check request content type, or just read the requestXml? Would cause requestXml related exception if other content was sent // // Parsing the calling XML should already be wrapped by SOAP fault handling // requestXml := XmlTreeCreateClass.CreateRead(FTransportLayer, FXmlParseAttributes); requestBodyNode := GMCheckGetXmlNodeByPath(requestXml.Obj.RootNode, [cStrSoapEnv, cStrSoapBody]); GMQueryInterface(requestBodyNode.Obj.SubNodes.First, IGMXmlNode, requestMethodNode); if requestMethodNode = nil then raise EGMSoapException.ObjError(RStrNoSOAPMethodNode, Self, {$I %CurrentRoutine%}); Result.SOAPMethodName := requestMethodNode.Obj.Name; Result.LogMessage := TracePrefix + ' "' + requestMethodNode.Obj.Name + '"'; if Length(requestMethodNode.Obj.Name) <= 0 then raise EGMSoapException.ObjError(RStrEmptySoapMethodName, Self, {$I %CurrentRoutine%}); soapMethod := FindSoapMethod(requestMethodNode.Obj.Name); if not Assigned(soapMethod) then raise EGMSoapException.ObjError(GMFormat(RStrSoapMethodNotImpl, [requestMethodNode.Obj.Name]), Self, {$I %CurrentRoutine%}); soapMethod(requestMethodNode, responseBodyNode); Result.HttpStatusCode := HTTP_STATUS_OK; except // if DoTracing then GMTraceException(GMExceptObject); responseXml := XmlTreeCreateClass.CreateWrite; // responseXml.Obj.RootNode.Obj.SubNodes.Clear; errMsg := InsertFaultNode(InsertSoapBody(InsertSoapEnvelope(responseXml.Obj.RootNode, False)), ExceptObject); if Length(Result.LogMessage) <= 0 then Result.LogMessage := errMsg; Result.HttpStatusCode := HTTP_STATUS_SERVER_ERROR; end; responseXml.Obj.SaveToStream(FTransportLayer); end; //procedure TGMSoapServerPort.SendResponseContent(const ATransportLayer: ISequentialStream); //begin // //end; end.