auth4: add auth_context_create_for_netlogon()
[sfrench/samba-autobuild/.git] / source4 / auth / ntlm / auth.c
index 69cbff6e9a54c1f6bc184e9ef127b36f1d575b61..926bf48e19262b1b99b3c64db7f42318453cc382 100644 (file)
 #include "param/param.h"
 #include "dsdb/samdb/samdb.h"
 #include "libcli/wbclient/wbclient.h"
+#include "lib/util/samba_modules.h"
+#include "auth/credentials/credentials.h"
+#include "system/kerberos.h"
+#include "auth/kerberos/kerberos.h"
+#include "auth/kerberos/kerberos_util.h"
+#include "libds/common/roles.h"
+
+static NTSTATUS auth_generate_session_info_wrapper(struct auth4_context *auth_context,
+                                                  TALLOC_CTX *mem_ctx,
+                                                  void *server_returned_info,
+                                                  const char *original_user_name,
+                                                  uint32_t session_info_flags,
+                                                  struct auth_session_info **session_info);
 
 /***************************************************************************
  Set a fixed challenge
@@ -42,22 +55,12 @@ _PUBLIC_ NTSTATUS auth_context_set_challenge(struct auth4_context *auth_ctx, con
        return NT_STATUS_OK;
 }
 
-/***************************************************************************
- Set a fixed challenge
-***************************************************************************/
-_PUBLIC_ bool auth_challenge_may_be_modified(struct auth4_context *auth_ctx)
-{
-       return auth_ctx->challenge.may_be_modified;
-}
-
 /****************************************************************************
  Try to get a challenge out of the various authentication modules.
  Returns a const char of length 8 bytes.
 ****************************************************************************/
 _PUBLIC_ NTSTATUS auth_get_challenge(struct auth4_context *auth_ctx, uint8_t chal[8])
 {
-       NTSTATUS nt_status;
-       struct auth_method_context *method;
 
        if (auth_ctx->challenge.data.length == 8) {
                DEBUG(5, ("auth_get_challenge: returning previous challenge by module %s (normal)\n", 
@@ -66,29 +69,12 @@ _PUBLIC_ NTSTATUS auth_get_challenge(struct auth4_context *auth_ctx, uint8_t cha
                return NT_STATUS_OK;
        }
 
-       for (method = auth_ctx->methods; method; method = method->next) {
-               nt_status = method->ops->get_challenge(method, auth_ctx, chal);
-               if (NT_STATUS_EQUAL(nt_status, NT_STATUS_NOT_IMPLEMENTED)) {
-                       continue;
-               }
-
-               NT_STATUS_NOT_OK_RETURN(nt_status);
-
-               auth_ctx->challenge.data        = data_blob_talloc(auth_ctx, chal, 8);
-               NT_STATUS_HAVE_NO_MEMORY(auth_ctx->challenge.data.data);
-               auth_ctx->challenge.set_by      = method->ops->name;
-
-               break;
-       }
-
        if (!auth_ctx->challenge.set_by) {
                generate_random_buffer(chal, 8);
 
                auth_ctx->challenge.data                = data_blob_talloc(auth_ctx, chal, 8);
                NT_STATUS_HAVE_NO_MEMORY(auth_ctx->challenge.data.data);
                auth_ctx->challenge.set_by              = "random";
-
-               auth_ctx->challenge.may_be_modified     = true;
        }
 
        DEBUG(10,("auth_get_challenge: challenge set by %s\n",
@@ -103,24 +89,35 @@ PAC isn't available, and for tokenGroups in the DSDB stack.
 
  Supply either a principal or a DN
 ****************************************************************************/
-_PUBLIC_ NTSTATUS auth_get_user_info_dc_principal(TALLOC_CTX *mem_ctx,
-                                                struct auth4_context *auth_ctx,
-                                                const char *principal,
-                                                struct ldb_dn *user_dn,
-                                                struct auth_user_info_dc **user_info_dc)
+static NTSTATUS auth_generate_session_info_principal(struct auth4_context *auth_ctx,
+                                                 TALLOC_CTX *mem_ctx,
+                                                 const char *principal,
+                                                 struct ldb_dn *user_dn,
+                                                  uint32_t session_info_flags,
+                                                  struct auth_session_info **session_info)
 {
        NTSTATUS nt_status;
        struct auth_method_context *method;
+       struct auth_user_info_dc *user_info_dc;
 
        for (method = auth_ctx->methods; method; method = method->next) {
                if (!method->ops->get_user_info_dc_principal) {
                        continue;
                }
 
-               nt_status = method->ops->get_user_info_dc_principal(mem_ctx, auth_ctx, principal, user_dn, user_info_dc);
+               nt_status = method->ops->get_user_info_dc_principal(mem_ctx, auth_ctx, principal, user_dn, &user_info_dc);
                if (NT_STATUS_EQUAL(nt_status, NT_STATUS_NOT_IMPLEMENTED)) {
                        continue;
                }
+               if (!NT_STATUS_IS_OK(nt_status)) {
+                       return nt_status;
+               }
+
+               nt_status = auth_generate_session_info_wrapper(auth_ctx, mem_ctx, 
+                                                              user_info_dc,
+                                                              user_info_dc->info->account_name,
+                                                              session_info_flags, session_info);
+               talloc_free(user_info_dc);
 
                return nt_status;
        }
@@ -187,6 +184,42 @@ _PUBLIC_ NTSTATUS auth_check_password(struct auth4_context *auth_ctx,
        return status;
 }
 
+static NTSTATUS auth_check_password_wrapper(struct auth4_context *auth_ctx,
+                                           TALLOC_CTX *mem_ctx,
+                                           const struct auth_usersupplied_info *user_info,
+                                           void **server_returned_info,
+                                           DATA_BLOB *user_session_key, DATA_BLOB *lm_session_key)
+{
+       struct auth_user_info_dc *user_info_dc;
+       NTSTATUS status;
+
+       status = auth_check_password(auth_ctx, mem_ctx, user_info,
+                                    &user_info_dc);
+       if (!NT_STATUS_IS_OK(status)) {
+               return status;
+       }
+
+       *server_returned_info = user_info_dc;
+
+       if (user_session_key) {
+               DEBUG(10, ("Got NT session key of length %u\n",
+                          (unsigned)user_info_dc->user_session_key.length));
+               *user_session_key = user_info_dc->user_session_key;
+               talloc_steal(mem_ctx, user_session_key->data);
+               user_info_dc->user_session_key = data_blob_null;
+       }
+
+       if (lm_session_key) {
+               DEBUG(10, ("Got LM session key of length %u\n",
+                          (unsigned)user_info_dc->lm_session_key.length));
+               *lm_session_key = user_info_dc->lm_session_key;
+               talloc_steal(mem_ctx, lm_session_key->data);
+               user_info_dc->lm_session_key = data_blob_null;
+       }
+
+       return NT_STATUS_OK;
+}
+
 struct auth_check_password_state {
        struct auth4_context *auth_ctx;
        const struct auth_usersupplied_info *user_info;
@@ -233,7 +266,6 @@ _PUBLIC_ struct tevent_req *auth_check_password_send(TALLOC_CTX *mem_ctx,
        /* if all the modules say 'not for me' this is reasonable */
        NTSTATUS nt_status;
        uint8_t chal[8];
-       struct auth_usersupplied_info *user_info_tmp;
        struct tevent_immediate *im;
 
        DEBUG(3,("auth_check_password_send: "
@@ -251,8 +283,15 @@ _PUBLIC_ struct tevent_req *auth_check_password_send(TALLOC_CTX *mem_ctx,
        state->user_info        = user_info;
 
        if (!user_info->mapped_state) {
-               nt_status = map_user_info(auth_ctx->sam_ctx, req, lpcfg_workgroup(auth_ctx->lp_ctx),
-                                         user_info, &user_info_tmp);
+               int server_role = lpcfg_server_role(auth_ctx->lp_ctx);
+               struct auth_usersupplied_info *user_info_tmp;
+
+               nt_status = map_user_info(
+                       auth_ctx->sam_ctx, req,
+                       server_role == ROLE_ACTIVE_DIRECTORY_DC,
+                       lpcfg_workgroup(auth_ctx->lp_ctx),
+                       user_info, &user_info_tmp);
+
                if (tevent_req_nterror(req, nt_status)) {
                        return tevent_req_post(req, ev);
                }
@@ -314,6 +353,11 @@ static void auth_check_password_async_trigger(struct tevent_context *ev,
 
        for (method=state->auth_ctx->methods; method; method = method->next) {
 
+               if (state->user_info->flags & USER_INFO_LOCAL_SAM_ONLY
+                   && !(method->ops->flags & AUTH_METHOD_LOCAL_SAM)) {
+                       continue;
+               }
+
                /* we fill in state->method here so debug messages in
                   the callers know which method failed */
                state->method = method;
@@ -342,9 +386,13 @@ static void auth_check_password_async_trigger(struct tevent_context *ev,
        }
 
        if (NT_STATUS_EQUAL(status, NT_STATUS_NOT_IMPLEMENTED)) {
-               /* don't expose the NT_STATUS_NOT_IMPLEMENTED
-                  internals */
-               status = NT_STATUS_NO_SUCH_USER;
+               if (!(state->user_info->flags & USER_INFO_LOCAL_SAM_ONLY)) {
+                       /* don't expose the NT_STATUS_NOT_IMPLEMENTED
+                        * internals, except when the caller is only probing
+                        * one method, as they may do the fallback 
+                        */
+                       status = NT_STATUS_NO_SUCH_USER;
+               }
        }
 
        if (tevent_req_nterror(req, status)) {
@@ -406,45 +454,93 @@ _PUBLIC_ NTSTATUS auth_check_password_recv(struct tevent_req *req,
        return NT_STATUS_OK;
 }
 
-/* Wrapper because we don't want to expose all callers to needing to
- * know that session_info is generated from the main ldb, and because
- * we need to break a depenency loop between the DCE/RPC layer and the
- * generation of unix tokens via IRPC */
-static NTSTATUS auth_generate_session_info_wrapper(TALLOC_CTX *mem_ctx,
-                                                  struct auth4_context *auth_context,
-                                                  struct auth_user_info_dc *user_info_dc,
-                                                  uint32_t session_info_flags,
-                                                  struct auth_session_info **session_info)
+ /* Wrapper because we don't want to expose all callers to needing to
+  * know that session_info is generated from the main ldb, and because
+  * we need to break a depenency loop between the DCE/RPC layer and the
+  * generation of unix tokens via IRPC */
+static NTSTATUS auth_generate_session_info_wrapper(struct auth4_context *auth_context,
+                                                  TALLOC_CTX *mem_ctx,
+                                                  void *server_returned_info,
+                                                  const char *original_user_name,
+                                                  uint32_t session_info_flags,
+                                                  struct auth_session_info **session_info)
 {
-       NTSTATUS status = auth_generate_session_info(mem_ctx, auth_context->lp_ctx,
-                                                    auth_context->sam_ctx, user_info_dc,
-                                                    session_info_flags, session_info);
+       NTSTATUS status;
+       struct auth_user_info_dc *user_info_dc = talloc_get_type_abort(server_returned_info, struct auth_user_info_dc);
+
+       if (user_info_dc->info->authenticated) {
+               session_info_flags |= AUTH_SESSION_INFO_AUTHENTICATED;
+       }
+
+       status = auth_generate_session_info(mem_ctx, auth_context->lp_ctx,
+                                           auth_context->sam_ctx, user_info_dc,
+                                           session_info_flags, session_info);
+       if (!NT_STATUS_IS_OK(status)) {
+               return status;
+       }
 
        if ((session_info_flags & AUTH_SESSION_INFO_UNIX_TOKEN)
            && NT_STATUS_IS_OK(status)) {
-               struct wbc_context *wbc_ctx = wbc_init(auth_context,
-                                                      auth_context->msg_ctx,
-                                                      auth_context->event_ctx);
-               if (!wbc_ctx) {
-                       TALLOC_FREE(*session_info);
-                       DEBUG(1, ("Cannot contact winbind to provide unix token\n"));
-                       return NT_STATUS_INVALID_SERVER_STATE;
-               }
-               status = auth_session_info_fill_unix(wbc_ctx, auth_context->lp_ctx,
-                                                    *session_info);
+               status = auth_session_info_fill_unix(auth_context->event_ctx,
+                                                    auth_context->lp_ctx,
+                                                    original_user_name, *session_info);
                if (!NT_STATUS_IS_OK(status)) {
                        TALLOC_FREE(*session_info);
                }
-               TALLOC_FREE(wbc_ctx);
        }
        return status;
 }
 
+/* Wrapper because we don't want to expose all callers to needing to
+ * know anything about the PAC or auth subsystem internal structures
+ * before we output a struct auth session_info */
+static NTSTATUS auth_generate_session_info_pac(struct auth4_context *auth_ctx,
+                                              TALLOC_CTX *mem_ctx,
+                                              struct smb_krb5_context *smb_krb5_context,
+                                              DATA_BLOB *pac_blob,
+                                              const char *principal_name,
+                                              const struct tsocket_address *remote_address,
+                                              uint32_t session_info_flags,
+                                              struct auth_session_info **session_info)
+{
+       NTSTATUS status;
+       struct auth_user_info_dc *user_info_dc;
+       TALLOC_CTX *tmp_ctx;
+
+       if (!pac_blob) {
+               return auth_generate_session_info_principal(auth_ctx, mem_ctx, principal_name,
+                                                      NULL, session_info_flags, session_info);
+       }
+
+       tmp_ctx = talloc_named(mem_ctx, 0, "gensec_gssapi_session_info context");
+       NT_STATUS_HAVE_NO_MEMORY(tmp_ctx);
+
+       status = kerberos_pac_blob_to_user_info_dc(tmp_ctx,
+                                                  *pac_blob,
+                                                  smb_krb5_context->krb5_context,
+                                                  &user_info_dc, NULL, NULL);
+       if (!NT_STATUS_IS_OK(status)) {
+               talloc_free(tmp_ctx);
+               return status;
+       }
+
+       if (user_info_dc->info->authenticated) {
+               session_info_flags |= AUTH_SESSION_INFO_AUTHENTICATED;
+       }
+
+       status = auth_generate_session_info_wrapper(auth_ctx, mem_ctx, 
+                                                   user_info_dc,
+                                                   user_info_dc->info->account_name,
+                                                   session_info_flags, session_info);
+       talloc_free(tmp_ctx);
+       return status;
+}
+
 /***************************************************************************
  Make a auth_info struct for the auth subsystem
  - Allow the caller to specify the methods to use, including optionally the SAM to use
 ***************************************************************************/
-_PUBLIC_ NTSTATUS auth_context_create_methods(TALLOC_CTX *mem_ctx, const char **methods, 
+_PUBLIC_ NTSTATUS auth_context_create_methods(TALLOC_CTX *mem_ctx, const char * const *methods, 
                                              struct tevent_context *ev,
                                              struct imessaging_context *msg,
                                              struct loadparm_context *lp_ctx,
@@ -461,10 +557,8 @@ _PUBLIC_ NTSTATUS auth_context_create_methods(TALLOC_CTX *mem_ctx, const char **
                return NT_STATUS_INTERNAL_ERROR;
        }
 
-       ctx = talloc(mem_ctx, struct auth4_context);
+       ctx = talloc_zero(mem_ctx, struct auth4_context);
        NT_STATUS_HAVE_NO_MEMORY(ctx);
-       ctx->challenge.set_by           = NULL;
-       ctx->challenge.may_be_modified  = false;
        ctx->challenge.data             = data_blob(NULL, 0);
        ctx->methods                    = NULL;
        ctx->event_ctx                  = ev;
@@ -491,15 +585,14 @@ _PUBLIC_ NTSTATUS auth_context_create_methods(TALLOC_CTX *mem_ctx, const char **
                }
                method->auth_ctx        = ctx;
                method->depth           = i;
-               DLIST_ADD_END(ctx->methods, method, struct auth_method_context *);
+               DLIST_ADD_END(ctx->methods, method);
        }
 
-       ctx->check_password = auth_check_password;
-       ctx->get_challenge = auth_get_challenge;
-       ctx->set_challenge = auth_context_set_challenge;
-       ctx->challenge_may_be_modified = auth_challenge_may_be_modified;
-       ctx->get_user_info_dc_principal = auth_get_user_info_dc_principal;
+       ctx->check_ntlm_password = auth_check_password_wrapper;
+       ctx->get_ntlm_challenge = auth_get_challenge;
+       ctx->set_ntlm_challenge = auth_context_set_challenge;
        ctx->generate_session_info = auth_generate_session_info_wrapper;
+       ctx->generate_session_info_pac = auth_generate_session_info_pac;
 
        *auth_ctx = ctx;
 
@@ -519,10 +612,11 @@ const char **auth_methods_from_lp(TALLOC_CTX *mem_ctx, struct loadparm_context *
                break;
        case ROLE_DOMAIN_BDC:
        case ROLE_DOMAIN_PDC:
+       case ROLE_ACTIVE_DIRECTORY_DC:
                auth_methods = str_list_make(mem_ctx, "anonymous sam_ignoredomain winbind", NULL);
                break;
        }
-       return (const char **) auth_methods;
+       return discard_const_p(const char *, auth_methods);
 }
 
 /***************************************************************************
@@ -551,30 +645,13 @@ _PUBLIC_ NTSTATUS auth_context_create(TALLOC_CTX *mem_ctx,
        return status;
 }
 
-/* Create an auth context from an open LDB.
-
-   This allows us not to re-open the LDB when we need to do a some authentication logic (such as tokenGroups)
-
- */
-NTSTATUS auth_context_create_from_ldb(TALLOC_CTX *mem_ctx, struct ldb_context *ldb, struct auth4_context **auth_ctx)
+_PUBLIC_ NTSTATUS auth_context_create_for_netlogon(TALLOC_CTX *mem_ctx,
+                                                  struct tevent_context *ev,
+                                                  struct imessaging_context *msg,
+                                                  struct loadparm_context *lp_ctx,
+                                                  struct auth4_context **auth_ctx)
 {
-       NTSTATUS status;
-       const char **auth_methods;
-       struct loadparm_context *lp_ctx = talloc_get_type_abort(ldb_get_opaque(ldb, "loadparm"), struct loadparm_context);
-       struct tevent_context *ev = ldb_get_event_context(ldb);
-
-       TALLOC_CTX *tmp_ctx = talloc_new(mem_ctx);
-       if (!tmp_ctx) {
-               return NT_STATUS_NO_MEMORY;
-       }
-
-       auth_methods = auth_methods_from_lp(tmp_ctx, lp_ctx);
-       if (!auth_methods) {
-               return NT_STATUS_INVALID_PARAMETER;
-       }
-       status = auth_context_create_methods(mem_ctx, auth_methods, ev, NULL, lp_ctx, ldb, auth_ctx);
-       talloc_free(tmp_ctx);
-       return status;
+       return auth_context_create(mem_ctx, ev, msg, lp_ctx, auth_ctx);
 }
 
 /* the list of currently registered AUTH backends */