2001-12-20 10:42:00 +03:00
/*
Samba Unix / Linux SMB client library
net lookup command
Copyright ( C ) 2001 Andrew Tridgell ( tridge @ samba . org )
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
the Free Software Foundation ; either version 2 of the License , or
( at your option ) any later version .
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 .
You should have received a copy of the GNU General Public License
along with this program ; if not , write to the Free Software
Foundation , Inc . , 675 Mass Ave , Cambridge , MA 0213 9 , USA . */
# include "includes.h"
# include "../utils/net.h"
int net_lookup_usage ( int argc , const char * * argv )
{
d_printf (
2003-09-05 08:40:10 +04:00
" net lookup [host] HOSTNAME[#<type>] \n \t gives IP for a hostname \n \n "
2002-07-15 14:35:28 +04:00
" net lookup ldap [domain] \n \t gives IP of domain's ldap server \n \n "
" net lookup kdc [realm] \n \t gives IP of realm's kerberos KDC \n \n "
" net lookup dc [domain] \n \t gives IP of domains Domain Controllers \n \n "
" net lookup master [domain|wg] \n \t give IP of master browser \n \n "
) ;
2001-12-20 10:42:00 +03:00
return - 1 ;
}
/* lookup a hostname giving an IP */
static int net_lookup_host ( int argc , const char * * argv )
{
struct in_addr ip ;
int name_type = 0x20 ;
2003-09-05 08:40:10 +04:00
const char * name = argv [ 0 ] ;
char * p ;
2001-12-20 10:42:00 +03:00
2003-09-05 08:40:10 +04:00
if ( argc = = 0 )
return net_lookup_usage ( argc , argv ) ;
2001-12-20 10:42:00 +03:00
2003-09-05 08:40:10 +04:00
p = strchr_m ( name , ' # ' ) ;
if ( p ) {
* p = ' \0 ' ;
sscanf ( + + p , " %x " , & name_type ) ;
}
if ( ! resolve_name ( name , & ip , name_type ) ) {
2001-12-20 10:42:00 +03:00
/* we deliberately use DEBUG() here to send it to stderr
so scripts aren ' t mucked up */
2003-09-05 08:40:10 +04:00
DEBUG ( 0 , ( " Didn't find %s#%02x \n " , name , name_type ) ) ;
2001-12-20 10:42:00 +03:00
return - 1 ;
}
d_printf ( " %s \n " , inet_ntoa ( ip ) ) ;
return 0 ;
}
2002-07-15 14:35:28 +04:00
static void print_ldap_srvlist ( char * srvlist )
{
char * cur , * next ;
struct in_addr ip ;
BOOL printit ;
cur = srvlist ;
do {
next = strchr ( cur , ' : ' ) ;
if ( next ) * next + + = ' \0 ' ;
printit = resolve_name ( cur , & ip , 0x20 ) ;
cur = next ;
next = cur ? strchr ( cur , ' ' ) : NULL ;
if ( next )
* next + + = ' \0 ' ;
if ( printit )
d_printf ( " %s:%s \n " , inet_ntoa ( ip ) , cur ? cur : " " ) ;
cur = next ;
} while ( next ) ;
}
static int net_lookup_ldap ( int argc , const char * * argv )
{
# ifdef HAVE_LDAP
2002-08-17 18:45:04 +04:00
char * srvlist ;
const char * domain ;
2002-11-06 08:14:15 +03:00
int rc ;
struct in_addr addr ;
2002-07-15 14:35:28 +04:00
struct hostent * hostent ;
if ( argc > 0 )
domain = argv [ 0 ] ;
else
domain = opt_target_workgroup ;
DEBUG ( 9 , ( " Lookup up ldap for domain %s \n " , domain ) ) ;
rc = ldap_domain2hostlist ( domain , & srvlist ) ;
if ( ( rc = = LDAP_SUCCESS ) & & srvlist ) {
print_ldap_srvlist ( srvlist ) ;
return 0 ;
}
DEBUG ( 9 , ( " Looking up DC for domain %s \n " , domain ) ) ;
2002-11-06 08:14:15 +03:00
if ( ! get_pdc_ip ( domain , & addr ) )
2002-07-15 14:35:28 +04:00
return - 1 ;
2002-11-06 08:14:15 +03:00
hostent = gethostbyaddr ( ( char * ) & addr . s_addr , sizeof ( addr . s_addr ) ,
2002-07-15 14:35:28 +04:00
AF_INET ) ;
if ( ! hostent )
return - 1 ;
DEBUG ( 9 , ( " Found DC with DNS name %s \n " , hostent - > h_name ) ) ;
domain = strchr ( hostent - > h_name , ' . ' ) ;
if ( ! domain )
return - 1 ;
domain + + ;
DEBUG ( 9 , ( " Looking up ldap for domain %s \n " , domain ) ) ;
rc = ldap_domain2hostlist ( domain , & srvlist ) ;
if ( ( rc = = LDAP_SUCCESS ) & & srvlist ) {
print_ldap_srvlist ( srvlist ) ;
return 0 ;
}
return - 1 ;
# endif
DEBUG ( 1 , ( " No LDAP support \n " ) ) ;
return - 1 ;
}
static int net_lookup_dc ( int argc , const char * * argv )
{
2003-06-25 21:41:05 +04:00
struct ip_service * ip_list ;
struct in_addr addr ;
2002-07-15 14:35:28 +04:00
char * pdc_str = NULL ;
2002-08-17 18:45:04 +04:00
const char * domain = opt_target_workgroup ;
2002-07-15 14:35:28 +04:00
int count , i ;
if ( argc > 0 )
domain = argv [ 0 ] ;
/* first get PDC */
2002-11-06 08:14:15 +03:00
if ( ! get_pdc_ip ( domain , & addr ) )
2002-07-15 14:35:28 +04:00
return - 1 ;
2002-11-06 08:14:15 +03:00
asprintf ( & pdc_str , " %s " , inet_ntoa ( addr ) ) ;
2002-07-15 14:35:28 +04:00
d_printf ( " %s \n " , pdc_str ) ;
2001-12-20 10:42:00 +03:00
2003-06-25 21:41:05 +04:00
if ( ! get_sorted_dc_list ( domain , & ip_list , & count , False ) ) {
2002-07-15 14:35:28 +04:00
SAFE_FREE ( pdc_str ) ;
return 0 ;
}
for ( i = 0 ; i < count ; i + + ) {
2003-06-25 21:41:05 +04:00
char * dc_str = inet_ntoa ( ip_list [ i ] . ip ) ;
2002-07-15 14:35:28 +04:00
if ( ! strequal ( pdc_str , dc_str ) )
d_printf ( " %s \n " , dc_str ) ;
}
SAFE_FREE ( pdc_str ) ;
return 0 ;
}
static int net_lookup_master ( int argc , const char * * argv )
{
struct in_addr master_ip ;
2002-08-17 18:45:04 +04:00
const char * domain = opt_target_workgroup ;
2002-07-15 14:35:28 +04:00
if ( argc > 0 )
domain = argv [ 0 ] ;
if ( ! find_master_ip ( domain , & master_ip ) )
return - 1 ;
d_printf ( " %s \n " , inet_ntoa ( master_ip ) ) ;
return 0 ;
}
static int net_lookup_kdc ( int argc , const char * * argv )
{
# ifdef HAVE_KRB5
krb5_error_code rc ;
krb5_context ctx ;
struct sockaddr_in * addrs ;
int num_kdcs , i ;
krb5_data realm ;
char * * realms ;
rc = krb5_init_context ( & ctx ) ;
if ( rc ) {
DEBUG ( 1 , ( " krb5_init_context failed (%s) \n " ,
error_message ( rc ) ) ) ;
return - 1 ;
}
if ( argc > 0 ) {
realm . data = ( krb5_pointer ) argv [ 0 ] ;
realm . length = strlen ( argv [ 0 ] ) ;
} else if ( lp_realm ( ) & & * lp_realm ( ) ) {
realm . data = ( krb5_pointer ) lp_realm ( ) ;
realm . length = strlen ( realm . data ) ;
} else {
rc = krb5_get_host_realm ( ctx , NULL , & realms ) ;
if ( rc ) {
DEBUG ( 1 , ( " krb5_gethost_realm failed (%s) \n " ,
error_message ( rc ) ) ) ;
return - 1 ;
}
realm . data = ( krb5_pointer ) * realms ;
realm . length = strlen ( realm . data ) ;
}
2003-01-30 07:40:12 +03:00
rc = krb5_locate_kdc ( ctx , & realm , & addrs , & num_kdcs , 0 ) ;
2002-07-15 14:35:28 +04:00
if ( rc ) {
DEBUG ( 1 , ( " krb5_locate_kdc failed (%s) \n " , error_message ( rc ) ) ) ;
return - 1 ;
}
for ( i = 0 ; i < num_kdcs ; i + + )
if ( addrs [ i ] . sin_family = = AF_INET )
d_printf ( " %s:%hd \n " , inet_ntoa ( addrs [ i ] . sin_addr ) ,
ntohs ( addrs [ i ] . sin_port ) ) ;
return 0 ;
# endif
DEBUG ( 1 , ( " No kerberos support \n " ) ) ;
return - 1 ;
}
2002-08-17 18:45:04 +04:00
2001-12-20 10:42:00 +03:00
/* lookup hosts or IP addresses using internal samba lookup fns */
int net_lookup ( int argc , const char * * argv )
{
2003-09-05 08:40:10 +04:00
int i ;
struct functable table [ ] = {
2001-12-20 10:42:00 +03:00
{ " HOST " , net_lookup_host } ,
2002-07-15 14:35:28 +04:00
{ " LDAP " , net_lookup_ldap } ,
{ " DC " , net_lookup_dc } ,
{ " MASTER " , net_lookup_master } ,
{ " KDC " , net_lookup_kdc } ,
2001-12-20 10:42:00 +03:00
{ NULL , NULL }
} ;
2003-09-05 08:40:10 +04:00
if ( argc < 1 ) {
d_printf ( " \n Usage: \n " ) ;
return net_lookup_usage ( argc , argv ) ;
}
for ( i = 0 ; table [ i ] . funcname ; i + + ) {
if ( StrCaseCmp ( argv [ 0 ] , table [ i ] . funcname ) = = 0 )
return table [ i ] . fn ( argc - 1 , argv + 1 ) ;
}
/* Default to lookup a hostname so 'net lookup foo#1b' can be
used instead of ' net lookup host foo # 1 b ' . The host syntax
is a bit confusing as non # 00 names can ' t really be
considered hosts as such . */
return net_lookup_host ( argc , argv ) ;
2001-12-20 10:42:00 +03:00
}