2006-08-24 15:43:32 +00:00
/*
Samba Unix / Linux Dynamic DNS Update
net ads commands
Copyright ( C ) Krishna Ganugapati ( krishnag @ centeris . com ) 2006
Copyright ( C ) Gerald Carter 2006
This program is free software ; you can redistribute it and / or modify
it under the terms of the GNU General Public License as published by
2007-07-09 19:25:36 +00:00
the Free Software Foundation ; either version 3 of the License , or
2006-08-24 15:43:32 +00:00
( at your option ) any later version .
2010-07-18 13:39:51 +02:00
2006-08-24 15:43:32 +00:00
This program is distributed in the hope that it will be useful ,
but WITHOUT ANY WARRANTY ; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE . See the
GNU General Public License for more details .
2010-07-18 13:39:51 +02:00
2006-08-24 15:43:32 +00:00
You should have received a copy of the GNU General Public License
2007-07-10 00:52:41 +00:00
along with this program . If not , see < http : //www.gnu.org/licenses/>.
2006-08-24 15:43:32 +00:00
*/
# include "includes.h"
# include "utils/net.h"
2011-03-04 13:16:40 +01:00
# include "../lib/addns/dns.h"
2012-09-19 15:35:15 +02:00
# include "utils/net_dns.h"
2006-08-24 15:43:32 +00:00
# if defined(WITH_DNS_UPDATES)
2007-07-25 18:45:57 +00:00
2006-08-24 15:43:32 +00:00
/*********************************************************************
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2006-12-14 17:00:10 +00:00
DNS_ERROR DoDNSUpdate ( char * pszServerName ,
2006-11-17 21:46:26 +00:00
const char * pszDomainName , const char * pszHostName ,
2012-09-25 11:08:48 +02:00
const struct sockaddr_storage * sslist , size_t num_addrs ,
2016-06-26 16:26:53 -07:00
uint32_t flags , bool remove_host )
2006-08-24 15:43:32 +00:00
{
2006-11-17 21:46:26 +00:00
DNS_ERROR err ;
struct dns_connection * conn ;
TALLOC_CTX * mem_ctx ;
OM_uint32 minor ;
struct dns_update_request * req , * resp ;
2006-08-24 15:43:32 +00:00
2012-09-25 11:09:45 +02:00
DEBUG ( 10 , ( " DoDNSUpdate called with flags: 0x%08x \n " , flags ) ) ;
if ( ! ( flags & DNS_UPDATE_SIGNED ) & &
! ( flags & DNS_UPDATE_UNSIGNED ) & &
! ( flags & DNS_UPDATE_PROBE ) ) {
return ERROR_DNS_INVALID_PARAMETER ;
}
2016-06-26 16:26:53 -07:00
if ( ! remove_host & & ( ( num_addrs < = 0 ) | | ! sslist ) ) {
2006-11-17 21:46:26 +00:00
return ERROR_DNS_INVALID_PARAMETER ;
}
if ( ! ( mem_ctx = talloc_init ( " DoDNSUpdate " ) ) ) {
return ERROR_DNS_NO_MEMORY ;
2006-08-24 15:43:32 +00:00
}
2010-07-18 13:39:51 +02:00
2006-12-31 06:50:44 +00:00
err = dns_open_connection ( pszServerName , DNS_TCP , mem_ctx , & conn ) ;
2006-11-17 21:46:26 +00:00
if ( ! ERR_DNS_IS_OK ( err ) ) {
goto error ;
}
2012-09-25 11:09:45 +02:00
if ( flags & DNS_UPDATE_PROBE ) {
2006-11-17 21:46:26 +00:00
2012-09-25 11:09:45 +02:00
/*
* Probe if everything ' s fine
*/
2006-11-17 21:46:26 +00:00
2012-09-25 11:09:45 +02:00
err = dns_create_probe ( mem_ctx , pszDomainName , pszHostName ,
num_addrs , sslist , & req ) ;
if ( ! ERR_DNS_IS_OK ( err ) ) goto error ;
2006-11-17 21:46:26 +00:00
2012-09-25 11:09:45 +02:00
err = dns_update_transaction ( mem_ctx , conn , req , & resp ) ;
if ( ! ERR_DNS_IS_OK ( err ) ) {
DEBUG ( 3 , ( " DoDNSUpdate: failed to probe DNS \n " ) ) ;
2018-05-17 11:53:18 +02:00
goto error ;
2012-09-25 11:09:45 +02:00
}
if ( ( dns_response_code ( resp - > flags ) = = DNS_NO_ERROR ) & &
( flags & DNS_UPDATE_PROBE_SUFFICIENT ) ) {
TALLOC_FREE ( mem_ctx ) ;
return ERROR_DNS_SUCCESS ;
}
2006-11-17 21:46:26 +00:00
}
2012-09-25 11:09:45 +02:00
if ( flags & DNS_UPDATE_UNSIGNED ) {
/*
* First try without signing
*/
2006-08-24 15:43:32 +00:00
2012-09-25 11:09:45 +02:00
err = dns_create_update_request ( mem_ctx , pszDomainName , pszHostName ,
sslist , num_addrs , & req ) ;
if ( ! ERR_DNS_IS_OK ( err ) ) goto error ;
2006-08-24 15:43:32 +00:00
2012-09-25 11:09:45 +02:00
err = dns_update_transaction ( mem_ctx , conn , req , & resp ) ;
if ( ! ERR_DNS_IS_OK ( err ) ) {
DEBUG ( 3 , ( " DoDNSUpdate: unsigned update failed \n " ) ) ;
2018-05-23 17:35:15 +02:00
goto error ;
2012-09-25 11:09:45 +02:00
}
2006-11-17 21:46:26 +00:00
2012-09-25 11:09:45 +02:00
if ( ( dns_response_code ( resp - > flags ) = = DNS_NO_ERROR ) & &
( flags & DNS_UPDATE_UNSIGNED_SUFFICIENT ) ) {
TALLOC_FREE ( mem_ctx ) ;
return ERROR_DNS_SUCCESS ;
}
2006-11-17 21:46:26 +00:00
}
/*
* Okay , we have to try with signing
*/
2012-09-25 11:09:45 +02:00
if ( flags & DNS_UPDATE_SIGNED ) {
2006-11-17 21:46:26 +00:00
gss_ctx_id_t gss_context ;
char * keyname ;
2012-09-25 11:09:45 +02:00
err = dns_create_update_request ( mem_ctx , pszDomainName , pszHostName ,
sslist , num_addrs , & req ) ;
if ( ! ERR_DNS_IS_OK ( err ) ) goto error ;
2006-11-17 21:46:26 +00:00
if ( ! ( keyname = dns_generate_keyname ( mem_ctx ) ) ) {
err = ERROR_DNS_NO_MEMORY ;
goto error ;
}
err = dns_negotiate_sec_ctx ( pszDomainName , pszServerName ,
2006-12-14 16:27:45 +00:00
keyname , & gss_context , DNS_SRV_ANY ) ;
/* retry using the Windows 2000 DNS hack */
if ( ! ERR_DNS_IS_OK ( err ) ) {
err = dns_negotiate_sec_ctx ( pszDomainName , pszServerName ,
keyname , & gss_context ,
DNS_SRV_WIN2000 ) ;
}
2010-07-18 13:39:51 +02:00
2006-12-14 16:27:45 +00:00
if ( ! ERR_DNS_IS_OK ( err ) )
goto error ;
2006-11-17 21:46:26 +00:00
err = dns_sign_update ( req , gss_context , keyname ,
" gss.microsoft.com " , time ( NULL ) , 3600 ) ;
gss_delete_sec_context ( & minor , & gss_context , GSS_C_NO_BUFFER ) ;
if ( ! ERR_DNS_IS_OK ( err ) ) goto error ;
err = dns_update_transaction ( mem_ctx , conn , req , & resp ) ;
if ( ! ERR_DNS_IS_OK ( err ) ) goto error ;
err = ( dns_response_code ( resp - > flags ) = = DNS_NO_ERROR ) ?
ERROR_DNS_SUCCESS : ERROR_DNS_UPDATE_FAILED ;
2012-09-25 11:09:45 +02:00
if ( ! ERR_DNS_IS_OK ( err ) ) {
DEBUG ( 3 , ( " DoDNSUpdate: signed update failed \n " ) ) ;
}
2006-08-24 15:43:32 +00:00
}
error :
2006-11-17 21:46:26 +00:00
TALLOC_FREE ( mem_ctx ) ;
return err ;
2006-08-24 15:43:32 +00:00
}
/*********************************************************************
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2008-01-08 13:11:12 -08:00
int get_my_ip_address ( struct sockaddr_storage * * pp_ss )
2006-08-24 15:43:32 +00:00
{
int i , n ;
2008-01-08 13:11:12 -08:00
struct sockaddr_storage * list = NULL ;
2006-08-24 15:43:32 +00:00
int count = 0 ;
2008-10-03 10:51:54 -05:00
/* Honor the configured list of interfaces to register */
2006-08-24 15:43:32 +00:00
2008-10-03 10:51:54 -05:00
load_interfaces ( ) ;
n = iface_count ( ) ;
2008-01-08 13:11:12 -08:00
if ( n < = 0 ) {
return - 1 ;
}
if ( ( list = SMB_MALLOC_ARRAY ( struct sockaddr_storage , n ) ) = = NULL ) {
2006-08-24 15:43:32 +00:00
return - 1 ;
}
for ( i = 0 ; i < n ; i + + ) {
2008-10-03 10:51:54 -05:00
const struct sockaddr_storage * nic_sa_storage = NULL ;
if ( ( nic_sa_storage = iface_n_sockaddr_storage ( i ) ) = = NULL )
continue ;
/* Don't register loopback addresses */
2011-09-25 13:24:50 -07:00
if ( is_loopback_addr ( ( const struct sockaddr * ) nic_sa_storage ) ) {
2008-01-08 13:11:12 -08:00
continue ;
}
2008-10-03 10:51:54 -05:00
2011-09-24 18:18:14 -07:00
/* Don't register link-local addresses */
if ( is_linklocal_addr ( nic_sa_storage ) ) {
continue ;
}
2008-10-03 10:51:54 -05:00
memcpy ( & list [ count + + ] , nic_sa_storage , sizeof ( struct sockaddr_storage ) ) ;
2006-08-24 15:43:32 +00:00
}
2008-01-09 11:44:40 +01:00
* pp_ss = list ;
2006-08-24 15:43:32 +00:00
return count ;
}
2006-11-17 21:46:26 +00:00
DNS_ERROR do_gethostbyname ( const char * server , const char * host )
{
2015-04-23 12:36:28 -07:00
struct dns_connection * conn = NULL ;
2006-11-17 21:46:26 +00:00
struct dns_request * req , * resp ;
DNS_ERROR err ;
2016-06-18 13:29:36 -07:00
int ans = 0 ;
2006-11-17 21:46:26 +00:00
2006-12-31 06:50:44 +00:00
err = dns_open_connection ( server , DNS_UDP , NULL , & conn ) ;
2016-06-18 13:29:36 -07:00
if ( ! ERR_DNS_IS_OK ( err ) ) {
goto error ;
}
2006-11-17 21:46:26 +00:00
err = dns_create_query ( conn , host , QTYPE_A , DNS_CLASS_IN , & req ) ;
2016-06-18 13:29:36 -07:00
if ( ! ERR_DNS_IS_OK ( err ) ) {
goto error ;
}
2006-11-17 21:46:26 +00:00
err = dns_transaction ( conn , conn , req , & resp ) ;
2016-06-18 13:29:36 -07:00
if ( ! ERR_DNS_IS_OK ( err ) ) {
goto error ;
}
if ( resp - > num_answers = = 0 ) {
printf ( " %s " , " No answers! \n " ) ;
goto error ;
}
for ( ans = 0 ; ans < resp - > num_answers ; ans + + ) {
struct in_addr resp_ip ;
if ( ans > 0 )
printf ( " %s " , " " ) ;
resp_ip . s_addr = * ( ( uint32_t * ) resp - > answers [ ans ] - > data ) ;
printf ( " %s " , inet_ntoa ( resp_ip ) ) ;
}
printf ( " %s " , " \n " ) ;
2006-11-17 21:46:26 +00:00
error :
TALLOC_FREE ( conn ) ;
return err ;
}
2006-08-24 15:43:32 +00:00
# endif /* defined(WITH_DNS_UPDATES) */