2020-07-07 14:33:11 +02:00
< ? php
/**
2020-09-06 11:38:10 +02:00
* Invoice Ninja ( https :// invoiceninja . com ) .
2020-07-07 14:33:11 +02:00
*
* @ link https :// github . com / invoiceninja / invoiceninja source repository
*
2021-01-03 22:54:54 +01:00
* @ copyright Copyright ( c ) 2021. Invoice Ninja LLC ( https :// invoiceninja . com )
2020-07-07 14:33:11 +02:00
*
2021-06-16 08:58:16 +02:00
* @ license https :// www . elastic . co / licensing / elastic - license
2020-07-07 14:33:11 +02:00
*/
namespace App\Services\Invoice ;
2020-07-08 04:20:44 +02:00
use App\DataMapper\InvoiceItem ;
2020-07-07 14:33:11 +02:00
use App\Events\Payment\PaymentWasCreated ;
use App\Factory\PaymentFactory ;
2020-10-12 06:10:34 +02:00
use App\Models\Credit ;
2020-07-07 14:33:11 +02:00
use App\Models\Invoice ;
use App\Models\Payment ;
2020-09-04 00:01:17 +02:00
use App\Models\PaymentHash ;
2020-10-26 05:06:58 +01:00
use App\Models\PaymentType ;
2020-07-07 14:33:11 +02:00
use App\Services\AbstractService ;
2020-10-08 05:31:02 +02:00
use App\Utils\Ninja ;
2020-09-04 00:01:17 +02:00
use Illuminate\Support\Str ;
2020-07-07 14:33:11 +02:00
class AutoBillInvoice extends AbstractService
{
private $invoice ;
2020-09-06 11:38:10 +02:00
private $client ;
2020-07-07 14:33:11 +02:00
2020-10-08 02:28:23 +02:00
private $used_credit = [];
2020-10-07 13:03:53 +02:00
2020-07-07 14:33:11 +02:00
public function __construct ( Invoice $invoice )
{
$this -> invoice = $invoice ;
2020-09-06 11:38:10 +02:00
2020-07-07 14:33:11 +02:00
$this -> client = $invoice -> client ;
}
public function run ()
{
2021-05-26 08:04:38 +02:00
$is_partial = false ;
2020-10-07 13:03:53 +02:00
/* Is the invoice payable? */
2021-01-27 22:31:31 +01:00
if ( ! $this -> invoice -> isPayable ())
2020-07-08 04:20:44 +02:00
return $this -> invoice ;
2020-10-28 11:10:49 +01:00
2020-10-07 13:03:53 +02:00
/* Mark the invoice as sent */
2020-07-15 08:08:57 +02:00
$this -> invoice = $this -> invoice -> service () -> markSent () -> save ();
2020-10-07 13:03:53 +02:00
/* Mark the invoice as paid if there is no balance */
2021-01-29 12:00:30 +01:00
if (( int ) $this -> invoice -> balance == 0 )
2020-07-13 01:29:44 +02:00
return $this -> invoice -> service () -> markPaid () -> save ();
2021-01-29 12:00:30 +01:00
2020-10-09 03:59:59 +02:00
//if the credits cover the payments, we stop here, build the payment with credits and exit early
2021-01-29 12:00:30 +01:00
if ( $this -> client -> getSetting ( 'use_credits_payment' ) != 'off' )
2020-10-28 11:10:49 +01:00
$this -> applyCreditPayment ();
2020-07-07 14:33:11 +02:00
2020-10-07 13:03:53 +02:00
/* Determine $amount */
2020-11-25 15:19:52 +01:00
if ( $this -> invoice -> partial > 0 ) {
2021-05-26 08:04:38 +02:00
$is_partial = true ;
$invoice_total = $this -> invoice -> amount ;
2020-09-04 00:01:17 +02:00
$amount = $this -> invoice -> partial ;
2020-11-25 15:19:52 +01:00
} elseif ( $this -> invoice -> balance > 0 ) {
2020-09-04 00:01:17 +02:00
$amount = $this -> invoice -> balance ;
2020-11-25 15:19:52 +01:00
} else {
2020-10-23 06:18:16 +02:00
return $this -> invoice ;
2020-11-25 15:19:52 +01:00
}
2020-10-07 13:03:53 +02:00
2020-10-12 06:10:34 +02:00
info ( " balance remains to be paid!! " );
2020-10-12 06:30:53 +02:00
2021-01-27 22:31:31 +01:00
/* Retrieve the Client Gateway Token */
2020-10-07 13:03:53 +02:00
$gateway_token = $this -> getGateway ( $amount );
/* Bail out if no payment methods available */
2021-01-29 12:00:30 +01:00
if ( ! $gateway_token || ! $gateway_token -> gateway -> driver ( $this -> client ) -> token_billing )
2020-10-07 13:03:53 +02:00
return $this -> invoice ;
2021-01-29 12:00:30 +01:00
2020-10-07 13:03:53 +02:00
/* $gateway fee */
2021-01-27 22:31:31 +01:00
//$fee = $gateway_token->gateway->calcGatewayFee($amount, $gateway_token->gateway_type_id, $this->invoice->uses_inclusive_taxes);
$this -> invoice = $this -> invoice -> service () -> addGatewayFee ( $gateway_token -> gateway , $gateway_token -> gateway_type_id , $amount ) -> save ();
2021-05-26 08:04:38 +02:00
if ( $is_partial )
$fee = $this -> invoice -> amount - $invoice_total ;
else
$fee = $this -> invoice -> amount - $amount ;
2020-07-08 04:20:44 +02:00
2020-10-07 13:03:53 +02:00
/* Build payment hash */
2020-09-04 00:01:17 +02:00
$payment_hash = PaymentHash :: create ([
'hash' => Str :: random ( 128 ),
2021-01-17 11:33:05 +01:00
'data' => [ 'invoices' => [[ 'invoice_id' => $this -> invoice -> hashed_id , 'amount' => $amount ]]],
2020-09-04 00:01:17 +02:00
'fee_total' => $fee ,
'fee_invoice_id' => $this -> invoice -> id ,
]);
2020-07-08 04:20:44 +02:00
2020-10-07 13:03:53 +02:00
$payment = $gateway_token -> gateway
-> driver ( $this -> client )
2021-01-27 22:31:31 +01:00
-> setPaymentHash ( $payment_hash )
2020-10-07 13:03:53 +02:00
-> tokenBilling ( $gateway_token , $payment_hash );
2020-08-30 12:47:32 +02:00
2021-08-05 02:46:03 +02:00
if ( $payment ){
info ( " Auto Bill payment captured for " . $this -> invoice -> number );
}
2020-07-15 07:05:02 +02:00
return $this -> invoice ;
2020-07-07 14:33:11 +02:00
}
2020-10-08 02:28:23 +02:00
/**
* If the credits on file cover the invoice amount
* the we create a matching payment using credits only
2020-10-28 11:10:49 +01:00
*
2020-10-08 02:28:23 +02:00
* @ return Invoice $invoice
*/
private function finalizePaymentUsingCredits ()
{
$amount = array_sum ( array_column ( $this -> used_credit , 'amount' ));
$payment = PaymentFactory :: create ( $this -> invoice -> company_id , $this -> invoice -> user_id );
$payment -> amount = $amount ;
2020-10-12 06:10:34 +02:00
$payment -> applied = $amount ;
2020-10-08 05:31:02 +02:00
$payment -> client_id = $this -> invoice -> client_id ;
$payment -> currency_id = $this -> invoice -> client -> getSetting ( 'currency_id' );
2020-10-08 02:28:23 +02:00
$payment -> date = now ();
2020-10-08 05:31:02 +02:00
$payment -> status_id = Payment :: STATUS_COMPLETED ;
2020-10-26 03:06:24 +01:00
$payment -> type_id = PaymentType :: CREDIT ;
2020-10-08 05:31:02 +02:00
$payment -> service () -> applyNumber () -> save ();
2020-10-08 02:28:23 +02:00
$payment -> invoices () -> attach ( $this -> invoice -> id , [ 'amount' => $amount ]);
2021-01-24 07:44:14 +01:00
$this -> invoice
-> service ()
-> setStatus ( Invoice :: STATUS_PAID )
-> save ();
2020-10-08 02:28:23 +02:00
2020-11-25 15:19:52 +01:00
foreach ( $this -> used_credit as $credit ) {
$current_credit = Credit :: find ( $credit [ 'credit_id' ]);
2021-01-24 07:44:14 +01:00
$payment -> credits ()
-> attach ( $current_credit -> id , [ 'amount' => $credit [ 'amount' ]]);
2020-10-28 11:10:49 +01:00
2020-11-25 15:19:52 +01:00
info ( " adjusting credit balance { $current_credit -> balance } by this amount " . $credit [ 'amount' ]);
2020-10-23 06:18:16 +02:00
2021-01-24 07:44:14 +01:00
$current_credit -> service ()
2021-01-24 10:00:45 +01:00
-> adjustBalance ( $credit [ 'amount' ] *- 1 )
2021-01-24 07:44:14 +01:00
-> updatePaidToDate ( $credit [ 'amount' ])
-> setCalculatedStatus ()
-> save ();
2020-10-23 06:18:16 +02:00
2020-11-25 15:19:52 +01:00
}
2020-10-08 02:28:23 +02:00
2020-11-25 15:19:52 +01:00
$payment -> ledger ()
2020-10-08 02:28:23 +02:00
-> updatePaymentBalance ( $amount * - 1 )
-> save ();
2020-11-25 15:19:52 +01:00
$this -> invoice -> client -> service ()
2020-10-08 02:28:23 +02:00
-> updateBalance ( $amount * - 1 )
-> updatePaidToDate ( $amount )
-> adjustCreditBalance ( $amount * - 1 )
-> save ();
2020-11-25 15:19:52 +01:00
$this -> invoice -> ledger ()
2021-01-21 05:42:30 +01:00
-> updateInvoiceBalance ( $amount * - 1 , " Invoice { $this -> invoice -> number } payment using Credit { $current_credit -> number } " )
-> updateCreditBalance ( $amount * - 1 , " Credit { $current_credit -> number } used to pay down Invoice { $this -> invoice -> number } " )
2020-10-08 02:28:23 +02:00
-> save ();
2020-11-25 15:19:52 +01:00
event ( new PaymentWasCreated ( $payment , $payment -> company , Ninja :: eventVars ()));
2020-10-08 05:31:02 +02:00
2021-01-24 07:44:14 +01:00
return $this -> invoice
-> service ()
-> setCalculatedStatus ()
-> save ();
2020-10-08 02:28:23 +02:00
}
2020-10-07 13:03:53 +02:00
/**
* Applies credits to a payment prior to push
* to the payment gateway
2020-10-28 11:10:49 +01:00
*
2020-10-07 13:03:53 +02:00
* @ return $this
*/
2020-10-28 11:10:49 +01:00
private function applyCreditPayment ()
2020-10-07 13:03:53 +02:00
{
$available_credits = $this -> client
-> credits
-> where ( 'is_deleted' , false )
-> where ( 'balance' , '>' , 0 )
-> sortBy ( 'created_at' );
$available_credit_balance = $available_credits -> sum ( 'balance' );
2020-10-23 06:18:16 +02:00
info ( " available credit balance = { $available_credit_balance } " );
2020-11-25 15:19:52 +01:00
if (( int ) $available_credit_balance == 0 ) {
2020-10-07 13:03:53 +02:00
return ;
2020-11-25 15:19:52 +01:00
}
2020-10-07 13:03:53 +02:00
$is_partial_amount = false ;
if ( $this -> invoice -> partial > 0 ) {
$is_partial_amount = true ;
}
2020-10-08 05:31:02 +02:00
$this -> used_credit = [];
2020-10-07 13:03:53 +02:00
2020-11-25 15:19:52 +01:00
foreach ( $available_credits as $key => $credit ) {
if ( $is_partial_amount ) {
2020-10-08 01:36:06 +02:00
//more credit than needed
2021-01-24 10:28:18 +01:00
if ( $credit -> balance > $this -> invoice -> partial ) {
2020-10-08 02:28:23 +02:00
$this -> used_credit [ $key ][ 'credit_id' ] = $credit -> id ;
$this -> used_credit [ $key ][ 'amount' ] = $this -> invoice -> partial ;
$this -> invoice -> balance -= $this -> invoice -> partial ;
2021-01-24 07:44:14 +01:00
$this -> invoice -> paid_to_date += $this -> invoice -> partial ;
2020-10-08 01:36:06 +02:00
$this -> invoice -> partial = 0 ;
break ;
2020-11-25 15:19:52 +01:00
} else {
2020-10-08 02:28:23 +02:00
$this -> used_credit [ $key ][ 'credit_id' ] = $credit -> id ;
$this -> used_credit [ $key ][ 'amount' ] = $credit -> balance ;
2020-10-08 01:36:06 +02:00
$this -> invoice -> partial -= $credit -> balance ;
2020-10-08 02:28:23 +02:00
$this -> invoice -> balance -= $credit -> balance ;
2021-01-24 07:44:14 +01:00
$this -> invoice -> paid_to_date += $credit -> balance ;
2020-10-08 01:36:06 +02:00
}
2020-11-25 15:19:52 +01:00
} else {
2020-10-08 01:36:06 +02:00
//more credit than needed
2021-01-24 10:28:18 +01:00
if ( $credit -> balance > $this -> invoice -> balance ) {
2020-10-08 05:31:02 +02:00
$this -> used_credit [ $key ][ 'credit_id' ] = $credit -> id ;
$this -> used_credit [ $key ][ 'amount' ] = $this -> invoice -> balance ;
2021-01-24 07:44:14 +01:00
$this -> invoice -> paid_to_date += $this -> invoice -> balance ;
2020-10-08 01:36:06 +02:00
$this -> invoice -> balance = 0 ;
2021-01-24 07:44:14 +01:00
2020-10-08 01:36:06 +02:00
break ;
2020-11-25 15:19:52 +01:00
} else {
2020-10-08 05:31:02 +02:00
$this -> used_credit [ $key ][ 'credit_id' ] = $credit -> id ;
$this -> used_credit [ $key ][ 'amount' ] = $credit -> balance ;
2020-10-08 01:36:06 +02:00
$this -> invoice -> balance -= $credit -> balance ;
2021-01-24 07:44:14 +01:00
$this -> invoice -> paid_to_date += $credit -> balance ;
2020-10-08 01:36:06 +02:00
}
}
}
2020-10-07 13:03:53 +02:00
2020-10-23 06:18:16 +02:00
$this -> finalizePaymentUsingCredits ();
2020-10-07 13:03:53 +02:00
return $this ;
}
2020-10-12 06:10:34 +02:00
private function applyPaymentToCredit ( $credit , $amount ) : Credit
2020-10-07 13:03:53 +02:00
{
$credit_item = new InvoiceItem ;
$credit_item -> type_id = '1' ;
$credit_item -> product_key = ctrans ( 'texts.credit' );
$credit_item -> notes = ctrans ( 'texts.credit_payment' , [ 'invoice_number' => $this -> invoice -> number ]);
$credit_item -> quantity = 1 ;
$credit_item -> cost = $amount * - 1 ;
$credit_items = $credit -> line_items ;
$credit_items [] = $credit_item ;
$credit -> line_items = $credit_items ;
$credit = $credit -> calc () -> getCredit ();
2020-10-12 06:10:34 +02:00
$credit -> save ();
2020-10-07 13:03:53 +02:00
2020-10-12 06:10:34 +02:00
return $credit ;
2020-10-07 13:03:53 +02:00
}
2020-07-15 07:05:02 +02:00
/**
* Harvests a client gateway token which passes the
2020-09-06 11:38:10 +02:00
* necessary filters for an $amount .
*
2020-07-15 07:05:02 +02:00
* @ param float $amount The amount to charge
* @ return ClientGatewayToken The client gateway token
*/
2021-01-29 12:00:30 +01:00
// private function
2021-01-17 11:33:05 +01:00
// {
// $gateway_tokens = $this->client->gateway_tokens()->orderBy('is_default', 'DESC')->get();
// foreach ($gateway_tokens as $gateway_token) {
// if ($this->validGatewayLimits($gateway_token, $amount)) {
// return $gateway_token;
// }
// }
// }
public function getGateway ( $amount )
2020-07-07 14:33:11 +02:00
{
2020-07-13 00:28:19 +02:00
2021-01-17 11:33:05 +01:00
//get all client gateway tokens and set the is_default one to the first record
//$gateway_tokens = $this->client->gateway_tokens()->orderBy('is_default', 'DESC');
$gateway_tokens = $this -> client -> gateway_tokens ;
$filtered_gateways = $gateway_tokens -> filter ( function ( $gateway_token ) use ( $amount ) {
2021-01-29 12:00:30 +01:00
2021-01-17 11:33:05 +01:00
$company_gateway = $gateway_token -> gateway ;
//check if fees and limits are set
2021-01-29 12:00:30 +01:00
if ( isset ( $company_gateway -> fees_and_limits ) && property_exists ( $company_gateway -> fees_and_limits , $gateway_token -> gateway_type_id ))
2021-01-17 11:33:05 +01:00
{
//if valid we keep this gateway_token
2021-01-29 12:00:30 +01:00
if ( $this -> invoice -> client -> validGatewayForAmount ( $company_gateway -> fees_and_limits -> { $gateway_token -> gateway_type_id }, $amount ))
2021-01-17 11:33:05 +01:00
return true ;
else
return false ;
2021-01-29 12:00:30 +01:00
2020-07-15 07:05:02 +02:00
}
2021-01-17 11:33:05 +01:00
return true ; //if no fees_and_limits set then we automatically must add this gateway
});
if ( $filtered_gateways -> count () >= 1 )
return $filtered_gateways -> first ();
return false ;
2020-07-08 02:18:13 +02:00
}
2020-07-07 16:50:51 +02:00
2021-01-17 11:33:05 +01:00
2020-07-08 05:07:07 +02:00
/**
2020-09-06 11:38:10 +02:00
* Adds a gateway fee to the invoice .
*
2020-07-08 05:07:07 +02:00
* @ param float $fee The fee amount .
2020-10-28 11:10:49 +01:00
* @ return AutoBillInvoice
2020-07-08 05:07:07 +02:00
*/
2020-07-08 04:20:44 +02:00
private function addFeeToInvoice ( float $fee )
{
2020-07-08 05:07:07 +02:00
//todo if we increase the invoice balance here, we will also need to adjust UP the client balance and ledger?
$starting_amount = $this -> invoice -> amount ;
2020-07-08 04:20:44 +02:00
$item = new InvoiceItem ;
$item -> quantity = 1 ;
$item -> cost = $fee ;
$item -> notes = ctrans ( 'texts.online_payment_surcharge' );
$item -> type_id = 3 ;
2020-09-06 11:38:10 +02:00
$items = ( array ) $this -> invoice -> line_items ;
2020-07-08 04:20:44 +02:00
$items [] = $item ;
$this -> invoice -> line_items = $items ;
$this -> invoice -> save ();
$this -> invoice = $this -> invoice -> calc () -> getInvoice () -> save ();
2020-09-06 11:38:10 +02:00
if ( $starting_amount != $this -> invoice -> amount && $this -> invoice -> status_id != Invoice :: STATUS_DRAFT ) {
2020-07-08 05:07:07 +02:00
$this -> invoice -> client -> service () -> updateBalance ( $this -> invoice -> amount - $starting_amount ) -> save ();
2021-01-21 05:42:30 +01:00
$this -> invoice -> ledger () -> updateInvoiceBalance ( $this -> invoice -> amount - $starting_amount , " Invoice { $this -> invoice -> number } balance updated after stale gateway fee removed " ) -> save ();
2020-07-08 05:07:07 +02:00
}
2020-07-08 08:54:16 +02:00
2020-07-08 05:07:07 +02:00
return $this ;
}
2020-07-07 14:33:11 +02:00
}