(* * Licensed to the Apache Software Foundation (ASF) under one * or more contributor license agreements. See the NOTICE file * distributed with this work for additional information * regarding copyright ownership. The ASF licenses this file * to you under the Apache License, Version 2.0 (the * "License"); you may not use this file except in compliance * with the License. You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, * software distributed under the License is distributed on an * "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY * KIND, either express or implied. See the License for the * specific language governing permissions and limitations * under the License. *) unit Thrift.Server; {$I Thrift.Defines.inc} {$I-} // prevent annoying errors with default log delegate and no console interface uses {$IFDEF OLD_UNIT_NAMES} Windows, SysUtils, {$ELSE} Winapi.Windows, System.SysUtils, {$ENDIF} Thrift, Thrift.Protocol, Thrift.Transport; type IServerEvents = interface ['{9E2A99C5-EE85-40B2-9A52-2D1722B18176}'] // Called before the server begins. procedure PreServe; // Called when the server transport is ready to accept requests procedure PreAccept; // Called when a new client has connected and the server is about to being processing. function CreateProcessingContext( const input, output : IProtocol) : IProcessorEvents; end; IServer = interface ['{ADC46F2D-8199-4D1C-96D2-87FD54351723}'] procedure Serve; procedure Stop; function GetServerEvents : IServerEvents; procedure SetServerEvents( const value : IServerEvents); property ServerEvents : IServerEvents read GetServerEvents write SetServerEvents; end; TServerImpl = class abstract( TInterfacedObject, IServer ) public type TLogDelegate = reference to procedure( const str: string); protected FProcessor : IProcessor; FServerTransport : IServerTransport; FInputTransportFactory : ITransportFactory; FOutputTransportFactory : ITransportFactory; FInputProtocolFactory : IProtocolFactory; FOutputProtocolFactory : IProtocolFactory; FLogDelegate : TLogDelegate; FServerEvents : IServerEvents; class procedure DefaultLogDelegate( const str: string); function GetServerEvents : IServerEvents; procedure SetServerEvents( const value : IServerEvents); procedure Serve; virtual; abstract; procedure Stop; virtual; abstract; public constructor Create( const AProcessor :IProcessor; const AServerTransport: IServerTransport; const AInputTransportFactory : ITransportFactory; const AOutputTransportFactory : ITransportFactory; const AInputProtocolFactory : IProtocolFactory; const AOutputProtocolFactory : IProtocolFactory; const ALogDelegate : TLogDelegate ); overload; constructor Create( const AProcessor :IProcessor; const AServerTransport: IServerTransport ); overload; constructor Create( const AProcessor :IProcessor; const AServerTransport: IServerTransport; const ALogDelegate: TLogDelegate ); overload; constructor Create( const AProcessor :IProcessor; const AServerTransport: IServerTransport; const ATransportFactory : ITransportFactory ); overload; constructor Create( const AProcessor :IProcessor; const AServerTransport: IServerTransport; const ATransportFactory : ITransportFactory; const AProtocolFactory : IProtocolFactory ); overload; end; TSimpleServer = class( TServerImpl) private FStop : Boolean; public constructor Create( const AProcessor: IProcessor; const AServerTransport: IServerTransport); overload; constructor Create( const AProcessor: IProcessor; const AServerTransport: IServerTransport; ALogDel: TServerImpl.TLogDelegate); overload; constructor Create( const AProcessor: IProcessor; const AServerTransport: IServerTransport; const ATransportFactory: ITransportFactory); overload; constructor Create( const AProcessor: IProcessor; const AServerTransport: IServerTransport; const ATransportFactory: ITransportFactory; const AProtocolFactory: IProtocolFactory); overload; procedure Serve; override; procedure Stop; override; end; implementation { TServerImpl } constructor TServerImpl.Create( const AProcessor: IProcessor; const AServerTransport: IServerTransport; const ALogDelegate: TLogDelegate); var InputFactory, OutputFactory : IProtocolFactory; InputTransFactory, OutputTransFactory : ITransportFactory; begin InputFactory := TBinaryProtocolImpl.TFactory.Create; OutputFactory := TBinaryProtocolImpl.TFactory.Create; InputTransFactory := TTransportFactoryImpl.Create; OutputTransFactory := TTransportFactoryImpl.Create; //no inherited; Create( AProcessor, AServerTransport, InputTransFactory, OutputTransFactory, InputFactory, OutputFactory, ALogDelegate ); end; constructor TServerImpl.Create(const AProcessor: IProcessor; const AServerTransport: IServerTransport); var InputFactory, OutputFactory : IProtocolFactory; InputTransFactory, OutputTransFactory : ITransportFactory; begin InputFactory := TBinaryProtocolImpl.TFactory.Create; OutputFactory := TBinaryProtocolImpl.TFactory.Create; InputTransFactory := TTransportFactoryImpl.Create; OutputTransFactory := TTransportFactoryImpl.Create; //no inherited; Create( AProcessor, AServerTransport, InputTransFactory, OutputTransFactory, InputFactory, OutputFactory, DefaultLogDelegate ); end; constructor TServerImpl.Create(const AProcessor: IProcessor; const AServerTransport: IServerTransport; const ATransportFactory: ITransportFactory); var InputProtocolFactory : IProtocolFactory; OutputProtocolFactory : IProtocolFactory; begin InputProtocolFactory := TBinaryProtocolImpl.TFactory.Create; OutputProtocolFactory := TBinaryProtocolImpl.TFactory.Create; //no inherited; Create( AProcessor, AServerTransport, ATransportFactory, ATransportFactory, InputProtocolFactory, OutputProtocolFactory, DefaultLogDelegate); end; constructor TServerImpl.Create(const AProcessor: IProcessor; const AServerTransport: IServerTransport; const AInputTransportFactory, AOutputTransportFactory: ITransportFactory; const AInputProtocolFactory, AOutputProtocolFactory: IProtocolFactory; const ALogDelegate : TLogDelegate); begin inherited Create; FProcessor := AProcessor; FServerTransport := AServerTransport; FInputTransportFactory := AInputTransportFactory; FOutputTransportFactory := AOutputTransportFactory; FInputProtocolFactory := AInputProtocolFactory; FOutputProtocolFactory := AOutputProtocolFactory; FLogDelegate := ALogDelegate; end; class procedure TServerImpl.DefaultLogDelegate( const str: string); begin try Writeln( str); if IoResult <> 0 then OutputDebugString(PChar(str)); except OutputDebugString(PChar(str)); end; end; constructor TServerImpl.Create( const AProcessor: IProcessor; const AServerTransport: IServerTransport; const ATransportFactory: ITransportFactory; const AProtocolFactory: IProtocolFactory); begin //no inherited; Create( AProcessor, AServerTransport, ATransportFactory, ATransportFactory, AProtocolFactory, AProtocolFactory, DefaultLogDelegate); end; function TServerImpl.GetServerEvents : IServerEvents; begin result := FServerEvents; end; procedure TServerImpl.SetServerEvents( const value : IServerEvents); begin // if you need more than one, provide a specialized IServerEvents implementation FServerEvents := value; end; { TSimpleServer } constructor TSimpleServer.Create( const AProcessor: IProcessor; const AServerTransport: IServerTransport); var InputProtocolFactory : IProtocolFactory; OutputProtocolFactory : IProtocolFactory; InputTransportFactory : ITransportFactory; OutputTransportFactory : ITransportFactory; begin InputProtocolFactory := TBinaryProtocolImpl.TFactory.Create; OutputProtocolFactory := TBinaryProtocolImpl.TFactory.Create; InputTransportFactory := TTransportFactoryImpl.Create; OutputTransportFactory := TTransportFactoryImpl.Create; inherited Create( AProcessor, AServerTransport, InputTransportFactory, OutputTransportFactory, InputProtocolFactory, OutputProtocolFactory, DefaultLogDelegate); end; constructor TSimpleServer.Create( const AProcessor: IProcessor; const AServerTransport: IServerTransport; ALogDel: TServerImpl.TLogDelegate); var InputProtocolFactory : IProtocolFactory; OutputProtocolFactory : IProtocolFactory; InputTransportFactory : ITransportFactory; OutputTransportFactory : ITransportFactory; begin InputProtocolFactory := TBinaryProtocolImpl.TFactory.Create; OutputProtocolFactory := TBinaryProtocolImpl.TFactory.Create; InputTransportFactory := TTransportFactoryImpl.Create; OutputTransportFactory := TTransportFactoryImpl.Create; inherited Create( AProcessor, AServerTransport, InputTransportFactory, OutputTransportFactory, InputProtocolFactory, OutputProtocolFactory, ALogDel); end; constructor TSimpleServer.Create( const AProcessor: IProcessor; const AServerTransport: IServerTransport; const ATransportFactory: ITransportFactory); begin inherited Create( AProcessor, AServerTransport, ATransportFactory, ATransportFactory, TBinaryProtocolImpl.TFactory.Create, TBinaryProtocolImpl.TFactory.Create, DefaultLogDelegate); end; constructor TSimpleServer.Create( const AProcessor: IProcessor; const AServerTransport: IServerTransport; const ATransportFactory: ITransportFactory; const AProtocolFactory: IProtocolFactory); begin inherited Create( AProcessor, AServerTransport, ATransportFactory, ATransportFactory, AProtocolFactory, AProtocolFactory, DefaultLogDelegate); end; procedure TSimpleServer.Serve; var client : ITransport; InputTransport : ITransport; OutputTransport : ITransport; InputProtocol : IProtocol; OutputProtocol : IProtocol; context : IProcessorEvents; begin try FServerTransport.Listen; except on E: Exception do begin FLogDelegate( E.ToString); end; end; if FServerEvents <> nil then FServerEvents.PreServe; client := nil; while (not FStop) do begin try // clean up any old instances before waiting for clients InputTransport := nil; OutputTransport := nil; InputProtocol := nil; OutputProtocol := nil; // close any old connections before before waiting for new clients if client <> nil then try try client.Close; finally client := nil; end; except // catch all, we can't do much about it at this point end; client := FServerTransport.Accept( procedure begin if FServerEvents <> nil then FServerEvents.PreAccept; end); if client = nil then begin if FStop then Abort // silent exception else raise TTransportExceptionUnknown.Create('ServerTransport.Accept() may not return NULL'); end; FLogDelegate( 'Client Connected!'); InputTransport := FInputTransportFactory.GetTransport( client ); OutputTransport := FOutputTransportFactory.GetTransport( client ); InputProtocol := FInputProtocolFactory.GetProtocol( InputTransport ); OutputProtocol := FOutputProtocolFactory.GetProtocol( OutputTransport ); if FServerEvents <> nil then context := FServerEvents.CreateProcessingContext( InputProtocol, OutputProtocol) else context := nil; while not FStop do begin if context <> nil then context.Processing( client); if not FProcessor.Process( InputProtocol, OutputProtocol, context) then Break; end; except on E: TTransportException do begin if FStop then FLogDelegate('TSimpleServer was shutting down, caught ' + E.ToString) else FLogDelegate( E.ToString); end; on E: Exception do begin FLogDelegate( E.ToString); end; end; if context <> nil then begin context.CleanupContext; context := nil; end; if InputTransport <> nil then begin InputTransport.Close; end; if OutputTransport <> nil then begin OutputTransport.Close; end; end; if FStop then begin try FServerTransport.Close; except on E: TTransportException do begin FLogDelegate('TServerTranport failed on close: ' + E.Message); end; end; FStop := False; end; end; procedure TSimpleServer.Stop; begin FStop := True; FServerTransport.Close; end; end.