|
|
|
@@ -7,17 +7,12 @@ use Illuminate\Notifications\Notifiable;
|
|
|
|
|
use Illuminate\Foundation\Auth\User as Authenticatable;
|
|
|
|
|
use Illuminate\Support\Collection;
|
|
|
|
|
use Illuminate\Database\Eloquent\Collection as DatabaseCollection;
|
|
|
|
|
use Illuminate\Support\Facades\Cache;
|
|
|
|
|
use Illuminate\Support\Facades\DB;
|
|
|
|
|
use Illuminate\Support\Facades\Session;
|
|
|
|
|
use Laravel\Passport\HasApiTokens;
|
|
|
|
|
use Leenooks\Traits\ScopeActive;
|
|
|
|
|
use Leenooks\Traits\UserSwitch;
|
|
|
|
|
|
|
|
|
|
use App\Interfaces\IDs;
|
|
|
|
|
use App\Models\Scopes\SiteScope;
|
|
|
|
|
use App\Notifications\ResetPassword as ResetPasswordNotification;
|
|
|
|
|
use App\Traits\SiteID;
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* Class User
|
|
|
|
@@ -27,14 +22,12 @@ use App\Traits\SiteID;
|
|
|
|
|
*/
|
|
|
|
|
class User extends Authenticatable implements IDs
|
|
|
|
|
{
|
|
|
|
|
use HasFactory,HasApiTokens,Notifiable,UserSwitch,SiteID,ScopeActive;
|
|
|
|
|
use HasFactory,HasApiTokens,Notifiable,UserSwitch,ScopeActive;
|
|
|
|
|
|
|
|
|
|
private const CACHE_TIME = 3600;
|
|
|
|
|
|
|
|
|
|
protected $dates = [
|
|
|
|
|
'created_at',
|
|
|
|
|
'updated_at',
|
|
|
|
|
'last_access'
|
|
|
|
|
protected $casts = [
|
|
|
|
|
'last_access' => 'datetime:Y-m-d H:i:s',
|
|
|
|
|
];
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
@@ -96,27 +89,25 @@ class User extends Authenticatable implements IDs
|
|
|
|
|
/* RELATIONS */
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* The accounts that this user manages
|
|
|
|
|
*
|
|
|
|
|
* @return \Illuminate\Database\Eloquent\Relations\HasMany
|
|
|
|
|
* @note This cannot be loaded with "with"?
|
|
|
|
|
* This user's accounts
|
|
|
|
|
*/
|
|
|
|
|
public function accounts()
|
|
|
|
|
{
|
|
|
|
|
return $this->hasMany(Account::class)
|
|
|
|
|
->orWhereIn('id',$this->rtm_accounts()->pluck('id'))
|
|
|
|
|
->active();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* This users invoices
|
|
|
|
|
* The accounts that this user manages
|
|
|
|
|
*
|
|
|
|
|
* @return \Illuminate\Database\Eloquent\Relations\HasManyThrough
|
|
|
|
|
* @deprecated Accounts have invoices, not users
|
|
|
|
|
* @return \Illuminate\Database\Eloquent\Relations\HasMany
|
|
|
|
|
* @note This cannot be preloaded with load() or with() - $this is a blank object
|
|
|
|
|
*/
|
|
|
|
|
public function invoices()
|
|
|
|
|
public function accounts_all()
|
|
|
|
|
{
|
|
|
|
|
return $this->hasManyThrough(Invoice::class,Account::class)
|
|
|
|
|
return $this->hasMany(Account::class)
|
|
|
|
|
->when((! is_null($this->rtm) && $this->id),
|
|
|
|
|
fn($query)=>$query->orWhereIn('rtm_id',$this->rtm?->children_all()->pluck('id')))
|
|
|
|
|
->active();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
@@ -132,37 +123,20 @@ class User extends Authenticatable implements IDs
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* Return the routes to market account for this user
|
|
|
|
|
*
|
|
|
|
|
* @return \Illuminate\Database\Eloquent\Relations\HasOneThrough
|
|
|
|
|
*/
|
|
|
|
|
public function rtm()
|
|
|
|
|
{
|
|
|
|
|
return $this->hasOneThrough(Rtm::class,Account::class);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* The services this user has
|
|
|
|
|
*
|
|
|
|
|
* @return \Illuminate\Database\Eloquent\Relations\HasManyThrough
|
|
|
|
|
* @deprecated Accounts have services, not users
|
|
|
|
|
*/
|
|
|
|
|
public function services()
|
|
|
|
|
{
|
|
|
|
|
return $this->hasManyThrough(Service::class,Account::class)
|
|
|
|
|
->with(['product.type'])
|
|
|
|
|
->active();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* Supplier configuration for this user
|
|
|
|
|
*
|
|
|
|
|
* @return \Illuminate\Database\Eloquent\Relations\BelongsToMany
|
|
|
|
|
* @deprecated Move to account->suppliers()
|
|
|
|
|
* @deprecated To move to account->suppliers()
|
|
|
|
|
*/
|
|
|
|
|
public function suppliers()
|
|
|
|
|
{
|
|
|
|
|
return $this->belongsToMany(Supplier::class)
|
|
|
|
|
->where('supplier_user.site_id',$this->site_id)
|
|
|
|
|
->withPivot('id','created_at');
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
@@ -179,7 +153,7 @@ class User extends Authenticatable implements IDs
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* Logged in users full name
|
|
|
|
|
* Logged in user's full name
|
|
|
|
|
*
|
|
|
|
|
* @return string
|
|
|
|
|
*/
|
|
|
|
@@ -188,25 +162,9 @@ class User extends Authenticatable implements IDs
|
|
|
|
|
return sprintf('%s %s',$this->firstname,$this->lastname);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* Return my accounts, but only those accounts with the same group_id
|
|
|
|
|
*
|
|
|
|
|
* @note Users can only manage accounts with the same group ID, thereby ensuring they dont see different
|
|
|
|
|
* pricing options - since prices can be controlled by groups
|
|
|
|
|
* @return Collection
|
|
|
|
|
*/
|
|
|
|
|
public function getMyAccountsAttribute(): Collection
|
|
|
|
|
{
|
|
|
|
|
$result = $this->accounts->where('user_id',$this->id);
|
|
|
|
|
|
|
|
|
|
if (! $result->count())
|
|
|
|
|
return $result;
|
|
|
|
|
|
|
|
|
|
return $this->isReseller() ? $result : $result->groupBy('group.id')->first();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* Return a friendly string of this persons role
|
|
|
|
|
*
|
|
|
|
|
* @return string
|
|
|
|
|
*/
|
|
|
|
|
public function getRoleAttribute(): string
|
|
|
|
@@ -259,22 +217,6 @@ class User extends Authenticatable implements IDs
|
|
|
|
|
|
|
|
|
|
/* METHODS */
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* Show this user's clients with service movements
|
|
|
|
|
*
|
|
|
|
|
* A service movement, is an active service where the status is not ACTIVE
|
|
|
|
|
*
|
|
|
|
|
* @return DatabaseCollection
|
|
|
|
|
*/
|
|
|
|
|
public function client_service_movements(): DatabaseCollection
|
|
|
|
|
{
|
|
|
|
|
return Service::active()
|
|
|
|
|
->serviceUserAuthorised($this)
|
|
|
|
|
->where('order_status','!=','ACTIVE')
|
|
|
|
|
->with(['account','product'])
|
|
|
|
|
->get();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* Determine if the user is an admin of the user with $id
|
|
|
|
|
*
|
|
|
|
@@ -283,7 +225,11 @@ class User extends Authenticatable implements IDs
|
|
|
|
|
*/
|
|
|
|
|
public function isAdmin(User $user=NULL): bool
|
|
|
|
|
{
|
|
|
|
|
return $user->exists AND $this->isReseller() AND $this->accounts->pluck('user_id')->contains($user->id);
|
|
|
|
|
return $user->exists
|
|
|
|
|
&& $this->isReseller()
|
|
|
|
|
&& $this->accounts_all
|
|
|
|
|
->pluck('user_id')
|
|
|
|
|
->contains($user->id);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
@@ -311,9 +257,11 @@ class User extends Authenticatable implements IDs
|
|
|
|
|
*
|
|
|
|
|
* @param bool $future
|
|
|
|
|
* @return DatabaseCollection
|
|
|
|
|
* @deprecated This should be done in accounts
|
|
|
|
|
*/
|
|
|
|
|
public function next_invoice_items(bool $future=FALSE): DatabaseCollection
|
|
|
|
|
public function next_invoice_items(bool $future=FALSE): Collection
|
|
|
|
|
{
|
|
|
|
|
return collect();
|
|
|
|
|
$result = new DatabaseCollection;
|
|
|
|
|
$this->services->load(['invoice_items.taxes']);
|
|
|
|
|
|
|
|
|
@@ -339,117 +287,6 @@ class User extends Authenticatable implements IDs
|
|
|
|
|
return $result;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* Return an SQL query that will return a list of invoices
|
|
|
|
|
*
|
|
|
|
|
* @return \Illuminate\Database\Query\Builder
|
|
|
|
|
*/
|
|
|
|
|
private function query_invoice_items()
|
|
|
|
|
{
|
|
|
|
|
return DB::table('invoice_items')
|
|
|
|
|
->select([
|
|
|
|
|
'invoice_id',
|
|
|
|
|
DB::raw('invoice_items.id AS invoice_item_id'),
|
|
|
|
|
DB::raw('COALESCE(invoice_items.discount_amt,0) AS discount'),
|
|
|
|
|
DB::raw('quantity*price_base AS base'),
|
|
|
|
|
DB::raw('invoice_item_taxes.amount AS tax'),
|
|
|
|
|
])
|
|
|
|
|
->leftjoin('invoice_item_taxes',['invoice_item_taxes.invoice_item_id'=>'invoice_items.id'])
|
|
|
|
|
->where('invoice_items.active',TRUE);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* Return an SQL query that will return payment summaries by invoices.
|
|
|
|
|
*
|
|
|
|
|
* @return \Illuminate\Database\Query\Builder
|
|
|
|
|
*/
|
|
|
|
|
private function query_payment_items()
|
|
|
|
|
{
|
|
|
|
|
return DB::table('payment_items')
|
|
|
|
|
->select([
|
|
|
|
|
'payment_id',
|
|
|
|
|
'invoice_id',
|
|
|
|
|
DB::raw('SUM(amount) AS allocate'),
|
|
|
|
|
])
|
|
|
|
|
->where('amount','>',0)
|
|
|
|
|
->groupBy(['invoice_id','payment_id']);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* Return an SQL query that will summarise invoices with payments
|
|
|
|
|
*
|
|
|
|
|
* @return \Illuminate\Database\Query\Builder
|
|
|
|
|
* @todo change this to just return outstanding invoices as a collection.
|
|
|
|
|
* @deprecated Use account->invoiceSummary
|
|
|
|
|
*/
|
|
|
|
|
public function query_invoice_summary()
|
|
|
|
|
{
|
|
|
|
|
$invoices = (new Invoice)
|
|
|
|
|
->select([
|
|
|
|
|
'invoice_id',
|
|
|
|
|
DB::raw('SUM(discount) AS discount'),
|
|
|
|
|
DB::raw('SUM(base) AS base'),
|
|
|
|
|
DB::raw('SUM(tax) AS tax'),
|
|
|
|
|
DB::raw('base+tax-discount AS total'),
|
|
|
|
|
DB::raw('0 AS payments'),
|
|
|
|
|
DB::raw('0 AS payment_fees'),
|
|
|
|
|
])
|
|
|
|
|
->from($this->query_invoice_items(),'II')
|
|
|
|
|
->join('invoices',['invoices.id'=>'II.invoice_id'])
|
|
|
|
|
->whereIN('account_id',$this->accounts->pluck('id'))
|
|
|
|
|
->where('invoices.active',TRUE)
|
|
|
|
|
->groupBy(['invoice_id','base','tax','discount']);
|
|
|
|
|
|
|
|
|
|
$payments = (new Payment)
|
|
|
|
|
->select([
|
|
|
|
|
'invoice_id',
|
|
|
|
|
DB::raw('0 AS discount'),
|
|
|
|
|
DB::raw('0 AS base'),
|
|
|
|
|
DB::raw('0 AS tax'),
|
|
|
|
|
DB::raw('0 AS total'),
|
|
|
|
|
DB::raw('SUM(allocate) AS payments'),
|
|
|
|
|
DB::raw('SUM(fees_amt) AS payment_fees'),
|
|
|
|
|
])
|
|
|
|
|
->from($this->query_payment_items(),'PI')
|
|
|
|
|
->join('payments',['payments.id'=>'PI.payment_id'])
|
|
|
|
|
->whereIN('account_id',$this->accounts->pluck('id'))
|
|
|
|
|
//->where('payments.active',TRUE) // @todo To implement
|
|
|
|
|
->groupBy(['invoice_id']);
|
|
|
|
|
|
|
|
|
|
$summary = (new Invoice)
|
|
|
|
|
->withoutGlobalScope(SiteScope::class)
|
|
|
|
|
->select([
|
|
|
|
|
'invoice_id',
|
|
|
|
|
DB::raw('SUM(discount) AS discount'),
|
|
|
|
|
DB::raw('SUM(base) AS invoice_base'),
|
|
|
|
|
DB::raw('SUM(tax) AS invoice_tax'),
|
|
|
|
|
DB::raw('SUM(total) AS invoice_total'),
|
|
|
|
|
DB::raw('SUM(payments) AS payments'),
|
|
|
|
|
DB::raw('SUM(payment_fees) AS payment_fees'),
|
|
|
|
|
])
|
|
|
|
|
->from($invoices->unionAll($payments),'invoices')
|
|
|
|
|
->groupBy(['invoice_id']);
|
|
|
|
|
|
|
|
|
|
return (new Invoice)
|
|
|
|
|
->select([
|
|
|
|
|
'account_id',
|
|
|
|
|
'id',
|
|
|
|
|
'due_at',
|
|
|
|
|
'created_at',
|
|
|
|
|
'discount',
|
|
|
|
|
'invoice_base',
|
|
|
|
|
'invoice_tax',
|
|
|
|
|
'invoice_total',
|
|
|
|
|
'payments',
|
|
|
|
|
'payment_fees',
|
|
|
|
|
DB::raw('invoice_total-payments AS balance'),
|
|
|
|
|
])
|
|
|
|
|
->join('invoices',['invoices.id'=>'invoice_id'])
|
|
|
|
|
->with(['items.taxes'])
|
|
|
|
|
->from($summary,'summary')
|
|
|
|
|
->groupBy(['id','account_id','discount','invoice_base','invoice_tax','invoice_total','payments','payment_fees']);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* Determine what the logged in user's role is
|
|
|
|
|
* + Wholesaler - aka Super User
|
|
|
|
@@ -460,67 +297,8 @@ class User extends Authenticatable implements IDs
|
|
|
|
|
*/
|
|
|
|
|
public function role()
|
|
|
|
|
{
|
|
|
|
|
// Cache our role for this session
|
|
|
|
|
$cache_key = sprintf('%s:%s:%s',$this->id,__METHOD__,Session::getId());
|
|
|
|
|
|
|
|
|
|
return Cache::remember($cache_key,self::CACHE_TIME,function() {
|
|
|
|
|
// Get the RTM for our accounts
|
|
|
|
|
$rtms = Rtm::whereIn('account_id',$this->accounts->pluck('id'))->get();
|
|
|
|
|
|
|
|
|
|
// If I have no parent, I am the wholesaler
|
|
|
|
|
if ($rtms->whereNull('parent_id')->count())
|
|
|
|
|
return 'wholesaler';
|
|
|
|
|
|
|
|
|
|
// If I exist in the RTM table, I'm a reseller
|
|
|
|
|
else if ($rtms->count())
|
|
|
|
|
return 'reseller';
|
|
|
|
|
|
|
|
|
|
// Otherwise a client
|
|
|
|
|
else
|
|
|
|
|
return 'customer';
|
|
|
|
|
});
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* Return the accounts that this user can manage
|
|
|
|
|
* This method is a helper to User::accounts() - use $user->accounts instead
|
|
|
|
|
*
|
|
|
|
|
* @return Collection
|
|
|
|
|
*/
|
|
|
|
|
private function rtm_accounts(): Collection
|
|
|
|
|
{
|
|
|
|
|
return Account::whereIn('rtm_id',$this->rtm_list()->pluck('id'))
|
|
|
|
|
->get();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* Return the RTM hierarchy that this user can manage
|
|
|
|
|
*
|
|
|
|
|
* @param Rtm|null $rtm
|
|
|
|
|
* @return Collection
|
|
|
|
|
*/
|
|
|
|
|
public function rtm_list(Rtm $rtm=NULL): Collection
|
|
|
|
|
{
|
|
|
|
|
// If this user doesnt manage any accounts
|
|
|
|
|
if (! $this->exists || ! $this->rtm)
|
|
|
|
|
return collect();
|
|
|
|
|
|
|
|
|
|
$list = collect();
|
|
|
|
|
|
|
|
|
|
// Add this RTM to the list
|
|
|
|
|
if (! $rtm) {
|
|
|
|
|
$list->push($this->rtm);
|
|
|
|
|
$children = $this->rtm->children;
|
|
|
|
|
|
|
|
|
|
} else {
|
|
|
|
|
$list->push($rtm);
|
|
|
|
|
$children =$rtm->children;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Capture any children
|
|
|
|
|
foreach ($children as $child)
|
|
|
|
|
$list->push($this->rtm_list($child));
|
|
|
|
|
|
|
|
|
|
return $rtm ? $list : $list->flatten()->unique(function($item) { return $item->id; });
|
|
|
|
|
return $this->rtm
|
|
|
|
|
? ($this->rtm->parent_id ? 'reseller' : 'wholesaler')
|
|
|
|
|
: 'customer';
|
|
|
|
|
}
|
|
|
|
|
}
|