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); }