1
0
mirror of https://github.com/invoiceninja/invoiceninja.git synced 2024-09-20 08:21:34 +02:00
invoiceninja/app/Console/Commands/SendRecurringInvoices.php

193 lines
5.7 KiB
PHP
Raw Normal View History

2017-01-30 20:40:43 +01:00
<?php
2015-03-16 22:45:25 +01:00
2017-01-30 20:40:43 +01:00
namespace App\Console\Commands;
2017-03-23 10:39:44 +01:00
use App\Models\Account;
2017-01-30 20:40:43 +01:00
use App\Models\Invoice;
2017-06-26 14:57:40 +02:00
use App\Models\RecurringExpense;
use App\Ninja\Repositories\InvoiceRepository;
2017-06-26 14:57:40 +02:00
use App\Ninja\Repositories\RecurringExpenseRepository;
2016-05-25 04:49:06 +02:00
use App\Services\PaymentService;
2018-03-27 14:59:48 +02:00
use App\Jobs\SendInvoiceEmail;
2017-01-30 20:40:43 +01:00
use DateTime;
use Illuminate\Console\Command;
2017-05-01 14:46:57 +02:00
use Symfony\Component\Console\Input\InputOption;
2017-05-29 12:43:28 +02:00
use Auth;
2017-07-12 11:37:19 +02:00
use Exception;
use Utils;
2015-03-16 22:45:25 +01:00
/**
2017-01-30 20:40:43 +01:00
* Class SendRecurringInvoices.
*/
2015-03-16 22:45:25 +01:00
class SendRecurringInvoices extends Command
{
/**
* @var string
*/
2015-03-16 22:45:25 +01:00
protected $name = 'ninja:send-invoices';
/**
* @var string
*/
2015-03-16 22:45:25 +01:00
protected $description = 'Send recurring invoices';
/**
* @var InvoiceRepository
*/
protected $invoiceRepo;
/**
* @var PaymentService
*/
2016-05-25 04:49:06 +02:00
protected $paymentService;
2015-03-16 22:45:25 +01:00
/**
* SendRecurringInvoices constructor.
2017-01-30 20:40:43 +01:00
*
* @param InvoiceRepository $invoiceRepo
2017-01-30 20:40:43 +01:00
* @param PaymentService $paymentService
*/
2018-03-27 14:59:48 +02:00
public function __construct(InvoiceRepository $invoiceRepo, PaymentService $paymentService, RecurringExpenseRepository $recurringExpenseRepo)
2015-03-16 22:45:25 +01:00
{
parent::__construct();
$this->invoiceRepo = $invoiceRepo;
2016-05-25 04:49:06 +02:00
$this->paymentService = $paymentService;
2017-06-26 14:57:40 +02:00
$this->recurringExpenseRepo = $recurringExpenseRepo;
2015-03-16 22:45:25 +01:00
}
public function fire()
{
2017-10-24 09:59:26 +02:00
$this->info(date('r') . ' Running SendRecurringInvoices...');
2015-03-16 22:45:25 +01:00
2017-05-01 14:17:52 +02:00
if ($database = $this->option('database')) {
config(['database.default' => $database]);
}
2017-06-26 11:45:42 +02:00
$this->resetCounters();
$this->createInvoices();
$this->billInvoices();
$this->createExpenses();
2017-10-24 09:59:26 +02:00
$this->info(date('r') . ' Done');
2017-06-26 11:45:42 +02:00
}
private function resetCounters()
{
2017-03-23 10:39:44 +01:00
$accounts = Account::where('reset_counter_frequency_id', '>', 0)
->orderBy('id', 'asc')
->get();
foreach ($accounts as $account) {
$account->checkCounterReset();
}
2017-06-26 11:45:42 +02:00
}
private function createInvoices()
{
$today = new DateTime();
2017-03-23 10:39:44 +01:00
2015-03-16 22:45:25 +01:00
$invoices = Invoice::with('account.timezone', 'invoice_items', 'client', 'user')
->whereRaw('is_deleted IS FALSE AND deleted_at IS NULL AND is_recurring IS TRUE AND is_public IS TRUE AND frequency_id > 0 AND start_date <= ? AND (end_date IS NULL OR end_date >= ?)', [$today, $today])
2015-10-20 10:23:38 +02:00
->orderBy('id', 'asc')
->get();
$this->info($invoices->count() . ' recurring invoice(s) found');
2015-03-16 22:45:25 +01:00
foreach ($invoices as $recurInvoice) {
$shouldSendToday = $recurInvoice->shouldSendToday();
2017-01-30 17:05:31 +01:00
if (! $shouldSendToday) {
2015-10-20 10:23:38 +02:00
continue;
}
2016-06-20 16:14:43 +02:00
$this->info('Processing Invoice: '. $recurInvoice->id);
2017-03-23 10:39:44 +01:00
$account = $recurInvoice->account;
$account->loadLocalizationSettings($recurInvoice->client);
2018-01-16 13:57:37 +01:00
Auth::loginUsingId($recurInvoice->activeUser()->id);
2015-08-11 16:38:36 +02:00
2017-07-12 11:37:19 +02:00
try {
$invoice = $this->invoiceRepo->createRecurringInvoice($recurInvoice);
2018-03-13 21:05:57 +01:00
if ($invoice && ! $invoice->isPaid() && $account->auto_email_invoice) {
2017-12-03 15:40:02 +01:00
$this->info('Not billed - Sending Invoice');
2018-03-27 14:59:48 +02:00
dispatch(new SendInvoiceEmail($invoice, $invoice->user_id));
2017-12-03 15:40:02 +01:00
} elseif ($invoice) {
$this->info('Successfully billed invoice');
2017-07-12 11:37:19 +02:00
}
} catch (Exception $exception) {
$this->info('Error: ' . $exception->getMessage());
Utils::logError($exception);
2015-08-11 16:38:36 +02:00
}
2017-07-12 11:37:19 +02:00
2017-05-29 12:43:28 +02:00
Auth::logout();
2015-03-16 22:45:25 +01:00
}
2017-06-26 11:45:42 +02:00
}
private function billInvoices()
{
$today = new DateTime();
2015-03-16 22:45:25 +01:00
2016-05-25 04:49:06 +02:00
$delayedAutoBillInvoices = Invoice::with('account.timezone', 'recurring_invoice', 'invoice_items', 'client', 'user')
->whereRaw('is_deleted IS FALSE AND deleted_at IS NULL AND is_recurring IS FALSE AND is_public IS TRUE
2016-05-26 21:22:09 +02:00
AND balance > 0 AND due_date = ? AND recurring_invoice_id IS NOT NULL',
[$today->format('Y-m-d')])
2016-05-25 04:49:06 +02:00
->orderBy('invoices.id', 'asc')
->get();
$this->info($delayedAutoBillInvoices->count() . ' due recurring invoice instance(s) found');
2016-05-25 04:49:06 +02:00
/** @var Invoice $invoice */
2016-05-25 04:49:06 +02:00
foreach ($delayedAutoBillInvoices as $invoice) {
2016-06-20 16:14:43 +02:00
if ($invoice->isPaid()) {
continue;
2016-05-25 04:49:06 +02:00
}
2016-06-20 16:14:43 +02:00
if ($invoice->getAutoBillEnabled() && $invoice->client->autoBillLater()) {
$this->info('Processing Autobill-delayed Invoice: ' . $invoice->id);
2018-01-16 13:59:06 +01:00
Auth::loginUsingId($invoice->activeUser()->id);
2016-05-25 04:49:06 +02:00
$this->paymentService->autoBillInvoice($invoice);
2017-05-29 12:43:28 +02:00
Auth::logout();
2016-05-25 04:49:06 +02:00
}
}
2017-06-26 11:45:42 +02:00
}
2016-05-25 04:49:06 +02:00
2017-06-26 11:45:42 +02:00
private function createExpenses()
{
$today = new DateTime();
2017-06-26 14:57:40 +02:00
$expenses = RecurringExpense::with('client')
2017-06-26 11:45:42 +02:00
->whereRaw('is_deleted IS FALSE AND deleted_at IS NULL AND start_date <= ? AND (end_date IS NULL OR end_date >= ?)', [$today, $today])
->orderBy('id', 'asc')
->get();
$this->info($expenses->count() . ' recurring expenses(s) found');
2017-06-26 11:45:42 +02:00
foreach ($expenses as $expense) {
$shouldSendToday = $expense->shouldSendToday();
if (! $shouldSendToday) {
continue;
}
$this->info('Processing Expense: '. $expense->id);
$this->recurringExpenseRepo->createRecurringExpense($expense);
}
2015-03-16 22:45:25 +01:00
}
/**
* @return array
*/
2015-03-16 22:45:25 +01:00
protected function getArguments()
{
return [];
2015-03-16 22:45:25 +01:00
}
/**
* @return array
*/
2015-03-16 22:45:25 +01:00
protected function getOptions()
{
2017-05-01 14:17:52 +02:00
return [
['database', null, InputOption::VALUE_OPTIONAL, 'Database', null],
];
2015-03-16 22:45:25 +01:00
}
}