mirror of
				https://github.com/asterisk/asterisk.git
				synced 2025-10-26 14:27:14 +00:00 
			
		
		
		
	The verification process will now load a full certificate chain retrieved via the X5U URL instead of loading only the end user cert. * Renamed crypto_load_cert_from_file() and crypto_load_cert_from_memory() to crypto_load_cert_chain_from_file() and crypto_load_cert_chain_from_memory() respectively. * The two load functions now continue to load certs from the file or memory PEMs and store them in a separate stack of untrusted certs specific to the current verification context. * crypto_is_cert_trusted() now uses the stack of untrusted certs that were extracted from the PEM in addition to any untrusted certs that were passed in from the configuration (and any CA certs passed in from the config of course). Resolves: #1272 UserNote: The STIR/SHAKEN verification process will now load a full certificate chain retrieved via the X5U URL instead of loading only the end user cert.
		
			
				
	
	
		
			1137 lines
		
	
	
		
			36 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			1137 lines
		
	
	
		
			36 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| /*
 | |
|  * Asterisk -- An open source telephony toolkit.
 | |
|  *
 | |
|  * Copyright (C) 2023, Sangoma Technologies Corporation
 | |
|  *
 | |
|  * George Joseph <gjoseph@sangoma.com>
 | |
|  *
 | |
|  * See http://www.asterisk.org for more information about
 | |
|  * the Asterisk project. Please do not directly contact
 | |
|  * any of the maintainers of this project for assistance;
 | |
|  * the project provides a web site, mailing lists and IRC
 | |
|  * channels for your use.
 | |
|  *
 | |
|  * This program is free software, distributed under the terms of
 | |
|  * the GNU General Public License Version 2. See the LICENSE file
 | |
|  * at the top of the source tree.
 | |
|  */
 | |
| #include <curl/curl.h>
 | |
| #include <sys/stat.h>
 | |
| 
 | |
| #include <jwt.h>
 | |
| #include <regex.h>
 | |
| 
 | |
| #include "asterisk.h"
 | |
| 
 | |
| #define _TRACE_PREFIX_ "v",__LINE__, ""
 | |
| 
 | |
| #include "asterisk/channel.h"
 | |
| #include "asterisk/cli.h"
 | |
| #include "asterisk/config.h"
 | |
| #include "asterisk/module.h"
 | |
| #include "asterisk/sorcery.h"
 | |
| #include "asterisk/astdb.h"
 | |
| #include "asterisk/conversions.h"
 | |
| #include "asterisk/utils.h"
 | |
| #include "asterisk/paths.h"
 | |
| #include "asterisk/logger.h"
 | |
| #include "asterisk/acl.h"
 | |
| #include "asterisk/time.h"
 | |
| #include "asterisk/localtime.h"
 | |
| #include "asterisk/crypto.h"
 | |
| #include "asterisk/json.h"
 | |
| 
 | |
| #include "stir_shaken.h"
 | |
| 
 | |
| #define AST_DB_FAMILY "STIR_SHAKEN"
 | |
| 
 | |
| static regex_t url_match_regex;
 | |
| 
 | |
| /* Certificates should begin with this */
 | |
| #define BEGIN_CERTIFICATE_STR "-----BEGIN CERTIFICATE-----"
 | |
| 
 | |
| static const char *vs_rc_map[] = {
 | |
| 	[AST_STIR_SHAKEN_VS_SUCCESS] = "success",
 | |
| 	[AST_STIR_SHAKEN_VS_DISABLED] = "disabled",
 | |
| 	[AST_STIR_SHAKEN_VS_INVALID_ARGUMENTS] = "invalid_arguments",
 | |
| 	[AST_STIR_SHAKEN_VS_INTERNAL_ERROR] = "internal_error",
 | |
| 	[AST_STIR_SHAKEN_VS_NO_IDENTITY_HDR] = "missing_identity_hdr",
 | |
| 	[AST_STIR_SHAKEN_VS_NO_DATE_HDR] = "missing_date_hdr",
 | |
| 	[AST_STIR_SHAKEN_VS_DATE_HDR_PARSE_FAILURE] = "date_hdr_parse_failure",
 | |
| 	[AST_STIR_SHAKEN_VS_DATE_HDR_EXPIRED] = "date_hdr_range_error",
 | |
| 	[AST_STIR_SHAKEN_VS_NO_JWT_HDR] = "missing_jwt_hdr",
 | |
| 	[AST_STIR_SHAKEN_VS_CERT_CACHE_MISS] = "cert_cache_miss",
 | |
| 	[AST_STIR_SHAKEN_VS_CERT_CACHE_INVALID] = "cert_cache_invalid",
 | |
| 	[AST_STIR_SHAKEN_VS_CERT_CACHE_EXPIRED] = "cert_cache_expired",
 | |
| 	[AST_STIR_SHAKEN_VS_CERT_RETRIEVAL_FAILURE] = "cert_retrieval_failure",
 | |
| 	[AST_STIR_SHAKEN_VS_CERT_CONTENTS_INVALID] = "cert_contents_invalid",
 | |
| 	[AST_STIR_SHAKEN_VS_CERT_NOT_TRUSTED] = "cert_not_trusted",
 | |
| 	[AST_STIR_SHAKEN_VS_CERT_DATE_INVALID] = "cert_date_failure",
 | |
| 	[AST_STIR_SHAKEN_VS_CERT_NO_TN_AUTH_EXT] = "cert_no_tn_auth_ext",
 | |
| 	[AST_STIR_SHAKEN_VS_CERT_NO_SPC_IN_TN_AUTH_EXT] = "cert_no_spc_in_auth_ext",
 | |
| 	[AST_STIR_SHAKEN_VS_NO_RAW_KEY] = "no_raw_key",
 | |
| 	[AST_STIR_SHAKEN_VS_SIGNATURE_VALIDATION] = "signature_validation",
 | |
| 	[AST_STIR_SHAKEN_VS_NO_IAT] = "missing_iat",
 | |
| 	[AST_STIR_SHAKEN_VS_IAT_EXPIRED] = "iat_range_error",
 | |
| 	[AST_STIR_SHAKEN_VS_INVALID_OR_NO_PPT] = "invalid_or_no_ppt",
 | |
| 	[AST_STIR_SHAKEN_VS_INVALID_OR_NO_ALG] = "invalid_or_no_alg",
 | |
| 	[AST_STIR_SHAKEN_VS_INVALID_OR_NO_TYP] = "invalid_or_no_typ",
 | |
| 	[AST_STIR_SHAKEN_VS_INVALID_OR_NO_GRANTS] = "invalid_or_no_grants",
 | |
| 	[AST_STIR_SHAKEN_VS_INVALID_OR_NO_ATTEST] = "invalid_or_no_attest",
 | |
| 	[AST_STIR_SHAKEN_VS_NO_ORIGID] = "missing_origid",
 | |
| 	[AST_STIR_SHAKEN_VS_NO_ORIG_TN] = "missing_orig_tn",
 | |
| 	[AST_STIR_SHAKEN_VS_CID_ORIG_TN_MISMATCH] = "cid_orig_tn_mismatch",
 | |
| 	[AST_STIR_SHAKEN_VS_NO_DEST_TN] = "missing_dest_tn",
 | |
| 	[AST_STIR_SHAKEN_VS_INVALID_HEADER] = "invalid_header",
 | |
| 	[AST_STIR_SHAKEN_VS_INVALID_GRANT] = "invalid_grant",
 | |
| 	[AST_STIR_SHAKEN_VS_INVALID_OR_NO_CID] = "invalid_or_no_callerid",
 | |
| };
 | |
| 
 | |
| const char *vs_response_code_to_str(
 | |
| 	enum ast_stir_shaken_vs_response_code vs_rc)
 | |
| {
 | |
| 	return ARRAY_IN_BOUNDS(vs_rc, vs_rc_map) ?
 | |
| 		vs_rc_map[vs_rc] : NULL;
 | |
| }
 | |
| 
 | |
| static void cleanup_cert_from_astdb_and_fs(
 | |
| 	struct ast_stir_shaken_vs_ctx *ctx)
 | |
| {
 | |
| 	if (ast_db_exists(ctx->hash_family, "path") || ast_db_exists(ctx->hash_family, "expiration")) {
 | |
| 		ast_db_deltree(ctx->hash_family, NULL);
 | |
| 	}
 | |
| 
 | |
| 	if (ast_db_exists(ctx->url_family, ctx->public_url)) {
 | |
| 		ast_db_del(ctx->url_family, ctx->public_url);
 | |
| 	}
 | |
| 
 | |
| 	/* Remove the actual file from the system */
 | |
| 	remove(ctx->filename);
 | |
| }
 | |
| 
 | |
| static int add_cert_expiration_to_astdb(struct ast_stir_shaken_vs_ctx *cert,
 | |
| 	const char *cache_control_header, const char *expires_header)
 | |
| {
 | |
| 	RAII_VAR(struct verification_cfg *, cfg, vs_get_cfg(), ao2_cleanup);
 | |
| 
 | |
| 	char time_buf[32];
 | |
| 	time_t current_time = time(NULL);
 | |
| 	time_t max_age_hdr = 0;
 | |
| 	time_t expires_hdr = 0;
 | |
| 	ASN1_TIME *notAfter = NULL;
 | |
| 	time_t cert_expires = 0;
 | |
| 	time_t config_expires = 0;
 | |
| 	time_t expires = 0;
 | |
| 	int rc = 0;
 | |
| 
 | |
| 	config_expires = current_time + cfg->vcfg_common.max_cache_entry_age;
 | |
| 
 | |
| 	if (!ast_strlen_zero(cache_control_header)) {
 | |
| 		char *str_max_age;
 | |
| 
 | |
| 		str_max_age = strstr(cache_control_header, "s-maxage");
 | |
| 		if (!str_max_age) {
 | |
| 			str_max_age = strstr(cache_control_header, "max-age");
 | |
| 		}
 | |
| 
 | |
| 		if (str_max_age) {
 | |
| 			unsigned int m;
 | |
| 			char *equal = strchr(str_max_age, '=');
 | |
| 			if (equal && !ast_str_to_uint(equal + 1, &m)) {
 | |
| 				max_age_hdr = current_time + m;
 | |
| 			}
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	if (!ast_strlen_zero(expires_header)) {
 | |
| 		struct ast_tm expires_time;
 | |
| 
 | |
| 		ast_strptime(expires_header, "%a, %d %b %Y %T %z", &expires_time);
 | |
| 		expires_time.tm_isdst = -1;
 | |
| 		expires_hdr = ast_mktime(&expires_time, "GMT").tv_sec;
 | |
| 	}
 | |
| 
 | |
| 	notAfter = X509_get_notAfter(cert->xcert);
 | |
| 	cert_expires = crypto_asn_time_as_time_t(notAfter);
 | |
| 
 | |
| 	/*
 | |
| 	 * ATIS-1000074 says:
 | |
| 	 * The STI-VS shall implement the cache behavior described in
 | |
| 	 * [Ref 10]. If the HTTP response does not include any recognized
 | |
| 	 * caching directives or indicates caching for less than 24 hours,
 | |
| 	 * then the STI-VS should cache the HTTP response for 24 hours.
 | |
| 	 *
 | |
| 	 * Basically, they're saying "cache for 24 hours unless the HTTP
 | |
| 	 * response says to cache for longer."  Instead of the fixed 24
 | |
| 	 * hour minumum, however, we'll use max_cache_entry_age instead.
 | |
| 	 *
 | |
| 	 * We got all the possible values of expires so let's find the
 | |
| 	 * highest value greater than the configured max_cache_entry_age.
 | |
| 	 */
 | |
| 
 | |
| 	/* The default */
 | |
| 	expires = config_expires;
 | |
| 
 | |
| 	if (max_age_hdr > expires) {
 | |
| 		expires = max_age_hdr;
 | |
| 	}
 | |
| 
 | |
| 	if (expires_hdr > expires) {
 | |
| 		expires = expires_hdr;
 | |
| 	}
 | |
| 
 | |
| 	/*
 | |
| 	 * However...  Don't cache for longer than the
 | |
| 	 * certificate is actually valid.
 | |
| 	 */
 | |
| 	if (cert_expires && cert_expires < expires) {
 | |
| 		expires = cert_expires;
 | |
| 	}
 | |
| 
 | |
| 	snprintf(time_buf, sizeof(time_buf), "%ld", expires);
 | |
| 
 | |
| 	rc = ast_db_put(cert->hash_family, "expiration", time_buf);
 | |
| 	if (rc == 0) {
 | |
| 		strcpy(cert->expiration, time_buf); /* safe */
 | |
| 	}
 | |
| 
 | |
| 	return rc;
 | |
| }
 | |
| 
 | |
| static int add_cert_key_to_astdb(struct ast_stir_shaken_vs_ctx *cert,
 | |
| 	const char *cache_control_hdr, const char *expires_hdr)
 | |
| {
 | |
| 	int rc = 0;
 | |
| 
 | |
| 	rc = ast_db_put(cert->url_family, cert->public_url, cert->hash);
 | |
| 	if (rc) {
 | |
| 		return rc;
 | |
| 	}
 | |
| 	rc = ast_db_put(cert->hash_family, "path", cert->filename);
 | |
| 	if (rc) {
 | |
| 		ast_db_del(cert->url_family, cert->public_url);
 | |
| 		return rc;
 | |
| 	}
 | |
| 
 | |
| 	rc = add_cert_expiration_to_astdb(cert, cache_control_hdr, expires_hdr);
 | |
| 	if (rc) {
 | |
| 		ast_db_del(cert->url_family, cert->public_url);
 | |
| 		ast_db_del(cert->hash_family, "path");
 | |
| 	}
 | |
| 
 | |
| 	return rc;
 | |
| }
 | |
| 
 | |
| static int is_cert_cache_entry_expired(char *expiration)
 | |
| {
 | |
| 	struct timeval current_time = ast_tvnow();
 | |
| 	struct timeval expires = { .tv_sec = 0, .tv_usec = 0 };
 | |
| 	int res = 0;
 | |
| 	SCOPE_ENTER(3, "Checking for cache expiration: %s\n", expiration);
 | |
| 
 | |
| 	if (ast_strlen_zero(expiration)) {
 | |
| 		SCOPE_EXIT_RTN_VALUE(1, "No expiration date provided\n");
 | |
| 	}
 | |
| 
 | |
| 	if (ast_str_to_ulong(expiration, (unsigned long *)&expires.tv_sec)) {
 | |
| 		SCOPE_EXIT_RTN_VALUE(1, "Couldn't convert expiration string '%s' to ulong",
 | |
| 			expiration);
 | |
| 	}
 | |
| 	ast_trace(2, "Expiration comparison: exp: %" PRIu64 "  curr: %" PRIu64 "  Diff: %" PRIu64 ".\n",
 | |
| 		expires.tv_sec, current_time.tv_sec, expires.tv_sec - current_time.tv_sec);
 | |
| 
 | |
| 	res = (ast_tvcmp(current_time, expires) == -1 ? 0 : 1);
 | |
| 	SCOPE_EXIT_RTN_VALUE(res , "entry was %sexpired\n", res ? "" : "not ");
 | |
| }
 | |
| 
 | |
| #define ASN1_TAG_TNAUTH_SPC 0
 | |
| #define ASN1_TAG_TNAUTH_TN_RANGE 1
 | |
| #define ASN1_TAG_TNAUTH_TN 2
 | |
| 
 | |
| #define IS_GET_OBJ_ERR(ret) (ret & 0x80)
 | |
| 
 | |
| static enum ast_stir_shaken_vs_response_code
 | |
| 	check_tn_auth_list(struct ast_stir_shaken_vs_ctx * ctx)
 | |
| {
 | |
| 	ASN1_OCTET_STRING *tn_exten;
 | |
| 	const unsigned char* octet_str_data = NULL;
 | |
| 	long xlen;
 | |
| 	int tag, xclass;
 | |
| 	int ret;
 | |
| 	SCOPE_ENTER(3, "%s: Checking TNAuthList in cert '%s'\n", ctx->tag, ctx->public_url);
 | |
| 
 | |
| 	tn_exten = crypto_get_cert_extension_data(ctx->xcert, get_tn_auth_nid(), NULL);
 | |
| 	if (!tn_exten) {
 | |
| 		SCOPE_EXIT_LOG_RTN_VALUE(AST_STIR_SHAKEN_VS_CERT_NO_TN_AUTH_EXT,
 | |
| 			LOG_ERROR, "%s: Cert '%s' doesn't have a TNAuthList extension\n",
 | |
| 			ctx->tag, ctx->public_url);
 | |
| 	}
 | |
| 	octet_str_data = tn_exten->data;
 | |
| 
 | |
| 	/* The first call to ASN1_get_object should return a SEQUENCE */
 | |
| 	ret = ASN1_get_object(&octet_str_data, &xlen, &tag, &xclass, tn_exten->length);
 | |
| 	if (IS_GET_OBJ_ERR(ret)) {
 | |
| 		crypto_log_openssl(LOG_ERROR, "%s: Cert '%s' has malformed TNAuthList extension\n",
 | |
| 			ctx->tag, ctx->public_url);
 | |
| 		SCOPE_EXIT_RTN_VALUE(AST_STIR_SHAKEN_VS_CERT_NO_TN_AUTH_EXT);
 | |
| 	}
 | |
| 
 | |
| 	if (ret != V_ASN1_CONSTRUCTED || tag != V_ASN1_SEQUENCE) {
 | |
| 		SCOPE_EXIT_LOG_RTN_VALUE(AST_STIR_SHAKEN_VS_CERT_NO_TN_AUTH_EXT,
 | |
| 			LOG_ERROR, "%s: Cert '%s' has malformed TNAuthList extension (tag %d != V_ASN1_SEQUENCE)\n",
 | |
| 			ctx->tag, ctx->public_url, tag);
 | |
| 	}
 | |
| 
 | |
| 	/*
 | |
| 	 * The second call to ASN1_get_object should return one of
 | |
| 	 * the following tags defined in RFC8226 section 9:
 | |
| 	 *
 | |
| 	 * ASN1_TAG_TNAUTH_SPC 0
 | |
| 	 * ASN1_TAG_TNAUTH_TN_RANGE 1
 | |
| 	 * ASN1_TAG_TNAUTH_TN 2
 | |
| 	 *
 | |
| 	 * ATIS-1000080 however limits this to only ASN1_TAG_TNAUTH_SPC
 | |
| 	 *
 | |
| 	 */
 | |
| 	ret = ASN1_get_object(&octet_str_data, &xlen, &tag, &xclass, tn_exten->length);
 | |
| 	if (IS_GET_OBJ_ERR(ret)) {
 | |
| 		crypto_log_openssl(LOG_ERROR, "%s: Cert '%s' has malformed TNAuthList extension\n",
 | |
| 			ctx->tag, ctx->public_url);
 | |
| 		SCOPE_EXIT_RTN_VALUE(AST_STIR_SHAKEN_VS_CERT_NO_TN_AUTH_EXT);
 | |
| 	}
 | |
| 
 | |
| 	if (ret != V_ASN1_CONSTRUCTED || tag != ASN1_TAG_TNAUTH_SPC) {
 | |
| 		SCOPE_EXIT_LOG_RTN_VALUE(AST_STIR_SHAKEN_VS_CERT_NO_SPC_IN_TN_AUTH_EXT,
 | |
| 			LOG_ERROR, "%s: Cert '%s' has malformed TNAuthList extension (tag %d != ASN1_TAG_TNAUTH_SPC(0))\n",
 | |
| 			ctx->tag, ctx->public_url, tag);
 | |
| 	}
 | |
| 
 | |
| 	/* The third call to ASN1_get_object should contain the SPC */
 | |
| 	ret = ASN1_get_object(&octet_str_data, &xlen, &tag, &xclass, tn_exten->length);
 | |
| 	if (ret != 0) {
 | |
| 		SCOPE_EXIT_LOG_RTN_VALUE(AST_STIR_SHAKEN_VS_CERT_NO_SPC_IN_TN_AUTH_EXT,
 | |
| 			LOG_ERROR, "%s: Cert '%s' has malformed TNAuthList extension (no SPC)\n",
 | |
| 			ctx->tag, ctx->public_url);
 | |
| 	}
 | |
| 
 | |
| 	if (ast_string_field_set(ctx, cert_spc, (char *)octet_str_data) != 0) {
 | |
| 		SCOPE_EXIT_RTN_VALUE(AST_STIR_SHAKEN_VS_INTERNAL_ERROR);
 | |
| 	}
 | |
| 
 | |
| 	SCOPE_EXIT_RTN_VALUE(AST_STIR_SHAKEN_VS_SUCCESS, "%s: Cert '%s' with SPC: %s CN: %s has valid TNAuthList\n",
 | |
| 		ctx->tag, ctx->public_url, ctx->cert_spc, ctx->cert_cn);
 | |
| }
 | |
| #undef IS_GET_OBJ_ERR
 | |
| 
 | |
| static enum ast_stir_shaken_vs_response_code check_cert(
 | |
| 	struct ast_stir_shaken_vs_ctx * ctx)
 | |
| {
 | |
| 	RAII_VAR(char *, CN, NULL, ast_free);
 | |
| 	int res = 0;
 | |
| 	const char *err_msg;
 | |
| 	SCOPE_ENTER(3, "%s: Validating cert '%s'\n", ctx->tag, ctx->public_url);
 | |
| 
 | |
| 	CN = crypto_get_cert_subject(ctx->xcert, "CN");
 | |
| 	if (!CN) {
 | |
| 		CN = crypto_get_cert_subject(ctx->xcert, NULL);
 | |
| 		SCOPE_EXIT_LOG_RTN_VALUE(AST_STIR_SHAKEN_VS_CERT_CONTENTS_INVALID,
 | |
| 			LOG_ERROR, "%s: Cert '%s' has no commonName(CN) in Subject '%s'\n",
 | |
| 			ctx->tag, ctx->public_url, CN);
 | |
| 	}
 | |
| 
 | |
| 	res = ast_string_field_set(ctx, cert_cn, CN);
 | |
| 	if (res != 0) {
 | |
| 		SCOPE_EXIT_RTN_VALUE(AST_STIR_SHAKEN_VS_INTERNAL_ERROR);
 | |
| 	}
 | |
| 
 | |
| 	ast_trace(3,"%s: Checking ctx against CA ctx\n", ctx->tag);
 | |
| 	res = crypto_is_cert_trusted(ctx->eprofile->vcfg_common.tcs, ctx->xcert,
 | |
| 		ctx->cert_chain, &err_msg);
 | |
| 	if (!res) {
 | |
| 		SCOPE_EXIT_LOG_RTN_VALUE(AST_STIR_SHAKEN_VS_CERT_NOT_TRUSTED,
 | |
| 			LOG_ERROR, "%s: Cert '%s' not trusted: %s\n",
 | |
| 			ctx->tag, ctx->public_url, err_msg);
 | |
| 	}
 | |
| 
 | |
| 	ast_trace(3,"%s: Attempting to get the raw pubkey\n", ctx->tag);
 | |
| 	ctx->raw_key_len = crypto_get_raw_pubkey_from_cert(ctx->xcert,
 | |
| 		&ctx->raw_key);
 | |
| 	if (ctx->raw_key_len <= 0) {
 | |
| 		SCOPE_EXIT_LOG_RTN_VALUE(AST_STIR_SHAKEN_VS_NO_RAW_KEY,
 | |
| 			LOG_ERROR, "%s: Unable to extract raw public key from '%s'\n",
 | |
| 			ctx->tag, ctx->public_url);
 | |
| 	}
 | |
| 
 | |
| 	ast_trace(3,"%s: Checking cert '%s' validity dates\n",
 | |
| 		ctx->tag, ctx->public_url);
 | |
| 	if (!crypto_is_cert_time_valid(ctx->xcert, ctx->validity_check_time)) {
 | |
| 		SCOPE_EXIT_LOG_RTN_VALUE(AST_STIR_SHAKEN_VS_CERT_DATE_INVALID,
 | |
| 			LOG_ERROR, "%s: Cert '%s' dates not valid\n",
 | |
| 			ctx->tag, ctx->public_url);
 | |
| 	}
 | |
| 
 | |
| 	SCOPE_EXIT_RTN_VALUE(check_tn_auth_list(ctx),
 | |
| 		"%s: Cert '%s' with SPC: %s CN: %s is valid\n",
 | |
| 		ctx->tag, ctx->public_url, ctx->cert_spc, ctx->cert_cn);
 | |
| }
 | |
| 
 | |
| static enum ast_stir_shaken_vs_response_code retrieve_cert_from_url(
 | |
| 		struct ast_stir_shaken_vs_ctx *ctx)
 | |
| {
 | |
| 	FILE *cert_file;
 | |
| 	long http_code;
 | |
| 	int rc = 0;
 | |
| 	enum ast_stir_shaken_vs_response_code vs_rc;
 | |
| 	RAII_VAR(struct curl_header_data *, header_data,
 | |
| 		ast_calloc(1, sizeof(*header_data)), curl_header_data_free);
 | |
| 	RAII_VAR(struct curl_write_data *, write_data,
 | |
| 		ast_calloc(1, sizeof(*write_data)), curl_write_data_free);
 | |
| 	RAII_VAR(struct curl_open_socket_data *, open_socket_data,
 | |
| 		ast_calloc(1, sizeof(*open_socket_data)), curl_open_socket_data_free);
 | |
| 
 | |
| 	const char *cache_control;
 | |
| 	const char *expires;
 | |
| 	SCOPE_ENTER(2, "%s: Attempting to retrieve '%s' from net\n",
 | |
| 		ctx->tag, ctx->public_url);
 | |
| 
 | |
| 	if (!header_data || !write_data || !open_socket_data) {
 | |
| 		SCOPE_EXIT_LOG_RTN_VALUE(AST_STIR_SHAKEN_VS_INTERNAL_ERROR,
 | |
| 			LOG_ERROR, "%s: Unable to allocate memory for curl '%s' transaction\n",
 | |
| 			ctx->tag, ctx->public_url);
 | |
| 	}
 | |
| 
 | |
| 	header_data->debug_info = ast_strdup(ctx->public_url);
 | |
| 	write_data->debug_info = ast_strdup(ctx->public_url);
 | |
| 	write_data->max_download_bytes = 8192;
 | |
| 	write_data->stream_buffer = NULL;
 | |
| 	open_socket_data->debug_info = ast_strdup(ctx->public_url);
 | |
| 	open_socket_data->acl = ctx->eprofile->vcfg_common.acl;
 | |
| 
 | |
| 	if (!header_data->debug_info || !write_data->debug_info ||
 | |
| 		!open_socket_data->debug_info) {
 | |
| 		SCOPE_EXIT_LOG_RTN_VALUE(AST_STIR_SHAKEN_VS_INTERNAL_ERROR,
 | |
| 			LOG_ERROR, "%s: Unable to allocate memory for curl '%s' transaction\n",
 | |
| 			ctx->tag, ctx->public_url);
 | |
| 	}
 | |
| 
 | |
| 	http_code = curler(ctx->public_url,
 | |
| 		ctx->eprofile->vcfg_common.curl_timeout,
 | |
| 		write_data, header_data, open_socket_data);
 | |
| 
 | |
| 	if (http_code / 100 != 2) {
 | |
| 		SCOPE_EXIT_LOG_RTN_VALUE(AST_STIR_SHAKEN_VS_CERT_RETRIEVAL_FAILURE,
 | |
| 			LOG_ERROR, "%s: Failed to retrieve cert %s: code %ld\n",
 | |
| 			ctx->tag, ctx->public_url, http_code);
 | |
| 	}
 | |
| 
 | |
| 	if (!ast_begins_with(write_data->stream_buffer, BEGIN_CERTIFICATE_STR)) {
 | |
| 		SCOPE_EXIT_LOG_RTN_VALUE(AST_STIR_SHAKEN_VS_CERT_CONTENTS_INVALID,
 | |
| 			LOG_ERROR, "%s: Cert '%s' contains invalid data\n",
 | |
| 			ctx->tag, ctx->public_url);
 | |
| 	}
 | |
| 
 | |
| 	ctx->xcert = crypto_load_cert_chain_from_memory(write_data->stream_buffer,
 | |
| 		write_data->stream_bytes_downloaded, &ctx->cert_chain);
 | |
| 	if (!ctx->xcert) {
 | |
| 		SCOPE_EXIT_LOG_RTN_VALUE(AST_STIR_SHAKEN_VS_CERT_CONTENTS_INVALID,
 | |
| 			LOG_ERROR, "%s: Cert '%s' was not parseable as an X509 certificate\n",
 | |
| 			ctx->tag, ctx->public_url);
 | |
| 	}
 | |
| 
 | |
| 	vs_rc = check_cert(ctx);
 | |
| 	if (vs_rc != AST_STIR_SHAKEN_VS_SUCCESS) {
 | |
| 		X509_free(ctx->xcert);
 | |
| 		ctx->xcert = NULL;
 | |
| 		SCOPE_EXIT_RTN_VALUE(vs_rc, "%s: Cert '%s' failed validity checks\n",
 | |
| 			ctx->tag, ctx->public_url);
 | |
| 	}
 | |
| 
 | |
| 	cert_file = fopen(ctx->filename, "w");
 | |
| 	if (!cert_file) {
 | |
| 		X509_free(ctx->xcert);
 | |
| 		ctx->xcert = NULL;
 | |
| 		SCOPE_EXIT_LOG_RTN_VALUE(AST_STIR_SHAKEN_VS_INTERNAL_ERROR,
 | |
| 			LOG_ERROR, "%s: Failed to write cert %s: file '%s' %s (%d)\n",
 | |
| 			ctx->tag, ctx->public_url, ctx->filename, strerror(errno), errno);
 | |
| 	}
 | |
| 
 | |
| 	rc = fputs(write_data->stream_buffer, cert_file);
 | |
| 	fclose(cert_file);
 | |
| 	if (rc == EOF) {
 | |
| 		X509_free(ctx->xcert);
 | |
| 		ctx->xcert = NULL;
 | |
| 		SCOPE_EXIT_LOG_RTN_VALUE(AST_STIR_SHAKEN_VS_INTERNAL_ERROR,
 | |
| 			LOG_ERROR, "%s: Failed to write cert %s: file '%s' %s (%d)\n",
 | |
| 			ctx->tag, ctx->public_url, ctx->filename, strerror(errno), errno);
 | |
| 	}
 | |
| 
 | |
| 	ast_trace(2, "%s: Cert '%s' written to file '%s'\n",
 | |
| 		ctx->tag, ctx->public_url, ctx->filename);
 | |
| 
 | |
| 	ast_trace(2, "%s: Adding cert '%s' to astdb",
 | |
| 		ctx->tag, ctx->public_url);
 | |
| 	cache_control = ast_variable_find_in_list(header_data->headers, "cache-control");
 | |
| 	expires = ast_variable_find_in_list(header_data->headers, "expires");
 | |
| 
 | |
| 	rc = add_cert_key_to_astdb(ctx, cache_control, expires);
 | |
| 	if (rc) {
 | |
| 		X509_free(ctx->xcert);
 | |
| 		ctx->xcert = NULL;
 | |
| 		SCOPE_EXIT_LOG_RTN_VALUE(AST_STIR_SHAKEN_VS_INTERNAL_ERROR,
 | |
| 			LOG_ERROR, "%s: Unable to add cert '%s' to ASTDB\n",
 | |
| 			ctx->tag, ctx->public_url);
 | |
| 	}
 | |
| 
 | |
| 	SCOPE_EXIT_RTN_VALUE(AST_STIR_SHAKEN_VS_SUCCESS,
 | |
| 		"%s: Cert '%s' successfully retrieved from internet and cached\n",
 | |
| 		ctx->tag, ctx->public_url);
 | |
| }
 | |
| 
 | |
| static enum ast_stir_shaken_vs_response_code
 | |
| 	retrieve_cert_from_cache(struct ast_stir_shaken_vs_ctx *ctx)
 | |
| {
 | |
| 	int rc = 0;
 | |
| 	enum ast_stir_shaken_vs_response_code vs_rc;
 | |
| 
 | |
| 	SCOPE_ENTER(2, "%s: Attempting to retrieve cert '%s' from cache\n",
 | |
| 		ctx->tag, ctx->public_url);
 | |
| 
 | |
| 	if (!ast_db_exists(ctx->hash_family, "path")) {
 | |
| 		cleanup_cert_from_astdb_and_fs(ctx);
 | |
| 		SCOPE_EXIT_RTN_VALUE(AST_STIR_SHAKEN_VS_CERT_CACHE_MISS,
 | |
| 			"%s: No cert found in astdb for '%s'\n",
 | |
| 			ctx->tag, ctx->public_url);
 | |
| 	}
 | |
| 
 | |
| 	rc = ast_db_get(ctx->hash_family, "expiration", ctx->expiration, sizeof(ctx->expiration));
 | |
| 	if (rc) {
 | |
| 		cleanup_cert_from_astdb_and_fs(ctx);
 | |
| 		SCOPE_EXIT_RTN_VALUE(AST_STIR_SHAKEN_VS_CERT_CACHE_MISS,
 | |
| 			"%s: No cert found in astdb for '%s'\n",
 | |
| 			ctx->tag, ctx->public_url);
 | |
| 	}
 | |
| 
 | |
| 	if (!ast_file_is_readable(ctx->filename)) {
 | |
| 		cleanup_cert_from_astdb_and_fs(ctx);
 | |
| 		SCOPE_EXIT_RTN_VALUE(AST_STIR_SHAKEN_VS_CERT_CACHE_MISS,
 | |
| 			"%s: Cert file '%s' was not found or was not readable for '%s'\n",
 | |
| 			ctx->tag, ctx->filename, ctx->public_url);
 | |
| 	}
 | |
| 
 | |
| 	if (is_cert_cache_entry_expired(ctx->expiration)) {
 | |
| 		cleanup_cert_from_astdb_and_fs(ctx);
 | |
| 		SCOPE_EXIT_RTN_VALUE(AST_STIR_SHAKEN_VS_CERT_CACHE_EXPIRED,
 | |
| 			"%s: Cert file '%s' cache entry was expired for '%s'\n",
 | |
| 			ctx->tag, ctx->filename, ctx->public_url);
 | |
| 	}
 | |
| 
 | |
| 	ctx->xcert = crypto_load_cert_chain_from_file(ctx->filename, &ctx->cert_chain);
 | |
| 	if (!ctx->xcert) {
 | |
| 		cleanup_cert_from_astdb_and_fs(ctx);
 | |
| 		SCOPE_EXIT_RTN_VALUE(AST_STIR_SHAKEN_VS_CERT_CONTENTS_INVALID,
 | |
| 			"%s: Cert file '%s' was not parseable as an X509 certificate for '%s'\n",
 | |
| 			ctx->tag, ctx->filename, ctx->public_url);
 | |
| 	}
 | |
| 
 | |
| 	vs_rc = check_cert(ctx);
 | |
| 	if (vs_rc != AST_STIR_SHAKEN_VS_SUCCESS) {
 | |
| 		X509_free(ctx->xcert);
 | |
| 		ctx->xcert = NULL;
 | |
| 		SCOPE_EXIT_RTN_VALUE(vs_rc, "%s: Cert '%s' failed validity checks\n",
 | |
| 			ctx->tag, ctx->public_url);
 | |
| 	}
 | |
| 
 | |
| 	SCOPE_EXIT_RTN_VALUE(AST_STIR_SHAKEN_VS_SUCCESS,
 | |
| 		"%s: Cert '%s' successfully retrieved from cache\n",
 | |
| 		ctx->tag, ctx->public_url);
 | |
| }
 | |
| 
 | |
| static enum ast_stir_shaken_vs_response_code ctx_populate(
 | |
| 	struct ast_stir_shaken_vs_ctx *ctx)
 | |
| {
 | |
| 	char hash[41];
 | |
| 
 | |
| 	ast_sha1_hash(hash, ctx->public_url);
 | |
| 	if (ast_string_field_set(ctx, hash, hash) != 0) {
 | |
| 		return AST_STIR_SHAKEN_VS_INTERNAL_ERROR;
 | |
| 	}
 | |
| 
 | |
| 	if (ast_string_field_build(ctx, filename, "%s/%s.pem",
 | |
| 		ctx->eprofile->vcfg_common.cert_cache_dir, hash) != 0) {
 | |
| 		return AST_STIR_SHAKEN_VS_INTERNAL_ERROR;
 | |
| 	}
 | |
| 
 | |
| 	if (ast_string_field_build(ctx, hash_family, "%s/hash/%s",
 | |
| 		AST_DB_FAMILY, hash) != 0) {
 | |
| 		return AST_STIR_SHAKEN_VS_INTERNAL_ERROR;
 | |
| 	}
 | |
| 
 | |
| 	if (ast_string_field_build(ctx, url_family, "%s/url", AST_DB_FAMILY) != 0) {
 | |
| 		return AST_STIR_SHAKEN_VS_INTERNAL_ERROR;
 | |
| 	}
 | |
| 
 | |
| 	return AST_STIR_SHAKEN_VS_SUCCESS;
 | |
| }
 | |
| 
 | |
| static enum ast_stir_shaken_vs_response_code
 | |
| 	retrieve_verification_cert(struct ast_stir_shaken_vs_ctx *ctx)
 | |
| {
 | |
| 	enum ast_stir_shaken_vs_response_code rc = AST_STIR_SHAKEN_VS_SUCCESS;
 | |
| 	SCOPE_ENTER(3, "%s: Retrieving cert '%s'\n", ctx->tag, ctx->public_url);
 | |
| 
 | |
| 	ast_trace(1, "%s: Checking cache for cert '%s'\n", ctx->tag, ctx->public_url);
 | |
| 	rc = retrieve_cert_from_cache(ctx);
 | |
| 	if (rc == AST_STIR_SHAKEN_VS_SUCCESS) {
 | |
| 		SCOPE_EXIT_RTN_VALUE(rc, "%s: Using cert '%s' from cache\n",
 | |
| 			ctx->tag, ctx->public_url);;
 | |
| 	}
 | |
| 
 | |
| 	ast_trace(1, "%s: No valid cert for '%s' available in cache\n",
 | |
| 		ctx->tag, ctx->public_url);
 | |
| 	ast_trace(1, "%s: Retrieving cert directly from url '%s'\n",
 | |
| 		ctx->tag, ctx->public_url);
 | |
| 
 | |
| 	rc = retrieve_cert_from_url(ctx);
 | |
| 	if (rc == AST_STIR_SHAKEN_VS_SUCCESS) {
 | |
| 		SCOPE_EXIT_RTN_VALUE(rc, "%s: Using cert '%s' from internet\n",
 | |
| 			ctx->tag, ctx->public_url);
 | |
| 	}
 | |
| 
 | |
| 	SCOPE_EXIT_LOG_RTN_VALUE(rc, LOG_ERROR,
 | |
| 		"%s: Unable to retrieve cert '%s' from cache or internet\n",
 | |
| 		ctx->tag, ctx->public_url);
 | |
| }
 | |
| 
 | |
| enum ast_stir_shaken_vs_response_code
 | |
| 	ast_stir_shaken_vs_ctx_add_identity_hdr(
 | |
| 	struct ast_stir_shaken_vs_ctx * ctx, const char *identity_hdr)
 | |
| {
 | |
| 	return ast_string_field_set(ctx, identity_hdr, identity_hdr) == 0 ?
 | |
| 		AST_STIR_SHAKEN_VS_SUCCESS : AST_STIR_SHAKEN_VS_INTERNAL_ERROR;
 | |
| }
 | |
| 
 | |
| enum ast_stir_shaken_vs_response_code
 | |
| 	ast_stir_shaken_vs_ctx_add_date_hdr(struct ast_stir_shaken_vs_ctx * ctx,
 | |
| 	const char *date_hdr)
 | |
| {
 | |
| 	return ast_string_field_set(ctx, date_hdr, date_hdr) == 0 ?
 | |
| 		AST_STIR_SHAKEN_VS_SUCCESS : AST_STIR_SHAKEN_VS_INTERNAL_ERROR;
 | |
| }
 | |
| 
 | |
| enum stir_shaken_failure_action_enum
 | |
| 	ast_stir_shaken_vs_get_failure_action(
 | |
| 		struct ast_stir_shaken_vs_ctx *ctx)
 | |
| {
 | |
| 	return ctx->eprofile->vcfg_common.stir_shaken_failure_action;
 | |
| }
 | |
| 
 | |
| int ast_stir_shaken_vs_get_use_rfc9410_responses(
 | |
| 		struct ast_stir_shaken_vs_ctx *ctx)
 | |
| {
 | |
| 	return ctx->eprofile->vcfg_common.use_rfc9410_responses;
 | |
| }
 | |
| 
 | |
| const char *ast_stir_shaken_vs_get_caller_id(
 | |
| 		struct ast_stir_shaken_vs_ctx *ctx)
 | |
| {
 | |
| 	return ctx->caller_id;
 | |
| }
 | |
| 
 | |
| void ast_stir_shaken_vs_ctx_set_response_code(
 | |
| 	struct ast_stir_shaken_vs_ctx *ctx,
 | |
| 	enum ast_stir_shaken_vs_response_code vs_rc)
 | |
| {
 | |
| 	ctx->failure_reason = vs_rc;
 | |
| }
 | |
| 
 | |
| static void ctx_destructor(void *obj)
 | |
| {
 | |
| 	struct ast_stir_shaken_vs_ctx *ctx = obj;
 | |
| 
 | |
| 	ao2_cleanup(ctx->eprofile);
 | |
| 	ast_free(ctx->raw_key);
 | |
| 	ast_string_field_free_memory(ctx);
 | |
| 	X509_free(ctx->xcert);
 | |
| 	sk_X509_free(ctx->cert_chain);
 | |
| }
 | |
| 
 | |
| enum ast_stir_shaken_vs_response_code
 | |
| 	ast_stir_shaken_vs_ctx_create(const char *caller_id,
 | |
| 		struct ast_channel *chan, const char *profile_name,
 | |
| 		const char *tag, struct ast_stir_shaken_vs_ctx **ctxout)
 | |
| {
 | |
| 	RAII_VAR(struct ast_stir_shaken_vs_ctx *, ctx, NULL, ao2_cleanup);
 | |
| 	RAII_VAR(struct profile_cfg *, profile, NULL, ao2_cleanup);
 | |
| 	RAII_VAR(struct verification_cfg *, vs, NULL, ao2_cleanup);
 | |
| 	RAII_VAR(char *, canon_caller_id , canonicalize_tn_alloc(caller_id), ast_free);
 | |
| 
 | |
| 	const char *t = S_OR(tag, S_COR(chan, ast_channel_name(chan), ""));
 | |
| 	SCOPE_ENTER(3, "%s: Enter\n", t);
 | |
| 
 | |
| 	vs = vs_get_cfg();
 | |
| 	if (vs->global_disable) {
 | |
| 		SCOPE_EXIT_RTN_VALUE(AST_STIR_SHAKEN_VS_DISABLED,
 | |
| 			"%s: Globally disabled\n", t);
 | |
| 	}
 | |
| 
 | |
| 	if (ast_strlen_zero(profile_name)) {
 | |
| 		SCOPE_EXIT_RTN_VALUE(AST_STIR_SHAKEN_VS_DISABLED,
 | |
| 			"%s: Disabled due to missing profile name\n", t);
 | |
| 	}
 | |
| 
 | |
| 	profile = eprofile_get_cfg(profile_name);
 | |
| 	if (!profile) {
 | |
| 		SCOPE_EXIT_LOG_RTN_VALUE(AST_STIR_SHAKEN_VS_DISABLED,
 | |
| 		LOG_ERROR, "%s: No profile for profile name '%s'.  Call will continue\n", tag,
 | |
| 			profile_name);
 | |
| 	}
 | |
| 
 | |
| 	if (!PROFILE_ALLOW_VERIFY(profile)) {
 | |
| 		SCOPE_EXIT_RTN_VALUE(AST_STIR_SHAKEN_VS_DISABLED,
 | |
| 			"%s: Disabled by profile '%s'\n", t, profile_name);
 | |
| 	}
 | |
| 
 | |
| 	if (ast_strlen_zero(tag)) {
 | |
| 		SCOPE_EXIT_LOG_RTN_VALUE(AST_STIR_SHAKEN_VS_INVALID_ARGUMENTS,
 | |
| 			LOG_ERROR, "%s: Must provide tag\n", t);
 | |
| 	}
 | |
| 
 | |
| 	ctx = ao2_alloc_options(sizeof(*ctx), ctx_destructor,
 | |
| 		AO2_ALLOC_OPT_LOCK_NOLOCK);
 | |
| 	if (!ctx) {
 | |
| 		SCOPE_EXIT_RTN_VALUE(AST_STIR_SHAKEN_VS_INTERNAL_ERROR);
 | |
| 	}
 | |
| 	if (ast_string_field_init(ctx, 1024) != 0) {
 | |
| 		SCOPE_EXIT_RTN_VALUE(AST_STIR_SHAKEN_VS_INTERNAL_ERROR);
 | |
| 	}
 | |
| 
 | |
| 	if (ast_string_field_set(ctx, tag, tag) != 0) {
 | |
| 		SCOPE_EXIT_RTN_VALUE(AST_STIR_SHAKEN_VS_INTERNAL_ERROR);
 | |
| 	}
 | |
| 
 | |
| 	ctx->chan = chan;
 | |
| 	if (ast_string_field_set(ctx, caller_id, canon_caller_id) != 0) {
 | |
| 		SCOPE_EXIT_RTN_VALUE(AST_STIR_SHAKEN_VS_INTERNAL_ERROR);
 | |
| 	}
 | |
| 
 | |
| 	/* Transfer references to ctx */
 | |
| 	ctx->eprofile = profile;
 | |
| 	profile = NULL;
 | |
| 
 | |
| 	ao2_ref(ctx, +1);
 | |
| 	*ctxout = ctx;
 | |
| 	SCOPE_EXIT_RTN_VALUE(AST_STIR_SHAKEN_VS_SUCCESS, "%s: Done\n", t);
 | |
| }
 | |
| 
 | |
| static enum ast_stir_shaken_vs_response_code check_date_header(
 | |
| 	struct ast_stir_shaken_vs_ctx * ctx)
 | |
| {
 | |
| 	struct ast_tm date_hdr_tm;
 | |
| 	struct timeval date_hdr_timeval;
 | |
| 	struct timeval current_timeval;
 | |
| 	char *remainder;
 | |
| 	char timezone[80] = { 0 };
 | |
| 	int64_t time_diff;
 | |
| 	SCOPE_ENTER(3, "%s: Checking date header: '%s'\n",
 | |
| 		ctx->tag, ctx->date_hdr);
 | |
| 
 | |
| 	if (ast_strlen_zero(ctx->date_hdr)) {
 | |
| 		if (ctx->eprofile->vcfg_common.ignore_sip_date_header) {
 | |
| 			SCOPE_EXIT_RTN_VALUE(AST_STIR_SHAKEN_VS_SUCCESS,
 | |
| 				"%s: ignore_sip_date_header set\n", ctx->tag);
 | |
| 		}
 | |
| 		SCOPE_EXIT_LOG_RTN_VALUE(AST_STIR_SHAKEN_VS_NO_DATE_HDR,
 | |
| 			LOG_ERROR, "%s: No date header provided\n", ctx->tag);
 | |
| 	}
 | |
| 
 | |
| 	if (!(remainder = ast_strptime(ctx->date_hdr, "%a, %d %b %Y %T", &date_hdr_tm))) {
 | |
| 		SCOPE_EXIT_LOG_RTN_VALUE(AST_STIR_SHAKEN_VS_DATE_HDR_PARSE_FAILURE,
 | |
| 			LOG_ERROR, "%s: Failed to parse: '%s'\n",
 | |
| 			ctx->tag, ctx->date_hdr);
 | |
| 	}
 | |
| 
 | |
| 	sscanf(remainder, "%79s", timezone);
 | |
| 
 | |
| 	if (ast_strlen_zero(timezone)) {
 | |
| 		SCOPE_EXIT_LOG_RTN_VALUE(AST_STIR_SHAKEN_VS_DATE_HDR_PARSE_FAILURE,
 | |
| 			LOG_ERROR, "%s: A timezone is required: '%s'\n",
 | |
| 			ctx->tag, ctx->date_hdr);
 | |
| 	}
 | |
| 
 | |
| 	date_hdr_timeval = ast_mktime(&date_hdr_tm, timezone);
 | |
| 	ctx->date_hdr_time = date_hdr_timeval.tv_sec;
 | |
| 	current_timeval = ast_tvnow();
 | |
| 
 | |
| 	time_diff = ast_tvdiff_ms(current_timeval, date_hdr_timeval);
 | |
| 	ast_trace(3, "%zu  %zu  %zu %d\n", current_timeval.tv_sec,
 | |
| 		date_hdr_timeval.tv_sec,
 | |
| 		(current_timeval.tv_sec - date_hdr_timeval.tv_sec), (int)time_diff);
 | |
| 	if (time_diff < 0) {
 | |
| 		/* An INVITE from the future! */
 | |
| 		SCOPE_EXIT_LOG_RTN_VALUE(AST_STIR_SHAKEN_VS_DATE_HDR_EXPIRED,
 | |
| 			LOG_ERROR, "%s: Future date: '%s'\n",
 | |
| 			ctx->tag, ctx->date_hdr);
 | |
| 	} else if (time_diff > (ctx->eprofile->vcfg_common.max_date_header_age * 1000)) {
 | |
| 		SCOPE_EXIT_LOG_RTN_VALUE(AST_STIR_SHAKEN_VS_DATE_HDR_EXPIRED,
 | |
| 			LOG_ERROR, "%s: More than %u seconds old: '%s'\n",
 | |
| 			ctx->tag, ctx->eprofile->vcfg_common.max_date_header_age, ctx->date_hdr);
 | |
| 	}
 | |
| 
 | |
| 	SCOPE_EXIT_RTN_VALUE(AST_STIR_SHAKEN_VS_SUCCESS,
 | |
| 		"%s: Success: '%s'\n", ctx->tag, ctx->date_hdr);
 | |
| }
 | |
| 
 | |
| #define FULL_URL_REGEX   "^([a-zA-Z]+)://(([^@]+@[^:]+):)?(([^:/?]+)|([0-9.]+)|([[][0-9a-fA-F:]+[]]))(:([0-9]+))?(/([^#\\?]+))?(\\?([^#]+))?(#(.*))?"
 | |
| #define FULL_URL_REGEX_GROUPS 15
 | |
| /*
 | |
|  * Broken down...
 | |
|  * ^([a-zA-Z]+)           must start with scheme    group 1
 | |
|  * ://
 | |
|  * (([^@]+@[^:]+):)?      optional user@pass        group 3
 | |
|  * (                      start hostname group      group 4
 | |
|  * ([^:/?]+)              normal fqdn               group 5
 | |
|  * |([0-9.]+)             OR IPv4 address           group 6
 | |
|  * |([[][0-9a-fA-F:]+[]]) OR IPv6 address           group 7
 | |
|  * )                      end hostname group
 | |
|  * (:([0-9]+))?           optional port             group 9
 | |
|  * (/([^#\?]+))?          optional path             group 11
 | |
|  * (\?([^#]+))?           optional query string     group 13
 | |
|  * (#([^?]+))?            optional fagment          group 15
 | |
|  *
 | |
|  * If you change the regex, make sure FULL_URL_REGEX_GROUPS is updated.
 | |
|  */
 | |
| #define URL_MATCH_SCHEME   1
 | |
| #define URL_MATCH_USERPASS 3
 | |
| #define URL_MATCH_HOST     4
 | |
| #define URL_MATCH_PORT     9
 | |
| #define URL_MATCH_PATH     11
 | |
| #define URL_MATCH_QUERY    13
 | |
| #define URL_MATCH_FRAGMENT 15
 | |
| 
 | |
| #define get_match_string(__x5u, __pmatch, __i) \
 | |
| ({ \
 | |
|  	char *__match = NULL; \
 | |
|  	if (__pmatch[__i].rm_so >= 0) { \
 | |
| 		regoff_t __len = __pmatch[__i].rm_eo - __pmatch[__i].rm_so; \
 | |
| 		const char *__start = __x5u + __pmatch[__i].rm_so; \
 | |
| 		__match = ast_alloca(__len + 1); \
 | |
| 		ast_copy_string(__match, __start, __len + 1); \
 | |
| 	} \
 | |
| 	__match; \
 | |
| })
 | |
| 
 | |
| #define DUMP_X5U_MATCH() \
 | |
| {\
 | |
| 	int i; \
 | |
| 	if (TRACE_ATLEAST(4)) { \
 | |
| 		ast_trace(-1, "%s: x5u: %s\n", ctx->tag, x5u); \
 | |
| 		for (i=0;i<FULL_URL_REGEX_GROUPS;i++) { \
 | |
| 			const char *m = get_match_string(x5u, pmatch, i); \
 | |
| 			if (m) { \
 | |
| 				ast_trace(-1, "%s: %2d %s\n", ctx->tag, i, m); \
 | |
| 			} \
 | |
| 		} \
 | |
| 	} \
 | |
| }
 | |
| 
 | |
| static int check_x5u_url(struct ast_stir_shaken_vs_ctx * ctx,
 | |
| 	const char *x5u)
 | |
| {
 | |
| 	int max_groups = url_match_regex.re_nsub + 1;
 | |
| 	regmatch_t pmatch[max_groups];
 | |
| 	int rc;
 | |
| 	SCOPE_ENTER(3, "%s: Checking x5u '%s'\n", ctx->tag, x5u);
 | |
| 
 | |
| 	rc = regexec(&url_match_regex, x5u, max_groups, pmatch, 0);
 | |
| 	if (rc) {
 | |
| 		char regex_error[512];
 | |
| 		regerror(rc, &url_match_regex, regex_error, sizeof(regex_error));
 | |
| 		SCOPE_EXIT_LOG_RTN_VALUE(AST_STIR_SHAKEN_VS_INVALID_OR_NO_X5U, LOG_ERROR,
 | |
| 			"%s: x5u '%s' in Identity header failed basic URL validation: %s\n",
 | |
| 			ctx->tag, x5u, regex_error);
 | |
| 	}
 | |
| 
 | |
| 	if (ctx->eprofile->vcfg_common.relax_x5u_port_scheme_restrictions
 | |
| 		!= relax_x5u_port_scheme_restrictions_YES) {
 | |
| 		const char *scheme = get_match_string(x5u, pmatch, URL_MATCH_SCHEME);
 | |
| 		const char *port = get_match_string(x5u, pmatch, URL_MATCH_PORT);
 | |
| 
 | |
| 		if (!ast_strings_equal(scheme, "https")) {
 | |
| 			DUMP_X5U_MATCH();
 | |
| 			SCOPE_EXIT_LOG_RTN_VALUE(AST_STIR_SHAKEN_VS_INVALID_OR_NO_X5U, LOG_ERROR,
 | |
| 				"%s: x5u '%s': scheme '%s' not https\n",
 | |
| 				ctx->tag, x5u, scheme);
 | |
| 		}
 | |
| 		if (!ast_strlen_zero(port)) {
 | |
| 			if (!ast_strings_equal(port, "443")
 | |
| 				&& !ast_strings_equal(port, "8443")) {
 | |
| 				DUMP_X5U_MATCH();
 | |
| 				SCOPE_EXIT_LOG_RTN_VALUE(AST_STIR_SHAKEN_VS_INVALID_OR_NO_X5U, LOG_ERROR,
 | |
| 					"%s: x5u '%s': port '%s' not port 443 or 8443\n",
 | |
| 					ctx->tag, x5u, port);
 | |
| 			}
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	if (ctx->eprofile->vcfg_common.relax_x5u_path_restrictions
 | |
| 		!= relax_x5u_path_restrictions_YES) {
 | |
| 		const char *userpass = get_match_string(x5u, pmatch, URL_MATCH_USERPASS);
 | |
| 		const char *qs = get_match_string(x5u, pmatch, URL_MATCH_QUERY);
 | |
| 		const char *frag = get_match_string(x5u, pmatch, URL_MATCH_FRAGMENT);
 | |
| 
 | |
| 		if (!ast_strlen_zero(userpass) || !ast_strlen_zero(qs)
 | |
| 			|| !ast_strlen_zero(frag)) {
 | |
| 			DUMP_X5U_MATCH();
 | |
| 			SCOPE_EXIT_LOG_RTN_VALUE(AST_STIR_SHAKEN_VS_INVALID_OR_NO_X5U, LOG_ERROR,
 | |
| 				"%s: x5u '%s' contains user:password, query parameters or fragment\n",
 | |
| 				ctx->tag, x5u);
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	return 0;
 | |
| }
 | |
| 
 | |
| enum ast_stir_shaken_vs_response_code
 | |
| 	ast_stir_shaken_vs_verify(struct ast_stir_shaken_vs_ctx * ctx)
 | |
| {
 | |
| 	RAII_VAR(char *, jwt_encoded, NULL, ast_free);
 | |
| 	RAII_VAR(jwt_t *, jwt, NULL, jwt_free);
 | |
| 	RAII_VAR(struct ast_json *, grants, NULL, ast_json_unref);
 | |
| 	char *p = NULL;
 | |
| 	char *grants_str = NULL;
 | |
| 	const char *x5u;
 | |
| 	const char *ppt_header = NULL;
 | |
| 	const char *grant = NULL;
 | |
| 	time_t now_s = time(NULL);
 | |
| 	time_t iat;
 | |
| 	struct ast_json *grant_obj = NULL;
 | |
| 	int len;
 | |
| 	int rc;
 | |
| 	enum ast_stir_shaken_vs_response_code vs_rc;
 | |
| 	SCOPE_ENTER(3, "%s: Verifying\n", ctx ? ctx->tag : "NULL");
 | |
| 
 | |
| 	if (!ctx) {
 | |
| 		SCOPE_EXIT_LOG_RTN_VALUE(AST_STIR_SHAKEN_VS_INTERNAL_ERROR, LOG_ERROR,
 | |
| 			"%s: No context object!\n", "NULL");
 | |
| 	}
 | |
| 
 | |
| 	if (ast_strlen_zero(ctx->identity_hdr)) {
 | |
| 		SCOPE_EXIT_LOG_RTN_VALUE(AST_STIR_SHAKEN_VS_INTERNAL_ERROR, LOG_ERROR,
 | |
| 			"%s: No identity header in ctx\n", ctx->tag);
 | |
| 	}
 | |
| 
 | |
| 	p = strchr(ctx->identity_hdr, ';');
 | |
| 	len = p - ctx->identity_hdr + 1;
 | |
| 	jwt_encoded = ast_malloc(len);
 | |
| 	if (!jwt_encoded) {
 | |
| 		SCOPE_EXIT_LOG_RTN_VALUE(AST_STIR_SHAKEN_VS_INTERNAL_ERROR, LOG_ERROR,
 | |
| 			"%s: Failed to allocate memory for encoded jwt\n", ctx->tag);
 | |
| 	}
 | |
| 
 | |
| 	memcpy(jwt_encoded, ctx->identity_hdr, len);
 | |
| 	jwt_encoded[len - 1] = '\0';
 | |
| 
 | |
| 	jwt_decode(&jwt, jwt_encoded, NULL, 0);
 | |
| 
 | |
| 	ppt_header = jwt_get_header(jwt, "ppt");
 | |
| 	if (!ppt_header || strcmp(ppt_header, STIR_SHAKEN_PPT)) {
 | |
| 		SCOPE_EXIT_RTN_VALUE(AST_STIR_SHAKEN_VS_INVALID_OR_NO_PPT, "%s: %s\n",
 | |
| 			ctx->tag, vs_response_code_to_str(AST_STIR_SHAKEN_VS_INVALID_OR_NO_PPT));
 | |
| 	}
 | |
| 
 | |
| 	vs_rc = check_date_header(ctx);
 | |
| 	if (vs_rc != AST_STIR_SHAKEN_VS_SUCCESS) {
 | |
| 		SCOPE_EXIT_LOG_RTN_VALUE(vs_rc, LOG_ERROR,
 | |
| 			"%s: Date header verification failed\n", ctx->tag);
 | |
| 	}
 | |
| 
 | |
| 	x5u = jwt_get_header(jwt, "x5u");
 | |
| 	if (ast_strlen_zero(x5u)) {
 | |
| 		SCOPE_EXIT_LOG_RTN_VALUE(AST_STIR_SHAKEN_VS_INVALID_OR_NO_X5U, LOG_ERROR,
 | |
| 			"%s: No x5u in Identity header\n", ctx->tag);
 | |
| 	}
 | |
| 
 | |
| 	vs_rc = check_x5u_url(ctx, x5u);
 | |
| 	if (vs_rc != AST_STIR_SHAKEN_VS_SUCCESS) {
 | |
| 		SCOPE_EXIT_RTN_VALUE(vs_rc,
 | |
| 			"%s: x5u URL verification failed\n", ctx->tag);
 | |
| 	}
 | |
| 
 | |
| 	ast_trace(3, "%s: Decoded enough to get x5u: '%s'\n", ctx->tag, x5u);
 | |
| 	if (ast_string_field_set(ctx, public_url, x5u) != 0) {
 | |
| 		SCOPE_EXIT_LOG_RTN_VALUE(AST_STIR_SHAKEN_VS_INTERNAL_ERROR, LOG_ERROR,
 | |
| 			"%s: Failed to set public_url '%s'\n", ctx->tag, x5u);
 | |
| 	}
 | |
| 
 | |
| 	iat = jwt_get_grant_int(jwt, "iat");
 | |
| 	if (iat == 0) {
 | |
| 		SCOPE_EXIT_LOG_RTN_VALUE(AST_STIR_SHAKEN_VS_NO_IAT, LOG_ERROR,
 | |
| 			"%s: No 'iat' in Identity header\n", ctx->tag);
 | |
| 	}
 | |
| 	ast_trace(1, "date_hdr: %zu  iat: %zu\n",
 | |
| 		ctx->date_hdr_time, iat);
 | |
| 
 | |
| 	if (iat + ctx->eprofile->vcfg_common.max_iat_age < now_s) {
 | |
| 		SCOPE_EXIT_RTN_VALUE(AST_STIR_SHAKEN_VS_IAT_EXPIRED,
 | |
| 			"%s: iat %ld older than %u seconds\n", ctx->tag,
 | |
| 			iat, ctx->eprofile->vcfg_common.max_iat_age);
 | |
| 	}
 | |
| 	ctx->validity_check_time = iat;
 | |
| 
 | |
| 	vs_rc = ctx_populate(ctx);
 | |
| 	if (vs_rc != AST_STIR_SHAKEN_VS_SUCCESS) {
 | |
| 		SCOPE_EXIT_LOG_RTN_VALUE(vs_rc, LOG_ERROR,
 | |
| 			"%s: Unable to populate ctx\n", ctx->tag);
 | |
| 	}
 | |
| 
 | |
| 	vs_rc = retrieve_verification_cert(ctx);
 | |
| 	if (vs_rc != AST_STIR_SHAKEN_VS_SUCCESS) {
 | |
| 		SCOPE_EXIT_LOG_RTN_VALUE(vs_rc, LOG_ERROR,
 | |
| 			"%s: Could not get valid cert from '%s'\n", ctx->tag, ctx->public_url);
 | |
| 	}
 | |
| 
 | |
| 	jwt_free(jwt);
 | |
| 	jwt = NULL;
 | |
| 
 | |
| 	rc = jwt_decode(&jwt, jwt_encoded, ctx->raw_key, ctx->raw_key_len);
 | |
| 	if (rc != 0) {
 | |
| 		SCOPE_EXIT_LOG_RTN_VALUE(AST_STIR_SHAKEN_VS_SIGNATURE_VALIDATION,
 | |
| 			LOG_ERROR, "%s: Signature validation failed for '%s'\n",
 | |
| 			ctx->tag, ctx->public_url);
 | |
| 	}
 | |
| 
 | |
| 	ast_trace(1, "%s: Decoding succeeded\n", ctx->tag);
 | |
| 
 | |
| 	ppt_header = jwt_get_header(jwt, "alg");
 | |
| 	if (!ppt_header || strcmp(ppt_header, STIR_SHAKEN_ENCRYPTION_ALGORITHM)) {
 | |
| 		SCOPE_EXIT_RTN_VALUE(AST_STIR_SHAKEN_VS_INVALID_OR_NO_ALG,
 | |
| 			"%s: %s\n", ctx->tag,
 | |
| 			vs_response_code_to_str(AST_STIR_SHAKEN_VS_INVALID_OR_NO_ALG));
 | |
| 	}
 | |
| 
 | |
| 	ppt_header = jwt_get_header(jwt, "ppt");
 | |
| 	if (!ppt_header || strcmp(ppt_header, STIR_SHAKEN_PPT)) {
 | |
| 		SCOPE_EXIT_RTN_VALUE(AST_STIR_SHAKEN_VS_INVALID_OR_NO_PPT,
 | |
| 			"%s: %s\n", ctx->tag,
 | |
| 			vs_response_code_to_str(AST_STIR_SHAKEN_VS_INVALID_OR_NO_PPT));
 | |
| 	}
 | |
| 
 | |
| 	ppt_header = jwt_get_header(jwt, "typ");
 | |
| 	if (!ppt_header || strcmp(ppt_header, STIR_SHAKEN_TYPE)) {
 | |
| 		SCOPE_EXIT_RTN_VALUE(AST_STIR_SHAKEN_VS_INVALID_OR_NO_TYP,
 | |
| 			"%s: %s\n", ctx->tag,
 | |
| 			vs_response_code_to_str(AST_STIR_SHAKEN_VS_INVALID_OR_NO_TYP));
 | |
| 	}
 | |
| 
 | |
| 	grants_str = jwt_get_grants_json(jwt, NULL);
 | |
| 	if (ast_strlen_zero(grants_str)) {
 | |
| 		SCOPE_EXIT_RTN_VALUE(AST_STIR_SHAKEN_VS_INVALID_OR_NO_GRANTS,
 | |
| 			"%s: %s\n", ctx->tag,
 | |
| 			vs_response_code_to_str(AST_STIR_SHAKEN_VS_INVALID_OR_NO_GRANTS));
 | |
| 	}
 | |
| 	ast_trace(1, "grants: %s\n", grants_str);
 | |
| 	grants = ast_json_load_string(grants_str, NULL);
 | |
| 	ast_std_free(grants_str);
 | |
| 	if (!grants) {
 | |
| 		SCOPE_EXIT_RTN_VALUE(AST_STIR_SHAKEN_VS_INVALID_OR_NO_GRANTS,
 | |
| 			"%s: %s\n", ctx->tag,
 | |
| 			vs_response_code_to_str(AST_STIR_SHAKEN_VS_INVALID_OR_NO_GRANTS));
 | |
| 	}
 | |
| 
 | |
| 	grant = ast_json_object_string_get(grants, "attest");
 | |
| 	if (ast_strlen_zero(grant)) {
 | |
| 		SCOPE_EXIT_RTN_VALUE(AST_STIR_SHAKEN_VS_INVALID_OR_NO_ATTEST,
 | |
| 			"%s: No 'attest' in Identity header\n", ctx->tag);
 | |
| 	}
 | |
| 	if (grant[0] < 'A' || grant[0] > 'C') {
 | |
| 		SCOPE_EXIT_RTN_VALUE(AST_STIR_SHAKEN_VS_INVALID_OR_NO_ATTEST,
 | |
| 			"%s: Invalid attest value '%s'\n", ctx->tag, grant);
 | |
| 	}
 | |
| 	ast_string_field_set(ctx, attestation, grant);
 | |
| 	ast_trace(1, "got attest: %s\n", grant);
 | |
| 
 | |
| 	grant_obj = ast_json_object_get(grants, "dest");
 | |
| 	if (!grant_obj) {
 | |
| 		SCOPE_EXIT_RTN_VALUE(AST_STIR_SHAKEN_VS_NO_DEST_TN,
 | |
| 			"%s: No 'dest' in Identity header\n", ctx->tag);
 | |
| 	}
 | |
| 	if (TRACE_ATLEAST(3)) {
 | |
| 		char *otn = ast_json_dump_string(grant_obj);
 | |
| 		ast_trace(1, "got dest: %s\n", otn);
 | |
| 		ast_json_free(otn);
 | |
| 	}
 | |
| 
 | |
| 	grant_obj = ast_json_object_get(grants, "orig");
 | |
| 	if (!grant_obj) {
 | |
| 		SCOPE_EXIT_RTN_VALUE(AST_STIR_SHAKEN_VS_NO_ORIG_TN,
 | |
| 			"%s: No 'orig' in Identity header\n", ctx->tag);
 | |
| 	}
 | |
| 	if (TRACE_ATLEAST(3)) {
 | |
| 		char *otn = ast_json_dump_string(grant_obj);
 | |
| 		ast_trace(1, "got orig: %s\n", otn);
 | |
| 		ast_json_free(otn);
 | |
| 	}
 | |
| 	grant = ast_json_object_string_get(grant_obj, "tn");
 | |
| 	if (!grant) {
 | |
| 		SCOPE_EXIT_RTN_VALUE(AST_STIR_SHAKEN_VS_NO_ORIG_TN,
 | |
| 			"%s: No 'orig.tn' in Indentity header\n", ctx->tag);
 | |
| 	}
 | |
| 	ast_string_field_set(ctx, orig_tn, grant);
 | |
| 	if (strcmp(ctx->caller_id, ctx->orig_tn) != 0) {
 | |
| 		SCOPE_EXIT_RTN_VALUE(AST_STIR_SHAKEN_VS_CID_ORIG_TN_MISMATCH,
 | |
| 			"%s: Mismatched cid '%s' and orig_tn '%s'\n", ctx->tag,
 | |
| 			ctx->caller_id, grant);
 | |
| 	}
 | |
| 
 | |
| 	grant = ast_json_object_string_get(grants, "origid");
 | |
| 	if (ast_strlen_zero(grant)) {
 | |
| 		SCOPE_EXIT_RTN_VALUE(AST_STIR_SHAKEN_VS_NO_ORIGID,
 | |
| 			"%s: No 'origid' in Identity header\n", ctx->tag);
 | |
| 	}
 | |
| 
 | |
| 	SCOPE_EXIT_RTN_VALUE(AST_STIR_SHAKEN_VS_SUCCESS,
 | |
| 		"%s: verification succeeded\n", ctx->tag);
 | |
| }
 | |
| 
 | |
| int vs_reload()
 | |
| {
 | |
| 	vs_config_reload();
 | |
| 
 | |
| 	return 0;
 | |
| }
 | |
| 
 | |
| int vs_unload()
 | |
| {
 | |
| 	vs_config_unload();
 | |
| 	if (url_match_regex.re_nsub > 0) {
 | |
| 		regfree(&url_match_regex);
 | |
| 	}
 | |
| 
 | |
| 	return 0;
 | |
| }
 | |
| 
 | |
| int vs_load()
 | |
| {
 | |
| 	int rc = 0;
 | |
| 
 | |
| 	if (vs_config_load()) {
 | |
| 		return AST_MODULE_LOAD_DECLINE;
 | |
| 	}
 | |
| 
 | |
| 	rc = regcomp(&url_match_regex, FULL_URL_REGEX, REG_EXTENDED);
 | |
| 	if (rc) {
 | |
| 		char regex_error[512];
 | |
| 		regerror(rc, &url_match_regex, regex_error, sizeof(regex_error));
 | |
| 		ast_log(LOG_ERROR, "Verification service URL regex failed to compile: %s\n", regex_error);
 | |
| 		vs_unload();
 | |
| 		return AST_MODULE_LOAD_DECLINE;
 | |
| 	}
 | |
| 	if (url_match_regex.re_nsub != FULL_URL_REGEX_GROUPS) {
 | |
| 		ast_log(LOG_ERROR, "The verification service URL regex was updated without updating FULL_URL_REGEX_GROUPS\n");
 | |
| 		vs_unload();
 | |
| 		return AST_MODULE_LOAD_DECLINE;
 | |
| 	}
 | |
| 
 | |
| 	return AST_MODULE_LOAD_SUCCESS;
 | |
| }
 |