mirror of
https://github.com/invoiceninja/invoiceninja.git
synced 2024-11-16 16:13:20 +01:00
171 lines
6.3 KiB
PHP
171 lines
6.3 KiB
PHP
<?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\Events\Expense\ExpenseWasCreated;
|
|
use App\Factory\ExpenseFactory;
|
|
use App\Helpers\Mail\Mailbox\Imap\ImapMailbox;
|
|
use App\Libraries\MultiDB;
|
|
use App\Models\Company;
|
|
use App\Models\Vendor;
|
|
use App\Repositories\ExpenseRepository;
|
|
use App\Utils\Ninja;
|
|
use App\Utils\TempFile;
|
|
use App\Utils\Traits\MakesHash;
|
|
use App\Utils\Traits\SavesDocuments;
|
|
use Illuminate\Bus\Queueable;
|
|
use Illuminate\Contracts\Queue\ShouldQueue;
|
|
use Illuminate\Foundation\Bus\Dispatchable;
|
|
use Illuminate\Queue\InteractsWithQueue;
|
|
use Illuminate\Queue\SerializesModels;
|
|
|
|
/*Multi Mailer implemented*/
|
|
|
|
class ExpenseMailboxJob implements ShouldQueue
|
|
{
|
|
use Dispatchable, InteractsWithQueue, Queueable, SerializesModels, MakesHash, SavesDocuments;
|
|
|
|
public $tries = 4; //number of retries
|
|
|
|
public $deleteWhenMissingModels = true;
|
|
private array $imap_companies;
|
|
private array $imap_credentials;
|
|
private $expense_repo;
|
|
|
|
public function __construct()
|
|
{
|
|
$this->credentials = [];
|
|
|
|
$this->getImapCredentials();
|
|
|
|
$this->expense_repo = new ExpenseRepository(); // @turbo124 @todo is this the right aproach? should it be handled just with the model?
|
|
}
|
|
|
|
public function handle()
|
|
{
|
|
|
|
//multiDB environment, need to
|
|
if (sizeOf($this->imap_credentials) == 0)
|
|
return;
|
|
|
|
foreach ($this->imap_companies as $companyId) {
|
|
$company = MultiDB::findAndSetDbByCompanyId($companyId);
|
|
if (!$company) {
|
|
nlog("processing of an imap_mailbox skipped because of unknown companyId: " . $companyId);
|
|
return;
|
|
}
|
|
|
|
try {
|
|
nlog("start importing expenses from imap-server of company: " . $companyId);
|
|
$this->handleImapCompany($company);
|
|
|
|
} catch (\Exception $e) {
|
|
nlog("processing of an imap_mailbox failed upnormally: " . $companyId . " message: " . $e->getMessage()); // @turbo124 @todo should this be handled in an other way?
|
|
}
|
|
}
|
|
|
|
}
|
|
|
|
private function getImapCredentials()
|
|
{
|
|
$servers = array_map('trim', explode(",", config('ninja.inbound_expense.imap.servers')));
|
|
$ports = array_map('trim', explode(",", config('ninja.inbound_expense.imap.ports')));
|
|
$users = array_map('trim', explode(",", config('ninja.inbound_expense.imap.users')));
|
|
$passwords = array_map('trim', explode(",", config('ninja.inbound_expense.imap.passwords')));
|
|
$companies = array_map('trim', explode(",", config('ninja.inbound_expense.imap.companies')));
|
|
|
|
if (sizeOf($servers) != sizeOf($ports) || sizeOf($servers) != sizeOf($users) || sizeOf($servers) != sizeOf($passwords) || sizeOf($servers) != sizeOf($companies))
|
|
throw new \Exception('invalid configuration inbound_expense.imap (wrong element-count)');
|
|
|
|
foreach ($companies as $index => $companyId) {
|
|
if ($servers[$index] == '') // if property is empty, ignore => this happens exspecialy when no config is provided and it enabled us to set a single default company for env (usefull on self-hosted)
|
|
continue;
|
|
|
|
$this->imap_credentials[$companyId] = [
|
|
"server" => $servers[$index],
|
|
"port" => $ports[$index] != '' ? $ports[$index] : null,
|
|
"user" => $users[$index],
|
|
"password" => $passwords[$index],
|
|
];
|
|
$this->imap_companies[] = $companyId;
|
|
}
|
|
}
|
|
|
|
private function handleImapCompany(Company $company)
|
|
{
|
|
nlog("importing expenses for company: " . $company->id);
|
|
|
|
$credentials = $this->imap_credentials[$company->id];
|
|
$imapMailbox = new ImapMailbox($credentials->server, $credentials->port, $credentials->user, $credentials->password);
|
|
|
|
$emails = $imapMailbox->getUnprocessedEmails();
|
|
|
|
foreach ($emails as $mail) {
|
|
|
|
try {
|
|
|
|
$sender = $mail->getSender();
|
|
|
|
$vendor = Vendor::where('expense_sender_email', $sender)->first();
|
|
if ($vendor == null)
|
|
$vendor = Vendor::where($sender, 'LIKE', "CONCAT('%',expense_sender_domain)")->first();
|
|
if ($vendor == null)
|
|
$vendor = Vendor::where("email", $sender)->first();
|
|
|
|
$documents = []; // TODO: $mail->getAttachments() + save email as document (.html)
|
|
|
|
$data = [
|
|
"vendor_id" => $vendor !== null ? $vendor->id : null,
|
|
"date" => $mail->getDate(),
|
|
"public_notes" => $mail->getSubject(),
|
|
"private_notes" => $mail->getCompleteBodyText(),
|
|
"documents" => $documents, // FIXME: https://github.com/ddeboer/imap?tab=readme-ov-file#message-attachments
|
|
];
|
|
|
|
$expense = ExpenseFactory::create($company->company->id, $company->company->owner()->id);
|
|
|
|
$expense->vendor_id = $vendor !== null ? $vendor->id : null;
|
|
$expense->public_notes = $mail->getSubject();
|
|
$expense->private_notes = $mail->getBodyText();
|
|
$expense->date = $mail->getDate();
|
|
|
|
// add html_message as document to the expense
|
|
$documents[] = TempFile::UploadedFileFromRaw($mail->getBodyHtml(), "E-Mail.html", "text/html");
|
|
|
|
$this->saveDocuments($documents, $expense);
|
|
|
|
$expense->saveQuietly();
|
|
|
|
event(new ExpenseWasCreated($expense, $expense->company, Ninja::eventVars(null)));
|
|
event('eloquent.created: App\Models\Expense', $expense);
|
|
|
|
$mail->markAsSeen();
|
|
$imapMailbox->moveProcessed($mail);
|
|
|
|
} catch (\Exception $e) {
|
|
$imapMailbox->moveFailed($mail);
|
|
|
|
nlog("processing of an email failed upnormally: " . $company->id . " message: " . $e->getMessage()); // @turbo124 @todo should this be handled in an other way?
|
|
}
|
|
|
|
}
|
|
}
|
|
|
|
public function backoff()
|
|
{
|
|
// return [5, 10, 30, 240];
|
|
return [rand(5, 10), rand(30, 40), rand(60, 79), rand(160, 400)];
|
|
|
|
}
|
|
|
|
}
|