public int?CreateDonationForBankAccountErrorRefund(StripeRefund refund) { if (refund.Data[0].BalanceTransaction == null || !"payment_failure_refund".Equals(refund.Data[0].BalanceTransaction.Type)) { _logger.Error(string.Format("Balance transaction was not set, or was not a payment_failure_refund for refund ID {0}", refund.Data[0].Id)); return(null); } if (refund.Data[0].Charge == null || string.IsNullOrWhiteSpace(refund.Data[0].Charge.Id)) { _logger.Error(string.Format("No associated Charge for Refund {0}", refund.Data[0].Id)); return(null); } MpDonation donation; try { donation = _mpDonationRepository.GetDonationByProcessorPaymentId(refund.Data[0].Charge.Id, true); } catch (DonationNotFoundException) { _logger.Error(string.Format("No Donation with payment processor ID {0} in MP for Refund {1}", refund.Data[0].Charge.Id, refund.Data[0].Id)); return(null); } var donationAndDist = new MpDonationAndDistributionRecord { Anonymous = false, ChargeId = refund.Data[0].Id, CheckNumber = null, CheckScannerBatchName = null, DonationAmt = -(int.Parse(refund.Data[0].Amount) / Constants.StripeDecimalConversionValue), DonationStatus = (int)DonationStatus.Declined, DonorAcctId = string.Empty, DonorId = _bankErrorRefundDonorId, FeeAmt = refund.Data[0].BalanceTransaction.Fee, PledgeId = null, RecurringGift = false, ProcessorId = string.Empty, ProgramId = donation.Distributions[0].donationDistributionProgram, PymtType = MinistryPlatform.Translation.Enum.PaymentType.GetPaymentType(donation.paymentTypeId).name, RecurringGiftId = null, RegisteredDonor = false, SetupDate = refund.Data[0].BalanceTransaction.Created, Notes = string.Format("Reversed from DonationID {0}", donation.donationId) }; foreach (var distribution in donation.Distributions) { donationAndDist.Distributions.Add(new MpDonationDistribution { donationDistributionAmt = -distribution.donationDistributionAmt, donationDistributionProgram = distribution.donationDistributionProgram, PledgeId = distribution.PledgeId }); } // Create the refund donation and distribution(s), but do NOT send email return(_mpDonorRepository.CreateDonationAndDistributionRecord(donationAndDist, false)); }
private IHttpActionResult CreateDonationAndDistributionAuthenticated(String token, CreateDonationDTO dto) { var isPayment = (dto.TransactionType != null && dto.TransactionType.Equals("PAYMENT")); try { if (isPayment) { //check if invoice exists before create Stripe Charge if (dto.InvoiceId != null && !_invoiceRepository.InvoiceExists(dto.InvoiceId.Value)) { var apiError = new ApiErrorDto("Invoice Not Found", new InvoiceNotFoundException(dto.InvoiceId.Value)); throw new HttpResponseException(apiError.HttpResponseMessage); } } var contactId = _authenticationService.GetContactId(token); var donor = _mpDonorService.GetContactDonor(contactId); var charge = _stripeService.ChargeCustomer(donor.ProcessorId, dto.Amount, donor.DonorId, isPayment); var fee = charge.BalanceTransaction != null ? charge.BalanceTransaction.Fee : null; int?pledgeId = null; if (dto.PledgeCampaignId != null && dto.PledgeDonorId != null) { var pledge = _mpPledgeService.GetPledgeByCampaignAndDonor(dto.PledgeCampaignId.Value, dto.PledgeDonorId.Value); if (pledge != null) { pledgeId = pledge.PledgeId; } } if (!isPayment) { var donationAndDistribution = new MpDonationAndDistributionRecord { DonationAmt = dto.Amount, FeeAmt = fee, DonorId = donor.DonorId, ProgramId = dto.ProgramId, PledgeId = pledgeId, ChargeId = charge.Id, PymtType = dto.PaymentType, ProcessorId = donor.ProcessorId, SetupDate = DateTime.Now, RegisteredDonor = true, Anonymous = dto.Anonymous, SourceUrl = dto.SourceUrl, PredefinedAmount = dto.PredefinedAmount }; var donationId = _mpDonorService.CreateDonationAndDistributionRecord(donationAndDistribution, !dto.TripDeposit); if (!dto.GiftMessage.IsNullOrWhiteSpace() && pledgeId != null) { SendMessageFromDonor(pledgeId.Value, donationId, dto.GiftMessage); } var response = new DonationDTO { ProgramId = dto.ProgramId, Amount = (int)dto.Amount, Id = donationId.ToString(), Email = donor.Email }; return(Ok(response)); } else //Payment flow (non-contribution transaction) { if (!ModelState.IsValid) { var errors = ModelState.Values.SelectMany(val => val.Errors).Aggregate("", (current, err) => current + err.Exception.Message); var dataError = new ApiErrorDto("Payment data Invalid", new InvalidOperationException("Invalid Payment Data" + errors)); throw new HttpResponseException(dataError.HttpResponseMessage); } try { var invoiceId = dto.InvoiceId != null ? dto.InvoiceId.Value : 0; var payment = new MpDonationAndDistributionRecord { DonationAmt = dto.Amount, PymtType = dto.PaymentType, ProcessorId = charge.Id, ContactId = contactId, InvoiceId = invoiceId, FeeAmt = fee }; var paymentReturn = _paymentService.PostPayment(payment); var response = new DonationDTO { Amount = (int)dto.Amount, Email = donor.Email, PaymentId = paymentReturn.PaymentId }; return(Ok(response)); } catch (InvoiceNotFoundException e) { var apiError = new ApiErrorDto("Invoice Not Found", e); throw new HttpResponseException(apiError.HttpResponseMessage); } catch (ContactNotFoundException e) { var apiError = new ApiErrorDto("Contact Not Found", e); throw new HttpResponseException(apiError.HttpResponseMessage); } catch (PaymentTypeNotFoundException e) { var apiError = new ApiErrorDto("PaymentType Not Found", e); throw new HttpResponseException(apiError.HttpResponseMessage); } catch (Exception e) { var apiError = new ApiErrorDto("SavePayment Failed", e); throw new HttpResponseException(apiError.HttpResponseMessage); } } } catch (PaymentProcessorException stripeException) { return(stripeException.GetStripeResult()); } catch (Exception exception) { var apiError = new ApiErrorDto("Donation/Payment Post Failed", exception); throw new HttpResponseException(apiError.HttpResponseMessage); } }
public CheckScannerBatch CreateDonationsForBatch(CheckScannerBatch batchDetails) { var checks = _checkScannerDao.GetChecksForBatch(batchDetails.Name); foreach (var check in checks) { if (check.Exported) { var previousError = string.IsNullOrWhiteSpace(check.Error) ? string.Empty : string.Format("Previous Error: {0}", check.Error); var msg = string.Format("Not exporting check {0} on batch {1}, it was already exported. {2}", check.Id, batchDetails.Name, previousError); _logger.Info(msg); check.Error = msg; batchDetails.ErrorChecks.Add(check); continue; } try { var contactDonor = CreateDonor(check); var account = _mpDonorService.DecryptCheckValue(check.AccountNumber); var routing = _mpDonorService.DecryptCheckValue(check.RoutingNumber); var encryptedKey = _mpDonorService.CreateHashedAccountAndRoutingNumber(account, routing); string donorAccountId = ""; if (contactDonor.Account.HasPaymentProcessorInfo() == false) { var stripeCustomer = _paymentService.CreateCustomer(null, contactDonor.DonorId + " Scanned Checks"); var stripeCustomerSource = _paymentService.AddSourceToCustomer(stripeCustomer.id, contactDonor.Account.Token); donorAccountId = _mpDonorService.CreateDonorAccount(null, routing, account.Right(4), encryptedKey, contactDonor.DonorId, stripeCustomerSource.id, stripeCustomer.id).ToString(); contactDonor.Account = new MpDonorAccount { DonorAccountId = int.Parse(donorAccountId), ProcessorId = stripeCustomer.id, ProcessorAccountId = stripeCustomerSource.id }; } else { donorAccountId = contactDonor.Account.DonorAccountId.ToString(); } //Always use the customer ID and source ID from the Donor Account, if it exists var charge = _paymentService.ChargeCustomer(contactDonor.Account.ProcessorId, contactDonor.Account.ProcessorAccountId, check.Amount, contactDonor.DonorId, check.CheckNumber); var fee = charge.BalanceTransaction != null ? charge.BalanceTransaction.Fee : null; // Mark the check as exported now, so we don't double-charge a community member. // If the CreateDonationAndDistributionRecord fails, we'll still consider it exported, but // it will be in error, and will have to be manually resolved. check.Exported = true; var programId = batchDetails.ProgramId == null ? null : batchDetails.ProgramId + ""; var donationAndDistribution = new MpDonationAndDistributionRecord { DonationAmt = check.Amount, FeeAmt = fee, DonorId = contactDonor.DonorId, ProgramId = programId, ChargeId = charge.Id, PymtType = "check", ProcessorId = contactDonor.Account.ProcessorId, SetupDate = check.CheckDate ?? (check.ScanDate ?? DateTime.Now), RegisteredDonor = contactDonor.RegisteredUser, DonorAcctId = donorAccountId, CheckScannerBatchName = batchDetails.Name, CheckNumber = (check.CheckNumber ?? string.Empty).TrimStart(' ', '0').Right(MinistryPlatformCheckNumberMaxLength) }; var donationId = _mpDonorService.CreateDonationAndDistributionRecord(donationAndDistribution, false); check.DonationId = donationId; _checkScannerDao.UpdateCheckStatus(check.Id, true); batchDetails.Checks.Add(check); } catch (Exception e) { check.Error = e.ToString(); check.AccountNumber = _mpDonorService.DecryptCheckValue(check.AccountNumber); check.RoutingNumber = _mpDonorService.DecryptCheckValue(check.RoutingNumber); batchDetails.ErrorChecks.Add(check); _checkScannerDao.UpdateCheckStatus(check.Id, check.Exported, check.Error); } } batchDetails.Status = BatchStatus.Exported; _checkScannerDao.UpdateBatchStatus(batchDetails.Name, batchDetails.Status); return(batchDetails); }