2022-10-24 01:04:22 +01:00
|
|
|
<?php
|
|
|
|
|
|
|
|
namespace App\Services;
|
|
|
|
|
|
|
|
use App\Absence;
|
|
|
|
use App\Exceptions\AbsenceNotActionableException;
|
|
|
|
use App\Notifications\AbsenceRequestApproved;
|
|
|
|
use App\Notifications\AbsenceRequestCancelled;
|
|
|
|
use App\Notifications\AbsenceRequestDeclined;
|
|
|
|
use App\Notifications\AbsenceRequestEnded;
|
|
|
|
use App\Notifications\NewAbsenceRequest;
|
|
|
|
use App\User;
|
|
|
|
use Carbon\Carbon;
|
|
|
|
use Illuminate\Contracts\Auth\Authenticatable;
|
|
|
|
use Illuminate\Http\Request;
|
|
|
|
use Illuminate\Support\Facades\Auth;
|
|
|
|
use Illuminate\Support\Facades\Log;
|
|
|
|
|
|
|
|
class AbsenceService
|
|
|
|
{
|
|
|
|
/**
|
|
|
|
* Determines whether someone already has an active leave of absence request
|
|
|
|
*
|
2023-01-15 00:04:00 +00:00
|
|
|
* @param User $user The user to check
|
2022-10-24 01:04:22 +01:00
|
|
|
* @return bool Their status
|
|
|
|
*/
|
2023-01-15 00:04:00 +00:00
|
|
|
public function hasActiveRequest(Authenticatable $user): bool
|
|
|
|
{
|
2022-10-24 01:04:22 +01:00
|
|
|
$absences = Absence::where('requesterID', $user->id)->get();
|
|
|
|
|
|
|
|
foreach ($absences as $absence) {
|
|
|
|
// Or we could adjust the query (using a model scope) to only return valid absences;
|
|
|
|
// If there are any, refuse to store more, but this approach also works
|
|
|
|
// A model scope that only returns cancelled, declined and ended absences could also be implemented for future use
|
2023-01-15 00:04:00 +00:00
|
|
|
if (in_array($absence->getRawOriginal('status'), ['PENDING', 'APPROVED'])) {
|
2022-10-24 01:04:22 +01:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
public function createAbsence(Authenticatable $requester, Request $request)
|
|
|
|
{
|
|
|
|
$absence = Absence::create([
|
|
|
|
'requesterID' => $requester->id,
|
|
|
|
'start' => $request->start_date,
|
|
|
|
'predicted_end' => $request->predicted_end,
|
2023-01-15 00:04:00 +00:00
|
|
|
'available_assist' => $request->available_assist == 'on',
|
2022-10-24 01:04:22 +01:00
|
|
|
'reason' => $request->reason,
|
|
|
|
'status' => 'PENDING',
|
|
|
|
]);
|
|
|
|
|
2022-11-08 00:56:56 +00:00
|
|
|
User::whereHas('roles', function ($q) {
|
|
|
|
$q->where('name', 'admin');
|
|
|
|
})->get()->each(function ($user, $key) use ($absence) {
|
2022-11-19 02:59:25 +00:00
|
|
|
if ($user->isNot($absence->requester)) {
|
2022-11-12 22:09:42 +00:00
|
|
|
$user->notify((new NewAbsenceRequest($absence)));
|
|
|
|
}
|
2022-11-08 00:56:56 +00:00
|
|
|
});
|
|
|
|
|
2022-10-24 01:04:22 +01:00
|
|
|
Log::info('Processing new leave of absence request.', [
|
|
|
|
'requesting_user' => $requester->email,
|
|
|
|
'absenceid' => $absence->id,
|
2023-01-15 00:04:00 +00:00
|
|
|
'reason' => $request->reason,
|
2022-10-24 01:04:22 +01:00
|
|
|
]);
|
|
|
|
|
|
|
|
return $absence;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Sets an absence as Approved.
|
|
|
|
*
|
2023-01-15 00:04:00 +00:00
|
|
|
* @param Absence $absence The absence to approve.
|
2022-10-24 01:04:22 +01:00
|
|
|
* @return Absence The approved absence.
|
2023-01-15 00:04:00 +00:00
|
|
|
*
|
2022-10-24 01:04:22 +01:00
|
|
|
* @throws AbsenceNotActionableException
|
|
|
|
*/
|
|
|
|
public function approveAbsence(Absence $absence)
|
|
|
|
{
|
|
|
|
Log::info('An absence request has just been approved.', [
|
|
|
|
'absenceid' => $absence->id,
|
|
|
|
'reviewing_admim' => Auth::user()->email,
|
2023-01-15 00:04:00 +00:00
|
|
|
'new_status' => 'APPROVED',
|
2022-10-24 01:04:22 +01:00
|
|
|
]);
|
|
|
|
|
2022-11-12 22:09:42 +00:00
|
|
|
$absence->setApproved()
|
2022-10-24 01:04:22 +01:00
|
|
|
->requester->notify(new AbsenceRequestApproved($absence));
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Sets an absence as Declined.
|
|
|
|
*
|
2023-01-15 00:04:00 +00:00
|
|
|
* @param Absence $absence The absence to decline.
|
2022-10-24 01:04:22 +01:00
|
|
|
* @return Absence The declined absence.
|
2023-01-15 00:04:00 +00:00
|
|
|
*
|
2022-10-24 01:04:22 +01:00
|
|
|
* @throws AbsenceNotActionableException
|
|
|
|
*/
|
|
|
|
public function declineAbsence(Absence $absence)
|
|
|
|
{
|
|
|
|
Log::warning('An absence request has just been declined.', [
|
|
|
|
'absenceid' => $absence->id,
|
|
|
|
'reviewing_admim' => Auth::user()->email,
|
2023-01-15 00:04:00 +00:00
|
|
|
'new_status' => 'DECLINED',
|
2022-10-24 01:04:22 +01:00
|
|
|
]);
|
|
|
|
|
2022-11-12 22:09:42 +00:00
|
|
|
$absence->setDeclined()
|
2022-10-24 01:04:22 +01:00
|
|
|
->requester->notify(new AbsenceRequestDeclined($absence));
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Sets an absence as Cancelled.
|
|
|
|
*
|
2023-01-15 00:04:00 +00:00
|
|
|
* @param Absence $absence The absence to cancel.
|
2022-10-24 01:04:22 +01:00
|
|
|
* @return Absence The cancelled absence.
|
2023-01-15 00:04:00 +00:00
|
|
|
*
|
2022-10-24 01:04:22 +01:00
|
|
|
* @throws AbsenceNotActionableException
|
|
|
|
*/
|
|
|
|
public function cancelAbsence(Absence $absence)
|
|
|
|
{
|
|
|
|
Log::warning('An absence request has just been cancelled (only cancellable by requester).', [
|
|
|
|
'absenceid' => $absence->id,
|
2023-01-15 00:04:00 +00:00
|
|
|
'new_status' => 'CANCELLED',
|
2022-10-24 01:04:22 +01:00
|
|
|
]);
|
|
|
|
|
2022-11-12 22:09:42 +00:00
|
|
|
$absence->setCancelled()
|
2022-10-24 01:04:22 +01:00
|
|
|
->requester->notify(new AbsenceRequestCancelled($absence));
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Sets an absence as Ended.
|
|
|
|
*
|
2023-01-15 00:04:00 +00:00
|
|
|
* @param Absence $absence
|
2022-10-24 01:04:22 +01:00
|
|
|
* @return bool
|
|
|
|
*/
|
|
|
|
public function endAbsence(Absence $absence)
|
|
|
|
{
|
|
|
|
Log::info('An absence request has just expired.', [
|
|
|
|
'absenceid' => $absence->id,
|
2023-01-15 00:04:00 +00:00
|
|
|
'new_status' => 'ENDED',
|
2022-10-24 01:04:22 +01:00
|
|
|
]);
|
|
|
|
|
2022-11-12 22:09:42 +00:00
|
|
|
$absence->setEnded()
|
2022-10-24 01:04:22 +01:00
|
|
|
->requester->notify(new AbsenceRequestEnded($absence));
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Removes an absence
|
|
|
|
*
|
2023-01-15 00:04:00 +00:00
|
|
|
* @param Absence $absence The absence to remove.
|
2022-10-24 01:04:22 +01:00
|
|
|
* @return bool Whether the absence was removed.
|
|
|
|
*/
|
|
|
|
public function removeAbsence(Absence $absence): bool
|
|
|
|
{
|
|
|
|
Log::alert('An absence request has just been removed.', [
|
|
|
|
'absence_details' => $absence,
|
|
|
|
'reviewing_admim' => Auth::user()->email,
|
|
|
|
]);
|
|
|
|
|
|
|
|
return $absence->delete();
|
|
|
|
}
|
|
|
|
|
2022-11-19 03:05:55 +00:00
|
|
|
/**
|
|
|
|
* End all expired absences in the application
|
2023-01-15 00:04:00 +00:00
|
|
|
*
|
2022-11-19 03:05:55 +00:00
|
|
|
* @return void
|
|
|
|
*/
|
|
|
|
public function endExpired(): void
|
2022-10-24 01:04:22 +01:00
|
|
|
{
|
2023-01-15 00:04:00 +00:00
|
|
|
foreach (Absence::all() as $absence) {
|
2022-11-19 03:05:55 +00:00
|
|
|
// tell the absence we want to check for cancelability
|
2023-01-15 00:04:00 +00:00
|
|
|
if (! Carbon::parse($absence->predicted_end)->isFuture() && $absence->isActionable(true)) {
|
2022-10-24 01:04:22 +01:00
|
|
|
$this->endAbsence($absence);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|