2021-11-10 12:01:20 +01:00
/*
2005-03-21 02:08:38 +00:00
Unix SMB / CIFS implementation .
2005-09-29 00:02:38 +00:00
User credentials handling
2005-03-21 02:08:38 +00:00
Copyright ( C ) Jelmer Vernooij 2005
Copyright ( C ) Tim Potter 2001
2005-03-22 08:00:45 +00:00
Copyright ( C ) Andrew Bartlett < abartlet @ samba . org > 2005
2021-11-10 12:01:20 +01:00
2005-03-21 02:08:38 +00:00
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-10 02:07:03 +00:00
the Free Software Foundation ; either version 3 of the License , or
2005-03-21 02:08:38 +00:00
( at your option ) any later version .
2021-11-10 12:01:20 +01:00
2005-03-21 02:08:38 +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 .
2021-11-10 12:01:20 +01:00
2005-03-21 02:08:38 +00:00
You should have received a copy of the GNU General Public License
2007-07-10 02:07:03 +00:00
along with this program . If not , see < http : //www.gnu.org/licenses/>.
2005-03-21 02:08:38 +00:00
*/
# include "includes.h"
2024-04-10 13:02:39 +02:00
# include "lib/util/util_file.h"
2006-03-16 00:23:11 +00:00
# include "librpc/gen_ndr/samr.h" /* for struct samrPassword */
2006-11-07 00:48:36 +00:00
# include "auth/credentials/credentials.h"
2013-07-31 12:41:40 +02:00
# include "auth/credentials/credentials_internal.h"
2017-08-29 15:24:14 +02:00
# include "auth/gensec/gensec.h"
2006-03-14 15:03:25 +00:00
# include "libcli/auth/libcli_auth.h"
2011-10-09 23:27:44 +11:00
# include "tevent.h"
2007-09-08 12:42:09 +00:00
# include "param/param.h"
2010-10-11 01:20:42 +02:00
# include "system/filesys.h"
2021-11-10 12:06:51 +01:00
# include "system/passwd.h"
2005-03-21 02:08:38 +00:00
2005-03-27 06:55:03 +00:00
/**
* Create a new credentials structure
2021-11-10 12:01:20 +01:00
* @ param mem_ctx TALLOC_CTX parent for credentials structure
2005-03-27 06:55:03 +00:00
*/
2021-11-10 12:01:20 +01:00
_PUBLIC_ struct cli_credentials * cli_credentials_init ( TALLOC_CTX * mem_ctx )
2005-03-21 02:08:38 +00:00
{
2016-12-14 08:50:51 +01:00
struct cli_credentials * cred = talloc_zero ( mem_ctx , struct cli_credentials ) ;
2009-10-15 10:30:07 +02:00
if ( cred = = NULL ) {
2005-03-23 01:30:43 +00:00
return cred ;
}
2016-10-28 15:55:48 +02:00
cred - > winbind_separator = ' \\ ' ;
2020-08-19 15:46:11 +02:00
cred - > kerberos_state = CRED_USE_KERBEROS_DESIRED ;
2020-08-20 09:40:41 +02:00
2020-05-26 09:32:44 +02:00
cred - > signing_state = SMB_SIGNING_DEFAULT ;
2020-05-28 16:31:35 +02:00
/*
* The default value of lpcfg_client_ipc_signing ( ) is REQUIRED , so use
* the same value here .
*/
cred - > ipc_signing_state = SMB_SIGNING_REQUIRED ;
2020-05-28 16:10:52 +02:00
cred - > encryption_state = SMB_ENCRYPTION_DEFAULT ;
2020-05-28 16:31:35 +02:00
2005-03-23 01:30:43 +00:00
return cred ;
2005-03-22 08:00:45 +00:00
}
2005-03-21 21:22:07 +00:00
2020-09-04 12:21:21 +02:00
_PUBLIC_
struct cli_credentials * cli_credentials_init_server ( TALLOC_CTX * mem_ctx ,
struct loadparm_context * lp_ctx )
{
struct cli_credentials * server_creds = NULL ;
NTSTATUS status ;
2021-04-01 17:02:10 +02:00
bool ok ;
2020-09-04 12:21:21 +02:00
server_creds = cli_credentials_init ( mem_ctx ) ;
if ( server_creds = = NULL ) {
return NULL ;
}
2021-04-01 17:02:10 +02:00
ok = cli_credentials_set_conf ( server_creds , lp_ctx ) ;
if ( ! ok ) {
TALLOC_FREE ( server_creds ) ;
return NULL ;
}
2020-09-04 12:21:21 +02:00
status = cli_credentials_set_machine_account ( server_creds , lp_ctx ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
DEBUG ( 1 , ( " Failed to obtain server credentials: %s \n " ,
nt_errstr ( status ) ) ) ;
TALLOC_FREE ( server_creds ) ;
return NULL ;
}
return server_creds ;
}
2013-07-31 12:52:17 +02:00
_PUBLIC_ void cli_credentials_set_callback_data ( struct cli_credentials * cred ,
void * callback_data )
{
cred - > priv_data = callback_data ;
}
_PUBLIC_ void * _cli_credentials_callback_data ( struct cli_credentials * cred )
{
return cred - > priv_data ;
}
2007-03-02 23:24:27 +00:00
/**
* Create a new anonymous credential
2021-11-10 12:01:20 +01:00
* @ param mem_ctx TALLOC_CTX parent for credentials structure
2007-03-02 23:24:27 +00:00
*/
2008-04-02 04:53:27 +02:00
_PUBLIC_ struct cli_credentials * cli_credentials_init_anon ( TALLOC_CTX * mem_ctx )
2007-03-02 23:24:27 +00:00
{
struct cli_credentials * anon_credentials ;
anon_credentials = cli_credentials_init ( mem_ctx ) ;
cli_credentials_set_anonymous ( anon_credentials ) ;
return anon_credentials ;
}
2020-08-19 15:46:11 +02:00
_PUBLIC_ bool cli_credentials_set_kerberos_state ( struct cli_credentials * creds ,
enum credentials_use_kerberos kerberos_state ,
enum credentials_obtained obtained )
2006-01-28 12:15:24 +00:00
{
2020-08-19 15:46:11 +02:00
if ( obtained > = creds - > kerberos_state_obtained ) {
creds - > kerberos_state = kerberos_state ;
creds - > kerberos_state_obtained = obtained ;
return true ;
}
return false ;
2006-01-28 12:15:24 +00:00
}
2013-09-16 09:38:09 -07:00
_PUBLIC_ void cli_credentials_set_forced_sasl_mech ( struct cli_credentials * creds ,
const char * sasl_mech )
{
TALLOC_FREE ( creds - > forced_sasl_mech ) ;
creds - > forced_sasl_mech = talloc_strdup ( creds , sasl_mech ) ;
}
2010-09-16 14:12:37 +10:00
_PUBLIC_ void cli_credentials_set_krb_forwardable ( struct cli_credentials * creds ,
enum credentials_krb_forwardable krb_forwardable )
{
creds - > krb_forwardable = krb_forwardable ;
}
2008-04-02 04:53:27 +02:00
_PUBLIC_ enum credentials_use_kerberos cli_credentials_get_kerberos_state ( struct cli_credentials * creds )
2006-01-28 12:15:24 +00:00
{
2020-08-19 15:46:11 +02:00
return creds - > kerberos_state ;
2006-01-28 12:15:24 +00:00
}
2024-06-18 18:53:48 +02:00
_PUBLIC_ enum credentials_obtained cli_credentials_get_kerberos_state_obtained ( struct cli_credentials * creds )
{
return creds - > kerberos_state_obtained ;
}
2013-09-16 09:38:09 -07:00
_PUBLIC_ const char * cli_credentials_get_forced_sasl_mech ( struct cli_credentials * creds )
{
return creds - > forced_sasl_mech ;
}
2010-09-16 14:12:37 +10:00
_PUBLIC_ enum credentials_krb_forwardable cli_credentials_get_krb_forwardable ( struct cli_credentials * creds )
{
return creds - > krb_forwardable ;
}
2020-08-20 10:50:30 +02:00
_PUBLIC_ bool cli_credentials_set_gensec_features ( struct cli_credentials * creds ,
uint32_t gensec_features ,
enum credentials_obtained obtained )
2006-12-12 21:47:56 +00:00
{
2020-08-20 10:50:30 +02:00
if ( obtained > = creds - > gensec_features_obtained ) {
creds - > gensec_features_obtained = obtained ;
creds - > gensec_features = gensec_features ;
return true ;
}
return false ;
2006-12-12 21:47:56 +00:00
}
2024-05-28 12:32:58 +02:00
_PUBLIC_ bool cli_credentials_add_gensec_features (
struct cli_credentials * creds ,
uint32_t gensec_features ,
enum credentials_obtained obtained )
{
return cli_credentials_set_gensec_features (
creds , creds - > gensec_features | gensec_features , obtained ) ;
}
2008-04-02 04:53:27 +02:00
_PUBLIC_ uint32_t cli_credentials_get_gensec_features ( struct cli_credentials * creds )
2006-12-12 21:47:56 +00:00
{
return creds - > gensec_features ;
}
2024-03-13 17:50:34 +01:00
/**
* @ brief Find out how the username was obtained .
*
* @ param cred A credentials context .
*
* @ return The obtained information for the username .
*/
_PUBLIC_ enum credentials_obtained
cli_credentials_get_username_obtained ( struct cli_credentials * cred )
{
return cred - > username_obtained ;
}
2006-01-28 12:15:24 +00:00
2005-03-27 06:55:03 +00:00
/**
* Obtain the username for this credentials context .
* @ param cred credentials context
* @ retval The username set on this context .
* @ note Return value will never be NULL except by programmer error .
*/
2008-04-02 04:53:27 +02:00
_PUBLIC_ const char * cli_credentials_get_username ( struct cli_credentials * cred )
2005-03-22 08:00:45 +00:00
{
2005-03-24 03:32:25 +00:00
if ( cred - > machine_account_pending ) {
2021-11-10 12:01:20 +01:00
cli_credentials_set_machine_account ( cred ,
2007-12-13 22:46:17 +01:00
cred - > machine_account_pending_lp_ctx ) ;
2005-03-24 03:32:25 +00:00
}
2021-11-10 12:01:20 +01:00
if ( cred - > username_obtained = = CRED_CALLBACK & &
2006-05-03 20:56:14 +00:00
! cred - > callback_running ) {
2007-10-06 22:16:19 +00:00
cred - > callback_running = true ;
2005-03-21 02:08:38 +00:00
cred - > username = cred - > username_cb ( cred ) ;
2007-10-06 22:16:19 +00:00
cred - > callback_running = false ;
2013-07-31 14:32:36 +02:00
if ( cred - > username_obtained = = CRED_CALLBACK ) {
cred - > username_obtained = CRED_CALLBACK_RESULT ;
cli_credentials_invalidate_ccache ( cred , cred - > username_obtained ) ;
}
2005-03-21 02:08:38 +00:00
}
2005-09-22 01:50:58 +00:00
return cred - > username ;
2005-03-21 02:08:38 +00:00
}
2020-08-31 18:52:44 +02:00
/**
* @ brief Obtain the username for this credentials context .
*
* @ param [ in ] cred The credential context .
*
* @ param [ in ] obtained A pointer to store the obtained information .
*
2023-03-14 08:50:34 +01:00
* return The user name or NULL if an error occurred .
2020-08-31 18:52:44 +02:00
*/
_PUBLIC_ const char *
cli_credentials_get_username_and_obtained ( struct cli_credentials * cred ,
enum credentials_obtained * obtained )
{
if ( obtained ! = NULL ) {
* obtained = cred - > username_obtained ;
}
return cli_credentials_get_username ( cred ) ;
}
2021-11-10 12:01:20 +01:00
_PUBLIC_ bool cli_credentials_set_username ( struct cli_credentials * cred ,
2005-08-30 01:19:41 +00:00
const char * val , enum credentials_obtained obtained )
2005-03-21 02:08:38 +00:00
{
if ( obtained > = cred - > username_obtained ) {
cred - > username = talloc_strdup ( cred , val ) ;
cred - > username_obtained = obtained ;
2007-05-22 05:21:59 +00:00
cli_credentials_invalidate_ccache ( cred , cred - > username_obtained ) ;
2007-10-06 22:16:19 +00:00
return true ;
2005-03-21 02:08:38 +00:00
}
2007-10-06 22:16:19 +00:00
return false ;
2005-03-21 02:08:38 +00:00
}
2011-10-09 23:27:44 +11:00
_PUBLIC_ bool cli_credentials_set_username_callback ( struct cli_credentials * cred ,
2005-08-30 01:19:41 +00:00
const char * ( * username_cb ) ( struct cli_credentials * ) )
{
if ( cred - > username_obtained < CRED_CALLBACK ) {
cred - > username_cb = username_cb ;
cred - > username_obtained = CRED_CALLBACK ;
2007-10-06 22:16:19 +00:00
return true ;
2005-08-30 01:19:41 +00:00
}
2007-10-06 22:16:19 +00:00
return false ;
2005-08-30 01:19:41 +00:00
}
2021-11-10 12:01:20 +01:00
_PUBLIC_ bool cli_credentials_set_bind_dn ( struct cli_credentials * cred ,
2005-12-18 05:01:15 +00:00
const char * bind_dn )
{
cred - > bind_dn = talloc_strdup ( cred , bind_dn ) ;
2007-10-06 22:16:19 +00:00
return true ;
2005-12-18 05:01:15 +00:00
}
/**
* Obtain the BIND DN for this credentials context .
* @ param cred credentials context
* @ retval The username set on this context .
2023-03-14 08:50:34 +01:00
* @ note Return value will be NULL if not specified explicitly
2005-12-18 05:01:15 +00:00
*/
2008-04-02 04:53:27 +02:00
_PUBLIC_ const char * cli_credentials_get_bind_dn ( struct cli_credentials * cred )
2005-12-18 05:01:15 +00:00
{
return cred - > bind_dn ;
}
2005-08-30 01:19:41 +00:00
2024-03-13 17:50:56 +01:00
/**
* @ brief Find out how the principal was obtained .
*
* @ param cred A credentials context .
*
* @ return The obtained information for the principal .
*/
_PUBLIC_ enum credentials_obtained
cli_credentials_get_principal_obtained ( struct cli_credentials * cred )
{
if ( cred - > machine_account_pending ) {
cli_credentials_set_machine_account ( cred ,
cred - > machine_account_pending_lp_ctx ) ;
}
if ( cred - > principal_obtained < cred - > username_obtained
| | cred - > principal_obtained < MAX ( cred - > domain_obtained , cred - > realm_obtained ) ) {
const char * effective_username = NULL ;
const char * effective_realm = NULL ;
enum credentials_obtained effective_obtained ;
/*
* We don ' t want to trigger a callbacks in
* cli_credentials_get_username ( )
* cli_credentials_get_domain ( )
* nor
* cli_credentials_get_realm ( )
*/
effective_username = cred - > username ;
if ( effective_username = = NULL | | strlen ( effective_username ) = = 0 ) {
return cred - > username_obtained ;
}
if ( cred - > domain_obtained > cred - > realm_obtained ) {
effective_realm = cred - > domain ;
effective_obtained = MIN ( cred - > domain_obtained ,
cred - > username_obtained ) ;
} else {
effective_realm = cred - > realm ;
effective_obtained = MIN ( cred - > realm_obtained ,
cred - > username_obtained ) ;
}
if ( effective_realm = = NULL | | strlen ( effective_realm ) = = 0 ) {
effective_realm = cred - > domain ;
effective_obtained = MIN ( cred - > domain_obtained ,
cred - > username_obtained ) ;
}
if ( effective_realm ! = NULL & & strlen ( effective_realm ) ! = 0 ) {
return effective_obtained ;
}
}
return cred - > principal_obtained ;
}
2005-08-29 04:30:22 +00:00
/**
* Obtain the client principal for this credentials context .
* @ param cred credentials context
* @ retval The username set on this context .
* @ note Return value will never be NULL except by programmer error .
*/
2016-12-29 15:26:00 +01:00
_PUBLIC_ char * cli_credentials_get_principal_and_obtained ( struct cli_credentials * cred , TALLOC_CTX * mem_ctx , enum credentials_obtained * obtained )
2005-08-29 04:30:22 +00:00
{
if ( cred - > machine_account_pending ) {
2007-12-13 22:46:17 +01:00
cli_credentials_set_machine_account ( cred ,
cred - > machine_account_pending_lp_ctx ) ;
2005-08-29 04:30:22 +00:00
}
2021-11-10 12:01:20 +01:00
if ( cred - > principal_obtained = = CRED_CALLBACK & &
2006-05-03 20:56:14 +00:00
! cred - > callback_running ) {
2007-10-06 22:16:19 +00:00
cred - > callback_running = true ;
2006-05-07 18:08:57 +00:00
cred - > principal = cred - > principal_cb ( cred ) ;
2007-10-06 22:16:19 +00:00
cred - > callback_running = false ;
2013-07-31 14:32:36 +02:00
if ( cred - > principal_obtained = = CRED_CALLBACK ) {
cred - > principal_obtained = CRED_CALLBACK_RESULT ;
cli_credentials_invalidate_ccache ( cred , cred - > principal_obtained ) ;
}
2005-08-29 04:30:22 +00:00
}
2010-05-01 10:33:08 +10:00
if ( cred - > principal_obtained < cred - > username_obtained
| | cred - > principal_obtained < MAX ( cred - > domain_obtained , cred - > realm_obtained ) ) {
2016-12-15 11:04:02 +01:00
const char * effective_username = NULL ;
const char * effective_realm = NULL ;
enum credentials_obtained effective_obtained ;
effective_username = cli_credentials_get_username ( cred ) ;
if ( effective_username = = NULL | | strlen ( effective_username ) = = 0 ) {
* obtained = cred - > username_obtained ;
return NULL ;
}
2005-09-20 21:29:29 +00:00
if ( cred - > domain_obtained > cred - > realm_obtained ) {
2016-12-15 11:04:02 +01:00
effective_realm = cli_credentials_get_domain ( cred ) ;
effective_obtained = MIN ( cred - > domain_obtained ,
cred - > username_obtained ) ;
2005-09-20 21:29:29 +00:00
} else {
2016-12-15 11:04:02 +01:00
effective_realm = cli_credentials_get_realm ( cred ) ;
effective_obtained = MIN ( cred - > realm_obtained ,
cred - > username_obtained ) ;
}
if ( effective_realm = = NULL | | strlen ( effective_realm ) = = 0 ) {
effective_realm = cli_credentials_get_domain ( cred ) ;
effective_obtained = MIN ( cred - > domain_obtained ,
cred - > username_obtained ) ;
}
if ( effective_realm ! = NULL & & strlen ( effective_realm ) ! = 0 ) {
* obtained = effective_obtained ;
2021-11-10 12:01:20 +01:00
return talloc_asprintf ( mem_ctx , " %s@%s " ,
2016-12-15 11:04:02 +01:00
effective_username ,
effective_realm ) ;
2005-09-20 21:29:29 +00:00
}
2005-08-29 04:30:22 +00:00
}
2010-05-01 10:33:08 +10:00
* obtained = cred - > principal_obtained ;
2013-07-31 12:33:25 +02:00
return talloc_strdup ( mem_ctx , cred - > principal ) ;
2005-08-29 04:30:22 +00:00
}
2010-05-01 10:33:08 +10:00
/**
* Obtain the client principal for this credentials context .
* @ param cred credentials context
* @ retval The username set on this context .
* @ note Return value will never be NULL except by programmer error .
*/
2016-12-29 15:26:00 +01:00
_PUBLIC_ char * cli_credentials_get_principal ( struct cli_credentials * cred , TALLOC_CTX * mem_ctx )
2010-05-01 10:33:08 +10:00
{
enum credentials_obtained obtained ;
return cli_credentials_get_principal_and_obtained ( cred , mem_ctx , & obtained ) ;
}
2021-11-10 12:01:20 +01:00
_PUBLIC_ bool cli_credentials_set_principal ( struct cli_credentials * cred ,
const char * val ,
2005-09-01 23:24:47 +00:00
enum credentials_obtained obtained )
2005-08-29 04:30:22 +00:00
{
if ( obtained > = cred - > principal_obtained ) {
cred - > principal = talloc_strdup ( cred , val ) ;
2016-10-06 08:16:57 +02:00
if ( cred - > principal = = NULL ) {
return false ;
}
2005-08-29 04:30:22 +00:00
cred - > principal_obtained = obtained ;
2016-10-06 08:16:57 +02:00
2007-05-22 05:21:59 +00:00
cli_credentials_invalidate_ccache ( cred , cred - > principal_obtained ) ;
2007-10-06 22:16:19 +00:00
return true ;
2005-08-29 04:30:22 +00:00
}
2007-10-06 22:16:19 +00:00
return false ;
2005-08-29 04:30:22 +00:00
}
r11995: A big kerberos-related update.
This merges Samba4 up to current lorikeet-heimdal, which includes a
replacement for some Samba-specific hacks.
In particular, the credentials system now supplies GSS client and
server credentials. These are imported into GSS with
gss_krb5_import_creds(). Unfortunetly this can't take an MEMORY
keytab, so we now create a FILE based keytab as provision and join
time.
Because the keytab is now created in advance, we don't spend .4s at
negprot doing sha1 s2k calls. Also, because the keytab is read in
real time, any change in the server key will be correctly picked up by
the the krb5 code.
To mark entries in the secrets which should be exported to a keytab,
there is a new kerberosSecret objectClass. The new routine
cli_credentials_update_all_keytabs() searches for these, and updates
the keytabs.
This is called in the provision.js via the ejs wrapper
credentials_update_all_keytabs().
We can now (in theory) use a system-provided /etc/krb5.keytab, if
krb5Keytab: FILE:/etc/krb5.keytab
is added to the secrets.ldb record. By default the attribute
privateKeytab: secrets.keytab
is set, pointing to allow the whole private directory to be moved
without breaking the internal links.
(This used to be commit 6b75573df49c6210e1b9d71e108a9490976bd41d)
2005-12-01 05:20:39 +00:00
/* Set a callback to get the principal. This could be a popup dialog,
* a terminal prompt or similar . */
2011-10-09 23:27:44 +11:00
_PUBLIC_ bool cli_credentials_set_principal_callback ( struct cli_credentials * cred ,
2005-08-30 01:19:41 +00:00
const char * ( * principal_cb ) ( struct cli_credentials * ) )
{
if ( cred - > principal_obtained < CRED_CALLBACK ) {
cred - > principal_cb = principal_cb ;
cred - > principal_obtained = CRED_CALLBACK ;
2007-10-06 22:16:19 +00:00
return true ;
2005-08-30 01:19:41 +00:00
}
2007-10-06 22:16:19 +00:00
return false ;
2005-08-30 01:19:41 +00:00
}
r11995: A big kerberos-related update.
This merges Samba4 up to current lorikeet-heimdal, which includes a
replacement for some Samba-specific hacks.
In particular, the credentials system now supplies GSS client and
server credentials. These are imported into GSS with
gss_krb5_import_creds(). Unfortunetly this can't take an MEMORY
keytab, so we now create a FILE based keytab as provision and join
time.
Because the keytab is now created in advance, we don't spend .4s at
negprot doing sha1 s2k calls. Also, because the keytab is read in
real time, any change in the server key will be correctly picked up by
the the krb5 code.
To mark entries in the secrets which should be exported to a keytab,
there is a new kerberosSecret objectClass. The new routine
cli_credentials_update_all_keytabs() searches for these, and updates
the keytabs.
This is called in the provision.js via the ejs wrapper
credentials_update_all_keytabs().
We can now (in theory) use a system-provided /etc/krb5.keytab, if
krb5Keytab: FILE:/etc/krb5.keytab
is added to the secrets.ldb record. By default the attribute
privateKeytab: secrets.keytab
is set, pointing to allow the whole private directory to be moved
without breaking the internal links.
(This used to be commit 6b75573df49c6210e1b9d71e108a9490976bd41d)
2005-12-01 05:20:39 +00:00
/* Some of our tools are 'anonymous by default'. This is a single
* function to determine if authentication has been explicitly
* requested */
2021-11-10 12:01:20 +01:00
_PUBLIC_ bool cli_credentials_authentication_requested ( struct cli_credentials * cred )
2005-08-29 04:30:22 +00:00
{
2017-08-29 15:24:14 +02:00
uint32_t gensec_features = 0 ;
2005-12-18 05:01:15 +00:00
if ( cred - > bind_dn ) {
2007-10-06 22:16:19 +00:00
return true ;
2005-12-18 05:01:15 +00:00
}
2013-09-17 13:09:07 -07:00
/*
* If we forced the mech we clearly want authentication . E . g . to use
* SASL / EXTERNAL which has no credentials .
*/
if ( cred - > forced_sasl_mech ) {
return true ;
}
2006-06-14 04:00:42 +00:00
if ( cli_credentials_is_anonymous ( cred ) ) {
2007-10-06 22:16:19 +00:00
return false ;
2005-12-14 07:22:25 +00:00
}
2005-08-29 04:38:37 +00:00
if ( cred - > principal_obtained > = CRED_SPECIFIED ) {
2007-10-06 22:16:19 +00:00
return true ;
2005-08-29 04:30:22 +00:00
}
if ( cred - > username_obtained > = CRED_SPECIFIED ) {
2007-10-06 22:16:19 +00:00
return true ;
2005-08-29 04:30:22 +00:00
}
2007-02-19 16:43:56 +00:00
2020-08-20 09:40:41 +02:00
if ( cli_credentials_get_kerberos_state ( cred ) = = CRED_USE_KERBEROS_REQUIRED ) {
2007-10-06 22:16:19 +00:00
return true ;
2007-02-19 16:43:56 +00:00
}
2017-08-29 15:24:14 +02:00
gensec_features = cli_credentials_get_gensec_features ( cred ) ;
if ( gensec_features & GENSEC_FEATURE_NTLM_CCACHE ) {
return true ;
}
if ( gensec_features & GENSEC_FEATURE_SIGN ) {
return true ;
}
if ( gensec_features & GENSEC_FEATURE_SEAL ) {
return true ;
}
2007-10-06 22:16:19 +00:00
return false ;
2005-08-29 04:30:22 +00:00
}
2005-03-27 06:55:03 +00:00
/**
* Obtain the password for this credentials context .
* @ param cred credentials context
* @ retval If set , the cleartext password , otherwise NULL
*/
2008-04-02 04:53:27 +02:00
_PUBLIC_ const char * cli_credentials_get_password ( struct cli_credentials * cred )
2005-03-21 02:08:38 +00:00
{
2005-03-24 03:32:25 +00:00
if ( cred - > machine_account_pending ) {
2007-12-13 22:46:17 +01:00
cli_credentials_set_machine_account ( cred ,
cred - > machine_account_pending_lp_ctx ) ;
2005-03-24 03:32:25 +00:00
}
2021-11-10 12:01:20 +01:00
if ( cred - > password_obtained = = CRED_CALLBACK & &
2016-12-14 10:02:10 +01:00
! cred - > callback_running & &
! cred - > password_will_be_nt_hash ) {
2013-09-21 01:49:08 +02:00
cred - > callback_running = true ;
2006-05-07 18:08:57 +00:00
cred - > password = cred - > password_cb ( cred ) ;
2013-09-21 01:49:08 +02:00
cred - > callback_running = false ;
2013-07-31 14:32:36 +02:00
if ( cred - > password_obtained = = CRED_CALLBACK ) {
cred - > password_obtained = CRED_CALLBACK_RESULT ;
cli_credentials_invalidate_ccache ( cred , cred - > password_obtained ) ;
}
2005-03-21 02:08:38 +00:00
}
return cred - > password ;
}
2022-04-14 13:46:18 +02:00
/**
* @ brief Find out how the password was obtained .
*
* @ param cred A credentials context .
*
* @ return The obtained information for the password .
*/
_PUBLIC_ enum credentials_obtained
cli_credentials_get_password_obtained ( struct cli_credentials * cred )
{
return cred - > password_obtained ;
}
2020-09-01 15:33:18 +02:00
/**
* @ brief Obtain the password for this credentials context .
*
* @ param [ in ] cred The credential context .
*
* @ param [ in ] obtained A pointer to store the obtained information .
*
2023-03-14 08:50:34 +01:00
* return The user name or NULL if an error occurred .
2020-09-01 15:33:18 +02:00
*/
_PUBLIC_ const char *
cli_credentials_get_password_and_obtained ( struct cli_credentials * cred ,
enum credentials_obtained * obtained )
{
2023-12-06 13:06:42 +01:00
const char * password = cli_credentials_get_password ( cred ) ;
2020-09-01 15:33:18 +02:00
if ( obtained ! = NULL ) {
* obtained = cred - > password_obtained ;
}
2023-12-06 13:06:42 +01:00
return password ;
2020-09-01 15:33:18 +02:00
}
r11995: A big kerberos-related update.
This merges Samba4 up to current lorikeet-heimdal, which includes a
replacement for some Samba-specific hacks.
In particular, the credentials system now supplies GSS client and
server credentials. These are imported into GSS with
gss_krb5_import_creds(). Unfortunetly this can't take an MEMORY
keytab, so we now create a FILE based keytab as provision and join
time.
Because the keytab is now created in advance, we don't spend .4s at
negprot doing sha1 s2k calls. Also, because the keytab is read in
real time, any change in the server key will be correctly picked up by
the the krb5 code.
To mark entries in the secrets which should be exported to a keytab,
there is a new kerberosSecret objectClass. The new routine
cli_credentials_update_all_keytabs() searches for these, and updates
the keytabs.
This is called in the provision.js via the ejs wrapper
credentials_update_all_keytabs().
We can now (in theory) use a system-provided /etc/krb5.keytab, if
krb5Keytab: FILE:/etc/krb5.keytab
is added to the secrets.ldb record. By default the attribute
privateKeytab: secrets.keytab
is set, pointing to allow the whole private directory to be moved
without breaking the internal links.
(This used to be commit 6b75573df49c6210e1b9d71e108a9490976bd41d)
2005-12-01 05:20:39 +00:00
/* Set a password on the credentials context, including an indication
* of ' how ' the password was obtained */
2021-11-10 12:01:20 +01:00
_PUBLIC_ bool cli_credentials_set_password ( struct cli_credentials * cred ,
const char * val ,
2005-08-30 01:19:41 +00:00
enum credentials_obtained obtained )
2005-03-21 02:08:38 +00:00
{
if ( obtained > = cred - > password_obtained ) {
2016-12-14 08:52:12 +01:00
cred - > lm_response = data_blob_null ;
cred - > nt_response = data_blob_null ;
cred - > nt_hash = NULL ;
cred - > password = NULL ;
cli_credentials_invalidate_ccache ( cred , obtained ) ;
2013-07-31 14:25:54 +02:00
cred - > password_tries = 0 ;
2016-12-14 08:52:12 +01:00
if ( val = = NULL ) {
cred - > password_obtained = obtained ;
return true ;
}
2016-12-14 10:02:10 +01:00
if ( cred - > password_will_be_nt_hash ) {
struct samr_Password * nt_hash = NULL ;
size_t val_len = strlen ( val ) ;
size_t converted ;
nt_hash = talloc ( cred , struct samr_Password ) ;
if ( nt_hash = = NULL ) {
return false ;
}
2024-05-24 17:16:14 +02:00
talloc_keep_secret ( nt_hash ) ;
2016-12-14 10:02:10 +01:00
converted = strhex_to_str ( ( char * ) nt_hash - > hash ,
sizeof ( nt_hash - > hash ) ,
val , val_len ) ;
if ( converted ! = sizeof ( nt_hash - > hash ) ) {
TALLOC_FREE ( nt_hash ) ;
return false ;
}
cred - > nt_hash = nt_hash ;
cred - > password_obtained = obtained ;
return true ;
}
2005-03-21 02:08:38 +00:00
cred - > password = talloc_strdup ( cred , val ) ;
2016-12-14 08:52:12 +01:00
if ( cred - > password = = NULL ) {
return false ;
2012-08-31 11:19:54 +10:00
}
2016-12-14 08:52:12 +01:00
/* Don't print the actual password in talloc memory dumps */
talloc_set_name_const ( cred - > password ,
" password set via cli_credentials_set_password " ) ;
2005-03-21 02:08:38 +00:00
cred - > password_obtained = obtained ;
2005-06-17 13:12:13 +00:00
2007-10-06 22:16:19 +00:00
return true ;
2005-06-17 13:12:13 +00:00
}
2007-10-06 22:16:19 +00:00
return false ;
2005-06-17 13:12:13 +00:00
}
2008-04-02 04:53:27 +02:00
_PUBLIC_ bool cli_credentials_set_password_callback ( struct cli_credentials * cred ,
2005-08-30 01:19:41 +00:00
const char * ( * password_cb ) ( struct cli_credentials * ) )
{
if ( cred - > password_obtained < CRED_CALLBACK ) {
2013-07-31 14:25:54 +02:00
cred - > password_tries = 3 ;
2005-08-30 01:19:41 +00:00
cred - > password_cb = password_cb ;
cred - > password_obtained = CRED_CALLBACK ;
2007-05-22 05:21:59 +00:00
cli_credentials_invalidate_ccache ( cred , cred - > password_obtained ) ;
2007-10-06 22:16:19 +00:00
return true ;
2005-08-30 01:19:41 +00:00
}
2007-10-06 22:16:19 +00:00
return false ;
2005-08-30 01:19:41 +00:00
}
2005-10-20 03:47:55 +00:00
/**
* Obtain the ' old ' password for this credentials context ( used for join accounts ) .
* @ param cred credentials context
* @ retval If set , the cleartext password , otherwise NULL
*/
2011-10-09 23:27:44 +11:00
_PUBLIC_ const char * cli_credentials_get_old_password ( struct cli_credentials * cred )
2005-10-20 03:47:55 +00:00
{
if ( cred - > machine_account_pending ) {
2007-12-13 22:46:17 +01:00
cli_credentials_set_machine_account ( cred ,
cred - > machine_account_pending_lp_ctx ) ;
2005-10-20 03:47:55 +00:00
}
return cred - > old_password ;
}
2021-11-10 12:01:20 +01:00
_PUBLIC_ bool cli_credentials_set_old_password ( struct cli_credentials * cred ,
const char * val ,
2005-10-20 03:47:55 +00:00
enum credentials_obtained obtained )
{
cred - > old_password = talloc_strdup ( cred , val ) ;
2012-08-31 11:19:54 +10:00
if ( cred - > old_password ) {
/* Don't print the actual password in talloc memory dumps */
talloc_set_name_const ( cred - > old_password , " password set via cli_credentials_set_old_password " ) ;
}
2015-02-09 09:04:42 +01:00
cred - > old_nt_hash = NULL ;
2007-10-06 22:16:19 +00:00
return true ;
2005-10-20 03:47:55 +00:00
}
2005-06-17 13:12:13 +00:00
/**
r11995: A big kerberos-related update.
This merges Samba4 up to current lorikeet-heimdal, which includes a
replacement for some Samba-specific hacks.
In particular, the credentials system now supplies GSS client and
server credentials. These are imported into GSS with
gss_krb5_import_creds(). Unfortunetly this can't take an MEMORY
keytab, so we now create a FILE based keytab as provision and join
time.
Because the keytab is now created in advance, we don't spend .4s at
negprot doing sha1 s2k calls. Also, because the keytab is read in
real time, any change in the server key will be correctly picked up by
the the krb5 code.
To mark entries in the secrets which should be exported to a keytab,
there is a new kerberosSecret objectClass. The new routine
cli_credentials_update_all_keytabs() searches for these, and updates
the keytabs.
This is called in the provision.js via the ejs wrapper
credentials_update_all_keytabs().
We can now (in theory) use a system-provided /etc/krb5.keytab, if
krb5Keytab: FILE:/etc/krb5.keytab
is added to the secrets.ldb record. By default the attribute
privateKeytab: secrets.keytab
is set, pointing to allow the whole private directory to be moved
without breaking the internal links.
(This used to be commit 6b75573df49c6210e1b9d71e108a9490976bd41d)
2005-12-01 05:20:39 +00:00
* Obtain the password , in the form MD4 ( unicode ( password ) ) for this credentials context .
*
* Sometimes we only have this much of the password , while the rest of
* the time this call avoids calling E_md4hash themselves .
*
2005-06-17 13:12:13 +00:00
* @ param cred credentials context
* @ retval If set , the cleartext password , otherwise NULL
*/
2013-08-09 10:15:05 +02:00
_PUBLIC_ struct samr_Password * cli_credentials_get_nt_hash ( struct cli_credentials * cred ,
TALLOC_CTX * mem_ctx )
2005-06-17 13:12:13 +00:00
{
2016-12-14 10:02:10 +01:00
enum credentials_obtained password_obtained ;
enum credentials_obtained ccache_threshold ;
enum credentials_obtained client_gss_creds_threshold ;
bool password_is_nt_hash ;
2014-12-16 13:58:11 +00:00
const char * password = NULL ;
2016-12-14 10:02:10 +01:00
struct samr_Password * nt_hash = NULL ;
2005-06-17 13:12:13 +00:00
2014-12-16 13:58:11 +00:00
if ( cred - > nt_hash ! = NULL ) {
2016-12-14 10:02:10 +01:00
/*
* If we already have a hash it ' s easy .
*/
goto return_hash ;
}
2013-08-09 10:15:05 +02:00
2016-12-14 10:02:10 +01:00
/*
* This is a bit tricky , with password_will_be_nt_hash
* we still need to get the value via the password_callback
* but if we did that we should not remember it ' s state
* in the long run so we need to undo it .
*/
2005-06-17 13:12:13 +00:00
2016-12-14 10:02:10 +01:00
password_obtained = cred - > password_obtained ;
ccache_threshold = cred - > ccache_threshold ;
client_gss_creds_threshold = cred - > client_gss_creds_threshold ;
password_is_nt_hash = cred - > password_will_be_nt_hash ;
2014-12-16 13:58:11 +00:00
2016-12-14 10:02:10 +01:00
cred - > password_will_be_nt_hash = false ;
2014-12-16 13:58:11 +00:00
password = cli_credentials_get_password ( cred ) ;
2013-08-09 10:15:05 +02:00
2016-12-14 10:02:10 +01:00
cred - > password_will_be_nt_hash = password_is_nt_hash ;
if ( password_is_nt_hash & & password_obtained = = CRED_CALLBACK ) {
/*
* We got the nt_hash as string via the callback ,
* so we need to undo the state change .
*
* And also don ' t remember it as plaintext password .
*/
cred - > client_gss_creds_threshold = client_gss_creds_threshold ;
cred - > ccache_threshold = ccache_threshold ;
cred - > password_obtained = password_obtained ;
cred - > password = NULL ;
}
if ( password = = NULL ) {
return NULL ;
}
nt_hash = talloc ( cred , struct samr_Password ) ;
if ( nt_hash = = NULL ) {
return NULL ;
}
2024-05-24 17:16:14 +02:00
talloc_keep_secret ( nt_hash ) ;
2016-12-14 10:02:10 +01:00
if ( password_is_nt_hash ) {
size_t password_len = strlen ( password ) ;
size_t converted ;
converted = strhex_to_str ( ( char * ) nt_hash - > hash ,
sizeof ( nt_hash - > hash ) ,
password , password_len ) ;
if ( converted ! = sizeof ( nt_hash - > hash ) ) {
TALLOC_FREE ( nt_hash ) ;
2017-09-05 16:43:18 +02:00
return NULL ;
2016-12-14 10:02:10 +01:00
}
} else {
2014-12-16 13:58:11 +00:00
E_md4hash ( password , nt_hash - > hash ) ;
2016-12-14 10:02:10 +01:00
}
cred - > nt_hash = nt_hash ;
nt_hash = NULL ;
2013-08-09 10:15:05 +02:00
2016-12-14 10:02:10 +01:00
return_hash :
nt_hash = talloc ( mem_ctx , struct samr_Password ) ;
if ( nt_hash = = NULL ) {
return NULL ;
2005-06-17 13:12:13 +00:00
}
2024-05-24 17:16:14 +02:00
talloc_keep_secret ( nt_hash ) ;
2013-08-09 10:15:05 +02:00
2016-12-14 10:02:10 +01:00
* nt_hash = * cred - > nt_hash ;
return nt_hash ;
2005-06-17 13:12:13 +00:00
}
2015-02-09 09:04:42 +01:00
/**
* Obtain the old password , in the form MD4 ( unicode ( password ) ) for this credentials context .
*
* Sometimes we only have this much of the password , while the rest of
* the time this call avoids calling E_md4hash themselves .
*
* @ param cred credentials context
* @ retval If set , the cleartext password , otherwise NULL
*/
_PUBLIC_ struct samr_Password * cli_credentials_get_old_nt_hash ( struct cli_credentials * cred ,
TALLOC_CTX * mem_ctx )
{
const char * old_password = NULL ;
if ( cred - > old_nt_hash ! = NULL ) {
struct samr_Password * nt_hash = talloc ( mem_ctx , struct samr_Password ) ;
if ( ! nt_hash ) {
return NULL ;
}
2024-05-24 17:16:14 +02:00
talloc_keep_secret ( nt_hash ) ;
2015-02-09 09:04:42 +01:00
* nt_hash = * cred - > old_nt_hash ;
return nt_hash ;
}
old_password = cli_credentials_get_old_password ( cred ) ;
if ( old_password ) {
struct samr_Password * nt_hash = talloc ( mem_ctx , struct samr_Password ) ;
if ( ! nt_hash ) {
return NULL ;
}
2024-05-24 17:16:14 +02:00
talloc_keep_secret ( nt_hash ) ;
2015-02-09 09:04:42 +01:00
E_md4hash ( old_password , nt_hash - > hash ) ;
return nt_hash ;
}
return NULL ;
}
2005-03-27 06:55:03 +00:00
/**
* Obtain the ' short ' or ' NetBIOS ' domain for this credentials context .
* @ param cred credentials context
2021-11-10 12:01:20 +01:00
* @ retval The domain set on this context .
2005-03-27 06:55:03 +00:00
* @ note Return value will never be NULL except by programmer error .
*/
2008-04-02 04:53:27 +02:00
_PUBLIC_ const char * cli_credentials_get_domain ( struct cli_credentials * cred )
2005-03-21 02:08:38 +00:00
{
2005-03-24 03:32:25 +00:00
if ( cred - > machine_account_pending ) {
2007-12-13 22:46:17 +01:00
cli_credentials_set_machine_account ( cred ,
cred - > machine_account_pending_lp_ctx ) ;
2005-03-24 03:32:25 +00:00
}
2021-11-10 12:01:20 +01:00
if ( cred - > domain_obtained = = CRED_CALLBACK & &
2006-05-03 20:56:14 +00:00
! cred - > callback_running ) {
2007-10-06 22:16:19 +00:00
cred - > callback_running = true ;
2005-03-21 02:08:38 +00:00
cred - > domain = cred - > domain_cb ( cred ) ;
2007-10-06 22:16:19 +00:00
cred - > callback_running = false ;
2013-07-31 14:32:36 +02:00
if ( cred - > domain_obtained = = CRED_CALLBACK ) {
cred - > domain_obtained = CRED_CALLBACK_RESULT ;
cli_credentials_invalidate_ccache ( cred , cred - > domain_obtained ) ;
}
2005-03-21 02:08:38 +00:00
}
return cred - > domain ;
}
2023-12-06 13:26:43 +01:00
/**
* @ brief Obtain the domain for this credential context .
*
* @ param [ in ] cred The credential context .
*
* @ param [ out ] obtained A pointer to store the obtained information .
*
* @ return The domain name or NULL if an error occurred .
*/
_PUBLIC_ const char * cli_credentials_get_domain_and_obtained (
struct cli_credentials * cred ,
enum credentials_obtained * obtained )
{
const char * domain = cli_credentials_get_domain ( cred ) ;
if ( obtained ! = NULL ) {
* obtained = cred - > domain_obtained ;
}
return domain ;
}
2005-03-21 02:08:38 +00:00
2021-11-10 12:01:20 +01:00
_PUBLIC_ bool cli_credentials_set_domain ( struct cli_credentials * cred ,
const char * val ,
2005-08-30 01:19:41 +00:00
enum credentials_obtained obtained )
2005-03-21 02:08:38 +00:00
{
if ( obtained > = cred - > domain_obtained ) {
2005-10-28 05:17:19 +00:00
/* it is important that the domain be in upper case,
* particularly for the sensitive NTLMv2
* calculations */
cred - > domain = strupper_talloc ( cred , val ) ;
2005-03-21 02:08:38 +00:00
cred - > domain_obtained = obtained ;
2021-11-10 12:01:20 +01:00
/* setting domain does not mean we have to invalidate ccache
2012-05-24 15:17:40 +03:00
* because domain in not used for Kerberos operations .
* If ccache invalidation is required , one will anyway specify
* a password to kinit , and that will force invalidation of the ccache
*/
2007-10-06 22:16:19 +00:00
return true ;
2005-03-21 02:08:38 +00:00
}
2007-10-06 22:16:19 +00:00
return false ;
2005-03-21 02:08:38 +00:00
}
2007-10-06 22:16:19 +00:00
bool cli_credentials_set_domain_callback ( struct cli_credentials * cred ,
2005-08-30 01:19:41 +00:00
const char * ( * domain_cb ) ( struct cli_credentials * ) )
{
if ( cred - > domain_obtained < CRED_CALLBACK ) {
cred - > domain_cb = domain_cb ;
cred - > domain_obtained = CRED_CALLBACK ;
2007-10-06 22:16:19 +00:00
return true ;
2005-08-30 01:19:41 +00:00
}
2007-10-06 22:16:19 +00:00
return false ;
2005-08-30 01:19:41 +00:00
}
2005-03-27 06:55:03 +00:00
/**
* Obtain the Kerberos realm for this credentials context .
* @ param cred credentials context
2021-11-10 12:01:20 +01:00
* @ retval The realm set on this context .
2005-03-27 06:55:03 +00:00
* @ note Return value will never be NULL except by programmer error .
*/
2008-04-02 04:53:27 +02:00
_PUBLIC_ const char * cli_credentials_get_realm ( struct cli_credentials * cred )
2021-11-10 12:01:20 +01:00
{
2005-03-24 03:32:25 +00:00
if ( cred - > machine_account_pending ) {
2007-12-13 22:46:17 +01:00
cli_credentials_set_machine_account ( cred ,
cred - > machine_account_pending_lp_ctx ) ;
2005-03-21 21:22:07 +00:00
}
2021-11-10 12:01:20 +01:00
if ( cred - > realm_obtained = = CRED_CALLBACK & &
2006-05-03 20:56:14 +00:00
! cred - > callback_running ) {
2007-10-06 22:16:19 +00:00
cred - > callback_running = true ;
2005-03-21 02:08:38 +00:00
cred - > realm = cred - > realm_cb ( cred ) ;
2007-10-06 22:16:19 +00:00
cred - > callback_running = false ;
2013-07-31 14:32:36 +02:00
if ( cred - > realm_obtained = = CRED_CALLBACK ) {
cred - > realm_obtained = CRED_CALLBACK_RESULT ;
cli_credentials_invalidate_ccache ( cred , cred - > realm_obtained ) ;
}
2005-03-21 02:08:38 +00:00
}
return cred - > realm ;
}
2005-05-10 09:51:55 +00:00
/**
* Set the realm for this credentials context , and force it to
2017-09-06 14:05:09 +02:00
* uppercase for the sanity of our local kerberos libraries
2005-05-10 09:51:55 +00:00
*/
2021-11-10 12:01:20 +01:00
_PUBLIC_ bool cli_credentials_set_realm ( struct cli_credentials * cred ,
const char * val ,
2005-08-30 01:19:41 +00:00
enum credentials_obtained obtained )
2005-03-21 02:08:38 +00:00
{
if ( obtained > = cred - > realm_obtained ) {
2005-05-10 09:51:55 +00:00
cred - > realm = strupper_talloc ( cred , val ) ;
2005-03-21 02:08:38 +00:00
cred - > realm_obtained = obtained ;
2007-05-22 05:21:59 +00:00
cli_credentials_invalidate_ccache ( cred , cred - > realm_obtained ) ;
2007-10-06 22:16:19 +00:00
return true ;
2005-03-21 02:08:38 +00:00
}
2007-10-06 22:16:19 +00:00
return false ;
2005-03-21 02:08:38 +00:00
}
2007-10-06 22:16:19 +00:00
bool cli_credentials_set_realm_callback ( struct cli_credentials * cred ,
2005-08-30 01:19:41 +00:00
const char * ( * realm_cb ) ( struct cli_credentials * ) )
{
if ( cred - > realm_obtained < CRED_CALLBACK ) {
cred - > realm_cb = realm_cb ;
cred - > realm_obtained = CRED_CALLBACK ;
2007-10-06 22:16:19 +00:00
return true ;
2005-08-30 01:19:41 +00:00
}
2007-10-06 22:16:19 +00:00
return false ;
2005-08-30 01:19:41 +00:00
}
2005-03-27 06:55:03 +00:00
/**
* Obtain the ' short ' or ' NetBIOS ' workstation name for this credentials context .
*
* @ param cred credentials context
2021-11-10 12:01:20 +01:00
* @ retval The workstation name set on this context .
2005-03-27 06:55:03 +00:00
* @ note Return value will never be NULL except by programmer error .
*/
2008-04-02 04:53:27 +02:00
_PUBLIC_ const char * cli_credentials_get_workstation ( struct cli_credentials * cred )
2005-03-21 02:08:38 +00:00
{
2021-11-10 12:01:20 +01:00
if ( cred - > workstation_obtained = = CRED_CALLBACK & &
2006-05-03 20:56:14 +00:00
! cred - > callback_running ) {
2007-10-06 22:16:19 +00:00
cred - > callback_running = true ;
2005-03-21 02:08:38 +00:00
cred - > workstation = cred - > workstation_cb ( cred ) ;
2007-10-06 22:16:19 +00:00
cred - > callback_running = false ;
2013-07-31 14:32:36 +02:00
if ( cred - > workstation_obtained = = CRED_CALLBACK ) {
cred - > workstation_obtained = CRED_CALLBACK_RESULT ;
}
2005-03-21 02:08:38 +00:00
}
return cred - > workstation ;
}
2021-11-10 12:01:20 +01:00
_PUBLIC_ bool cli_credentials_set_workstation ( struct cli_credentials * cred ,
const char * val ,
2005-08-30 01:19:41 +00:00
enum credentials_obtained obtained )
2005-03-21 02:08:38 +00:00
{
if ( obtained > = cred - > workstation_obtained ) {
cred - > workstation = talloc_strdup ( cred , val ) ;
cred - > workstation_obtained = obtained ;
2007-10-06 22:16:19 +00:00
return true ;
2005-03-21 02:08:38 +00:00
}
2007-10-06 22:16:19 +00:00
return false ;
2005-03-21 02:08:38 +00:00
}
2007-10-06 22:16:19 +00:00
bool cli_credentials_set_workstation_callback ( struct cli_credentials * cred ,
2005-08-30 01:19:41 +00:00
const char * ( * workstation_cb ) ( struct cli_credentials * ) )
{
if ( cred - > workstation_obtained < CRED_CALLBACK ) {
cred - > workstation_cb = workstation_cb ;
cred - > workstation_obtained = CRED_CALLBACK ;
2007-10-06 22:16:19 +00:00
return true ;
2005-08-30 01:19:41 +00:00
}
2007-10-06 22:16:19 +00:00
return false ;
2005-08-30 01:19:41 +00:00
}
2005-03-27 06:55:03 +00:00
/**
* Given a string , typically obtained from a - U argument , parse it into domain , username , realm and password fields
*
* The format accepted is [ domain \ \ ] user [ % password ] or user [ @ realm ] [ % password ]
*
* @ param credentials Credentials structure on which to set the password
* @ param data the string containing the username , password etc
* @ param obtained This enum describes how ' specified ' this password is
*/
2008-04-02 04:53:27 +02:00
_PUBLIC_ void cli_credentials_parse_string ( struct cli_credentials * credentials , const char * data , enum credentials_obtained obtained )
2005-03-21 02:08:38 +00:00
{
char * uname , * p ;
2021-03-22 18:11:33 +01:00
char * uname_free = NULL ;
2005-03-21 02:08:38 +00:00
2005-04-26 07:10:43 +00:00
if ( strcmp ( " % " , data ) = = 0 ) {
cli_credentials_set_anonymous ( credentials ) ;
return ;
}
2021-11-10 12:01:20 +01:00
uname = talloc_strdup ( credentials , data ) ;
2021-03-22 18:11:33 +01:00
uname_free = uname ;
2005-03-22 08:00:45 +00:00
if ( ( p = strchr_m ( uname , ' % ' ) ) ) {
2005-03-21 02:08:38 +00:00
* p = 0 ;
2005-03-22 08:00:45 +00:00
cli_credentials_set_password ( credentials , p + 1 , obtained ) ;
2005-03-21 02:08:38 +00:00
}
if ( ( p = strchr_m ( uname , ' @ ' ) ) ) {
2016-12-09 12:20:19 +01:00
/*
* We also need to set username and domain
* in order to undo the effect of
* cli_credentials_guess ( ) .
*/
2020-10-07 19:25:24 +03:00
cli_credentials_set_username ( credentials , uname , obtained ) ;
cli_credentials_set_domain ( credentials , " " , obtained ) ;
2020-11-04 19:20:58 +02:00
cli_credentials_set_principal ( credentials , uname , obtained ) ;
* p = 0 ;
2005-03-21 02:08:38 +00:00
cli_credentials_set_realm ( credentials , p + 1 , obtained ) ;
2021-03-22 18:11:33 +01:00
TALLOC_FREE ( uname_free ) ;
2005-08-29 04:30:22 +00:00
return ;
2016-10-28 15:55:48 +02:00
} else if ( ( p = strchr_m ( uname , ' \\ ' ) )
| | ( p = strchr_m ( uname , ' / ' ) )
| | ( p = strchr_m ( uname , credentials - > winbind_separator ) ) )
{
2016-12-14 16:47:57 +01:00
const char * domain = NULL ;
domain = uname ;
2005-03-21 02:08:38 +00:00
* p = 0 ;
2005-03-22 08:00:45 +00:00
uname = p + 1 ;
2016-12-14 16:47:57 +01:00
if ( obtained = = credentials - > realm_obtained & &
! strequal_m ( credentials - > domain , domain ) )
{
/*
* We need to undo a former set with the same level
* in order to get the expected result from
* cli_credentials_get_principal ( ) .
*
* But we only need to do that if the domain
* actually changes .
*/
cli_credentials_set_realm ( credentials , domain , obtained ) ;
}
cli_credentials_set_domain ( credentials , domain , obtained ) ;
}
if ( obtained = = credentials - > principal_obtained & &
! strequal_m ( credentials - > username , uname ) )
{
/*
* We need to undo a former set with the same level
* in order to get the expected result from
* cli_credentials_get_principal ( ) .
*
* But we only need to do that if the username
* actually changes .
*/
credentials - > principal_obtained = CRED_UNINITIALISED ;
credentials - > principal = NULL ;
2005-03-21 02:08:38 +00:00
}
2005-03-22 08:00:45 +00:00
cli_credentials_set_username ( credentials , uname , obtained ) ;
2021-03-22 18:11:33 +01:00
TALLOC_FREE ( uname_free ) ;
2005-03-21 02:08:38 +00:00
}
2006-05-03 20:23:19 +00:00
/**
2006-05-03 20:31:24 +00:00
* Given a a credentials structure , print it as a string
2006-05-03 20:23:19 +00:00
*
2006-05-03 20:31:24 +00:00
* The format output is [ domain \ \ ] user [ % password ] or user [ @ realm ] [ % password ]
2006-05-03 20:23:19 +00:00
*
* @ param credentials Credentials structure on which to set the password
2006-05-03 20:31:24 +00:00
* @ param mem_ctx The memory context to place the result on
2006-05-03 20:23:19 +00:00
*/
2016-12-29 15:26:00 +01:00
_PUBLIC_ char * cli_credentials_get_unparsed_name ( struct cli_credentials * credentials , TALLOC_CTX * mem_ctx )
2006-05-03 20:23:19 +00:00
{
const char * bind_dn = cli_credentials_get_bind_dn ( credentials ) ;
2016-12-29 15:26:00 +01:00
const char * domain = NULL ;
const char * username = NULL ;
char * name = NULL ;
2006-05-03 20:23:19 +00:00
if ( bind_dn ) {
2013-07-31 12:33:25 +02:00
name = talloc_strdup ( mem_ctx , bind_dn ) ;
2006-05-03 20:23:19 +00:00
} else {
cli_credentials_get_ntlm_username_domain ( credentials , mem_ctx , & username , & domain ) ;
if ( domain & & domain [ 0 ] ) {
2021-11-10 12:01:20 +01:00
name = talloc_asprintf ( mem_ctx , " %s \\ %s " ,
2006-05-03 20:23:19 +00:00
domain , username ) ;
} else {
2021-11-10 12:01:20 +01:00
name = talloc_asprintf ( mem_ctx , " %s " ,
2006-05-03 20:23:19 +00:00
username ) ;
}
}
return name ;
}
2021-04-01 17:02:10 +02:00
2005-03-27 06:55:03 +00:00
/**
* Specifies default values for domain , workstation and realm
* from the smb . conf configuration file
*
* @ param cred Credentials structure to fill in
2021-04-01 17:02:10 +02:00
*
* @ return true on success , false on error .
2005-03-27 06:55:03 +00:00
*/
2021-04-01 17:02:10 +02:00
_PUBLIC_ bool cli_credentials_set_conf ( struct cli_credentials * cred ,
struct loadparm_context * lp_ctx )
2005-03-24 03:32:25 +00:00
{
2016-10-28 15:55:48 +02:00
const char * sep = NULL ;
2016-12-15 11:04:02 +01:00
const char * realm = lpcfg_realm ( lp_ctx ) ;
2020-08-20 16:44:32 +02:00
enum credentials_client_protection protection =
lpcfg_client_protection ( lp_ctx ) ;
2021-04-01 17:02:10 +02:00
const char * workgroup = lpcfg_workgroup ( lp_ctx ) ;
const char * netbios_name = lpcfg_netbios_name ( lp_ctx ) ;
bool ok ;
2016-10-28 15:55:48 +02:00
2021-04-01 17:02:10 +02:00
( void ) cli_credentials_set_username ( cred , " " , CRED_UNINITIALISED ) ;
if ( workgroup ! = NULL & & strlen ( workgroup ) = = 0 ) {
workgroup = NULL ;
2011-10-18 08:41:46 +11:00
}
2021-04-01 17:02:10 +02:00
if ( workgroup ! = NULL ) {
if ( lpcfg_parm_is_cmdline ( lp_ctx , " workgroup " ) ) {
ok = cli_credentials_set_domain ( cred ,
workgroup ,
CRED_SPECIFIED ) ;
if ( ! ok ) {
DBG_ERR ( " Failed to set domain! \n " ) ;
return false ;
}
} else {
( void ) cli_credentials_set_domain ( cred ,
workgroup ,
CRED_SMB_CONF ) ;
}
}
if ( netbios_name ! = NULL & & strlen ( netbios_name ) = = 0 ) {
netbios_name = NULL ;
2011-10-18 08:41:46 +11:00
}
2021-04-01 17:02:10 +02:00
if ( netbios_name ! = NULL ) {
if ( lpcfg_parm_is_cmdline ( lp_ctx , " netbios name " ) ) {
ok = cli_credentials_set_workstation ( cred ,
netbios_name ,
CRED_SPECIFIED ) ;
if ( ! ok ) {
DBG_ERR ( " Failed to set workstation! \n " ) ;
return false ;
}
} else {
( void ) cli_credentials_set_workstation ( cred ,
netbios_name ,
CRED_SMB_CONF ) ;
}
}
2016-12-15 11:04:02 +01:00
if ( realm ! = NULL & & strlen ( realm ) = = 0 ) {
realm = NULL ;
}
2021-04-01 17:02:10 +02:00
if ( realm ! = NULL ) {
if ( lpcfg_parm_is_cmdline ( lp_ctx , " realm " ) ) {
ok = cli_credentials_set_realm ( cred ,
realm ,
CRED_SPECIFIED ) ;
if ( ! ok ) {
DBG_ERR ( " Failed to set realm! \n " ) ;
return false ;
}
} else {
( void ) cli_credentials_set_realm ( cred ,
realm ,
CRED_SMB_CONF ) ;
}
2011-10-18 08:41:46 +11:00
}
2016-10-28 15:55:48 +02:00
sep = lpcfg_winbind_separator ( lp_ctx ) ;
if ( sep ! = NULL & & sep [ 0 ] ! = ' \0 ' ) {
cred - > winbind_separator = * lpcfg_winbind_separator ( lp_ctx ) ;
}
2020-05-26 09:32:44 +02:00
if ( cred - > signing_state_obtained < = CRED_SMB_CONF ) {
/* Will be set to default for invalid smb.conf values */
cred - > signing_state = lpcfg_client_signing ( lp_ctx ) ;
2020-08-20 16:44:32 +02:00
if ( cred - > signing_state = = SMB_SIGNING_DEFAULT ) {
switch ( protection ) {
case CRED_CLIENT_PROTECTION_DEFAULT :
break ;
case CRED_CLIENT_PROTECTION_PLAIN :
cred - > signing_state = SMB_SIGNING_OFF ;
break ;
case CRED_CLIENT_PROTECTION_SIGN :
case CRED_CLIENT_PROTECTION_ENCRYPT :
cred - > signing_state = SMB_SIGNING_REQUIRED ;
break ;
}
}
2020-05-26 09:32:44 +02:00
cred - > signing_state_obtained = CRED_SMB_CONF ;
}
2020-05-28 16:31:35 +02:00
if ( cred - > ipc_signing_state_obtained < = CRED_SMB_CONF ) {
/* Will be set to required for invalid smb.conf values */
cred - > ipc_signing_state = lpcfg_client_ipc_signing ( lp_ctx ) ;
cred - > ipc_signing_state_obtained = CRED_SMB_CONF ;
}
2020-05-28 16:10:52 +02:00
if ( cred - > encryption_state_obtained < = CRED_SMB_CONF ) {
/* Will be set to default for invalid smb.conf values */
cred - > encryption_state = lpcfg_client_smb_encrypt ( lp_ctx ) ;
2020-08-20 16:44:32 +02:00
if ( cred - > encryption_state = = SMB_ENCRYPTION_DEFAULT ) {
switch ( protection ) {
case CRED_CLIENT_PROTECTION_DEFAULT :
break ;
case CRED_CLIENT_PROTECTION_PLAIN :
case CRED_CLIENT_PROTECTION_SIGN :
cred - > encryption_state = SMB_ENCRYPTION_OFF ;
break ;
case CRED_CLIENT_PROTECTION_ENCRYPT :
cred - > encryption_state = SMB_ENCRYPTION_REQUIRED ;
break ;
}
}
2020-05-28 16:10:52 +02:00
}
2020-08-19 15:46:11 +02:00
if ( cred - > kerberos_state_obtained < = CRED_SMB_CONF ) {
/* Will be set to default for invalid smb.conf values */
cred - > kerberos_state = lpcfg_client_use_kerberos ( lp_ctx ) ;
cred - > kerberos_state_obtained = CRED_SMB_CONF ;
}
2020-08-20 10:50:30 +02:00
if ( cred - > gensec_features_obtained < = CRED_SMB_CONF ) {
switch ( protection ) {
case CRED_CLIENT_PROTECTION_DEFAULT :
break ;
case CRED_CLIENT_PROTECTION_PLAIN :
cred - > gensec_features = 0 ;
break ;
case CRED_CLIENT_PROTECTION_SIGN :
cred - > gensec_features = GENSEC_FEATURE_SIGN ;
break ;
case CRED_CLIENT_PROTECTION_ENCRYPT :
cred - > gensec_features =
GENSEC_FEATURE_SIGN | GENSEC_FEATURE_SEAL ;
break ;
}
cred - > gensec_features_obtained = CRED_SMB_CONF ;
}
2021-04-01 17:02:10 +02:00
return true ;
2005-03-24 03:32:25 +00:00
}
2005-03-27 06:55:03 +00:00
/**
2021-11-10 12:01:20 +01:00
* Guess defaults for credentials from environment variables ,
2005-03-27 06:55:03 +00:00
* and from the configuration file
2021-11-10 12:01:20 +01:00
*
2005-03-27 06:55:03 +00:00
* @ param cred Credentials structure to fill in
*/
2021-04-27 16:15:30 +02:00
_PUBLIC_ bool cli_credentials_guess ( struct cli_credentials * cred ,
struct loadparm_context * lp_ctx )
2005-03-21 02:08:38 +00:00
{
2010-02-25 16:16:33 +11:00
const char * error_string ;
2021-04-27 16:11:48 +02:00
const char * env = NULL ;
2021-11-10 12:06:51 +01:00
struct passwd * pwd = NULL ;
2021-04-27 16:15:30 +02:00
bool ok ;
2005-03-21 02:08:38 +00:00
2007-12-03 23:33:09 +01:00
if ( lp_ctx ! = NULL ) {
2021-04-27 16:15:30 +02:00
ok = cli_credentials_set_conf ( cred , lp_ctx ) ;
if ( ! ok ) {
return false ;
}
2007-12-03 23:33:09 +01:00
}
2021-04-27 16:11:48 +02:00
2021-11-10 12:06:51 +01:00
pwd = getpwuid ( getuid ( ) ) ;
if ( pwd ! = NULL ) {
size_t len = strlen ( pwd - > pw_name ) ;
if ( len > 0 & & len < = 1024 ) {
( void ) cli_credentials_parse_string ( cred ,
pwd - > pw_name ,
CRED_GUESS_ENV ) ;
}
}
2021-04-27 16:11:48 +02:00
env = getenv ( " LOGNAME " ) ;
if ( env ! = NULL ) {
size_t len = strlen ( env ) ;
if ( len > 0 & & len < = 1024 ) {
2021-04-27 16:15:30 +02:00
( void ) cli_credentials_set_username ( cred ,
env ,
CRED_GUESS_ENV ) ;
2021-04-27 16:11:48 +02:00
}
2005-03-21 02:08:38 +00:00
}
2021-04-27 16:11:48 +02:00
env = getenv ( " USER " ) ;
if ( env ! = NULL ) {
size_t len = strlen ( env ) ;
if ( len > 0 & & len < = 1024 ) {
char * p = NULL ;
2021-04-27 16:15:30 +02:00
( void ) cli_credentials_parse_string ( cred ,
env ,
CRED_GUESS_ENV ) ;
2021-04-27 16:11:48 +02:00
if ( ( p = strchr_m ( env , ' % ' ) ) ) {
memset ( p , ' \0 ' , strlen ( cred - > password ) ) ;
}
2005-03-21 02:08:38 +00:00
}
}
2021-04-27 16:11:48 +02:00
env = getenv ( " PASSWD " ) ;
if ( env ! = NULL ) {
size_t len = strlen ( env ) ;
if ( len > 0 & & len < = 1024 ) {
2021-04-27 16:15:30 +02:00
( void ) cli_credentials_set_password ( cred ,
env ,
CRED_GUESS_ENV ) ;
2021-04-27 16:11:48 +02:00
}
2005-03-21 02:08:38 +00:00
}
2021-04-27 16:15:30 +02:00
env = getenv ( " PASSWD_FD " ) ;
2021-04-27 16:11:48 +02:00
if ( env ! = NULL ) {
size_t len = strlen ( env ) ;
if ( len > 0 & & len < = 1024 ) {
int fd = atoi ( env ) ;
2021-04-27 16:15:30 +02:00
( void ) cli_credentials_parse_password_fd ( cred ,
fd ,
CRED_GUESS_FILE ) ;
2021-04-27 16:11:48 +02:00
}
2005-03-21 02:08:38 +00:00
}
2021-04-27 16:11:48 +02:00
env = getenv ( " PASSWD_FILE " ) ;
if ( env ! = NULL ) {
size_t len = strlen ( env ) ;
if ( len > 0 & & len < = 4096 ) {
2021-04-27 16:15:30 +02:00
( void ) cli_credentials_parse_password_file ( cred ,
env ,
CRED_GUESS_FILE ) ;
2021-04-27 16:11:48 +02:00
}
2005-03-21 02:08:38 +00:00
}
2021-04-27 16:11:48 +02:00
2019-05-07 12:51:09 +12:00
if ( lp_ctx ! = NULL & &
2020-08-20 09:40:41 +02:00
cli_credentials_get_kerberos_state ( cred ) ! = CRED_USE_KERBEROS_DISABLED ) {
2021-04-27 16:15:30 +02:00
( void ) cli_credentials_set_ccache ( cred ,
lp_ctx ,
NULL ,
CRED_GUESS_FILE ,
& error_string ) ;
2006-05-03 20:23:19 +00:00
}
2021-04-27 16:15:30 +02:00
return true ;
2005-03-21 02:08:38 +00:00
}
2005-03-21 23:35:58 +00:00
2005-03-27 06:55:03 +00:00
/**
* Attach NETLOGON credentials for use with SCHANNEL
*/
2005-03-24 03:32:25 +00:00
2017-09-07 12:34:34 +02:00
_PUBLIC_ void cli_credentials_set_netlogon_creds (
struct cli_credentials * cred ,
const struct netlogon_creds_CredentialState * netlogon_creds )
2005-03-24 03:32:25 +00:00
{
2013-09-20 04:33:07 +02:00
TALLOC_FREE ( cred - > netlogon_creds ) ;
if ( netlogon_creds = = NULL ) {
return ;
}
cred - > netlogon_creds = netlogon_creds_copy ( cred , netlogon_creds ) ;
2005-03-24 03:32:25 +00:00
}
2005-03-27 06:55:03 +00:00
/**
2021-11-10 12:01:20 +01:00
* Return attached NETLOGON credentials
2005-03-27 06:55:03 +00:00
*/
2005-03-24 03:32:25 +00:00
2011-10-09 23:27:44 +11:00
_PUBLIC_ struct netlogon_creds_CredentialState * cli_credentials_get_netlogon_creds ( struct cli_credentials * cred )
2005-03-24 03:32:25 +00:00
{
return cred - > netlogon_creds ;
}
2021-11-10 12:01:20 +01:00
/**
2005-05-01 19:29:00 +00:00
* Set NETLOGON secure channel type
*/
2008-04-02 04:53:27 +02:00
_PUBLIC_ void cli_credentials_set_secure_channel_type ( struct cli_credentials * cred ,
2005-05-01 19:29:00 +00:00
enum netr_SchannelType secure_channel_type )
{
cred - > secure_channel_type = secure_channel_type ;
}
2010-05-17 13:41:01 +10:00
/**
2023-03-14 08:50:34 +01:00
* Return NETLOGON secure channel type
2010-05-17 13:41:01 +10:00
*/
_PUBLIC_ time_t cli_credentials_get_password_last_changed_time ( struct cli_credentials * cred )
{
return cred - > password_last_changed_time ;
}
2021-11-10 12:01:20 +01:00
/**
2010-05-17 13:41:01 +10:00
* Set NETLOGON secure channel type
*/
_PUBLIC_ void cli_credentials_set_password_last_changed_time ( struct cli_credentials * cred ,
time_t last_changed_time )
{
cred - > password_last_changed_time = last_changed_time ;
}
2005-05-01 19:29:00 +00:00
/**
2023-03-14 08:50:34 +01:00
* Return NETLOGON secure channel type
2005-05-01 19:29:00 +00:00
*/
2008-04-02 04:53:27 +02:00
_PUBLIC_ enum netr_SchannelType cli_credentials_get_secure_channel_type ( struct cli_credentials * cred )
2005-05-01 19:29:00 +00:00
{
return cred - > secure_channel_type ;
}
2005-03-27 06:55:03 +00:00
/**
* Fill in a credentials structure as the anonymous user
*/
2021-11-10 12:01:20 +01:00
_PUBLIC_ void cli_credentials_set_anonymous ( struct cli_credentials * cred )
2005-03-22 08:00:45 +00:00
{
cli_credentials_set_username ( cred , " " , CRED_SPECIFIED ) ;
cli_credentials_set_domain ( cred , " " , CRED_SPECIFIED ) ;
2005-03-22 10:33:53 +00:00
cli_credentials_set_password ( cred , NULL , CRED_SPECIFIED ) ;
2016-10-22 11:19:05 +02:00
cli_credentials_set_principal ( cred , NULL , CRED_SPECIFIED ) ;
2007-12-10 18:41:55 +01:00
cli_credentials_set_realm ( cred , NULL , CRED_SPECIFIED ) ;
2007-12-10 18:42:07 +01:00
cli_credentials_set_workstation ( cred , " " , CRED_UNINITIALISED ) ;
2020-08-19 15:46:11 +02:00
cli_credentials_set_kerberos_state ( cred ,
CRED_USE_KERBEROS_DISABLED ,
CRED_SPECIFIED ) ;
2005-03-22 08:00:45 +00:00
}
2005-03-27 06:55:03 +00:00
/**
* Describe a credentials context as anonymous or authenticated
2007-10-06 22:16:19 +00:00
* @ retval true if anonymous , false if a username is specified
2005-03-27 06:55:03 +00:00
*/
2008-04-02 04:53:27 +02:00
_PUBLIC_ bool cli_credentials_is_anonymous ( struct cli_credentials * cred )
2005-03-21 23:35:58 +00:00
{
r11995: A big kerberos-related update.
This merges Samba4 up to current lorikeet-heimdal, which includes a
replacement for some Samba-specific hacks.
In particular, the credentials system now supplies GSS client and
server credentials. These are imported into GSS with
gss_krb5_import_creds(). Unfortunetly this can't take an MEMORY
keytab, so we now create a FILE based keytab as provision and join
time.
Because the keytab is now created in advance, we don't spend .4s at
negprot doing sha1 s2k calls. Also, because the keytab is read in
real time, any change in the server key will be correctly picked up by
the the krb5 code.
To mark entries in the secrets which should be exported to a keytab,
there is a new kerberosSecret objectClass. The new routine
cli_credentials_update_all_keytabs() searches for these, and updates
the keytabs.
This is called in the provision.js via the ejs wrapper
credentials_update_all_keytabs().
We can now (in theory) use a system-provided /etc/krb5.keytab, if
krb5Keytab: FILE:/etc/krb5.keytab
is added to the secrets.ldb record. By default the attribute
privateKeytab: secrets.keytab
is set, pointing to allow the whole private directory to be moved
without breaking the internal links.
(This used to be commit 6b75573df49c6210e1b9d71e108a9490976bd41d)
2005-12-01 05:20:39 +00:00
const char * username ;
2021-11-10 12:01:20 +01:00
2009-10-09 14:30:51 -05:00
/* if bind dn is set it's not anonymous */
if ( cred - > bind_dn ) {
return false ;
}
r11995: A big kerberos-related update.
This merges Samba4 up to current lorikeet-heimdal, which includes a
replacement for some Samba-specific hacks.
In particular, the credentials system now supplies GSS client and
server credentials. These are imported into GSS with
gss_krb5_import_creds(). Unfortunetly this can't take an MEMORY
keytab, so we now create a FILE based keytab as provision and join
time.
Because the keytab is now created in advance, we don't spend .4s at
negprot doing sha1 s2k calls. Also, because the keytab is read in
real time, any change in the server key will be correctly picked up by
the the krb5 code.
To mark entries in the secrets which should be exported to a keytab,
there is a new kerberosSecret objectClass. The new routine
cli_credentials_update_all_keytabs() searches for these, and updates
the keytabs.
This is called in the provision.js via the ejs wrapper
credentials_update_all_keytabs().
We can now (in theory) use a system-provided /etc/krb5.keytab, if
krb5Keytab: FILE:/etc/krb5.keytab
is added to the secrets.ldb record. By default the attribute
privateKeytab: secrets.keytab
is set, pointing to allow the whole private directory to be moved
without breaking the internal links.
(This used to be commit 6b75573df49c6210e1b9d71e108a9490976bd41d)
2005-12-01 05:20:39 +00:00
if ( cred - > machine_account_pending ) {
2007-12-13 22:46:17 +01:00
cli_credentials_set_machine_account ( cred ,
cred - > machine_account_pending_lp_ctx ) ;
r11995: A big kerberos-related update.
This merges Samba4 up to current lorikeet-heimdal, which includes a
replacement for some Samba-specific hacks.
In particular, the credentials system now supplies GSS client and
server credentials. These are imported into GSS with
gss_krb5_import_creds(). Unfortunetly this can't take an MEMORY
keytab, so we now create a FILE based keytab as provision and join
time.
Because the keytab is now created in advance, we don't spend .4s at
negprot doing sha1 s2k calls. Also, because the keytab is read in
real time, any change in the server key will be correctly picked up by
the the krb5 code.
To mark entries in the secrets which should be exported to a keytab,
there is a new kerberosSecret objectClass. The new routine
cli_credentials_update_all_keytabs() searches for these, and updates
the keytabs.
This is called in the provision.js via the ejs wrapper
credentials_update_all_keytabs().
We can now (in theory) use a system-provided /etc/krb5.keytab, if
krb5Keytab: FILE:/etc/krb5.keytab
is added to the secrets.ldb record. By default the attribute
privateKeytab: secrets.keytab
is set, pointing to allow the whole private directory to be moved
without breaking the internal links.
(This used to be commit 6b75573df49c6210e1b9d71e108a9490976bd41d)
2005-12-01 05:20:39 +00:00
}
2015-05-07 14:12:03 +00:00
/* if principal is set, it's not anonymous */
if ( ( cred - > principal ! = NULL ) & & cred - > principal_obtained > = cred - > username_obtained ) {
return false ;
}
r11995: A big kerberos-related update.
This merges Samba4 up to current lorikeet-heimdal, which includes a
replacement for some Samba-specific hacks.
In particular, the credentials system now supplies GSS client and
server credentials. These are imported into GSS with
gss_krb5_import_creds(). Unfortunetly this can't take an MEMORY
keytab, so we now create a FILE based keytab as provision and join
time.
Because the keytab is now created in advance, we don't spend .4s at
negprot doing sha1 s2k calls. Also, because the keytab is read in
real time, any change in the server key will be correctly picked up by
the the krb5 code.
To mark entries in the secrets which should be exported to a keytab,
there is a new kerberosSecret objectClass. The new routine
cli_credentials_update_all_keytabs() searches for these, and updates
the keytabs.
This is called in the provision.js via the ejs wrapper
credentials_update_all_keytabs().
We can now (in theory) use a system-provided /etc/krb5.keytab, if
krb5Keytab: FILE:/etc/krb5.keytab
is added to the secrets.ldb record. By default the attribute
privateKeytab: secrets.keytab
is set, pointing to allow the whole private directory to be moved
without breaking the internal links.
(This used to be commit 6b75573df49c6210e1b9d71e108a9490976bd41d)
2005-12-01 05:20:39 +00:00
username = cli_credentials_get_username ( cred ) ;
2021-11-10 12:01:20 +01:00
2005-03-24 03:32:25 +00:00
/* Yes, it is deliberate that we die if we have a NULL pointer
2005-03-26 01:08:59 +00:00
* here - anonymous is " " , not NULL , which is ' never specified ,
2005-03-24 03:32:25 +00:00
* never guessed ' , ie programmer bug */
2005-08-29 04:30:22 +00:00
if ( ! username [ 0 ] ) {
2007-10-06 22:16:19 +00:00
return true ;
2005-08-29 04:30:22 +00:00
}
2005-10-14 15:55:51 +00:00
2007-10-06 22:16:19 +00:00
return false ;
2005-03-21 23:35:58 +00:00
}
2006-05-03 14:15:31 +00:00
/**
2021-11-10 12:01:20 +01:00
* Mark the current password for a credentials struct as wrong . This will
2006-05-03 14:15:31 +00:00
* cause the password to be prompted again ( if a callback is set ) .
*
2006-05-03 20:23:19 +00:00
* This will decrement the number of times the password can be tried .
2006-05-03 14:15:31 +00:00
*
* @ retval whether the credentials struct is finished
*/
2008-04-02 04:53:27 +02:00
_PUBLIC_ bool cli_credentials_wrong_password ( struct cli_credentials * cred )
2006-05-03 14:15:31 +00:00
{
if ( cred - > password_obtained ! = CRED_CALLBACK_RESULT ) {
2007-10-06 22:16:19 +00:00
return false ;
2006-05-03 14:15:31 +00:00
}
2013-07-31 14:25:54 +02:00
if ( cred - > password_tries = = 0 ) {
return false ;
}
cred - > password_tries - - ;
2006-05-03 14:15:31 +00:00
2013-07-31 14:25:54 +02:00
if ( cred - > password_tries = = 0 ) {
return false ;
}
cred - > password_obtained = CRED_CALLBACK ;
return true ;
2006-05-03 14:15:31 +00:00
}
2010-10-11 01:20:42 +02:00
2021-11-10 12:01:20 +01:00
_PUBLIC_ void cli_credentials_get_ntlm_username_domain ( struct cli_credentials * cred , TALLOC_CTX * mem_ctx ,
const char * * username ,
const char * * domain )
2010-10-11 01:20:42 +02:00
{
2019-04-01 15:59:10 +02:00
if ( cred - > principal_obtained > = cred - > username_obtained ) {
2010-10-11 01:20:42 +02:00
* domain = talloc_strdup ( mem_ctx , " " ) ;
* username = cli_credentials_get_principal ( cred , mem_ctx ) ;
} else {
* domain = cli_credentials_get_domain ( cred ) ;
* username = cli_credentials_get_username ( cred ) ;
}
}
/**
* Read a named file , and parse it for username , domain , realm and password
*
* @ param credentials Credentials structure on which to set the password
2021-11-10 12:01:20 +01:00
* @ param file a named file to read the details from
2010-10-11 01:20:42 +02:00
* @ param obtained This enum describes how ' specified ' this password is
*/
2021-11-10 12:01:20 +01:00
_PUBLIC_ bool cli_credentials_parse_file ( struct cli_credentials * cred , const char * file , enum credentials_obtained obtained )
2010-10-11 01:20:42 +02:00
{
uint16_t len = 0 ;
char * ptr , * val , * param ;
char * * lines ;
int i , numlines ;
2016-12-15 12:41:58 +01:00
const char * realm = NULL ;
const char * domain = NULL ;
const char * password = NULL ;
const char * username = NULL ;
2010-10-11 01:20:42 +02:00
lines = file_lines_load ( file , & numlines , 0 , NULL ) ;
if ( lines = = NULL )
{
/* fail if we can't open the credentials file */
d_printf ( " ERROR: Unable to open credentials file! \n " ) ;
return false ;
}
for ( i = 0 ; i < numlines ; i + + ) {
len = strlen ( lines [ i ] ) ;
if ( len = = 0 )
continue ;
/* break up the line into parameter & value.
* will need to eat a little whitespace possibly */
param = lines [ i ] ;
if ( ! ( ptr = strchr_m ( lines [ i ] , ' = ' ) ) )
continue ;
val = ptr + 1 ;
* ptr = ' \0 ' ;
/* eat leading white space */
while ( ( * val ! = ' \0 ' ) & & ( ( * val = = ' ' ) | | ( * val = = ' \t ' ) ) )
val + + ;
if ( strwicmp ( " password " , param ) = = 0 ) {
2016-12-15 12:41:58 +01:00
password = val ;
2010-10-11 01:20:42 +02:00
} else if ( strwicmp ( " username " , param ) = = 0 ) {
2016-12-15 12:41:58 +01:00
username = val ;
2010-10-11 01:20:42 +02:00
} else if ( strwicmp ( " domain " , param ) = = 0 ) {
2016-12-15 12:41:58 +01:00
domain = val ;
2010-10-11 01:20:42 +02:00
} else if ( strwicmp ( " realm " , param ) = = 0 ) {
2016-12-15 12:41:58 +01:00
realm = val ;
2010-10-11 01:20:42 +02:00
}
2016-12-15 12:41:58 +01:00
/*
* We need to readd ' = ' in order to let
* the strlen ( ) work in the last loop
* that clears the memory .
*/
* ptr = ' = ' ;
}
if ( realm ! = NULL & & strlen ( realm ) ! = 0 ) {
/*
* only overwrite with a valid string
*/
cli_credentials_set_realm ( cred , realm , obtained ) ;
}
if ( domain ! = NULL & & strlen ( domain ) ! = 0 ) {
/*
* only overwrite with a valid string
*/
cli_credentials_set_domain ( cred , domain , obtained ) ;
2010-10-11 01:20:42 +02:00
}
2016-12-15 12:41:58 +01:00
if ( password ! = NULL ) {
/*
* Here we allow " " .
*/
cli_credentials_set_password ( cred , password , obtained ) ;
}
if ( username ! = NULL ) {
/*
* The last " username " line takes preference
* if the string also contains domain , realm or
* password .
*/
cli_credentials_parse_string ( cred , username , obtained ) ;
}
for ( i = 0 ; i < numlines ; i + + ) {
len = strlen ( lines [ i ] ) ;
memset ( lines [ i ] , 0 , len ) ;
}
2010-10-11 01:20:42 +02:00
talloc_free ( lines ) ;
return true ;
}
/**
* Read a named file , and parse it for a password
*
* @ param credentials Credentials structure on which to set the password
2021-11-10 12:01:20 +01:00
* @ param file a named file to read the password from
2010-10-11 01:20:42 +02:00
* @ param obtained This enum describes how ' specified ' this password is
*/
_PUBLIC_ bool cli_credentials_parse_password_file ( struct cli_credentials * credentials , const char * file , enum credentials_obtained obtained )
{
int fd = open ( file , O_RDONLY , 0 ) ;
bool ret ;
if ( fd < 0 ) {
fprintf ( stderr , " Error opening password file %s: %s \n " ,
file , strerror ( errno ) ) ;
return false ;
}
ret = cli_credentials_parse_password_fd ( credentials , fd , obtained ) ;
close ( fd ) ;
2021-11-10 12:01:20 +01:00
2010-10-11 01:20:42 +02:00
return ret ;
}
/**
* Read a file descriptor , and parse it for a password ( eg from a file or stdin )
*
* @ param credentials Credentials structure on which to set the password
2021-11-10 12:01:20 +01:00
* @ param fd open file descriptor to read the password from
2010-10-11 01:20:42 +02:00
* @ param obtained This enum describes how ' specified ' this password is
*/
2021-11-10 12:01:20 +01:00
_PUBLIC_ bool cli_credentials_parse_password_fd ( struct cli_credentials * credentials ,
2010-10-11 01:20:42 +02:00
int fd , enum credentials_obtained obtained )
{
char * p ;
char pass [ 128 ] ;
2023-09-07 14:51:15 +02:00
if ( credentials - > password_obtained > = obtained ) {
return false ;
}
2010-10-11 01:20:42 +02:00
for ( p = pass , * p = ' \0 ' ; /* ensure that pass is null-terminated */
2023-02-28 16:55:06 +13:00
p & & p - pass < sizeof ( pass ) - 1 ; ) {
2010-10-11 01:20:42 +02:00
switch ( read ( fd , p , 1 ) ) {
case 1 :
if ( * p ! = ' \n ' & & * p ! = ' \0 ' ) {
* + + p = ' \0 ' ; /* advance p, and null-terminate pass */
break ;
}
2017-07-26 18:20:53 +02:00
FALL_THROUGH ;
2010-10-11 01:20:42 +02:00
case 0 :
if ( p - pass ) {
* p = ' \0 ' ; /* null-terminate it, just in case... */
p = NULL ; /* then force the loop condition to become false */
break ;
}
2018-03-01 11:04:49 +01:00
fprintf ( stderr ,
" Error reading password from file descriptor "
" %d: empty password \n " ,
fd ) ;
return false ;
2010-10-11 01:20:42 +02:00
default :
fprintf ( stderr , " Error reading password from file descriptor %d: %s \n " ,
fd , strerror ( errno ) ) ;
return false ;
}
}
cli_credentials_set_password ( credentials , pass , obtained ) ;
return true ;
}
2020-05-26 09:32:44 +02:00
/**
* @ brief Set the SMB signing state to request for a SMB connection .
*
* @ param [ in ] creds The credentials structure to update .
*
* @ param [ in ] signing_state The signing state to set .
*
* @ param obtained This way the described signing state was specified .
*
* @ return true if we could set the signing state , false otherwise .
*/
_PUBLIC_ bool cli_credentials_set_smb_signing ( struct cli_credentials * creds ,
enum smb_signing_setting signing_state ,
enum credentials_obtained obtained )
{
if ( obtained > = creds - > signing_state_obtained ) {
creds - > signing_state_obtained = obtained ;
creds - > signing_state = signing_state ;
return true ;
}
return false ;
}
/**
* @ brief Obtain the SMB signing state from a credentials structure .
*
* @ param [ in ] creds The credential structure to obtain the SMB signing state
* from .
*
2023-02-28 17:00:57 +13:00
* @ return The SMB signing state .
2020-05-26 09:32:44 +02:00
*/
_PUBLIC_ enum smb_signing_setting
cli_credentials_get_smb_signing ( struct cli_credentials * creds )
{
return creds - > signing_state ;
}
2010-10-11 01:20:42 +02:00
2020-05-28 16:31:35 +02:00
/**
* @ brief Set the SMB IPC signing state to request for a SMB connection .
*
* @ param [ in ] creds The credentials structure to update .
*
* @ param [ in ] signing_state The signing state to set .
*
* @ param obtained This way the described signing state was specified .
*
* @ return true if we could set the signing state , false otherwise .
*/
_PUBLIC_ bool
cli_credentials_set_smb_ipc_signing ( struct cli_credentials * creds ,
enum smb_signing_setting ipc_signing_state ,
enum credentials_obtained obtained )
{
if ( obtained > = creds - > ipc_signing_state_obtained ) {
creds - > ipc_signing_state_obtained = obtained ;
creds - > ipc_signing_state = ipc_signing_state ;
return true ;
}
return false ;
}
/**
* @ brief Obtain the SMB IPC signing state from a credentials structure .
*
* @ param [ in ] creds The credential structure to obtain the SMB IPC signing
* state from .
*
2023-02-28 17:00:57 +13:00
* @ return The SMB signing state .
2020-05-28 16:31:35 +02:00
*/
_PUBLIC_ enum smb_signing_setting
cli_credentials_get_smb_ipc_signing ( struct cli_credentials * creds )
{
return creds - > ipc_signing_state ;
}
2020-05-28 16:10:52 +02:00
/**
* @ brief Set the SMB encryption state to request for a SMB connection .
*
* @ param [ in ] creds The credentials structure to update .
*
* @ param [ in ] encryption_state The encryption state to set .
*
* @ param obtained This way the described encryption state was specified .
*
* @ return true if we could set the encryption state , false otherwise .
*/
_PUBLIC_ bool cli_credentials_set_smb_encryption ( struct cli_credentials * creds ,
enum smb_encryption_setting encryption_state ,
enum credentials_obtained obtained )
{
if ( obtained > = creds - > encryption_state_obtained ) {
creds - > encryption_state_obtained = obtained ;
creds - > encryption_state = encryption_state ;
return true ;
}
return false ;
}
2020-12-10 16:48:16 +01:00
static const char * obtained_to_str ( enum credentials_obtained obtained )
{
switch ( obtained ) {
case CRED_UNINITIALISED :
return " CRED_UNINITIALISED " ;
case CRED_SMB_CONF :
return " CRED_SMB_CONF " ;
case CRED_CALLBACK :
return " CRED_CALLBACK " ;
case CRED_GUESS_ENV :
return " CRED_GUESS_ENV " ;
case CRED_GUESS_FILE :
return " CRED_GUESS_FILE " ;
case CRED_CALLBACK_RESULT :
return " CRED_CALLBACK_RESULT " ;
case CRED_SPECIFIED :
return " CRED_SPECIFIED " ;
}
/* Never reached */
return " " ;
}
static const char * krb5_state_to_str ( enum credentials_use_kerberos krb5_state )
{
switch ( krb5_state ) {
case CRED_USE_KERBEROS_DISABLED :
return " CRED_USE_KERBEROS_DISABLED " ;
case CRED_USE_KERBEROS_DESIRED :
return " CRED_USE_KERBEROS_DESIRED " ;
case CRED_USE_KERBEROS_REQUIRED :
return " CRED_USE_KERBEROS_REQUIRED " ;
}
/* Never reached */
return " " ;
}
static const char * krb5_fwd_to_str ( enum credentials_krb_forwardable krb5_fwd )
{
switch ( krb5_fwd ) {
case CRED_AUTO_KRB_FORWARDABLE :
return " CRED_AUTO_KRB_FORWARDABLE " ;
case CRED_NO_KRB_FORWARDABLE :
return " CRED_NO_KRB_FORWARDABLE " ;
case CRED_FORCE_KRB_FORWARDABLE :
return " CRED_FORCE_KRB_FORWARDABLE " ;
}
/* Never reached */
return " " ;
}
static const char * signing_state_to_str ( enum smb_signing_setting signing_state )
{
switch ( signing_state ) {
case SMB_SIGNING_IPC_DEFAULT :
return " SMB_SIGNING_IPC_DEFAULT " ;
case SMB_SIGNING_DEFAULT :
return " SMB_SIGNING_DEFAULT " ;
case SMB_SIGNING_OFF :
return " SMB_SIGNING_OFF " ;
case SMB_SIGNING_IF_REQUIRED :
return " SMB_SIGNING_IF_REQUIRED " ;
case SMB_SIGNING_DESIRED :
return " SMB_SIGNING_DESIRED " ;
case SMB_SIGNING_REQUIRED :
return " SMB_SIGNING_REQUIRED " ;
}
/* Never reached */
return " " ;
}
static const char * encryption_state_to_str ( enum smb_encryption_setting encryption_state )
{
switch ( encryption_state ) {
case SMB_ENCRYPTION_DEFAULT :
return " SMB_ENCRYPTION_DEFAULT " ;
case SMB_ENCRYPTION_OFF :
return " SMB_ENCRYPTION_OFF " ;
case SMB_ENCRYPTION_IF_REQUIRED :
return " SMB_ENCRYPTION_IF_REQUIRED " ;
case SMB_ENCRYPTION_DESIRED :
return " SMB_ENCRYPTION_DESIRED " ;
case SMB_ENCRYPTION_REQUIRED :
return " SMB_ENCRYPTION_REQUIRED " ;
}
/* Never reached */
return " " ;
}
_PUBLIC_ void cli_credentials_dump ( struct cli_credentials * creds )
{
DBG_ERR ( " CLI_CREDENTIALS: \n " ) ;
DBG_ERR ( " \n " ) ;
DBG_ERR ( " Username: %s - %s \n " ,
creds - > username ,
obtained_to_str ( creds - > username_obtained ) ) ;
DBG_ERR ( " Workstation: %s - %s \n " ,
creds - > workstation ,
obtained_to_str ( creds - > workstation_obtained ) ) ;
DBG_ERR ( " Domain: %s - %s \n " ,
creds - > domain ,
obtained_to_str ( creds - > domain_obtained ) ) ;
DBG_ERR ( " Password: %s - %s \n " ,
creds - > password ! = NULL ? " *SECRET* " : " NULL " ,
obtained_to_str ( creds - > password_obtained ) ) ;
DBG_ERR ( " Old password: %s \n " ,
creds - > old_password ! = NULL ? " *SECRET* " : " NULL " ) ;
DBG_ERR ( " Password tries: %u \n " ,
creds - > password_tries ) ;
DBG_ERR ( " Realm: %s - %s \n " ,
creds - > realm ,
obtained_to_str ( creds - > realm_obtained ) ) ;
DBG_ERR ( " Principal: %s - %s \n " ,
creds - > principal ,
obtained_to_str ( creds - > principal_obtained ) ) ;
DBG_ERR ( " Salt principal: %s \n " ,
creds - > salt_principal ) ;
DBG_ERR ( " Impersonate principal: %s \n " ,
creds - > impersonate_principal ) ;
DBG_ERR ( " Self service: %s \n " ,
creds - > self_service ) ;
DBG_ERR ( " Target service: %s \n " ,
creds - > target_service ) ;
2020-08-19 15:46:11 +02:00
DBG_ERR ( " Kerberos state: %s - %s \n " ,
krb5_state_to_str ( creds - > kerberos_state ) ,
obtained_to_str ( creds - > kerberos_state_obtained ) ) ;
2020-12-10 16:48:16 +01:00
DBG_ERR ( " Kerberos forwardable ticket: %s \n " ,
krb5_fwd_to_str ( creds - > krb_forwardable ) ) ;
DBG_ERR ( " Signing state: %s - %s \n " ,
signing_state_to_str ( creds - > signing_state ) ,
obtained_to_str ( creds - > signing_state_obtained ) ) ;
DBG_ERR ( " IPC signing state: %s - %s \n " ,
signing_state_to_str ( creds - > ipc_signing_state ) ,
obtained_to_str ( creds - > ipc_signing_state_obtained ) ) ;
DBG_ERR ( " Encryption state: %s - %s \n " ,
encryption_state_to_str ( creds - > encryption_state ) ,
obtained_to_str ( creds - > encryption_state_obtained ) ) ;
DBG_ERR ( " Gensec features: %#X \n " ,
creds - > gensec_features ) ;
DBG_ERR ( " Forced sasl mech: %s \n " ,
creds - > forced_sasl_mech ) ;
DBG_ERR ( " CCACHE: %p - %s \n " ,
creds - > ccache ,
obtained_to_str ( creds - > ccache_obtained ) ) ;
DBG_ERR ( " CLIENT_GSS_CREDS: %p - %s \n " ,
creds - > client_gss_creds ,
obtained_to_str ( creds - > client_gss_creds_obtained ) ) ;
DBG_ERR ( " SERVER_GSS_CREDS: %p - %s \n " ,
creds - > server_gss_creds ,
obtained_to_str ( creds - > server_gss_creds_obtained ) ) ;
DBG_ERR ( " KEYTAB: %p - %s \n " ,
creds - > keytab ,
obtained_to_str ( creds - > keytab_obtained ) ) ;
DBG_ERR ( " KVNO: %u \n " ,
creds - > kvno ) ;
DBG_ERR ( " \n " ) ;
}
2020-05-28 16:10:52 +02:00
/**
* @ brief Obtain the SMB encryption state from a credentials structure .
*
* @ param [ in ] creds The credential structure to obtain the SMB encryption state
* from .
*
2023-02-28 17:00:57 +13:00
* @ return The SMB signing state .
2020-05-28 16:10:52 +02:00
*/
_PUBLIC_ enum smb_encryption_setting
cli_credentials_get_smb_encryption ( struct cli_credentials * creds )
{
return creds - > encryption_state ;
}