Add SendInvoiceDataReport command and InvoiceDataMail class with HTML view for invoice data reporting

This commit is contained in:
dhanabalan
2025-11-02 16:33:12 +05:30
parent 4f51643254
commit 9f38cda857
4 changed files with 432 additions and 2 deletions

View File

@@ -0,0 +1,201 @@
<?php
namespace App\Console\Commands;
use App\Mail\InvoiceDataMail;
use App\Models\Line;
use App\Models\Plant;
use App\Models\ProductionPlan;
use App\Models\ProductionQuantity;
use Illuminate\Console\Command;
use Illuminate\Support\Facades\Mail;
class SendInvoiceDataReport extends Command
{
/**
* The name and signature of the console command.
*
* @var string
*/
protected $signature = 'send:invoice-data-report {schedule_type} {plant}';
/**
* The console command description.
*
* @var string
*/
protected $description = 'Command description';
/**
* Execute the console command.
*/
public function handle()
{
$scheduleType = $this->argument('schedule_type');
$plantId = (int) $this->argument('plant'); // cast to int for safety
// Fetch mail rules based on schedule type
$mailRules = \App\Models\AlertMailRule::where('module', 'InvoiceValidation')
->where('rule_name', 'InvoiceDataMail')
->where('schedule_type', $scheduleType)
->where('plant', $plantId)
->get();
$emails = $mailRules->pluck('email')->unique()->toArray();
//$ccEmails = $mailRules->pluck('cc_email')->filter()->unique()->toArray();
//$this->info('Found emails: ' . implode(', ', $emails));
$plants = $plantId == 0
? Plant::all()
: Plant::where('id', $plantId)->get();
if ($plants->isEmpty()) {
$this->error("No valid plant(s) found.");
return;
}
if (strtolower($scheduleType) == 'daily') {
$startDate = now()->subDay()->setTime(8, 0, 0);
$endDate = now()->setTime(8, 0, 0);
} else {
$startDate = now()->setTime(8, 0, 0);
$endDate = now()->copy()->addDay()->setTime(8, 0, 0);
}
// $tableData = [];
// $no = 1;
// foreach ($plants as $plant) {
// $distributions = ['Direct Sale', 'Branch Sale', 'Internal Transfer'];
// foreach ($distributions as $selectedDistribution) {
// // Get invoice records
// $invoices = \App\Models\InvoiceDataValidation::where('plant_id', $plant->id)
// ->where('distribution_channel_desc', $selectedDistribution)
// ->whereBetween('document_date', [$startDate, $endDate])
// ->select('customer_code', 'document_number', 'document_date', 'customer_trade_name', 'customer_location')
// ->distinct('document_number')
// ->get();
// if ($invoices->isEmpty()) {
// $tableData[] = [
// 'no' => $no++,
// 'plant' => $plant->name,
// 'distribution_type' => $selectedDistribution,
// 'customer_code' => '-',
// 'document_number' => '-',
// 'document_date' => '-',
// 'customer_trade_name' => '-',
// 'customer_location' => '-',
// 'status' => 'No Invoices',
// ];
// continue;
// }
// // Fetch which invoices actually went out
// $wentOutInvoices = \App\Models\InvoiceOutValidation::where('plant_id', $plant->id)
// ->whereIn('qr_code', $invoices->pluck('document_number')->toArray())
// ->whereBetween('scanned_at', [$startDate, $endDate])
// ->distinct('qr_code')
// ->pluck('qr_code')
// ->toArray();
// foreach ($invoices as $inv) {
// $status = in_array($inv->document_number, $wentOutInvoices)
// ? 'Went Out'
// : 'Pending';
// $tableData[] = [
// 'no' => $no++,
// 'plant' => $plant->name,
// 'distribution_type' => $selectedDistribution,
// 'customer_code' => $inv->customer_code,
// 'document_number' => $inv->document_number,
// 'document_date' => $inv->document_date,
// 'customer_trade_name' => $inv->customer_trade_name,
// 'customer_location' => $inv->customer_location,
// 'status' => $status,
// ];
// }
// }
// }
$tableData = [];
$no = 1;
foreach ($plants as $plant) {
$distributions = ['Direct Sale', 'Branch Sale', 'Internal Transfer'];
foreach ($distributions as $selectedDistribution) {
$invoices = \App\Models\InvoiceDataValidation::where('plant_id', $plant->id)
->where('distribution_channel_desc', $selectedDistribution)
->whereBetween('document_date', [$startDate, $endDate])
->select('customer_code', 'document_number', 'document_date', 'customer_trade_name', 'customer_location')
->distinct('document_number')
->get();
if ($invoices->isEmpty()) {
continue;
}
$wentOutInvoices = \App\Models\InvoiceOutValidation::where('plant_id', $plant->id)
->whereIn('qr_code', $invoices->pluck('document_number')->toArray())
->whereBetween('scanned_at', [$startDate, $endDate])
->distinct('qr_code')
->pluck('qr_code')
->toArray();
$pendingInvoices = $invoices->filter(function ($inv) use ($wentOutInvoices) {
return !in_array($inv->document_number, $wentOutInvoices);
});
if ($pendingInvoices->isEmpty()) {
continue;
}
foreach ($pendingInvoices as $inv) {
$tableData[] = [
'no' => $no++,
'plant' => $plant->name,
'distribution_type' => $selectedDistribution,
'customer_code' => $inv->customer_code,
'document_number' => $inv->document_number,
'document_date' => $inv->document_date,
'customer_trade_name' => $inv->customer_trade_name,
'customer_location' => $inv->customer_location,
'status' => 'Pending',
];
}
}
}
$mail = new InvoiceDataMail($scheduleType, $tableData);
$contentVars = $mail->content()->with;
$this->info($contentVars['greeting'] ?? 'Invoice Data Report');
$this->table(
['No', 'Plant', 'Distribution Type', 'Customer Code', 'Document Number', 'Document Date', 'Trade Name', 'Location', 'Status'],
$tableData
);
$this->info($contentVars['wishes'] ?? '');
// Send Mail //->cc($ccEmails)
if (!empty($emails)) {
foreach ($emails as $email){
\Mail::to($email)->send(new InvoiceDataMail($scheduleType, $tableData));
}
$this->info("Invoice data report sent to " . count($emails) . " recipient(s).");
} else {
$this->warn('No recipients found for InvoiceDataMail.');
}
}
}

View File

@@ -0,0 +1,89 @@
<?php
namespace App\Mail;
use Illuminate\Bus\Queueable;
use Illuminate\Contracts\Queue\ShouldQueue;
use Illuminate\Mail\Mailable;
use Illuminate\Mail\Mailables\Content;
use Illuminate\Mail\Mailables\Envelope;
use Illuminate\Queue\SerializesModels;
use DateTime;
class InvoiceDataMail extends Mailable
{
use Queueable, SerializesModels;
public $tableData;
public $scheduleType;
/**
* Create a new message instance.
*/
public function __construct($scheduleType,$tableData = [])
{
$this->scheduleType = $scheduleType;
$this->tableData = $tableData ?? [];
}
/**
* Get the message envelope.
*/
public function envelope(): Envelope
{
return new Envelope(
subject: 'Invoice Data Mail',
);
}
/**
* Get the message content definition.
*/
public function content(): Content
{
$greeting = "Dear Sir/Madam,<br><br>Kindly find the attached invoice pending report status,";
if ($this->scheduleType == 'Daily') {
$fromDate = (new DateTime('yesterday 08:00'))->format('d/m/Y H:i') . ':000';
$toDate = (new DateTime('today 07:59'))->format('d/m/Y H:i') . ':999';
$reportPeriod = "The following report presents results from: $fromDate to $toDate.";
$greeting .= $reportPeriod;
}
if ($this->scheduleType == 'Hourly') {
$now = now();
$fromHour = $now->copy()->subHour()->format('H:i:s');
$toHour = $now->format('H:i:s');
$reportDate = $now->format('d/m/Y');
$greeting .= "The following report presents results from: $reportDate, $fromHour to $toHour.";
}
if ($this->scheduleType == 'Live') {
$now = now();
$fromMinute = $now->copy()->subMinute()->format('d/m/Y H:i:s');
$toMinute = $now->format('d/m/Y H:i:s');
$greeting .= "The following report presents results from: $fromMinute to $toMinute.";
}
return new Content(
view: 'mail.invoice_data_report',
with: [
'company' => "CRI Digital Manufacturing Solutions",
'greeting' => $greeting,
'tableData' => $this->tableData,
'wishes' => "Thanks & Regards,<br>CRI Digital Manufacturing Solutions"
],
);
}
/**
* Get the attachments for the message.
*
* @return array<int, \Illuminate\Mail\Mailables\Attachment>
*/
public function attachments(): array
{
return [];
}
}