2023-05-12 20:09:51 +10:00
|
|
|
<?php
|
|
|
|
|
2024-07-14 13:49:00 +10:00
|
|
|
namespace App\Console\Commands\Intuit;
|
2023-05-12 20:09:51 +10:00
|
|
|
|
|
|
|
use Illuminate\Console\Command;
|
|
|
|
use Intuit\Jobs\AccountingInvoiceUpdate;
|
|
|
|
use Intuit\Models\Invoice as AccInvoice;
|
2024-08-14 22:23:11 +10:00
|
|
|
use Intuit\Traits\ProviderTokenTrait;
|
2023-05-12 20:09:51 +10:00
|
|
|
|
2024-08-14 22:23:11 +10:00
|
|
|
use App\Models\Invoice;
|
2023-05-12 20:09:51 +10:00
|
|
|
|
2024-07-14 13:49:00 +10:00
|
|
|
class InvoiceAdd extends Command
|
2023-05-12 20:09:51 +10:00
|
|
|
{
|
2024-08-14 22:23:11 +10:00
|
|
|
use ProviderTokenTrait;
|
2024-07-14 13:49:00 +10:00
|
|
|
|
2023-05-12 20:09:51 +10:00
|
|
|
/**
|
|
|
|
* The name and signature of the console command.
|
|
|
|
*
|
|
|
|
* @var string
|
|
|
|
*/
|
2024-08-14 22:23:11 +10:00
|
|
|
protected $signature = 'intuit:invoice:add'
|
2024-07-14 13:49:00 +10:00
|
|
|
.' {id : Invoice ID}'
|
|
|
|
.' {user? : User Email}';
|
2023-05-12 20:09:51 +10:00
|
|
|
|
|
|
|
/**
|
|
|
|
* The console command description.
|
|
|
|
*
|
|
|
|
* @var string
|
|
|
|
*/
|
|
|
|
protected $description = 'Add an invoice to the accounting provider';
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Execute the console command.
|
|
|
|
*
|
|
|
|
* @return int
|
2024-07-14 13:49:00 +10:00
|
|
|
* @throws \Exception
|
2023-05-12 20:09:51 +10:00
|
|
|
*/
|
|
|
|
public function handle()
|
|
|
|
{
|
2024-08-14 22:23:11 +10:00
|
|
|
$to = $this->providerToken($this->argument('user'));
|
2023-05-12 20:09:51 +10:00
|
|
|
|
2024-08-14 22:23:11 +10:00
|
|
|
$io = Invoice::findOrFail($this->argument('id'));
|
2023-05-12 20:09:51 +10:00
|
|
|
|
|
|
|
// Check the customer exists
|
2024-08-14 22:23:11 +10:00
|
|
|
if ($io->account->providers->where('pivot.provider_oauth_id',$to->provider->id)->count() !== 1)
|
|
|
|
throw new \Exception(sprintf('Account [%d] for Invoice [%d] not defined',$io->account_id,$io->id));
|
2023-05-12 20:09:51 +10:00
|
|
|
|
2024-08-14 22:23:11 +10:00
|
|
|
$ao = $io->account->providers->where('pivot.provider_oauth_id',$to->provider->id)->pop();
|
2023-05-12 20:09:51 +10:00
|
|
|
|
|
|
|
// Some validation
|
|
|
|
if (! $ao->pivot->ref) {
|
2024-08-14 22:23:11 +10:00
|
|
|
$this->error(sprintf('Accounting not defined for account [%d]',$io->account_id));
|
|
|
|
|
|
|
|
return self::FAILURE;
|
2023-05-12 20:09:51 +10:00
|
|
|
}
|
|
|
|
|
|
|
|
$acc = new AccInvoice;
|
|
|
|
$acc->CustomerRef = (object)['value'=>$ao->pivot->ref];
|
2024-08-14 22:23:11 +10:00
|
|
|
$acc->DocNumber = $io->lid;
|
|
|
|
$acc->TxnDate = $io->created_at->format('Y-m-d');
|
|
|
|
$acc->DueDate = $io->due_at->format('Y-m-d');
|
2023-05-12 20:09:51 +10:00
|
|
|
|
|
|
|
$lines = collect();
|
|
|
|
$c = 0;
|
2024-08-14 22:23:11 +10:00
|
|
|
$subtotal = 0;
|
2023-05-12 20:09:51 +10:00
|
|
|
|
|
|
|
// @todo Group these by ItemRef and the Same Unit Price and Description, so that we can then use quantity to represent the number of them.
|
2024-08-14 22:23:11 +10:00
|
|
|
foreach ($io->items->groupBy(
|
|
|
|
fn($item)=>
|
|
|
|
sprintf('%s.%s.%s.%s',
|
|
|
|
$item->item_type_name,
|
|
|
|
$item->price_base,
|
2024-09-16 11:36:55 +10:00
|
|
|
$item->product?->provider_ref($to->provider),
|
2024-08-14 22:23:11 +10:00
|
|
|
$item->taxes->pluck('description')->join('|'))) as $os)
|
2023-05-12 20:09:51 +10:00
|
|
|
{
|
|
|
|
$key = $os->first();
|
|
|
|
|
|
|
|
// Some validation
|
2024-09-16 11:36:55 +10:00
|
|
|
if (! ($ref=$key->product?->provider_ref($to->provider))) {
|
2023-05-12 20:09:51 +10:00
|
|
|
$this->error(sprintf('Accounting not defined in product [%d]',$key->product_id));
|
2024-07-14 13:49:00 +10:00
|
|
|
|
|
|
|
return self::FAILURE;
|
2023-05-12 20:09:51 +10:00
|
|
|
}
|
|
|
|
|
|
|
|
if ($key->taxes->count() !== 1) {
|
|
|
|
$this->error(sprintf('Cannot handle when there is not just 1 tax line [%d]',$key->id));
|
2024-07-14 13:49:00 +10:00
|
|
|
|
|
|
|
return self::FAILURE;
|
2023-05-12 20:09:51 +10:00
|
|
|
}
|
|
|
|
|
|
|
|
$c++;
|
|
|
|
$line = new \stdClass;
|
|
|
|
$line->Id = $c;
|
|
|
|
$line->DetailType = 'SalesItemLineDetail';
|
|
|
|
$line->Description = $key->item_type_name;
|
|
|
|
$line->SalesItemLineDetail = (object)[
|
|
|
|
'Qty' => $os->sum('quantity'),
|
|
|
|
'UnitPrice' => $key->price_base,
|
|
|
|
'ItemRef' => ['value'=>$ref],
|
|
|
|
// @todo It is assumed there is only 1 tax category
|
2024-08-14 22:23:11 +10:00
|
|
|
'TaxCodeRef' => ['value'=>$tcf=$key->taxes->first()->tax->provider_ref($to->provider)],
|
2023-05-12 20:09:51 +10:00
|
|
|
];
|
2024-08-14 22:23:11 +10:00
|
|
|
$line->Amount = round($os->sum('quantity')*$key->price_base,2);
|
|
|
|
|
|
|
|
$subtotal += $line->Amount;
|
2023-05-12 20:09:51 +10:00
|
|
|
|
|
|
|
$lines->push($line);
|
|
|
|
}
|
|
|
|
|
|
|
|
$acc->Line = $lines;
|
|
|
|
|
2024-08-14 22:23:11 +10:00
|
|
|
// If our subtotal doesnt match, we need to add a tax line
|
|
|
|
if ($io->subtotal !== $subtotal) {
|
|
|
|
$acc->TxnTaxDetail = (object)[
|
|
|
|
'TotalTax' => $x=$io->total-$subtotal,
|
|
|
|
'TaxLine' => [
|
|
|
|
(object) [
|
|
|
|
'Amount' => $x,
|
|
|
|
'DetailType' => 'TaxLineDetail',
|
|
|
|
'TaxLineDetail' => (object)[
|
|
|
|
// @todo It is assumed there is only 1 tax category
|
|
|
|
'TaxRateRef' => (object)['value'=>$to->API()->getTaxCodeQuery($tcf)->getTaxRateRef()->first()],
|
|
|
|
'NetAmountTaxable' => $io->subtotal,
|
|
|
|
]
|
|
|
|
]
|
|
|
|
]
|
|
|
|
];
|
|
|
|
}
|
|
|
|
|
2023-05-12 20:09:51 +10:00
|
|
|
return AccountingInvoiceUpdate::dispatchSync($to,$acc);
|
|
|
|
}
|
|
|
|
}
|