Passa al contenuto principale

Autorizzazione

Come con qualsiasi framework, Flarum consente di limitare determinate azioni e contenuti a determinati utenti. Esistono 2 sistemi paralleli per questo:

  • Il processo di autorizzazione determina se un utente può eseguire una determinata azione.
  • La visibilità può essere applicata a una query di database per limitare in modo efficiente i record a cui gli utenti possono accedere. Questo è documentato nel nostro articolo modelli di visibilità.

Processo di autorizzazione

Il processo di autorizzazione viene utilizzato per verificare se una persona è autorizzata a eseguire determinate azioni. Ad esempio, vogliamo verificare se un utente è autorizzato prima di:

  • Poter accedere al pannello di amministrazione
  • Iniziare una discussione
  • Modificare un post
  • Aggiornare il profilo di un altro utente

Ciascuno di questi è determinato da criteri univoci: in alcuni casi è sufficiente un flag; altrimenti, potremmo aver bisogno di una logica personalizzata.

Visibility Scoping

Le richieste di autorizzazione vengono effettuate con 3 parametri, con logica contenuta in Flarum\User\Gate:

  1. L'attore: l'utente che tenta di eseguire l'azione
  2. L'abilità: una stringa che rappresenta l'azione che l'attore sta tentando
  3. Gli argomenti: di solito un'istanza di un modello di database che è l'oggetto dell'azione, ma potrebbe essere qualsiasi cosa.

Per prima cosa, eseguiamo l'intera richiesta (tutti e tre i parametri) attraverso tutte le policies registrate dalle estensioni e dal core. Le policy sono blocchi di logica forniti dal core e dalle estensioni che determinano se l'attore può eseguire l'azione sugli argomenti. I criteri possono restituire uno dei seguenti valori:

  • Flarum\User\Access\AbstractPolicy::ALLOW (tramite $this->allow())
  • Flarum\User\Access\AbstractPolicy::DENY (tramite $this->deny())
  • Flarum\User\Access\AbstractPolicy::FORCE_ALLOW (tramite $this->forceAllow())
  • Flarum\User\Access\AbstractPolicy::FORCE_DENY (tramite $this->forceDeny())

I risultati delle policy sono considerati prioritari FORCE_DENY > FORCE_ALLOW > DENY > ALLOW. Ad esempio, se viene restituita una singola policy FORCE_DENY, tutte le altre policy verranno ignorate. Se una policy restituisce DENY e altre 10 restituiscono ALLOW, la richiesta verrà rifiutata. Ciò consente di prendere decisioni indipendentemente dall'ordine in cui le estensioni vengono avviate. Le policy sono estremamente potenti: se l'accesso viene negato in fase di policy, questo sovrascriverà i permessi dei gruppi e i privilegi di amministratore.

In secondo luogo, se tutte le policy restituiscono null (o non restituiscono nulla), controlliamo se l'utente è in un gruppo che ha un permesso che consenta l'azione (nota che sia i permessi che le azioni sono rappresentati sotto forma di stringhe). In tal caso, autorizziamo l'azione. Guarda la Documentazione su gruppi e permessi per maggiori informazioni.

Quindi, se l'utente è nel gruppo admin, autorizzeremo l'azione.

Infine, poiché abbiamo esaurito tutti i controlli, daremo per scontato che l'utente non sia autorizzato e negheremo la richiesta.

Autorizzazioni del frontend

Il sistema di autorizzazione di Flarum è accessibile attraverso metodi pubblici delle classi Flarum\User\User. I più importanti sono elencati di seguito; altri sono documentati nelle documentazioni PHP API.

In questo esempio, useremo $actor come istanza di Flarum\User\User, 'viewForum' e 'reply' come esempi di abilità, e $discussion (istanza di Flarum\Discussion\Discussion) come esempio di argomento.

// Verifica se un utente può eseguire un'azione.
// ATTENZIONE: questo dovrebbe essere usato con cautela, poiché in realtà
// non viene eseguito attraverso il processo di autorizzazione, quindi non tiene conto delle policy.
$canDoSomething = $actor->can('viewForum');

// Verifica se un utente può eseguire un'azione su un argomento.
// Tuttavia, è utile nell'implementazione di criteri personalizzati.
$canDoSomething = $actor->can('reply', $discussion);

// Genera un'eccezione PermissionDeniedException se un utente non può eseguire un'azione.
$actpr->assertAdmin();

// Controlla se uno dei gruppi dell'utente dispone di un'autorizzazione.
$actor->assertCan('viewForum');
$actor->assertCan('reply', $discussion);

// Genera un'eccezione NotAuthenticatedException se l'utente non ha efettuato il login.
$actor->assertRegistered();

// Genera un'eccezione PermissionDeniedException se l'utente non è un amministratore.
$actorHasPermission = $actor->hasPermission(`viewForum`);

Policy personalizzate

I criteri ci consentono di utilizzare una logica personalizzata oltre a semplici gruppi e autorizzazioni quando si valuta l'autorizzazione per un'abilità con un soggetto. Per esempio:

  • Vogliamo consentire agli utenti di modificare i post anche se non sono moderatori, ma solo i propri post.
  • A seconda delle impostazioni, potremmo consentire agli utenti di rinominare le proprie discussioni a tempo indeterminato, per un breve periodo di tempo dopo la pubblicazione o per niente.

Come descritto qui, ad ogni controllo di autorizzazione, interroghiamo tutte le politiche registrate per il modello del target o qualsiasi classe genitore del modello del target. Se non viene fornito alcun target, verranno applicati i criteri egistrati come "global".

Quindi, come viene "verificato" un criterio?

Innanzitutto, controlliamo se la classe del criterio ha un metodo con lo stesso nome dell'abilità che viene valutata. In tal caso, lo eseguiamo con l'attore e il soggetto come parametri. Se quel metodo restituisce un valore non nullo, restituiamo quel risultato. In caso contrario, si passa alla fase successiva (non necessariamente al criterio successivo).

Quindi, controlliamo se la classe policy ha un metodo chiamato can. In tal caso, lo eseguiamo con l'attore, l'abilità e il soggetto e restituiamo il risultato.

Se "can" non esiste o restituisce null, abbiamo finito con questo criterio e procediamo a quello successivo.

È possibile utilizzare la CLI per generare automaticamente i criteri:

$ flarum-cli make backend policy

Come funziona

Diamo un'occhiata ad alcuni esempi tratti da Flarum Tags.

<?php
namespace Flarum\Tags\Access;

use Flarum\Tags\Tag;
use Flarum\User\Access\AbstractPolicy;
use Flarum\User\User;

class TagPolicy extends AbstractPolicy
{
/**
* @param User $actor
* @param Tag $tag
* @return bool|null
*/
public function startDiscussion(User $actor, Tag $tag)
{
if ($tag->is_restricted) {
return $actor->hasPermission('tag'.$tag->id.'.startDiscussion') ? $this->allow() : $this->deny();
}
}

/**
* @param User $actor
* @param Tag $tag
* @return bool|null
*/
public function addToDiscussion(User $actor, Tag $tag)
{
return $this->startDiscussion($actor, $tag);
}
}

Possiamo anche avere criteri globali, che vengono eseguiti quando $ user-> can () viene chiamato senza un'istanza del modello di destinazione. Di nuovo dai tag:

<?php

namespace Flarum\Tags\Access;

use Flarum\Settings\SettingsRepositoryInterface;
use Flarum\Tags\Tag;
use Flarum\User\Access\AbstractPolicy;
use Flarum\User\User;

class GlobalPolicy extends AbstractPolicy
{
/**
* @var SettingsRepositoryInterface
*/
protected $settings;

public function __construct(SettingsRepositoryInterface $settings)
{
$this->settings = $settings;
}

/**
* @param Flarum\User\User $actor
* @param string $ability
* @return bool|void
*/
public function can(User $actor, string $ability)
{
if (in_array($ability, ['viewForum', 'startDiscussion'])) {
$enoughPrimary = count(Tag::getIdsWhereCan($actor, $ability, true, false)) >= $this->settings->get('min_primary_tags');
$enoughSecondary = count(Tag::getIdsWhereCan($actor, $ability, false, true)) >= $this->settings->get('min_secondary_tags');

if ($enoughPrimary && $enoughSecondary) {
return $this->allow();
} else {
return $this->deny();
}
}
}
}

Come utilizzare le autorizzazioni

ed uno globale per il modelle Discussion:

use Flarum\Extend;
use Flarum\Tags\Tag;
use YourNamespace\Access;

return [
// Other extenders
(new Extend\Policy())
->modelPolicy(Tag::class, Access\TagPolicy::class)
->globalPolicy(Access\GlobalPolicy::class),
// Other extenders
];

Autorizzazioni del frontend

Di solito, vorrai utilizzare i risultati dell'autorizzazione nella logica del frontend. Ad esempio, se un utente non dispone dell'autorizzazione per visualizzare gli utenti di ricerca, non dovremmo inviare richieste a quell'endpoint. E se un utente non ha il permesso di modificare gli utenti, non dovremmo mostrare le voci di menu che consentono di effettuare tali modifiche.

Poiché non possiamo eseguire controlli di autorizzazione nel frontend, dobbiamo eseguirli nel backend e allegarli alla serializzazione dei dati che stiamo inviando. Permessi globali come (viewForum, viewUserList) possono essere inclusi in ForumSerializer, ma per l'autorizzazione specifica dell'oggetto, potremmo voler includere quelli con altri parametri. Ad esempio, quando restituiamo l'elenco delle discussioni, controlliamo se l'utente può rispondere, rinominare, modificare ed eliminarle oggetti, e memorizzare quei dati nel modello di discussione frontend. È quindi accessibile tramite discussion.canReply() o discussion.canEdit(), ma non c'è niente di magico qui: è solo un altro attributo inviato dal serializzatore.

Esistono in realtà due tipi di scoper: