• Delphi 实现Ping命令


    Delphi  实现Ping命令

    unit FtPing;
    
    interface
    
    
    uses
        Windows, SysUtils, Classes, Controls, Winsock, StdCtrls;
    
    resourcestring
        SICMPRunError = 'ICMP Run Error';
        SInitFailed = 'Init Failed. Maybe Winsock Verison Error';
        SNoResponse = '[%0:S] No Response';
        SInvalidAddr = 'IP Address Error';
        SPingResultString = '[%0:S]: Bytes:%1:D Time: %2:Dms  TTL:%3:D';
    
    type
    
        PCnIPOptionInformation = ^TCnIPOptionInformation;
        TCnIPOptionInformation = packed record
            TTL: Byte; // Time To Live (used for traceroute)
            TOS: Byte; // Type Of Service (usually 0)
            Flags: Byte; // IP header flags (usually 0)
            OptionsSize: Byte; // Size of options data (usually 0, max 40)
            OptionsData: PAnsiChar; // Options data buffer
        end;
    
        PCnIcmpEchoReply = ^TCnIcmpEchoReply;
        TCnIcmpEchoReply = packed record
            Address: DWORD; // replying address
            Status: DWORD; // IP status value (see below)
            RTT: DWORD; // Round Trip Time in milliseconds
            DataSize: Word; // reply data size
            Reserved: Word;
            Data: Pointer; // pointer to reply data buffer
            Options: TCnIPOptionInformation; // reply options
        end;
    
        TIpInfo = record
            Address: Int64;
            IP: string;
            Host: string;
        end;
    
        TOnReceive = procedure( Sender: TComponent; IPAddr, HostName: string; TTL, TOS: Byte ) of object;
    
        TOnError = procedure( Sender: TComponent; IPAddr, HostName: string; TTL, TOS: Byte; ErrorMsg: string ) of object;
    
        //==============================================================================
        // Ping 通讯类
        //==============================================================================
    
          { TFtPing }
    
        TFtPing = class( TComponent )
            {* 通过调用ICMP.DLL库中的函数来实现Ping功能。}
        private
            hICMP: THANDLE;
            FRemoteHost: string;
            FRemoteIP: string;
            FIPAddress: Int64;
            FTTL: Byte;
            FTimeOut: DWord;
            FPingCount: Integer;
            FDelay: Integer;
            FOnError: TOnError;
            FOnReceived: TOnReceive;
            FDataString: string;
            FWSAData: TWSAData;
            FIP: TIpInfo;
    
            procedure SetPingCount( const Value: Integer );
            procedure SetRemoteHost( const Value: string );
            procedure SetTimeOut( const Value: DWord );
            procedure SetTTL( const Value: Byte );
            procedure SetDataString( const Value: string );
            procedure SetRemoteIP( const Value: string );
            function PingIP_Host( const aIP: TIpInfo; const Data; Count: Cardinal; var aReply: string ): Integer;
            {* 以设定的数据Data(无类型缓冲区)Ping一次并返回结果。Count表示数据长度 }
            function GetReplyString( aResult: Integer; aIP: TIpInfo; pIPE: PCnIcmpEchoReply ): string;
            {* 返回结果字符串。}
            function GetDataString: string;
            function GetIPByName( const aName: string; var aIP: string ): Boolean;
            {* 通过机器名称获取IP地址}
            function SetIP( aIPAddr, aHost: string; var aIP: TIpInfo ): Boolean;
            {* 通过机器名称或IP地址填充完整IP信息}
        protected
    
        public
            constructor Create( AOwner: TComponent ); override;
            destructor Destroy; override;
    
            function IsOnline: Boolean;
    
            function Ping( var aReply: string ): Boolean;
            {* 进行循环Ping,循环次数在PingCount属性中指定。}
            function PingOnce( var aReply: string ): Boolean; overload;
            {* 以设定的数据Ping一次并返回结果。}
            function PingOnce( const aIP: string; var aReply: string ): Boolean; overload;
            {* 向指定IP进行一次Ping并返回结果。}
            function PingFromBuffer( var Buffer; Count: Longint; var aReply: string ): Boolean;
            {* 以参数Buffer的数据Ping一次并读取返回结果。}
        published
            property RemoteIP: string read FRemoteIP write SetRemoteIP;
            {* 要Ping的目标主机地址,只支持ip}
            property RemoteHost: string read FRemoteHost write SetRemoteHost;
            {* 要ping的目标主机名,有主机名存在时会覆盖 RemoteIP 的内容}
            property PingCount: Integer read FPingCount write SetPingCount default 4;
            {* 调用Ping方法时进行多少次数据发送,默认是4次。}
            property Delay: Integer read FDelay write FDelay default 0;
            {* 相邻两次 Ping 间的时间间隔,单位毫秒,默认 0 也就是不延时}
            property TTL: Byte read FTTL write SetTTL;
            {* 设置的TTL值,Time to Live}
            property TimeOut: DWord read FTimeOut write SetTimeOut;
            {* 设置的超时值}
            property DataString: string read GetDataString write SetDataString;
            {* 欲发送的数据,以字符串形式表示,默认为"CnPack Ping"。}
            property OnReceived: TOnReceive read FOnReceived write FOnReceived;
            {* Ping一次成功时返回数据所触发的事件}
            property OnError: TOnError read FOnError write FOnError;
            {* Ping出错时返回的内容和信息。包括目的未知、不可达、超时等。}
        end;
    
    implementation
    
    {$R-}
    
    const
        SCnPingData = 'FtPack Ping.';
        ICMPDLL = 'icmp.dll';
    
    type
    
        //==============================================================================
        // 辅助过程  从icmp.dll导入的函数
        //==============================================================================
    
        TIcmpCreateFile = function( ): THandle; stdcall;
    
        TIcmpCloseHandle = function( IcmpHandle: THandle ): Boolean; stdcall;
    
        TIcmpSendEcho = function( IcmpHandle: THandle;
            DestAddress: DWORD;
            RequestData: Pointer;
            RequestSize: Word;
            RequestOptions: PCnIPOptionInformation;
            ReplyBuffer: Pointer;
            ReplySize: DWord;
            TimeOut: DWord ): DWord; stdcall;
    
    var
        IcmpCreateFile: TIcmpCreateFile = nil;
        IcmpCloseHandle: TIcmpCloseHandle = nil;
        IcmpSendEcho: TIcmpSendEcho = nil;
    
        IcmpDllHandle: THandle = 0;
    
    procedure InitIcmpFunctions;
    begin
        IcmpDllHandle := LoadLibrary( ICMPDLL );
        if IcmpDllHandle <> 0 then
            begin
                @IcmpCreateFile := GetProcAddress( IcmpDllHandle, 'IcmpCreateFile' );
                @IcmpCloseHandle := GetProcAddress( IcmpDllHandle, 'IcmpCloseHandle' );
                @IcmpSendEcho := GetProcAddress( IcmpDllHandle, 'IcmpSendEcho' );
            end;
    end;
    
    procedure FreeIcmpFunctions;
    begin
        if IcmpDllHandle <> 0 then
            FreeLibrary( IcmpDllHandle );
    end;
    
    //==============================================================================
    // Ping 通讯类
    //==============================================================================
    
    { TFtPing }
    
    constructor TFtPing.Create( AOwner: TComponent );
    begin
        inherited Create( AOwner );
        FRemoteIP := '127.0.0.1';
        FTTL := 64;
        FPingCount := 4;
        FDelay := 0;
        FTimeOut := 10;
        FDataString := SCnPingData;
    
        hICMP := IcmpCreateFile( ); // 取得DLL句柄
        if hICMP = INVALID_HANDLE_VALUE then
            begin
                raise Exception.Create( SICMPRunError );
            end;
    end;
    
    
    
    destructor TFtPing.Destroy;
    begin
        if hICMP <> INVALID_HANDLE_VALUE then
            begin
                IcmpCloseHandle( hICMP );
            end;
        inherited Destroy;
    end;
    
    
    
    procedure TFtPing.SetPingCount( const Value: Integer );
    begin
        if Value > 0 then
            FPingCount := Value;
    end;
    
    
    
    procedure TFtPing.SetRemoteIP( const Value: string );
    begin
        if FRemoteIP <> Value then
            begin
                FRemoteIP := Value;
                if SetIP( FRemoteIP, '', FIP ) then
                    begin
                        FRemoteHost := FIP.Host;
                        FIPAddress := FIP.Address;
                    end;
            end;
    end;
    
    
    
    procedure TFtPing.SetRemoteHost( const Value: string );
    begin
        if FRemoteHost <> Value then
            begin
                // RemoteHost 更改的话,RemoteIP 自动清空
                FRemoteHost := Value;
                if SetIP( '', FRemoteHost, FIP ) then
                    begin
                        FRemoteIP := FIP.IP;
                        FIPAddress := FIP.Address;
                    end;
            end;
    end;
    
    
    procedure TFtPing.SetTimeOut( const Value: DWord );
    begin
        FTimeOut := Value;
    end;
    
    
    
    procedure TFtPing.SetTTL( const Value: Byte );
    begin
        FTTL := Value;
    end;
    
    
    
    procedure TFtPing.SetDataString( const Value: string );
    begin
        FDataString := Value;
    end;
    
    
    
    function TFtPing.GetDataString: string;
    begin
        if FDataString = '' then
            FDataString := SCnPingData;
        Result := FDataString;
    end;
    
    
    
    function TFtPing.IsOnline: Boolean;
    var
        sReply: string;
    begin
        SetIP( RemoteIP, RemoteHost, FIP );
        Result := PingIP_Host( FIP, pointer( FDataString )^, Length( DataString ), sReply ) >= 0;
    end;
    
    
    function TFtPing.Ping( var aReply: string ): Boolean;
    var
        iCount, iResult: Integer;
        sReply: string;
    begin
        aReply := '';
        iResult := 0;
        try
            SetIP( RemoteIP, RemoteHost, FIP );
            for iCount := 1 to PingCount do
                begin
                    iResult := PingIP_Host( FIP, Pointer( FDataString )^, Length( DataString ) * SizeOf( Char ), sReply );
                    aReply := aReply + #13#10 + sReply;
                    if iResult < 0 then
                        Break;
    
                    if FDelay > 0 then
                        Sleep( FDelay );
                end;
        finally
            Result := iResult >= 0;
        end;
    end;
    
    
    
    function TFtPing.PingOnce( var aReply: string ): Boolean;
    begin
        SetIP( RemoteIP, RemoteHost, FIP );
        Result := PingIP_Host( FIP, pointer( FDataString )^, Length( DataString ), aReply ) >= 0;
    end;
    
    
    function TFtPing.PingOnce( const aIP: string; var aReply: string ): Boolean;
    begin
        SetIP( aIP, aIP, FIP );
        Result := PingIP_Host( FIP, pointer( FDataString )^, Length( DataString ), aReply ) >= 0;
    end;
    
    
    
    function TFtPing.PingFromBuffer( var Buffer; Count: Integer; var aReply: string ): Boolean;
    begin
        SetIP( RemoteIP, RemoteHost, FIP );
        Result := PingIP_Host( FIP, Buffer, Count, aReply ) >= 0;
    end;
    
    
    
    function TFtPing.PingIP_Host( const aIP: TIpInfo; const Data; Count: Cardinal; var aReply: string ): Integer;
    var
        IPOpt: TCnIPOptionInformation; // 发送数据结构
        pReqData, pRevData: PAnsiChar;
        pCIER: PCnIcmpEchoReply;
    begin
        Result := -100;
        pReqData := nil;
    
        if Count <= 0 then
            begin
                aReply := GetReplyString( Result, aIP, nil );
                Exit;
            end;
        if aIP.Address = INADDR_NONE then
            begin
                Result := -1;
                aReply := GetReplyString( Result, aIP, nil );
                Exit;
            end;
    
        GetMem( pCIER, SizeOf( TCnICMPEchoReply ) + Count );
        GetMem( pRevData, Count );
        try
            FillChar( pCIER^, SizeOf( TCnICMPEchoReply ) + Count, 0 ); // 初始化接收数据结构
            pCIER^.Data := pRevData;
            GetMem( pReqData, Count );
            Move( Data, pReqData^, Count ); // 准备发送的数据
            FillChar( IPOpt, Sizeof( IPOpt ), 0 ); // 初始化发送数据结构
            IPOpt.TTL := FTTL;
    
            try //Ping开始
                if WSAStartup( MAKEWORD( 2, 0 ), FWSAData ) <> 0 then
                    raise Exception.Create( SInitFailed );
                if IcmpSendEcho( hICMP, //dll handle
                    aIP.Address, //target
                    pReqData, //data
                    Count, //data length
                    @IPOpt, //addree of ping option
                    pCIER,
                    SizeOf( TCnICMPEchoReply ) + Count, //pack size
                    FTimeOut //timeout value
                    ) <> 0 then
                    begin
                        Result := 0; // Ping正常返回
                        if Assigned( FOnReceived ) then
                            FOnReceived( Self, aIP.IP, aIP.Host, IPOpt.TTL, IPOpt.TOS );
                    end
                else
                    begin
                        Result := -2; // 没有响应
                        if Assigned( FOnError ) then
                            FOnError( Self, aIP.IP, aIP.Host, IPOpt.TTL, IPOpt.TOS, SNoResponse );
                    end;
            except
                on E: Exception do
                    begin
                        Result := -3; // 发生错误
                        if Assigned( FOnError ) then
                            FOnError( Self, aIP.IP, aIP.Host, IPOpt.TTL, IPOpt.TOS, E.Message );
                    end;
            end;
        finally
            WSACleanUP;
            aReply := GetReplyString( Result, aIP, pCIER );
            if pRevData <> nil then
                begin
                    FreeMem( pRevData ); // 释放内存
                    pCIER.Data := nil;
                end;
            if pReqData <> nil then
                FreeMem( pReqData ); //释放内存
            FreeMem( pCIER ); //释放内存
        end;
    end;
    
    
    
    function TFtPing.GetReplyString( aResult: Integer; aIP: TIpInfo;
        pIPE: PCnIcmpEchoReply ): string;
    var
        sHost: string;
    begin
        Result := SInvalidAddr;
        case aResult of
            -100: Result := SICMPRunError;
            -1: Result := SInvalidAddr;
            -2: Result := Format( SNoResponse, [ RemoteHost ] );
            else
                if pIPE <> nil then
                    begin
                        sHost := aIP.IP;
                        if aIP.Host <> '' then
                            sHost := aIP.Host + ': ' + sHost;
                        Result := ( Format( SPingResultString, [ sHost, pIPE^.DataSize, pIPE^.RTT,
                            pIPE^.Options.TTL ] ) );
                    end;
        end;
    end;
    
    
    function TFtPing.GetIPByName( const aName: string;
        var aIP: string ): Boolean;
    var
        pHost: PHostEnt;
        FWSAData: TWSAData;
        sName: array[ 0..255 ] of AnsiChar;
    begin
        Result := False;
        //    StrPCopy(sName, {$IFDEF DELPHI12_UP}AnsiString{$ENDIF}(aName));
        StrPCopy( sName, AnsiString( aName ) );
        aIP := '';
        if aName = '' then
            Exit;
    
        WSAStartup( $101, FWSAData );
        try
            pHost := GetHostByName( @sName );
            Result := pHost <> nil;
            if Result then
                //            aIP := {$IFDEF DELPHI12_UP}string{$ENDIF}(inet_ntoa(PInAddr(pHost^.h_addr_list^)^));
                aIP := string( inet_ntoa( PInAddr( pHost^.h_addr_list^ )^ ) );
        finally
            WSACleanup;
        end;
    end;
    
    
    
    function TFtPing.SetIP( aIPAddr, aHost: string; var aIP: TIpInfo ): Boolean;
    var
        pIPAddr: PAnsiChar;
    begin
        Result := False;
        aIP.Address := INADDR_NONE;
        aIP.IP := aIPAddr;
        aIP.Host := aHost;
        if aIP.IP = '' then
            begin
                if ( aIP.Host = '' ) or ( not GetIPByName( aIP.Host, aIP.IP ) ) then
                    Exit;
            end;
    
        GetMem( pIPAddr, Length( aIP.IP ) + 1 );
        try
            ZeroMemory( pIPAddr, Length( aIP.IP ) + 1 );
            //        StrPCopy(pIPAddr, {$IFDEF DELPHI12_UP}AnsiString{$ENDIF}(aIP.IP));
            StrPCopy( pIPAddr, AnsiString( aIP.IP ) );
            aIP.Address := inet_addr( PAnsiChar( pIPAddr ) ); // IP转换成无点整型
        finally
            FreeMem( pIPAddr ); // 释放申请的动态内存
        end;
        Result := aIP.Address <> INADDR_NONE;
    end;
    
    
    
    initialization
        InitIcmpFunctions;
    
    
    
    finalization
        FreeIcmpFunctions;
    
    end.
    
    
    {
      调用方法
    procedure TForm1.Button1Click( Sender: TObject );
    var
        FtPing: TFtPing;
        aReply: string;
    begin
        FtPing := TFtPing.Create( nil );
        try
            FtPing.RemoteIP := Edit1.Text;
            if FtPing.Ping( aReply ) then
                begin
                    Memo1.Lines.Add( '网络畅通!' )
                end
            else
                begin
                    Memo1.Lines.Add( '网络异常~~>|<~~' )
                end;
        finally
            FtPing.Free;
        end;
    end;
    
    }
    View Code

    正则取匹配IP地址

     Reg:=TPerlRegEx.Create;
          Reg.Subject:=pos.ServerUrl;
          Reg.RegEx:='((2[0-4]d|25[0-5]|[01]?dd?).){3}(2[0-4]d|25[0-5]|[01]?dd?)';
    
          if  reg.Match then
     IP:=Reg.MatchedText
    else  
     //TODO  没有获取到IP地址 

      

  • 相关阅读:
    041_form表单重置数据reset()
    040_下拉列表的显示与提交数值时,需要用到转义字符
    039_如何选取checkbox的id值?
    011_表单数据非空验证
    010_@ResposBody错误
    010_页面单击按钮失灵
    使用Maven创建 web项目
    java设计模式(八) 适配器模式
    设计模式 6大设计原则
    Java设计模式(七) 模板模式-使用钩子
  • 原文地址:https://www.cnblogs.com/stroll/p/11583338.html
Copyright © 2020-2023  润新知