mirror of
https://github.com/invoiceninja/invoiceninja.git
synced 2024-11-10 05:02:36 +01:00
207 lines
6.0 KiB
PHP
207 lines
6.0 KiB
PHP
<?php
|
|
|
|
namespace App\Console\Commands;
|
|
|
|
use App\Models\Account;
|
|
use App\Models\Invoice;
|
|
use App\Models\RecurringExpense;
|
|
use App\Ninja\Mailers\ContactMailer as Mailer;
|
|
use App\Ninja\Repositories\InvoiceRepository;
|
|
use App\Ninja\Repositories\RecurringExpenseRepository;
|
|
use App\Services\PaymentService;
|
|
use DateTime;
|
|
use Illuminate\Console\Command;
|
|
use Symfony\Component\Console\Input\InputOption;
|
|
use Auth;
|
|
use Exception;
|
|
use Utils;
|
|
|
|
/**
|
|
* Class SendRecurringInvoices.
|
|
*/
|
|
class SendRecurringInvoices extends Command
|
|
{
|
|
/**
|
|
* @var string
|
|
*/
|
|
protected $name = 'ninja:send-invoices';
|
|
|
|
/**
|
|
* @var string
|
|
*/
|
|
protected $description = 'Send recurring invoices';
|
|
|
|
/**
|
|
* @var Mailer
|
|
*/
|
|
protected $mailer;
|
|
|
|
/**
|
|
* @var InvoiceRepository
|
|
*/
|
|
protected $invoiceRepo;
|
|
|
|
/**
|
|
* @var PaymentService
|
|
*/
|
|
protected $paymentService;
|
|
|
|
/**
|
|
* SendRecurringInvoices constructor.
|
|
*
|
|
* @param Mailer $mailer
|
|
* @param InvoiceRepository $invoiceRepo
|
|
* @param PaymentService $paymentService
|
|
*/
|
|
public function __construct(Mailer $mailer, InvoiceRepository $invoiceRepo, PaymentService $paymentService, RecurringExpenseRepository $recurringExpenseRepo)
|
|
{
|
|
parent::__construct();
|
|
|
|
$this->mailer = $mailer;
|
|
$this->invoiceRepo = $invoiceRepo;
|
|
$this->paymentService = $paymentService;
|
|
$this->recurringExpenseRepo = $recurringExpenseRepo;
|
|
}
|
|
|
|
public function fire()
|
|
{
|
|
$this->info(date('r') . ' Running SendRecurringInvoices...');
|
|
|
|
if ($database = $this->option('database')) {
|
|
config(['database.default' => $database]);
|
|
}
|
|
|
|
$this->resetCounters();
|
|
$this->createInvoices();
|
|
$this->billInvoices();
|
|
$this->createExpenses();
|
|
|
|
$this->info(date('r') . ' Done');
|
|
}
|
|
|
|
private function resetCounters()
|
|
{
|
|
$accounts = Account::where('reset_counter_frequency_id', '>', 0)
|
|
->orderBy('id', 'asc')
|
|
->get();
|
|
|
|
foreach ($accounts as $account) {
|
|
$account->checkCounterReset();
|
|
}
|
|
}
|
|
|
|
private function createInvoices()
|
|
{
|
|
$today = new DateTime();
|
|
|
|
$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])
|
|
->orderBy('id', 'asc')
|
|
->get();
|
|
$this->info($invoices->count() . ' recurring invoice(s) found');
|
|
|
|
foreach ($invoices as $recurInvoice) {
|
|
$shouldSendToday = $recurInvoice->shouldSendToday();
|
|
|
|
if (! $shouldSendToday) {
|
|
continue;
|
|
}
|
|
|
|
$this->info('Processing Invoice: '. $recurInvoice->id);
|
|
|
|
$account = $recurInvoice->account;
|
|
$account->loadLocalizationSettings($recurInvoice->client);
|
|
Auth::loginUsingId($recurInvoice->activeUser()->id);
|
|
|
|
try {
|
|
$invoice = $this->invoiceRepo->createRecurringInvoice($recurInvoice);
|
|
if ($invoice && ! $invoice->isPaid()) {
|
|
$this->info('Not billed - Sending Invoice');
|
|
$this->mailer->sendInvoice($invoice);
|
|
} elseif ($invoice) {
|
|
$this->info('Successfully billed invoice');
|
|
}
|
|
} catch (Exception $exception) {
|
|
$this->info('Error: ' . $exception->getMessage());
|
|
Utils::logError($exception);
|
|
}
|
|
|
|
Auth::logout();
|
|
}
|
|
}
|
|
|
|
private function billInvoices()
|
|
{
|
|
$today = new DateTime();
|
|
|
|
$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
|
|
AND balance > 0 AND due_date = ? AND recurring_invoice_id IS NOT NULL',
|
|
[$today->format('Y-m-d')])
|
|
->orderBy('invoices.id', 'asc')
|
|
->get();
|
|
$this->info($delayedAutoBillInvoices->count() . ' due recurring invoice instance(s) found');
|
|
|
|
/** @var Invoice $invoice */
|
|
foreach ($delayedAutoBillInvoices as $invoice) {
|
|
if ($invoice->isPaid()) {
|
|
continue;
|
|
}
|
|
|
|
if ($invoice->getAutoBillEnabled() && $invoice->client->autoBillLater()) {
|
|
$this->info('Processing Autobill-delayed Invoice: ' . $invoice->id);
|
|
|
|
// TODO don't skip
|
|
if ($invoice->user->trashed()) {
|
|
$this->info('Archived user: skipping');
|
|
continue;
|
|
}
|
|
|
|
Auth::loginUsingId($invoice->activeUser()->id);
|
|
$this->paymentService->autoBillInvoice($invoice);
|
|
Auth::logout();
|
|
}
|
|
}
|
|
}
|
|
|
|
private function createExpenses()
|
|
{
|
|
$today = new DateTime();
|
|
|
|
$expenses = RecurringExpense::with('client')
|
|
->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');
|
|
|
|
foreach ($expenses as $expense) {
|
|
$shouldSendToday = $expense->shouldSendToday();
|
|
|
|
if (! $shouldSendToday) {
|
|
continue;
|
|
}
|
|
|
|
$this->info('Processing Expense: '. $expense->id);
|
|
$this->recurringExpenseRepo->createRecurringExpense($expense);
|
|
}
|
|
}
|
|
|
|
/**
|
|
* @return array
|
|
*/
|
|
protected function getArguments()
|
|
{
|
|
return [];
|
|
}
|
|
|
|
/**
|
|
* @return array
|
|
*/
|
|
protected function getOptions()
|
|
{
|
|
return [
|
|
['database', null, InputOption::VALUE_OPTIONAL, 'Database', null],
|
|
];
|
|
}
|
|
}
|