mirror of
				https://github.com/pelican-dev/panel.git
				synced 2025-11-04 06:26:52 +01:00 
			
		
		
		
	* Switch inserts to proper creates * Push `$token` to `$tokens[]` in `ToggleTwoFactorService` --------- Co-authored-by: RMartinOscar <40749467+RMartinOscar@users.noreply.github.com>
		
			
				
	
	
		
			254 lines
		
	
	
		
			6.5 KiB
		
	
	
	
		
			PHP
		
	
	
	
	
	
			
		
		
	
	
			254 lines
		
	
	
		
			6.5 KiB
		
	
	
	
		
			PHP
		
	
	
	
	
	
<?php
 | 
						|
 | 
						|
namespace App\Services\Activity;
 | 
						|
 | 
						|
use App\Models\User;
 | 
						|
use Illuminate\Support\Arr;
 | 
						|
use Throwable;
 | 
						|
use Webmozart\Assert\Assert;
 | 
						|
use Illuminate\Support\Collection;
 | 
						|
use App\Models\ActivityLog;
 | 
						|
use Illuminate\Database\Eloquent\Model;
 | 
						|
use Illuminate\Support\Facades\Request;
 | 
						|
use App\Models\Server;
 | 
						|
use Filament\Facades\Filament;
 | 
						|
use Illuminate\Database\ConnectionInterface;
 | 
						|
use Illuminate\Contracts\Auth\Factory as AuthFactory;
 | 
						|
 | 
						|
class ActivityLogService
 | 
						|
{
 | 
						|
    protected ?ActivityLog $activity = null;
 | 
						|
 | 
						|
    /** @var array<User> */
 | 
						|
    protected array $subjects = [];
 | 
						|
 | 
						|
    public function __construct(
 | 
						|
        protected AuthFactory $manager,
 | 
						|
        protected ActivityLogBatchService $batch,
 | 
						|
        protected ActivityLogTargetableService $targetable,
 | 
						|
        protected ConnectionInterface $connection
 | 
						|
    ) {}
 | 
						|
 | 
						|
    /**
 | 
						|
     * Sets the activity logger as having been caused by an anonymous
 | 
						|
     * user type.
 | 
						|
     */
 | 
						|
    public function anonymous(): self
 | 
						|
    {
 | 
						|
        $this->getActivity()->actor_id = null;
 | 
						|
        $this->getActivity()->actor_type = null;
 | 
						|
        $this->getActivity()->setRelation('actor', null);
 | 
						|
 | 
						|
        return $this;
 | 
						|
    }
 | 
						|
 | 
						|
    /**
 | 
						|
     * Sets the action for this activity log.
 | 
						|
     */
 | 
						|
    public function event(string $action): self
 | 
						|
    {
 | 
						|
        $this->getActivity()->event = $action;
 | 
						|
 | 
						|
        return $this;
 | 
						|
    }
 | 
						|
 | 
						|
    /**
 | 
						|
     * Set the description for this activity.
 | 
						|
     */
 | 
						|
    public function description(?string $description): self
 | 
						|
    {
 | 
						|
        $this->getActivity()->description = $description;
 | 
						|
 | 
						|
        return $this;
 | 
						|
    }
 | 
						|
 | 
						|
    /**
 | 
						|
     * Sets the subject model instance.
 | 
						|
     *
 | 
						|
     * @template T extends \Illuminate\Database\Eloquent\Model|\Illuminate\Contracts\Auth\Authenticatable
 | 
						|
     *
 | 
						|
     * @param  T|T[]|null  $subjects
 | 
						|
     */
 | 
						|
    public function subject(...$subjects): self
 | 
						|
    {
 | 
						|
        foreach (Arr::wrap($subjects) as $subject) {
 | 
						|
            if (is_null($subject)) {
 | 
						|
                continue;
 | 
						|
            }
 | 
						|
 | 
						|
            foreach ($this->subjects as $entry) {
 | 
						|
                // If this subject is already tracked in our array of subjects just skip over
 | 
						|
                // it and move on to the next one in the list.
 | 
						|
                if ($entry->is($subject)) {
 | 
						|
                    continue 2;
 | 
						|
                }
 | 
						|
            }
 | 
						|
 | 
						|
            $this->subjects[] = $subject;
 | 
						|
        }
 | 
						|
 | 
						|
        return $this;
 | 
						|
    }
 | 
						|
 | 
						|
    /**
 | 
						|
     * Sets the actor model instance.
 | 
						|
     */
 | 
						|
    public function actor(Model $actor): self
 | 
						|
    {
 | 
						|
        $this->getActivity()->actor()->associate($actor);
 | 
						|
 | 
						|
        return $this;
 | 
						|
    }
 | 
						|
 | 
						|
    /**
 | 
						|
     * Sets a custom property on the activity log instance.
 | 
						|
     *
 | 
						|
     * @param  string|array<string, mixed>  $key
 | 
						|
     * @param  mixed  $value
 | 
						|
     */
 | 
						|
    public function property($key, $value = null): self
 | 
						|
    {
 | 
						|
        $properties = $this->getActivity()->properties;
 | 
						|
        $this->activity->properties = is_array($key)
 | 
						|
            ? $properties->merge($key)
 | 
						|
            : $properties->put($key, $value);
 | 
						|
 | 
						|
        return $this;
 | 
						|
    }
 | 
						|
 | 
						|
    /**
 | 
						|
     * Attaches the instance request metadata to the activity log event.
 | 
						|
     */
 | 
						|
    public function withRequestMetadata(): self
 | 
						|
    {
 | 
						|
        return $this->property([
 | 
						|
            'ip' => Request::getClientIp(),
 | 
						|
            'useragent' => Request::userAgent(),
 | 
						|
        ]);
 | 
						|
    }
 | 
						|
 | 
						|
    /**
 | 
						|
     * Logs an activity log entry with the set values and then returns the
 | 
						|
     * model instance to the caller. If there is an exception encountered while
 | 
						|
     * performing this action it will be logged to the disk but will not interrupt
 | 
						|
     * the code flow.
 | 
						|
     */
 | 
						|
    public function log(?string $description = null): ActivityLog
 | 
						|
    {
 | 
						|
        $activity = $this->getActivity();
 | 
						|
 | 
						|
        if (!is_null($description)) {
 | 
						|
            $activity->description = $description;
 | 
						|
        }
 | 
						|
 | 
						|
        try {
 | 
						|
            return $this->save();
 | 
						|
        } catch (Throwable $exception) {
 | 
						|
            if (config('app.env') !== 'production') {
 | 
						|
                throw $exception;
 | 
						|
            }
 | 
						|
 | 
						|
            logger()->error($exception);
 | 
						|
        }
 | 
						|
 | 
						|
        return $activity;
 | 
						|
    }
 | 
						|
 | 
						|
    /**
 | 
						|
     * Returns a cloned instance of the service allowing for the creation of a base
 | 
						|
     * activity log with the ability to change values on the fly without impact.
 | 
						|
     */
 | 
						|
    public function clone(): self
 | 
						|
    {
 | 
						|
        return clone $this;
 | 
						|
    }
 | 
						|
 | 
						|
    /**
 | 
						|
     * Executes the provided callback within the scope of a database transaction
 | 
						|
     * and will only save the activity log entry if everything else successfully
 | 
						|
     * settles.
 | 
						|
     *
 | 
						|
     * @throws \Throwable
 | 
						|
     */
 | 
						|
    public function transaction(\Closure $callback): mixed
 | 
						|
    {
 | 
						|
        return $this->connection->transaction(function () use ($callback) {
 | 
						|
            $response = $callback($this);
 | 
						|
 | 
						|
            $this->save();
 | 
						|
 | 
						|
            return $response;
 | 
						|
        });
 | 
						|
    }
 | 
						|
 | 
						|
    /**
 | 
						|
     * Resets the instance and clears out the log.
 | 
						|
     */
 | 
						|
    public function reset(): void
 | 
						|
    {
 | 
						|
        $this->activity = null;
 | 
						|
        $this->subjects = [];
 | 
						|
    }
 | 
						|
 | 
						|
    /**
 | 
						|
     * Returns the current activity log instance.
 | 
						|
     */
 | 
						|
    protected function getActivity(): ActivityLog
 | 
						|
    {
 | 
						|
        if ($this->activity) {
 | 
						|
            return $this->activity;
 | 
						|
        }
 | 
						|
 | 
						|
        $this->activity = new ActivityLog([
 | 
						|
            'ip' => Request::ip(),
 | 
						|
            'batch_uuid' => $this->batch->uuid(),
 | 
						|
            'properties' => Collection::make([]),
 | 
						|
            'api_key_id' => $this->targetable->apiKeyId(),
 | 
						|
        ]);
 | 
						|
 | 
						|
        if ($subject = $this->targetable->subject()) {
 | 
						|
            $this->subject($subject);
 | 
						|
        } elseif ($tenant = Filament::getTenant()) {
 | 
						|
            if ($tenant instanceof Server) {
 | 
						|
                $this->subject($tenant);
 | 
						|
            }
 | 
						|
        }
 | 
						|
 | 
						|
        if ($actor = $this->targetable->actor()) {
 | 
						|
            $this->actor($actor);
 | 
						|
        } elseif ($user = $this->manager->guard()->user()) {
 | 
						|
            $this->actor($user);
 | 
						|
        }
 | 
						|
 | 
						|
        return $this->activity;
 | 
						|
    }
 | 
						|
 | 
						|
    /**
 | 
						|
     * Saves the activity log instance and attaches all the subject models.
 | 
						|
     *
 | 
						|
     * @throws \Throwable
 | 
						|
     */
 | 
						|
    protected function save(): ActivityLog
 | 
						|
    {
 | 
						|
        Assert::notNull($this->activity);
 | 
						|
 | 
						|
        $response = $this->connection->transaction(function () {
 | 
						|
            $this->activity->save();
 | 
						|
 | 
						|
            foreach ($this->subjects as $subject) {
 | 
						|
                $this->activity->subjects()->forceCreate([
 | 
						|
                    'subject_id' => $subject->getKey(),
 | 
						|
                    'subject_type' => $subject->getMorphClass(),
 | 
						|
                ]);
 | 
						|
            }
 | 
						|
 | 
						|
            return $this->activity;
 | 
						|
        });
 | 
						|
 | 
						|
        $this->activity = null;
 | 
						|
        $this->subjects = [];
 | 
						|
 | 
						|
        return $response;
 | 
						|
    }
 | 
						|
}
 |