1
0
mirror of https://github.com/bitwarden/server.git synced 2025-04-06 05:28:15 -05:00

Fix PayPal IPN Logging (#3768)

* Remove request logging, fix txn_id correlation

* Respond 400 when txn_id is missing

* More cleanup
This commit is contained in:
Alex Morask 2024-02-08 10:37:41 -05:00 committed by GitHub
parent d29755de5a
commit 6cc53b4739
No known key found for this signature in database
GPG Key ID: B5690EEEBB952194
5 changed files with 39 additions and 33 deletions

View File

@ -75,6 +75,14 @@ public class PayPalController : Controller
var transactionModel = new PayPalIPNTransactionModel(requestContent); var transactionModel = new PayPalIPNTransactionModel(requestContent);
_logger.LogInformation("PayPal IPN: Transaction Type = {Type}", transactionModel.TransactionType);
if (string.IsNullOrEmpty(transactionModel.TransactionId))
{
_logger.LogError("PayPal IPN: Transaction ID is missing");
return Ok();
}
var entityId = transactionModel.UserId ?? transactionModel.OrganizationId; var entityId = transactionModel.UserId ?? transactionModel.OrganizationId;
if (!entityId.HasValue) if (!entityId.HasValue)
@ -83,7 +91,7 @@ public class PayPalController : Controller
return BadRequest(); return BadRequest();
} }
var verified = await _payPalIPNClient.VerifyIPN(entityId.Value, requestContent); var verified = await _payPalIPNClient.VerifyIPN(transactionModel.TransactionId, requestContent);
if (!verified) if (!verified)
{ {

View File

@ -2,5 +2,5 @@
public interface IPayPalIPNClient public interface IPayPalIPNClient
{ {
Task<bool> VerifyIPN(Guid entityId, string formData); Task<bool> VerifyIPN(string transactionId, string formData);
} }

View File

@ -21,9 +21,9 @@ public class PayPalIPNClient : IPayPalIPNClient
_logger = logger; _logger = logger;
} }
public async Task<bool> VerifyIPN(Guid entityId, string formData) public async Task<bool> VerifyIPN(string transactionId, string formData)
{ {
LogInfo(entityId, $"Verifying IPN against {_ipnEndpoint}"); LogInfo(transactionId, $"Verifying IPN against {_ipnEndpoint}");
if (string.IsNullOrEmpty(formData)) if (string.IsNullOrEmpty(formData))
{ {
@ -34,8 +34,6 @@ public class PayPalIPNClient : IPayPalIPNClient
var requestContent = string.Concat("cmd=_notify-validate&", formData); var requestContent = string.Concat("cmd=_notify-validate&", formData);
LogInfo(entityId, $"Request Content: {requestContent}");
requestMessage.Content = new StringContent(requestContent, Encoding.UTF8, "application/x-www-form-urlencoded"); requestMessage.Content = new StringContent(requestContent, Encoding.UTF8, "application/x-www-form-urlencoded");
var response = await _httpClient.SendAsync(requestMessage); var response = await _httpClient.SendAsync(requestMessage);
@ -52,35 +50,35 @@ public class PayPalIPNClient : IPayPalIPNClient
}; };
} }
LogError(entityId, $"Unsuccessful Response | Status Code: {response.StatusCode} | Content: {responseContent}"); LogError(transactionId, $"Unsuccessful Response | Status Code: {response.StatusCode} | Content: {responseContent}");
return false; return false;
bool Verified() bool Verified()
{ {
LogInfo(entityId, "Verified"); LogInfo(transactionId, "Verified");
return true; return true;
} }
bool Invalid() bool Invalid()
{ {
LogError(entityId, "Verification Invalid"); LogError(transactionId, "Verification Invalid");
return false; return false;
} }
bool Unhandled(string content) bool Unhandled(string content)
{ {
LogWarning(entityId, $"Unhandled Response Content: {content}"); LogWarning(transactionId, $"Unhandled Response Content: {content}");
return false; return false;
} }
} }
private void LogInfo(Guid entityId, string message) private void LogInfo(string transactionId, string message)
=> _logger.LogInformation("Verify PayPal IPN ({RequestId}) | {Message}", entityId, message); => _logger.LogInformation("Verify PayPal IPN ({Id}) | {Message}", transactionId, message);
private void LogWarning(Guid entityId, string message) private void LogWarning(string transactionId, string message)
=> _logger.LogWarning("Verify PayPal IPN ({RequestId}) | {Message}", entityId, message); => _logger.LogWarning("Verify PayPal IPN ({Id}) | {Message}", transactionId, message);
private void LogError(Guid entityId, string message) private void LogError(string transactionId, string message)
=> _logger.LogError("Verify PayPal IPN ({RequestId}) | {Message}", entityId, message); => _logger.LogError("Verify PayPal IPN ({Id}) | {Message}", transactionId, message);
} }

View File

@ -129,7 +129,7 @@ public class PayPalControllerTests
var ipnBody = await PayPalTestIPN.GetAsync(IPNBody.SuccessfulPayment); var ipnBody = await PayPalTestIPN.GetAsync(IPNBody.SuccessfulPayment);
_payPalIPNClient.VerifyIPN(organizationId, ipnBody).Returns(false); _payPalIPNClient.VerifyIPN(Arg.Any<string>(), ipnBody).Returns(false);
var controller = ConfigureControllerContextWith(logger, _defaultWebhookKey, ipnBody); var controller = ConfigureControllerContextWith(logger, _defaultWebhookKey, ipnBody);
@ -154,7 +154,7 @@ public class PayPalControllerTests
var ipnBody = await PayPalTestIPN.GetAsync(IPNBody.UnsupportedTransactionType); var ipnBody = await PayPalTestIPN.GetAsync(IPNBody.UnsupportedTransactionType);
_payPalIPNClient.VerifyIPN(organizationId, ipnBody).Returns(true); _payPalIPNClient.VerifyIPN(Arg.Any<string>(), ipnBody).Returns(true);
var controller = ConfigureControllerContextWith(logger, _defaultWebhookKey, ipnBody); var controller = ConfigureControllerContextWith(logger, _defaultWebhookKey, ipnBody);
@ -183,7 +183,7 @@ public class PayPalControllerTests
var ipnBody = await PayPalTestIPN.GetAsync(IPNBody.SuccessfulPayment); var ipnBody = await PayPalTestIPN.GetAsync(IPNBody.SuccessfulPayment);
_payPalIPNClient.VerifyIPN(organizationId, ipnBody).Returns(true); _payPalIPNClient.VerifyIPN(Arg.Any<string>(), ipnBody).Returns(true);
var controller = ConfigureControllerContextWith(logger, _defaultWebhookKey, ipnBody); var controller = ConfigureControllerContextWith(logger, _defaultWebhookKey, ipnBody);
@ -212,7 +212,7 @@ public class PayPalControllerTests
var ipnBody = await PayPalTestIPN.GetAsync(IPNBody.RefundMissingParentTransaction); var ipnBody = await PayPalTestIPN.GetAsync(IPNBody.RefundMissingParentTransaction);
_payPalIPNClient.VerifyIPN(organizationId, ipnBody).Returns(true); _payPalIPNClient.VerifyIPN(Arg.Any<string>(), ipnBody).Returns(true);
var controller = ConfigureControllerContextWith(logger, _defaultWebhookKey, ipnBody); var controller = ConfigureControllerContextWith(logger, _defaultWebhookKey, ipnBody);
@ -241,7 +241,7 @@ public class PayPalControllerTests
var ipnBody = await PayPalTestIPN.GetAsync(IPNBody.ECheckPayment); var ipnBody = await PayPalTestIPN.GetAsync(IPNBody.ECheckPayment);
_payPalIPNClient.VerifyIPN(organizationId, ipnBody).Returns(true); _payPalIPNClient.VerifyIPN(Arg.Any<string>(), ipnBody).Returns(true);
var controller = ConfigureControllerContextWith(logger, _defaultWebhookKey, ipnBody); var controller = ConfigureControllerContextWith(logger, _defaultWebhookKey, ipnBody);
@ -270,7 +270,7 @@ public class PayPalControllerTests
var ipnBody = await PayPalTestIPN.GetAsync(IPNBody.NonUSDPayment); var ipnBody = await PayPalTestIPN.GetAsync(IPNBody.NonUSDPayment);
_payPalIPNClient.VerifyIPN(organizationId, ipnBody).Returns(true); _payPalIPNClient.VerifyIPN(Arg.Any<string>(), ipnBody).Returns(true);
var controller = ConfigureControllerContextWith(logger, _defaultWebhookKey, ipnBody); var controller = ConfigureControllerContextWith(logger, _defaultWebhookKey, ipnBody);
@ -299,7 +299,7 @@ public class PayPalControllerTests
var ipnBody = await PayPalTestIPN.GetAsync(IPNBody.SuccessfulPayment); var ipnBody = await PayPalTestIPN.GetAsync(IPNBody.SuccessfulPayment);
_payPalIPNClient.VerifyIPN(organizationId, ipnBody).Returns(true); _payPalIPNClient.VerifyIPN(Arg.Any<string>(), ipnBody).Returns(true);
_transactionRepository.GetByGatewayIdAsync( _transactionRepository.GetByGatewayIdAsync(
GatewayType.PayPal, GatewayType.PayPal,
@ -332,7 +332,7 @@ public class PayPalControllerTests
var ipnBody = await PayPalTestIPN.GetAsync(IPNBody.SuccessfulPayment); var ipnBody = await PayPalTestIPN.GetAsync(IPNBody.SuccessfulPayment);
_payPalIPNClient.VerifyIPN(organizationId, ipnBody).Returns(true); _payPalIPNClient.VerifyIPN(Arg.Any<string>(), ipnBody).Returns(true);
_transactionRepository.GetByGatewayIdAsync( _transactionRepository.GetByGatewayIdAsync(
GatewayType.PayPal, GatewayType.PayPal,
@ -367,7 +367,7 @@ public class PayPalControllerTests
var ipnBody = await PayPalTestIPN.GetAsync(IPNBody.SuccessfulPaymentForOrganizationCredit); var ipnBody = await PayPalTestIPN.GetAsync(IPNBody.SuccessfulPaymentForOrganizationCredit);
_payPalIPNClient.VerifyIPN(organizationId, ipnBody).Returns(true); _payPalIPNClient.VerifyIPN(Arg.Any<string>(), ipnBody).Returns(true);
_transactionRepository.GetByGatewayIdAsync( _transactionRepository.GetByGatewayIdAsync(
GatewayType.PayPal, GatewayType.PayPal,
@ -417,7 +417,7 @@ public class PayPalControllerTests
var ipnBody = await PayPalTestIPN.GetAsync(IPNBody.SuccessfulPaymentForUserCredit); var ipnBody = await PayPalTestIPN.GetAsync(IPNBody.SuccessfulPaymentForUserCredit);
_payPalIPNClient.VerifyIPN(userId, ipnBody).Returns(true); _payPalIPNClient.VerifyIPN(Arg.Any<string>(), ipnBody).Returns(true);
_transactionRepository.GetByGatewayIdAsync( _transactionRepository.GetByGatewayIdAsync(
GatewayType.PayPal, GatewayType.PayPal,
@ -467,7 +467,7 @@ public class PayPalControllerTests
var ipnBody = await PayPalTestIPN.GetAsync(IPNBody.SuccessfulRefund); var ipnBody = await PayPalTestIPN.GetAsync(IPNBody.SuccessfulRefund);
_payPalIPNClient.VerifyIPN(organizationId, ipnBody).Returns(true); _payPalIPNClient.VerifyIPN(Arg.Any<string>(), ipnBody).Returns(true);
_transactionRepository.GetByGatewayIdAsync( _transactionRepository.GetByGatewayIdAsync(
GatewayType.PayPal, GatewayType.PayPal,
@ -504,7 +504,7 @@ public class PayPalControllerTests
var ipnBody = await PayPalTestIPN.GetAsync(IPNBody.SuccessfulRefund); var ipnBody = await PayPalTestIPN.GetAsync(IPNBody.SuccessfulRefund);
_payPalIPNClient.VerifyIPN(organizationId, ipnBody).Returns(true); _payPalIPNClient.VerifyIPN(Arg.Any<string>(), ipnBody).Returns(true);
_transactionRepository.GetByGatewayIdAsync( _transactionRepository.GetByGatewayIdAsync(
GatewayType.PayPal, GatewayType.PayPal,
@ -545,7 +545,7 @@ public class PayPalControllerTests
var ipnBody = await PayPalTestIPN.GetAsync(IPNBody.SuccessfulRefund); var ipnBody = await PayPalTestIPN.GetAsync(IPNBody.SuccessfulRefund);
_payPalIPNClient.VerifyIPN(organizationId, ipnBody).Returns(true); _payPalIPNClient.VerifyIPN(Arg.Any<string>(), ipnBody).Returns(true);
_transactionRepository.GetByGatewayIdAsync( _transactionRepository.GetByGatewayIdAsync(
GatewayType.PayPal, GatewayType.PayPal,

View File

@ -34,7 +34,7 @@ public class PayPalIPNClientTests
[Fact] [Fact]
public async Task VerifyIPN_FormDataNull_ThrowsArgumentNullException() public async Task VerifyIPN_FormDataNull_ThrowsArgumentNullException()
=> await Assert.ThrowsAsync<ArgumentNullException>(() => _payPalIPNClient.VerifyIPN(Guid.NewGuid(), null)); => await Assert.ThrowsAsync<ArgumentNullException>(() => _payPalIPNClient.VerifyIPN(string.Empty, null));
[Fact] [Fact]
public async Task VerifyIPN_Unauthorized_ReturnsFalse() public async Task VerifyIPN_Unauthorized_ReturnsFalse()
@ -46,7 +46,7 @@ public class PayPalIPNClientTests
.WithFormData(new Dictionary<string, string> { { "cmd", "_notify-validate" }, { "form", "data" } }) .WithFormData(new Dictionary<string, string> { { "cmd", "_notify-validate" }, { "form", "data" } })
.Respond(HttpStatusCode.Unauthorized); .Respond(HttpStatusCode.Unauthorized);
var verified = await _payPalIPNClient.VerifyIPN(Guid.NewGuid(), formData); var verified = await _payPalIPNClient.VerifyIPN(string.Empty, formData);
Assert.False(verified); Assert.False(verified);
Assert.Equal(1, _mockHttpMessageHandler.GetMatchCount(request)); Assert.Equal(1, _mockHttpMessageHandler.GetMatchCount(request));
@ -62,7 +62,7 @@ public class PayPalIPNClientTests
.WithFormData(new Dictionary<string, string> { { "cmd", "_notify-validate" }, { "form", "data" } }) .WithFormData(new Dictionary<string, string> { { "cmd", "_notify-validate" }, { "form", "data" } })
.Respond("application/text", "INVALID"); .Respond("application/text", "INVALID");
var verified = await _payPalIPNClient.VerifyIPN(Guid.NewGuid(), formData); var verified = await _payPalIPNClient.VerifyIPN(string.Empty, formData);
Assert.False(verified); Assert.False(verified);
Assert.Equal(1, _mockHttpMessageHandler.GetMatchCount(request)); Assert.Equal(1, _mockHttpMessageHandler.GetMatchCount(request));
@ -78,7 +78,7 @@ public class PayPalIPNClientTests
.WithFormData(new Dictionary<string, string> { { "cmd", "_notify-validate" }, { "form", "data" } }) .WithFormData(new Dictionary<string, string> { { "cmd", "_notify-validate" }, { "form", "data" } })
.Respond("application/text", "VERIFIED"); .Respond("application/text", "VERIFIED");
var verified = await _payPalIPNClient.VerifyIPN(Guid.NewGuid(), formData); var verified = await _payPalIPNClient.VerifyIPN(string.Empty, formData);
Assert.True(verified); Assert.True(verified);
Assert.Equal(1, _mockHttpMessageHandler.GetMatchCount(request)); Assert.Equal(1, _mockHttpMessageHandler.GetMatchCount(request));