* *
* partial sync with the 3.4 fork codebase. These are the things
synces for the most part:
- copyright headers
- grammar/typos in comments and some readmes
- comment/whitespace/decorations
- variable scoping in C files
- DO CASE/SWITCH and some other alternate syntax usage
- minimal amount of human readable text in strings
- minor code updates
- HB_TRACE() void * casts for pointers and few other changes to
avoid C compiler warnings
- various other, minor code cleanups
- only Harbour/C code/headers were touched in src, utils, contrib,
include. No 3rd party code, no make files, and with just a few
exceptions, no 'tests' code was touched.
- certain components were not touched were 3.4 diverged too much
already, like f.e. hbmk2, hbssl, hbcurl, hbexpat
- the goal was that no actual program logic should be altered by
these changes. Except some possible minor exceptions, any such
change is probably a bug in this patch.
It's a massive patch, if you find anything broken after it, please
open an Issue with the details. Build test was done on macOS.
The goal is make it easier to see what actual code/logic was changed
in 3.4 compared to 3.2 and to make patches easier to apply in both
ways.
149 lines
4.2 KiB
Plaintext
149 lines
4.2 KiB
Plaintext
/* Copyright 2010 Mindaugas Kavaliauskas <dbtopas / at / dbtopas.lt> */
|
|
|
|
/*
|
|
* This module demonstrates a simple UDP Discovery Server
|
|
*
|
|
* If you run some service on the network (ex., hbnetio) you need to
|
|
* know server IP address and configure client to connect to this
|
|
* address. UDPDS helps client to find server address (or addresses
|
|
* of multiple servers) on local network. UDPDS should be run in
|
|
* parallel to real server (ex., hbnetio). Server part of UDPDS uses
|
|
* threads, so, it should be compiled in MT mode.
|
|
*
|
|
* Server functions:
|
|
* hb_udpds_Start( nPort, cName [, cVersion ] ) --> hServer
|
|
* hb_udpds_Stop( hServer )
|
|
*
|
|
* Client function:
|
|
* hb_udpds_Find( nPort, cName ) --> { { "ip_addr_1", "version_1" }, ... }
|
|
*
|
|
*/
|
|
|
|
#include "hbsocket.ch"
|
|
|
|
/* Client */
|
|
|
|
FUNCTION hb_udpds_Find( nPort, cName )
|
|
|
|
LOCAL hSocket, aRet, nEnd, nTime, cBuffer, nLen, aAddr
|
|
|
|
IF ! Empty( hSocket := hb_socketOpen( , HB_SOCKET_PT_DGRAM ) )
|
|
hb_socketSetBroadcast( hSocket, .T. )
|
|
cName := hb_StrToUTF8( cName )
|
|
IF s_sendBroadcastMessages( hSocket, nPort, hb_BChar( 5 ) + cName + hb_BChar( 0 ) )
|
|
nTime := hb_MilliSeconds()
|
|
nEnd := nTime + 100 /* 100ms delay is enough on LAN */
|
|
aRet := {}
|
|
DO WHILE nEnd > nTime
|
|
cBuffer := Space( 2000 )
|
|
nLen := hb_socketRecvFrom( hSocket, @cBuffer, , , @aAddr, nEnd - nTime )
|
|
IF hb_BLeft( cBuffer, hb_BLen( cName ) + 2 ) == hb_BChar( 6 ) + cName + hb_BChar( 0 ) .AND. ;
|
|
AScan( aRet, {| x | x[ 1 ] == aAddr[ 2 ] } ) == 0
|
|
AAdd( aRet, { aAddr[ 2 ], hb_BSubStr( cBuffer, hb_BLen( cName ) + 3, nLen - hb_BLen( cName ) - 2 ) } )
|
|
ENDIF
|
|
nTime := hb_MilliSeconds()
|
|
ENDDO
|
|
ENDIF
|
|
hb_socketClose( hSocket )
|
|
ENDIF
|
|
|
|
RETURN aRet
|
|
|
|
STATIC FUNCTION s_sendBroadcastMessages( hSocket, nPort, cMessage )
|
|
|
|
LOCAL lResult, cAddr
|
|
|
|
lResult := .F.
|
|
FOR EACH cAddr IN s_getBroadcastAddresses()
|
|
IF hb_socketSendTo( hSocket, cMessage, , , ;
|
|
{ HB_SOCKET_AF_INET, cAddr, nPort } ) == hb_BLen( cMessage )
|
|
lResult := .T.
|
|
ENDIF
|
|
NEXT
|
|
|
|
RETURN lResult
|
|
|
|
STATIC FUNCTION s_getBroadcastAddresses()
|
|
|
|
LOCAL aIF, cAddr
|
|
|
|
LOCAL lLo := .F.
|
|
LOCAL aAddrs := {}
|
|
|
|
FOR EACH aIF IN hb_socketGetIFaces()
|
|
IF Empty( cAddr := aIF[ HB_SOCKET_IFINFO_BROADCAST ] )
|
|
IF ! lLo .AND. aIF[ HB_SOCKET_IFINFO_ADDR ] == "127.0.0.1"
|
|
lLo := .T.
|
|
ENDIF
|
|
ELSEIF hb_AScan( aAddrs, cAddr,,, .T. ) == 0
|
|
AAdd( aAddrs, cAddr )
|
|
ENDIF
|
|
NEXT
|
|
IF Empty( aAddrs )
|
|
AAdd( aAddrs, "255.255.255.255" )
|
|
ENDIF
|
|
IF lLo
|
|
hb_AIns( aAddrs, 1, "127.0.0.1", .T. )
|
|
ENDIF
|
|
|
|
RETURN aAddrs
|
|
|
|
/* Server */
|
|
|
|
FUNCTION hb_udpds_Start( nPort, cName, cVersion )
|
|
|
|
LOCAL hSocket
|
|
|
|
IF ! Empty( hSocket := hb_socketOpen( , HB_SOCKET_PT_DGRAM ) )
|
|
IF hb_socketBind( hSocket, { HB_SOCKET_AF_INET, "0.0.0.0", nPort } )
|
|
hb_threadDetach( hb_threadStart( @UDPDS(), hSocket, cName, cVersion ) )
|
|
RETURN hSocket
|
|
ENDIF
|
|
hb_socketClose( hSocket )
|
|
ENDIF
|
|
|
|
RETURN NIL
|
|
|
|
PROCEDURE hb_udpds_Stop( hSocket )
|
|
|
|
hb_socketClose( hSocket )
|
|
|
|
RETURN
|
|
|
|
STATIC PROCEDURE UDPDS( hSocket, cName, cVersion )
|
|
|
|
LOCAL cBuffer, nLen, aAddr
|
|
|
|
cName := hb_StrToUTF8( cName )
|
|
cVersion := iif( HB_ISSTRING( cVersion ), hb_StrToUTF8( cVersion ), "" )
|
|
|
|
DO WHILE .T.
|
|
cBuffer := Space( 2000 )
|
|
BEGIN SEQUENCE WITH {| oErr | Break( oErr ) }
|
|
nLen := hb_socketRecvFrom( hSocket, @cBuffer, , , @aAddr, 1000 )
|
|
RECOVER
|
|
nLen := NIL
|
|
END SEQUENCE
|
|
IF nLen == NIL
|
|
EXIT
|
|
ENDIF
|
|
IF nLen == -1
|
|
IF hb_socketGetError() != HB_SOCKET_ERR_TIMEOUT
|
|
RETURN
|
|
ENDIF
|
|
ELSE
|
|
/*
|
|
* Communication protocol:
|
|
* Broadcast request: ENQ, ServerName, NUL
|
|
* Server response: ACK, ServerName, NUL, Version
|
|
*/
|
|
IF hb_BLeft( cBuffer, nLen ) == hb_BChar( 5 ) + cName + hb_BChar( 0 )
|
|
BEGIN SEQUENCE WITH __BreakBlock()
|
|
hb_socketSendTo( hSocket, hb_BChar( 6 ) + cName + hb_BChar( 0 ) + cVersion, , , aAddr )
|
|
END SEQUENCE
|
|
ENDIF
|
|
ENDIF
|
|
ENDDO
|
|
|
|
RETURN
|