/* * 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: 2025-10-29 * * 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 #include #include #include #include #include #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(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(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; } }