Blame SOURCES/dovecot-2.2.36-cve_2019_3814part1of3.patch

11c5e2
From eb5ffe2641febe0fa5e9038f2e216c130e1e7519 Mon Sep 17 00:00:00 2001
11c5e2
From: Aki Tuomi <aki.tuomi@open-xchange.com>
11c5e2
Date: Mon, 21 Jan 2019 11:36:30 +0200
11c5e2
Subject: [PATCH] login-common: Ensure we get username from certificate
11c5e2
11c5e2
---
11c5e2
 src/login-common/sasl-server.c | 42 ++++++++++++++++++++++++++++++++--
11c5e2
 1 file changed, 40 insertions(+), 2 deletions(-)
11c5e2
11c5e2
diff --git a/src/login-common/sasl-server.c b/src/login-common/sasl-server.c
11c5e2
index a833c9a6d4..9465da9657 100644
11c5e2
--- a/src/login-common/sasl-server.c
11c5e2
+++ b/src/login-common/sasl-server.c
11c5e2
@@ -321,6 +321,37 @@ authenticate_callback(struct auth_client_request *request,
11c5e2
 	}
11c5e2
 }
11c5e2
 
11c5e2
+static bool get_cert_username(struct client *client, const char **username_r,
11c5e2
+			      const char **error_r)
11c5e2
+{
11c5e2
+	/* no SSL */
11c5e2
+	if (client->ssl_proxy == NULL) {
11c5e2
+		*username_r = NULL;
11c5e2
+		return TRUE;
11c5e2
+	}
11c5e2
+
11c5e2
+	/* no client certificate */
11c5e2
+	if (!ssl_proxy_has_valid_client_cert(client->ssl_proxy)) {
11c5e2
+		*username_r = NULL;
11c5e2
+		return TRUE;
11c5e2
+	}
11c5e2
+
11c5e2
+	/* get peer name */
11c5e2
+	const char *username = ssl_proxy_get_peer_name(client->ssl_proxy);
11c5e2
+
11c5e2
+	/* if we wanted peer name, but it was not there, fail */
11c5e2
+	if (client->set->auth_ssl_username_from_cert &&
11c5e2
+	    (username == NULL || *username == '\0')) {
11c5e2
+		if (client->set->auth_ssl_require_client_cert) {
11c5e2
+			*error_r = "Missing username in certificate";
11c5e2
+			return FALSE;
11c5e2
+		}
11c5e2
+	}
11c5e2
+
11c5e2
+	*username_r = username;
11c5e2
+	return TRUE;
11c5e2
+}
11c5e2
+
11c5e2
 void sasl_server_auth_begin(struct client *client,
11c5e2
 			    const char *service, const char *mech_name,
11c5e2
 			    const char *initial_resp_base64,
11c5e2
@@ -359,8 +390,15 @@ void sasl_server_auth_begin(struct client *client,
11c5e2
 	info.mech = mech->name;
11c5e2
 	info.service = service;
11c5e2
 	info.session_id = client_get_session_id(client);
11c5e2
-	info.cert_username = client->ssl_proxy == NULL ? NULL :
11c5e2
-		ssl_proxy_get_peer_name(client->ssl_proxy);
11c5e2
+	if (client->set->auth_ssl_username_from_cert) {
11c5e2
+		const char *error;
11c5e2
+		if (!get_cert_username(client, &info.cert_username, &error)) {
11c5e2
+			client_log_err(client, t_strdup_printf("Cannot get username "
11c5e2
+							       "from certificate: %s", error));
11c5e2
+			sasl_server_auth_failed(client, "Unable to validate certificate");
11c5e2
+			return;
11c5e2
+		}
11c5e2
+	}
11c5e2
 	info.flags = client_get_auth_flags(client);
11c5e2
 	info.local_ip = client->local_ip;
11c5e2
 	info.remote_ip = client->ip;