1
0
mirror of https://github.com/invoiceninja/invoiceninja.git synced 2024-11-17 00:22:51 +01:00
invoiceninja/app/Jobs/Mail/ImapInboundExpensesJob.php

158 lines
5.2 KiB
PHP
Raw Normal View History

2023-12-10 16:06:33 +01:00
<?php
/**
* Invoice Ninja (https://invoiceninja.com).
*
* @link https://github.com/invoiceninja/invoiceninja source repository
*
* @copyright Copyright (c) 2023. Invoice Ninja LLC (https://invoiceninja.com)
*
* @license https://www.elastic.co/licensing/elastic-license
*/
namespace App\Jobs\Mail;
use App\DataMapper\Analytics\EmailFailure;
use App\DataMapper\Analytics\EmailSuccess;
use App\Events\Expense\ExpenseWasCreated;
use App\Events\Invoice\InvoiceWasEmailedAndFailed;
use App\Events\Payment\PaymentWasEmailedAndFailed;
use App\Factory\ExpenseFactory;
use App\Helpers\Mail\IncomingMailHandler;
use App\Jobs\Util\SystemLogger;
use App\Libraries\Google\Google;
use App\Libraries\MultiDB;
use App\Models\Account;
use App\Models\ClientContact;
use App\Models\Company;
use App\Models\Expense;
use App\Models\Invoice;
use App\Models\Payment;
use App\Models\SystemLog;
use App\Models\User;
use App\Models\Vendor;
use App\Repositories\ExpenseRepository;
use App\Utils\Ninja;
use App\Utils\Traits\MakesHash;
use GuzzleHttp\Exception\ClientException;
use Illuminate\Bus\Queueable;
use Illuminate\Contracts\Queue\ShouldQueue;
use Illuminate\Foundation\Bus\Dispatchable;
use Illuminate\Queue\InteractsWithQueue;
use Illuminate\Queue\SerializesModels;
use Illuminate\Support\Facades\App;
use Illuminate\Support\Facades\Cache;
use Illuminate\Support\Facades\Mail;
use Turbo124\Beacon\Facades\LightLogs;
/*Multi Mailer implemented*/
2023-12-14 08:32:15 +01:00
class InboundExpensesJob implements ShouldQueue
2023-12-10 16:06:33 +01:00
{
use Dispatchable, InteractsWithQueue, Queueable, SerializesModels, MakesHash;
public $tries = 4; //number of retries
public $deleteWhenMissingModels = true;
2023-12-14 08:32:15 +01:00
private array $imap_companies;
private array $imap_credentials;
2023-12-10 16:06:33 +01:00
private $expense_repo;
public function __construct()
{
2023-12-14 08:32:15 +01:00
$this->credentials = [];
2023-12-10 16:06:33 +01:00
2023-12-14 08:32:15 +01:00
$this->getImapCredentials();
2023-12-10 16:06:33 +01:00
2023-12-14 08:32:15 +01:00
$this->expense_repo = new ExpenseRepository();
2023-12-10 16:06:33 +01:00
}
public function handle()
{
//multiDB environment, need to
foreach (MultiDB::$dbs as $db) {
MultiDB::setDB($db);
nlog("importing expenses from imap-servers");
2023-12-14 08:32:15 +01:00
Account::with('companies')->cursor()->each(function ($account) {
$account->companies()->whereIn('id', $this->imap_companies)->cursor()->each(function ($company) {
2023-12-10 16:06:33 +01:00
$this->handleCompanyImap($company);
});
});
}
}
2023-12-14 08:32:15 +01:00
private function getImapCredentials()
{
$servers = explode(",", config('ninja.imap_inbound_expense.servers'));
$ports = explode(",", config('ninja.imap_inbound_expense.servers'));
$users = explode(",", config('ninja.imap_inbound_expense.servers'));
$passwords = explode(",", config('ninja.imap_inbound_expense.servers'));
$companies = explode(",", config('ninja.imap_inbound_expense.servers'));
if (sizeOf($servers) != sizeOf($ports) || sizeOf($servers) != sizeOf($users) || sizeOf($servers) != sizeOf($passwords) || sizeOf($servers) != sizeOf($companies))
throw new \Exception('invalid configuration imap_inbound_expenses (wrong element-count)');
foreach ($companies as $index => $companyId) {
$this->imap_credentials[$companyId] = [
"server" => $servers[$index],
"port" => $servers[$index],
"user" => $servers[$index],
"password" => $servers[$index],
];
$this->imap_companies[] = $companyId;
}
}
2023-12-10 16:06:33 +01:00
private function handleCompanyImap(Company $company)
{
2023-12-14 08:32:15 +01:00
$credentials = $this->imap_credentials[$company->id];
$incommingMails = new IncomingMailHandler($credentials->server, $credentials->port, $credentials->user, $credentials->password);
2023-12-10 16:06:33 +01:00
$emails = $incommingMails->getUnprocessedEmails();
foreach ($emails as $mail) {
$sender = $mail->getSender();
2023-12-14 08:32:15 +01:00
$vendor = Vendor::where('expense_sender_email', $sender)->orWhere($sender, 'LIKE', "CONCAT('%',expense_sender_domain)")->first();
2023-12-10 16:06:33 +01:00
if ($vendor !== null)
$vendor = Vendor::where("email", $sender)->first();
2023-12-14 08:32:15 +01:00
$documents = []; // TODO: $mail->getAttachments() + save email as document (.html)
2023-12-10 16:06:33 +01:00
$data = [
"vendor_id" => $vendor !== null ? $vendor->id : null,
"date" => $mail->getDate(),
"public_notes" => $mail->getSubject(),
"private_notes" => $mail->getCompleteBodyText(),
2023-12-14 08:32:15 +01:00
"documents" => $documents, // FIXME: https://github.com/ddeboer/imap?tab=readme-ov-file#message-attachments
2023-12-10 16:06:33 +01:00
];
$expense = $this->expense_repo->save($data, ExpenseFactory::create($company->company->id, $company->company->owner()->id)); // TODO: dont assign a new number at beginning
// TODO: check for recurring expense?! => maybe replace existing ?!
event(new ExpenseWasCreated($expense, $expense->company, Ninja::eventVars(null)));
event('eloquent.created: App\Models\Expense', $expense);
$mail->markAsSeen();
$incommingMails->moveProcessed($mail);
}
}
2023-12-14 08:32:15 +01:00
public function backoff()
{
// return [5, 10, 30, 240];
return [rand(5, 10), rand(30, 40), rand(60, 79), rand(160, 400)];
}
2023-12-10 16:06:33 +01:00
}