We have a scenario to decrypt emails which has attachments. We are using mimekit
library for the same. We also use mimekit
for the email encryption and it works properly.
In our case the encrypted email is only having an attachment, no such email body. There is an Azure Logic App
which gets the encrypted email from an Oiifce365 mailbox (using the built-in connector) and then it send the details to an Azure Function App
which runs the decryption logic. The decryption certificate is stored in Azure Key Vault
.
Below is the code we tried, and it shows exception saying
Unable to cast object of type 'Org.BouncyCastle.Asn1.DerApplicationSpecific' to type 'Org.BouncyCastle.Asn1.Asn1SequenceParser'.
[FunctionName("DecryptSMIME")]
public static async Task<IActionResult> Decrypt([HttpTrigger(AuthorizationLevel.Function, "post", Route = null)] HttpRequest req, ILogger log)
{
try
{
var temporarySecurityMimeContext = new TemporarySecureMimeContext();
// get decryption Cert pfx
var keyVaultClient = ServiceProvider.GetRequiredService<IKeyVaultClient>();
var decryptionCertBundle = keyVaultClient.GetSecretAsync("https://my-key-vault.vault.azure.net/secrets/Decryption-Certificate-Base64/d7a84b415a494c1ebaseae88cff50028").Result;
var decryptionCertBytes = Convert.FromBase64String(decryptionCertBundle.Value);
log.LogInformation($"Decoded length of decryption certificate: '{decryptionCertBytes.Length}'");
// get decryption Cert password
var decryptionCertPasswordBundle = keyVaultClient.GetSecretAsync("https://my-key-vault.vault.azure.net/secrets/Decryption-Certificate-Pass/34judc9f575f467a96d9483dfc8kf467").Result;
var decryptionCertPassword = decryptionCertPasswordBundle.Value;
using var stream = new MemoryStream(decryptionCertBytes);
temporarySecurityMimeContext.Import(stream, decryptionCertPassword);
log.LogInformation("Imported The Decryption certificate as MemoryStream");
using var encryptedContentStream = await GetMailAttachmentStreamAsync(req.Body, log) ;
log.LogInformation("Loading pkcs7-mime entity.");
ApplicationPkcs7Mime encryptedContent = (ApplicationPkcs7Mime)await MimeEntity.LoadAsync(ParserOptions.Default, ContentType.Parse(ParserOptions.Default, "application/pkcs7-mime; smime-type=enveloped-data; name=smime.p7m"), encryptedContentStream);
log.LogInformation("Decrypting pkcs7-mime entity.");
MimeEntity decryptedContent = encryptedContent.Decrypt();
return new OkObjectResult("OK");
}
catch (Exception ex)
{
log.LogError(ex, "Failed to decrypt the secure mime part in the request body.");
throw;
}
}
private static async Task<MemoryStream> GetMailAttachmentStreamAsync(Stream attachmentObjectStream, ILogger log)
{
var memoryStream = new MemoryStream();
await attachmentObjectStream.CopyToAsync(memoryStream);
memoryStream.Seek(0, SeekOrigin.Begin);
log.LogInformation($"Attachment Stream Processed. {memoryStream.Length} Bytes");
return memoryStream;
}
The certifcate is loading successfully. Also the email stream shows some data. However while running the decryption, it alsways shows the error. Any help will be helpful.