LCOV - code coverage report
Current view: top level - source3/auth - auth_sam.c (source / functions) Hit Total Coverage
Test: coverage report for v4-17-test 1498b464 Lines: 55 119 46.2 %
Date: 2024-06-13 04:01:37 Functions: 5 7 71.4 %

          Line data    Source code
       1             : /* 
       2             :    Unix SMB/CIFS implementation.
       3             :    Password and authentication handling
       4             :    Copyright (C) Andrew Tridgell              1992-2000
       5             :    Copyright (C) Luke Kenneth Casson Leighton 1996-2000
       6             :    Copyright (C) Andrew Bartlett              2001-2003
       7             :    Copyright (C) Gerald Carter                2003
       8             : 
       9             :    This program is free software; you can redistribute it and/or modify
      10             :    it under the terms of the GNU General Public License as published by
      11             :    the Free Software Foundation; either version 3 of the License, or
      12             :    (at your option) any later version.
      13             : 
      14             :    This program is distributed in the hope that it will be useful,
      15             :    but WITHOUT ANY WARRANTY; without even the implied warranty of
      16             :    MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
      17             :    GNU General Public License for more details.
      18             : 
      19             :    You should have received a copy of the GNU General Public License
      20             :    along with this program.  If not, see <http://www.gnu.org/licenses/>.
      21             : */
      22             : 
      23             : #include "includes.h"
      24             : #include "auth.h"
      25             : #include "passdb.h"
      26             : 
      27             : #undef DBGC_CLASS
      28             : #define DBGC_CLASS DBGC_AUTH
      29             : 
      30          54 : static NTSTATUS auth_sam_ignoredomain_auth(const struct auth_context *auth_context,
      31             :                                            void *my_private_data,
      32             :                                            TALLOC_CTX *mem_ctx,
      33             :                                            const struct auth_usersupplied_info *user_info,
      34             :                                            struct auth_serversupplied_info **server_info)
      35             : {
      36          54 :         if (!user_info || !auth_context) {
      37           0 :                 return NT_STATUS_UNSUCCESSFUL;
      38             :         }
      39             : 
      40          90 :         if (user_info->mapped.account_name == NULL ||
      41          54 :             user_info->mapped.account_name[0] == '\0')
      42             :         {
      43           2 :                 return NT_STATUS_NOT_IMPLEMENTED;
      44             :         }
      45             : 
      46          52 :         DBG_DEBUG("Check auth for: [%s]\\[%s]\n",
      47             :                   user_info->mapped.domain_name,
      48             :                   user_info->mapped.account_name);
      49             : 
      50          52 :         return check_sam_security(&auth_context->challenge, mem_ctx,
      51             :                                   user_info, server_info);
      52             : }
      53             : 
      54             : /* module initialisation */
      55        4206 : static NTSTATUS auth_init_sam_ignoredomain(
      56             :         struct auth_context *auth_context,
      57             :         const char *param,
      58             :         struct auth_methods **auth_method)
      59             : {
      60             :         struct auth_methods *result;
      61             : 
      62        4206 :         result = talloc_zero(auth_context, struct auth_methods);
      63        4206 :         if (result == NULL) {
      64           0 :                 return NT_STATUS_NO_MEMORY;
      65             :         }
      66        4206 :         result->auth = auth_sam_ignoredomain_auth;
      67        4206 :         result->name = "sam_ignoredomain";
      68             : 
      69        4206 :         *auth_method = result;
      70        4206 :         return NT_STATUS_OK;
      71             : }
      72             : 
      73             : 
      74             : /****************************************************************************
      75             : Check SAM security (above) but with a few extra checks.
      76             : ****************************************************************************/
      77             : 
      78         698 : static NTSTATUS auth_samstrict_auth(const struct auth_context *auth_context,
      79             :                                     void *my_private_data,
      80             :                                     TALLOC_CTX *mem_ctx,
      81             :                                     const struct auth_usersupplied_info *user_info,
      82             :                                     struct auth_serversupplied_info **server_info)
      83             : {
      84         698 :         const char *effective_domain = NULL;
      85             :         bool is_local_name, is_my_domain;
      86             : 
      87         698 :         if (!user_info || !auth_context) {
      88           0 :                 return NT_STATUS_LOGON_FAILURE;
      89             :         }
      90         698 :         effective_domain = user_info->mapped.domain_name;
      91             : 
      92        1065 :         if (user_info->mapped.account_name == NULL ||
      93         698 :             user_info->mapped.account_name[0] == '\0')
      94             :         {
      95           2 :                 return NT_STATUS_NOT_IMPLEMENTED;
      96             :         }
      97             : 
      98         696 :         if (lp_server_role() == ROLE_DOMAIN_MEMBER) {
      99         660 :                 const char *p = NULL;
     100             : 
     101         660 :                 p = strchr_m(user_info->mapped.account_name, '@');
     102         660 :                 if (p != NULL) {
     103             :                         /*
     104             :                          * This needs to go to the DC,
     105             :                          * even if @ is the last character
     106             :                          */
     107          28 :                         return NT_STATUS_NOT_IMPLEMENTED;
     108             :                 }
     109             :         }
     110             : 
     111         668 :         if (effective_domain == NULL) {
     112           0 :                 effective_domain = "";
     113             :         }
     114             : 
     115         668 :         DBG_DEBUG("Check auth for: [%s]\\[%s]\n",
     116             :                   effective_domain,
     117             :                   user_info->mapped.account_name);
     118             : 
     119             : 
     120         668 :         if (strequal(effective_domain, "") || strequal(effective_domain, ".")) {
     121             :                 /*
     122             :                  * An empty domain name or '.' should be handled
     123             :                  * as the local SAM name.
     124             :                  */
     125          56 :                 effective_domain = lp_netbios_name();
     126             :         }
     127             : 
     128         668 :         is_local_name = is_myname(effective_domain);
     129         668 :         is_my_domain  = strequal(effective_domain, lp_workgroup());
     130             : 
     131             :         /* check whether or not we service this domain/workgroup name */
     132             : 
     133         668 :         switch ( lp_server_role() ) {
     134         632 :                 case ROLE_STANDALONE:
     135             :                 case ROLE_DOMAIN_MEMBER:
     136         632 :                         if ( !is_local_name ) {
     137         174 :                                 DEBUG(6,("check_samstrict_security: %s is not one of my local names (%s)\n",
     138             :                                         effective_domain, (lp_server_role() == ROLE_DOMAIN_MEMBER
     139             :                                         ? "ROLE_DOMAIN_MEMBER" : "ROLE_STANDALONE") ));
     140         174 :                                 return NT_STATUS_NOT_IMPLEMENTED;
     141             :                         }
     142             : 
     143         458 :                         break;
     144          36 :                 case ROLE_DOMAIN_PDC:
     145             :                 case ROLE_DOMAIN_BDC:
     146             :                 case ROLE_IPA_DC:
     147          36 :                         if (!is_local_name && !is_my_domain) {
     148             :                                /* If we are running on a DC that has PASSDB module with domain
     149             :                                 * information, check if DNS forest name is matching the domain
     150             :                                 * name. This is the case of IPA domain controller when
     151             :                                 * trusted AD DCs attempt to authenticate IPA users using
     152             :                                 * the forest root domain (which is the only domain in IPA).
     153             :                                 */
     154           0 :                                 struct pdb_domain_info *dom_info = NULL;
     155             : 
     156           0 :                                 dom_info = pdb_get_domain_info(mem_ctx);
     157           0 :                                 if ((dom_info != NULL) && (dom_info->dns_forest != NULL)) {
     158           0 :                                         is_my_domain = strequal(user_info->mapped.domain_name,
     159           0 :                                                                 dom_info->dns_forest);
     160             :                                 }
     161             : 
     162           0 :                                 TALLOC_FREE(dom_info);
     163           0 :                                 if (!is_my_domain) {
     164           0 :                                         DEBUG(6,("check_samstrict_security: %s is not one "
     165             :                                                  "of my local names or domain name (DC)\n",
     166             :                                                  effective_domain));
     167           0 :                                         return NT_STATUS_NOT_IMPLEMENTED;
     168             :                                 }
     169             :                         }
     170             : 
     171          36 :                         break;
     172           0 :                 default: /* name is ok */
     173           0 :                         break;
     174             :         }
     175             : 
     176         494 :         return check_sam_security(&auth_context->challenge, mem_ctx,
     177             :                                   user_info, server_info);
     178             : }
     179             : 
     180             : /* module initialisation */
     181        4122 : static NTSTATUS auth_init_sam(
     182             :         struct auth_context *auth_context,
     183             :         const char *param,
     184             :         struct auth_methods **auth_method)
     185             : {
     186             :         struct auth_methods *result;
     187             : 
     188        4122 :         if (lp_server_role() == ROLE_ACTIVE_DIRECTORY_DC
     189           0 :             && !lp_parm_bool(-1, "server role check", "inhibit", false)) {
     190           0 :                 DEBUG(0, ("server role = 'active directory domain controller' not compatible with running the auth_sam module. \n"));
     191           0 :                 DEBUGADD(0, ("You should not set 'auth methods' when running the AD DC.\n"));
     192           0 :                 exit(1);
     193             :         }
     194             : 
     195        4122 :         result = talloc_zero(auth_context, struct auth_methods);
     196        4122 :         if (result == NULL) {
     197           0 :                 return NT_STATUS_NO_MEMORY;
     198             :         }
     199        4122 :         result->auth = auth_samstrict_auth;
     200        4122 :         result->name = "sam";
     201        4122 :         *auth_method = result;
     202        4122 :         return NT_STATUS_OK;
     203             : }
     204             : 
     205           0 : static NTSTATUS auth_sam_netlogon3_auth(const struct auth_context *auth_context,
     206             :                                         void *my_private_data,
     207             :                                         TALLOC_CTX *mem_ctx,
     208             :                                         const struct auth_usersupplied_info *user_info,
     209             :                                         struct auth_serversupplied_info **server_info)
     210             : {
     211           0 :         const char *effective_domain = NULL;
     212             :         bool is_my_domain;
     213             : 
     214           0 :         if (!user_info || !auth_context) {
     215           0 :                 return NT_STATUS_LOGON_FAILURE;
     216             :         }
     217           0 :         effective_domain = user_info->mapped.domain_name;
     218             : 
     219           0 :         if (user_info->mapped.account_name == NULL ||
     220           0 :             user_info->mapped.account_name[0] == '\0')
     221             :         {
     222           0 :                 return NT_STATUS_NOT_IMPLEMENTED;
     223             :         }
     224             : 
     225           0 :         if (effective_domain == NULL) {
     226           0 :                 effective_domain = "";
     227             :         }
     228             : 
     229           0 :         DBG_DEBUG("Check auth for: [%s]\\[%s]\n",
     230             :                   effective_domain,
     231             :                   user_info->mapped.account_name);
     232             : 
     233             :         /* check whether or not we service this domain/workgroup name */
     234             : 
     235           0 :         switch (lp_server_role()) {
     236           0 :         case ROLE_DOMAIN_PDC:
     237             :         case ROLE_DOMAIN_BDC:
     238             :         case ROLE_IPA_DC:
     239           0 :                 break;
     240           0 :         default:
     241           0 :                 DBG_ERR("Invalid server role\n");
     242           0 :                 return NT_STATUS_INVALID_SERVER_STATE;
     243             :         }
     244             : 
     245           0 :         if (strequal(effective_domain, "") || strequal(effective_domain, ".")) {
     246             :                 /*
     247             :                  * An empty domain name or '.' should be handled
     248             :                  * as the local SAM name.
     249             :                  */
     250           0 :                 effective_domain = lp_workgroup();
     251             :         }
     252             : 
     253           0 :         is_my_domain = strequal(user_info->mapped.domain_name, lp_workgroup());
     254           0 :         if (!is_my_domain) {
     255             :                /* If we are running on a DC that has PASSDB module with domain
     256             :                 * information, check if DNS forest name is matching the domain
     257             :                 * name. This is the case of IPA domain controller when
     258             :                 * trusted AD DCs attempt to authenticate IPA users using
     259             :                 * the forest root domain (which is the only domain in IPA).
     260             :                 */
     261           0 :                 struct pdb_domain_info *dom_info = NULL;
     262           0 :                 dom_info = pdb_get_domain_info(mem_ctx);
     263             : 
     264           0 :                 if ((dom_info != NULL) && (dom_info->dns_forest != NULL)) {
     265           0 :                         is_my_domain = strequal(user_info->mapped.domain_name,
     266           0 :                                                 dom_info->dns_forest);
     267             :                 }
     268             : 
     269           0 :                 TALLOC_FREE(dom_info);
     270             :         }
     271             : 
     272           0 :         if (!is_my_domain) {
     273           0 :                 DBG_INFO("%s is not our domain name (DC for %s)\n",
     274             :                          effective_domain, lp_workgroup());
     275           0 :                 return NT_STATUS_NOT_IMPLEMENTED;
     276             :         }
     277             : 
     278           0 :         return check_sam_security(&auth_context->challenge, mem_ctx,
     279             :                                   user_info, server_info);
     280             : }
     281             : 
     282             : /* module initialisation */
     283           0 : static NTSTATUS auth_init_sam_netlogon3(
     284             :         struct auth_context *auth_context,
     285             :         const char *param,
     286             :         struct auth_methods **auth_method)
     287             : {
     288             :         struct auth_methods *result;
     289             : 
     290           0 :         if (lp_server_role() == ROLE_ACTIVE_DIRECTORY_DC
     291           0 :             && !lp_parm_bool(-1, "server role check", "inhibit", false)) {
     292           0 :                 DEBUG(0, ("server role = 'active directory domain controller' "
     293             :                           "not compatible with running the auth_sam module.\n"));
     294           0 :                 DEBUGADD(0, ("You should not set 'auth methods' when "
     295             :                              "running the AD DC.\n"));
     296           0 :                 exit(1);
     297             :         }
     298             : 
     299           0 :         result = talloc_zero(auth_context, struct auth_methods);
     300           0 :         if (result == NULL) {
     301           0 :                 return NT_STATUS_NO_MEMORY;
     302             :         }
     303           0 :         result->auth = auth_sam_netlogon3_auth;
     304           0 :         result->name = "sam_netlogon3";
     305           0 :         *auth_method = result;
     306           0 :         return NT_STATUS_OK;
     307             : }
     308             : 
     309        5149 : NTSTATUS auth_sam_init(TALLOC_CTX *mem_ctx)
     310             : {
     311        5149 :         smb_register_auth(AUTH_INTERFACE_VERSION, "sam", auth_init_sam);
     312        5149 :         smb_register_auth(AUTH_INTERFACE_VERSION, "sam_ignoredomain", auth_init_sam_ignoredomain);
     313        5149 :         smb_register_auth(AUTH_INTERFACE_VERSION, "sam_netlogon3", auth_init_sam_netlogon3);
     314        5149 :         return NT_STATUS_OK;
     315             : }

Generated by: LCOV version 1.13