267 lines
		
	
	
		
			7.3 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			267 lines
		
	
	
		
			7.3 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| /*
 | |
|  * Copyright (c) 2016 MariaDB Corporation Ab
 | |
|  *
 | |
|  * Use of this software is governed by the Business Source License included
 | |
|  * in the LICENSE.TXT file and at www.mariadb.com/bsl11.
 | |
|  *
 | |
|  * Change Date: 2023-11-05
 | |
|  *
 | |
|  * On the date above, in accordance with the Business Source License, use
 | |
|  * of this software will be governed by version 2 or later of the General
 | |
|  * Public License.
 | |
|  */
 | |
| 
 | |
| #define MXS_MODULE_NAME "GSSAPIBackendAuth"
 | |
| 
 | |
| #include <maxscale/ccdefs.hh>
 | |
| #include <maxbase/alloc.h>
 | |
| #include <maxscale/authenticator.hh>
 | |
| #include <maxscale/dcb.hh>
 | |
| #include <maxscale/protocol/mysql.hh>
 | |
| #include <maxscale/server.hh>
 | |
| 
 | |
| #include "../gssapi_auth.hh"
 | |
| 
 | |
| /**
 | |
|  * @file gssapi_backend_auth.c - GSSAPI backend authenticator
 | |
|  */
 | |
| 
 | |
| void* gssapi_backend_auth_alloc(void* instance)
 | |
| {
 | |
|     gssapi_auth_t* rval = static_cast<gssapi_auth_t*>(MXS_MALLOC(sizeof(gssapi_auth_t)));
 | |
| 
 | |
|     if (rval)
 | |
|     {
 | |
|         rval->state = GSSAPI_AUTH_INIT;
 | |
|         rval->principal_name = NULL;
 | |
|         rval->principal_name_len = 0;
 | |
|         rval->sequence = 0;
 | |
|     }
 | |
| 
 | |
|     return rval;
 | |
| }
 | |
| 
 | |
| void gssapi_backend_auth_free(void* data)
 | |
| {
 | |
|     if (data)
 | |
|     {
 | |
|         gssapi_auth_t* auth = (gssapi_auth_t*)data;
 | |
|         MXS_FREE(auth->principal_name);
 | |
|         MXS_FREE(auth);
 | |
|     }
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * @brief Create a new GSSAPI token
 | |
|  * @param dcb Backend DCB
 | |
|  * @return True on success, false on error
 | |
|  */
 | |
| static bool send_new_auth_token(DCB* dcb)
 | |
| {
 | |
|     bool rval = false;
 | |
|     gssapi_auth_t* auth = (gssapi_auth_t*)dcb->authenticator_data;
 | |
|     MYSQL_session* ses = (MYSQL_session*)dcb->session->client_dcb->data;
 | |
|     GWBUF* buffer = gwbuf_alloc(MYSQL_HEADER_LEN + ses->auth_token_len);
 | |
| 
 | |
|     // This function actually just forwards the client's token to the backend server
 | |
| 
 | |
|     if (buffer)
 | |
|     {
 | |
|         uint8_t* data = (uint8_t*)GWBUF_DATA(buffer);
 | |
|         gw_mysql_set_byte3(data, ses->auth_token_len);
 | |
|         data += 3;
 | |
|         *data++ = ++auth->sequence;
 | |
|         memcpy(data, ses->auth_token, ses->auth_token_len);
 | |
| 
 | |
|         if (dcb_write(dcb, buffer))
 | |
|         {
 | |
|             rval = true;
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     return rval;
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * @brief Extract the principal name from the AuthSwitchRequest packet
 | |
|  *
 | |
|  * @param dcb Backend DCB
 | |
|  * @param buffer Buffer containing an AuthSwitchRequest packet
 | |
|  * @return True on success, false on error
 | |
|  */
 | |
| bool extract_principal_name(DCB* dcb, GWBUF* buffer)
 | |
| {
 | |
|     bool rval = false;
 | |
|     size_t buflen = gwbuf_length(buffer) - MYSQL_HEADER_LEN;
 | |
|     uint8_t databuf[buflen];
 | |
|     uint8_t* data = databuf;
 | |
|     gssapi_auth_t* auth = (gssapi_auth_t*)dcb->authenticator_data;
 | |
| 
 | |
|     /** Copy the payload and the current packet sequence number */
 | |
|     gwbuf_copy_data(buffer, MYSQL_HEADER_LEN, buflen, databuf);
 | |
|     gwbuf_copy_data(buffer, MYSQL_SEQ_OFFSET, 1, &auth->sequence);
 | |
| 
 | |
|     if (databuf[0] != MYSQL_REPLY_AUTHSWITCHREQUEST)
 | |
|     {
 | |
|         /** Server responded with something we did not expect. If it's an OK packet,
 | |
|          * it's possible that the server authenticated us as the anonymous user. This
 | |
|          * means that the server is not secure. */
 | |
|         MXS_ERROR("Server '%s' returned an unexpected authentication response.%s",
 | |
|                   dcb->server->name(),
 | |
|                   databuf[0] == MYSQL_REPLY_OK ?
 | |
|                   " Authentication was complete before it even started, "
 | |
|                   "anonymous users might not be disabled." : "");
 | |
|         return false;
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * The AuthSwitchRequest packet
 | |
|      *
 | |
|      * 0xfe        - Command byte
 | |
|      * string[NUL] - Auth plugin name
 | |
|      * string[EOF] - Auth plugin data
 | |
|      *
 | |
|      * Skip over the auth plugin name and copy the service principal name stored
 | |
|      * in the auth plugin data section.
 | |
|      */
 | |
|     while (*data && data < databuf + buflen)
 | |
|     {
 | |
|         data++;
 | |
|     }
 | |
| 
 | |
|     data++;
 | |
|     buflen -= data - databuf;
 | |
| 
 | |
|     if (buflen > 0)
 | |
|     {
 | |
|         uint8_t* principal = static_cast<uint8_t*>(MXS_MALLOC(buflen + 1));
 | |
| 
 | |
|         if (principal)
 | |
|         {
 | |
|             /** Store the principal name for later when we request the token
 | |
|              * from the GSSAPI server */
 | |
|             memcpy(principal, data, buflen);
 | |
|             principal[buflen] = '\0';
 | |
|             auth->principal_name = principal;
 | |
|             auth->principal_name_len = buflen;
 | |
|             rval = true;
 | |
|         }
 | |
|     }
 | |
|     else
 | |
|     {
 | |
|         MXS_ERROR("Backend server did not send any auth plugin data.");
 | |
|     }
 | |
| 
 | |
|     return rval;
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * @brief Extract data from a MySQL packet
 | |
|  * @param dcb Backend DCB
 | |
|  * @param buffer Buffer containing a complete packet
 | |
|  * @return True if authentication is ongoing or complete,
 | |
|  * false if authentication failed.
 | |
|  */
 | |
| static bool gssapi_backend_auth_extract(DCB* dcb, GWBUF* buffer)
 | |
| {
 | |
|     bool rval = false;
 | |
|     gssapi_auth_t* auth = (gssapi_auth_t*)dcb->authenticator_data;
 | |
| 
 | |
|     if (auth->state == GSSAPI_AUTH_INIT && extract_principal_name(dcb, buffer))
 | |
|     {
 | |
|         rval = true;
 | |
|     }
 | |
|     else if (auth->state == GSSAPI_AUTH_DATA_SENT)
 | |
|     {
 | |
|         /** Read authentication response */
 | |
|         if (mxs_mysql_is_ok_packet(buffer))
 | |
|         {
 | |
|             auth->state = GSSAPI_AUTH_OK;
 | |
|             rval = true;
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     return rval;
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * @brief Check whether the DCB supports SSL
 | |
|  * @param dcb Backend DCB
 | |
|  * @return True if DCB supports SSL
 | |
|  */
 | |
| static bool gssapi_backend_auth_connectssl(DCB* dcb)
 | |
| {
 | |
|     return dcb->server->ssl().context() != NULL;
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * @brief Authenticate the backend connection
 | |
|  * @param dcb Backend DCB
 | |
|  * @return MXS_AUTH_INCOMPLETE if authentication is ongoing, MXS_AUTH_SUCCEEDED
 | |
|  * if authentication is complete and MXS_AUTH_FAILED if authentication failed.
 | |
|  */
 | |
| static int gssapi_backend_auth_authenticate(DCB* dcb)
 | |
| {
 | |
|     int rval = MXS_AUTH_FAILED;
 | |
|     gssapi_auth_t* auth = (gssapi_auth_t*)dcb->authenticator_data;
 | |
| 
 | |
|     if (auth->state == GSSAPI_AUTH_INIT)
 | |
|     {
 | |
|         if (send_new_auth_token(dcb))
 | |
|         {
 | |
|             rval = MXS_AUTH_INCOMPLETE;
 | |
|             auth->state = GSSAPI_AUTH_DATA_SENT;
 | |
|         }
 | |
|     }
 | |
|     else if (auth->state == GSSAPI_AUTH_OK)
 | |
|     {
 | |
|         rval = MXS_AUTH_SUCCEEDED;
 | |
|     }
 | |
| 
 | |
|     return rval;
 | |
| }
 | |
| 
 | |
| extern "C"
 | |
| {
 | |
| /**
 | |
|  * Module handle entry point
 | |
|  */
 | |
| MXS_MODULE* MXS_CREATE_MODULE()
 | |
| {
 | |
|     static MXS_AUTHENTICATOR MyObject =
 | |
|     {
 | |
|         NULL,                                   /* No initialize entry point */
 | |
|         gssapi_backend_auth_alloc,              /* Allocate authenticator data */
 | |
|         gssapi_backend_auth_extract,            /* Extract data into structure   */
 | |
|         gssapi_backend_auth_connectssl,         /* Check if client supports SSL  */
 | |
|         gssapi_backend_auth_authenticate,       /* Authenticate user credentials */
 | |
|         NULL,                                   /* Client plugin will free shared data */
 | |
|         gssapi_backend_auth_free,               /* Free authenticator data */
 | |
|         NULL,                                   /* Load users from backend databases */
 | |
|         NULL,                                   /* No diagnostic */
 | |
|         NULL,
 | |
|         NULL                                /* No user reauthentication */
 | |
|     };
 | |
| 
 | |
|     static MXS_MODULE info =
 | |
|     {
 | |
|         MXS_MODULE_API_AUTHENTICATOR,
 | |
|         MXS_MODULE_GA,
 | |
|         MXS_AUTHENTICATOR_VERSION,
 | |
|         "GSSAPI backend authenticator",
 | |
|         "V1.0.0",
 | |
|         MXS_NO_MODULE_CAPABILITIES,
 | |
|         &MyObject,
 | |
|         NULL,       /* Process init. */
 | |
|         NULL,       /* Process finish. */
 | |
|         NULL,       /* Thread init. */
 | |
|         NULL,       /* Thread finish. */
 | |
|         {
 | |
|             {MXS_END_MODULE_PARAMS}
 | |
|         }
 | |
|     };
 | |
| 
 | |
|     return &info;
 | |
| }
 | |
| }
 | 
