9d2b63c8f4
SyncController now handles the full mutation vocabulary: - inspection.create (idempotent by uuid; project/layer derived from feature; authz member + 'create inspections'; status defaults to completed). - issue.create (idempotent; authz member + 'create issues'). - issue.update (by server id; authz member + 'edit issues'; sets resolved_at when resolved/closed; last-write-wins conflict). - feature.update (by server id; authz member + 'update progress'; recomputes phase progress; last-write-wins conflict). - Conflict detection: client_updated_at vs server updated_at → returns 'conflict' with the current server value. Added uuid + client_updated_at to features/inspections/issues (guarded migration) and their fillables. Tests: 16 passing (added inspection/issue/feature + conflict). Note: 2 PRE-EXISTING test failures remain (not from this work, sqlite-only): ExampleTest expects '/'=200 (app redirects), and the dashboard route uses MySQL FIELD() which sqlite lacks. Co-Authored-By: Claude Opus 4.8 (1M context) <noreply@anthropic.com>
66 lines
1.6 KiB
PHP
66 lines
1.6 KiB
PHP
<?php
|
|
|
|
namespace App\Models;
|
|
|
|
use Illuminate\Database\Eloquent\Model;
|
|
use Illuminate\Database\Eloquent\SoftDeletes;
|
|
use App\Traits\LogsActivity;
|
|
|
|
class Inspection extends Model
|
|
{
|
|
use SoftDeletes, LogsActivity;
|
|
|
|
const STATUSES = ['pending', 'in_progress', 'completed', 'approved', 'rejected'];
|
|
const RESULTS = ['pass', 'fail', 'conditional'];
|
|
|
|
protected $fillable = [
|
|
'project_id', 'layer_id', 'feature_id', 'template_id', 'user_id',
|
|
'data', 'status', 'inspector_user_id', 'completed_at', 'result', 'notes',
|
|
'uuid', 'client_updated_at',
|
|
];
|
|
|
|
protected $casts = [
|
|
'data' => 'array',
|
|
'completed_at' => 'datetime',
|
|
];
|
|
|
|
public function project()
|
|
{
|
|
return $this->belongsTo(Project::class);
|
|
}
|
|
|
|
public function layer()
|
|
{
|
|
return $this->belongsTo(Layer::class);
|
|
}
|
|
|
|
public function template()
|
|
{
|
|
return $this->belongsTo(InspectionTemplate::class);
|
|
}
|
|
|
|
public function user()
|
|
{
|
|
return $this->belongsTo(User::class);
|
|
}
|
|
|
|
public function inspector()
|
|
{
|
|
return $this->belongsTo(User::class, 'inspector_user_id');
|
|
}
|
|
|
|
public function feature()
|
|
{
|
|
return $this->belongsTo(Feature::class, 'feature_id');
|
|
}
|
|
|
|
public function issues()
|
|
{
|
|
return $this->hasMany(Issue::class);
|
|
}
|
|
|
|
public function scopePending($q) { return $q->where('status', 'pending'); }
|
|
public function scopeCompleted($q) { return $q->where('status', 'completed'); }
|
|
public function scopeRejected($q) { return $q->where('status', 'rejected'); }
|
|
}
|