* Add new panel * Add some basic resource pages * Wip * Wip terminal * Wip * Add new panel * Add some basic resource pages * Wip * [Sub-Users] Add Invite TODO: The logic with permissions * [Sub-Users] Fix Creation * [Cron] Add basics * Add basic auth and messages * Add basic buttons * WIP on issue/353 * WIP on issue/353 * Add Database page * Update Database Page * Start of Backup Page * Composer Update * Changes * Send input * Remove this includes * Better offline handling * Consolidate top nav config * Update Backups Page * Update Backups * Change name * Add Assign All, Layout Fixes. * conflict * update schedule pages * fix phpstan * update pint.json * add cron presets to schedule * fix tests * fix task creation * schedules: disable task creation if limit is reached & disable backup action if backup limit is 0 * update activity pages * update resources * Update Edit User TODO: actually save permissions when they're changed. TODO: Figure out why Control does not update it's state... but the rest do... * .... Sure it works. TODO: Update permissions when you save editing a sub user. * user: update canAccessPanel & canAccessTenant * add helper to convert bytes into readable format * very basic file explorer * files: fix some stuff & remove dummy data * files: better error handling * files: basic file editor * files: add some actions * File manager updates * files: fix paths * Revery Composer Upgrade, Fixes SQLite * fix: Pint (#517) feat: MenuItems to and from admin * Update File Editing Updated File Editing to its own page, Added Permission checks for file manager. Co-authored-by: Boy132 <Boy132@users.noreply.github.com> * add enum for editor langs * files: add upload & pull actions * fix build * files: handle images * Update to Filament v3.2.98 * files: add remaining actions * use `authorize` instead of `hidden` * fix canAccessTenant * update date columns * files: testing & fixes * Fix File Names Co-authored-by: lancepioch <git@lance.sh> * Combine Pull/Upload * Fix BulkDelete * Uncontained tabs * Hide Lang Selection, Move Actions * Update Monaco, more custom * Add livewire config livewire limits uploads to 12MB... who knows why... Fixed uploading a single files failing * files: fix record url * basic setup for settings & startup page * make abstract class for simple app pages * Basic Startup Page * Update nav sort * small cleanup * startup: fix shouldHideComponent & getSelectOptionsFromRules * startup: fix non editable fields & set default value * startup: add todo for save button * Save Variables after update & off click Variables update when the user clicks off the input. * Notifications are cool * Add rule validation * Sort variables by sortid * pint * Settings Page + Startup Changes * settings: cleanup * refactor: use server model for ServerFormPage (formerly known as SimplePage) * Use Repeater for variables * Add Network, Remove breadcrumbs * Add paginated to file explorer * Fix updating variables * Add link to go to new client area * fix after merge * Add graphs to console page Graphs still need to get the data from the web socket. * fix pint & phpstan * fix authorizeAccess for EditFiles and Startup page * Fix rules on startup page * Update console size * Fix node name * add "global search" to files list requires https://github.com/pelican-dev/wings/pull/44 * remove debug dummy data * update view action on ListServers * enable SPA mode for app panel * remove colors from app panel they are defined globally in AppServiceProvider * update global search ui a bit (to be replaced with a custom page that is similar to the list files table) * add own page for global search untested - and route needs cleanup (if possible) * fix File getRows * remove "path" from SearchFiles (for now) * fix caching for searched files * add title and breadcrumbs to global search page * make cpu & memory charts on console page working * fix phpstan * add missing import * cleanup console views & widgets * add overview stats to console * don't be so lazy, console! * make history working * decode data to get array * add missing On * fix json_decode * change polling to 1 sec * hide "0" cpu/ memory * add data to network chart * Remove data labels * fix data on network chart * fix data on network chart (2nd try) * WIP Network Stats * Remove test * Change MaxWidth * run pint * fix phpstan * Fix storeStats cast * make $data a string this time for real * update visible check for "admin" menu item * remove account widget * rebrand "Dashboard" to "Server List" WIP - doesn't look good but is somewhat working * fix canAccessPanel * separate server list into own panel * change path to avoid conflicts with old client area (and remove sidebar width) * display correct icon and color on server list entries * show total memory if server is offline * replace custom server list page with ListRecords page * fix tests * fix namespace * remove "open" button and make whole column clickable * Update EditProfile * run pint * fix access to server list * add new login page to panels * fix next_run_at for new schedules * use new DateTimeColumn * add own column for file bytes * return to server list when clicking title * fix console loading * handle server with "conflict state" * add banner if server is in "conflict state" * fix phpstan * update docker image select * fix permission checks on Settings & Startup pages * fix query for activity log page * fix activity log not being logged * adjust ListActivities * fix phpstan * fix pint * fix profile menu item link on server panel * add ip tooltip to activity logs (and role permission) * change backup icon * update navigation sort * general code cleanup * more cleanup * Disable Restart/Stop if server is offline * Change rename notification * Remove negation on abort_unless * Add notification on save * Single disabled closure & comment unused import * Add required to Server Name & Nullable to description * mutateFormDataBeforeSave doesn't work since we use forceFill * Fix web socket connection not existing. * Fix some subuser permissions * add permission checks to resources * do not allow self-deletion * Update editing file permissions * Fix of the previous fix * add service for subuser updating * Only allow save if they have file_update * Remove unused import * Update backup delete button * Add Delete, remove bulks * Update Database page * Use Allocation Permissions * add canAccess check to startup * Add Permission checks to Settings page * add service for subuser deletion * Remove Kill permission * Updates * fix move files * add redirects * fix phpstan * activity: remove properties from tans for now * If alias, use that, else ip --------- Co-authored-by: notCharles <charles@pelican.dev> Co-authored-by: Boy132 <mail@boy132.de> Co-authored-by: Senna <62171904+Poseidon281@users.noreply.github.com> Co-authored-by: Boy132 <Boy132@users.noreply.github.com> Co-authored-by: RMartinOscar <40749467+RMartinOscar@users.noreply.github.com>
		
			
				
	
	
		
			259 lines
		
	
	
		
			6.7 KiB
		
	
	
	
		
			PHP
		
	
	
	
	
	
			
		
		
	
	
			259 lines
		
	
	
		
			6.7 KiB
		
	
	
	
		
			PHP
		
	
	
	
	
	
<?php
 | 
						|
 | 
						|
namespace App\Services\Activity;
 | 
						|
 | 
						|
use Illuminate\Support\Arr;
 | 
						|
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\ActivityLogSubject;
 | 
						|
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;
 | 
						|
 | 
						|
    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  $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 $exception) {
 | 
						|
            if (config('app.env') !== 'production') {
 | 
						|
                /* @noinspection PhpUnhandledExceptionInspection */
 | 
						|
                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()) {
 | 
						|
            if ($user instanceof Model) {
 | 
						|
                $this->actor($user);
 | 
						|
            }
 | 
						|
        }
 | 
						|
 | 
						|
        return $this->activity;
 | 
						|
    }
 | 
						|
 | 
						|
    /**
 | 
						|
     * Saves the activity log instance and attaches all of the subject models.
 | 
						|
     *
 | 
						|
     * @throws \Throwable
 | 
						|
     */
 | 
						|
    protected function save(): ActivityLog
 | 
						|
    {
 | 
						|
        Assert::notNull($this->activity);
 | 
						|
 | 
						|
        $response = $this->connection->transaction(function () {
 | 
						|
            $this->activity->save();
 | 
						|
 | 
						|
            $subjects = Collection::make($this->subjects)
 | 
						|
                ->map(fn (Model $subject) => [
 | 
						|
                    'activity_log_id' => $this->activity->id,
 | 
						|
                    'subject_id' => $subject->getKey(),
 | 
						|
                    'subject_type' => $subject->getMorphClass(),
 | 
						|
                ])
 | 
						|
                ->values()
 | 
						|
                ->toArray();
 | 
						|
 | 
						|
            ActivityLogSubject::insert($subjects);
 | 
						|
 | 
						|
            return $this->activity;
 | 
						|
        });
 | 
						|
 | 
						|
        $this->activity = null;
 | 
						|
        $this->subjects = [];
 | 
						|
 | 
						|
        return $response;
 | 
						|
    }
 | 
						|
}
 |