Laravel Passport
- Introducción
- Instalación
- Configuración
- Emitiendo Tokens de Acceso
- Grant de Código de Autorización Con PKCE
- Tokens de Grant por Contraseña
- Tokens de Grant Implícito
- Tokens de Grant de Credenciales de Cliente
- Tokens de Acceso Personal
- Protegiendo Rutas
- Alcances de Token
- Consumiendo Tu API Con JavaScript
- Eventos
- Pruebas
Introducción
Laravel Passport proporciona una implementación completa del servidor OAuth2 para tu aplicación Laravel en cuestión de minutos. Passport se construye sobre el servidor OAuth2 de League que es mantenido por Andy Millington y Simon Hamp.
Esta documentación asume que ya estás familiarizado con OAuth2. Si no sabes nada sobre OAuth2, considera familiarizarte con la terminología y las características generales de OAuth2 antes de continuar.
¿Passport o Sanctum?
Antes de comenzar, es posible que desees determinar si tu aplicación se beneficiaría más de Laravel Passport o Laravel Sanctum. Si tu aplicación necesita absolutamente soportar OAuth2, entonces deberías usar Laravel Passport. Sin embargo, si estás intentando autenticar una aplicación de una sola página, una aplicación móvil, o emitir tokens API, deberías usar Laravel Sanctum. Laravel Sanctum no soporta OAuth2; sin embargo, ofrece una experiencia de desarrollo de autenticación de API mucho más simple.
Instalación
Puedes instalar Laravel Passport a través del comando Artisan install:api
:
php artisan install:api --passport
Este comando publicará y ejecutará las migraciones de base de datos necesarias para crear las tablas que tu aplicación necesita para almacenar clientes OAuth2 y tokens de acceso. El comando también creará las claves de cifrado requeridas para generar tokens de acceso seguros.
Además, este comando preguntará si te gustaría usar UUIDs como el valor de clave primaria del modelo Client
de Passport en lugar de enteros autoincrementales.
Después de ejecutar el comando install:api
, añade el rasgo Laravel\Passport\HasApiTokens
a tu modelo App\Models\User
. Este rasgo proporcionará algunos métodos auxiliares a tu modelo que te permitirán inspeccionar el token y los alcances del usuario autenticado:
<?php namespace App\Models; use Illuminate\Database\Eloquent\Factories\HasFactory; use Illuminate\Foundation\Auth\User as Authenticatable; use Illuminate\Notifications\Notifiable; use Laravel\Passport\HasApiTokens; class User extends Authenticatable { use HasApiTokens, HasFactory, Notifiable; }
Finalmente, en el archivo de configuración config/auth.php
de tu aplicación, deberías definir un guardia de autenticación api
y establecer la opción driver
en passport
. Esto indicará a tu aplicación que utilice el TokenGuard
de Passport al autenticar solicitudes API entrantes:
'guards' => [ 'web' => [ 'driver' => 'session', 'provider' => 'users', ], 'api' => [ 'driver' => 'passport', 'provider' => 'users', ], ],
Desplegando Passport
Al desplegar Passport en los servidores de tu aplicación por primera vez, es probable que necesites ejecutar el comando passport:keys
. Este comando genera las claves de cifrado que Passport necesita para generar tokens de acceso. Las claves generadas no se suelen mantener en el control de versiones:
php artisan passport:keys
Si es necesario, puedes definir la ruta desde donde se deben cargar las claves de Passport. Puedes usar el método Passport::loadKeysFrom
para lograr esto. Típicamente, este método debe llamarse desde el método boot
de la clase App\Providers\AppServiceProvider
de tu aplicación:
/** * Bootstrap any application services. */ public function boot(): void { Passport::loadKeysFrom(__DIR__.'/../secrets/oauth'); }
Cargando Claves Desde el Entorno
Alternativamente, puedes publicar el archivo de configuración de Passport utilizando el comando Artisan vendor:publish
:
php artisan vendor:publish --tag=passport-config
Después de que se haya publicado el archivo de configuración, puedes cargar las claves de cifrado de tu aplicación definiéndolas como variables de entorno:
PASSPORT_PRIVATE_KEY="-----BEGIN RSA PRIVATE KEY----- <private key here> -----END RSA PRIVATE KEY-----" PASSPORT_PUBLIC_KEY="-----BEGIN PUBLIC KEY----- <public key here> -----END PUBLIC KEY-----"
Actualizando Passport
Al actualizar a una nueva versión mayor de Passport, es importante que revises cuidadosamente la guía de actualización.
Configuración
Hashing de Secretos de Cliente
Si deseas que los secretos de tu cliente se almacenen en tu base de datos de forma hash, deberías llamar al método Passport::hashClientSecrets
en el método boot
de tu clase App\Providers\AppServiceProvider
:
use Laravel\Passport\Passport; Passport::hashClientSecrets();
Una vez habilitado, todos tus secretos de cliente solo serán visibles para el usuario inmediatamente después de que se creen. Dado que el valor del secreto de cliente en texto plano nunca se almacena en la base de datos, no es posible recuperar el valor del secreto si se pierde.
Duraciones de Token
Por defecto, Passport emite tokens de acceso de larga duración que expiran después de un año. Si deseas configurar un tiempo de vida de token más largo / corto, puedes usar los métodos tokensExpireIn
, refreshTokensExpireIn
y personalAccessTokensExpireIn
. Estos métodos deben ser llamados desde el método boot
de la clase App\Providers\AppServiceProvider
de tu aplicación:
/** * Bootstrap any application services. */ public function boot(): void { Passport::tokensExpireIn(now()->addDays(15)); Passport::refreshTokensExpireIn(now()->addDays(30)); Passport::personalAccessTokensExpireIn(now()->addMonths(6)); }
Las columnas expires_at
en las tablas de la base de datos de Passport son de solo lectura y solo para fines de visualización. Al emitir tokens, Passport almacena la información de expiración dentro de los tokens firmados y encriptados. Si necesitas invalidar un token, deberías revocarlo.
Sobrescribir Modelos Predeterminados
Puedes ampliar los modelos utilizados internamente por Passport definiendo tu propio modelo y extendiendo el modelo correspondiente de Passport:
use Laravel\Passport\Client as PassportClient; class Client extends PassportClient { // ... }
Después de definir tu modelo, puedes instruir a Passport para que use tu modelo personalizado a través de la clase Laravel\Passport\Passport
. Típicamente, debes informar a Passport sobre tus modelos personalizados en el método boot
de la clase App\Providers\AppServiceProvider
de tu aplicación:
use App\Models\Passport\AuthCode; use App\Models\Passport\Client; use App\Models\Passport\PersonalAccessClient; use App\Models\Passport\RefreshToken; use App\Models\Passport\Token; /** * Bootstrap any application services. */ public function boot(): void { Passport::useTokenModel(Token::class); Passport::useRefreshTokenModel(RefreshToken::class); Passport::useAuthCodeModel(AuthCode::class); Passport::useClientModel(Client::class); Passport::usePersonalAccessClientModel(PersonalAccessClient::class); }
Sobrescribir Rutas
A veces es posible que desees personalizar las rutas definidas por Passport. Para lograr esto, primero necesitas ignorar las rutas registradas por Passport añadiendo Passport::ignoreRoutes
al método register
del AppServiceProvider
de tu aplicación:
use Laravel\Passport\Passport; /** * Register any application services. */ public function register(): void { Passport::ignoreRoutes(); }
Entonces, puedes copiar las rutas definidas por Passport en su archivo de rutas a tu archivo routes/web.php
de la aplicación y modificarlas a tu gusto:
Route::group([ 'as' => 'passport.', 'prefix' => config('passport.path', 'oauth'), 'namespace' => '\Laravel\Passport\Http\Controllers', ], function () { // Passport routes... });
Emitiendo Tokens de Acceso
Usar OAuth2 a través de códigos de autorización es como la mayoría de los desarrolladores están familiarizados con OAuth2. Al usar códigos de autorización, una aplicación cliente redirigirá a un usuario a tu servidor, donde aprobarán o denegarán la solicitud para emitir un token de acceso al cliente.
Gestionando Clientes
Primero, los desarrolladores que estén creando aplicaciones que necesiten interactuar con la API de tu aplicación deberán registrar su aplicación con la tuya creando un "cliente". Típicamente, esto consiste en proporcionar el nombre de su aplicación y una URL a la que tu aplicación puede redirigir después de que los usuarios aprueben su solicitud de autorización.
El comando passport:client
La forma más sencilla de crear un cliente es utilizando el comando Artisan passport:client
. Este comando se puede usar para crear tus propios clientes para probar tu funcionalidad OAuth2. Cuando ejecutes el comando client
, Passport te pedirá más información sobre tu cliente y te proporcionará un ID de cliente y un secreto:
php artisan passport:client
Redirigir URLs
Si deseas permitir múltiples URL de redireccionamiento para tu cliente, puedes especificarlas usando una lista delimitada por comas cuando se te solicite la URL con el comando passport:client
. Cualquier URL que contenga comas debe estar codificada en URL:
http://example.com/callback,http://examplefoo.com/callback
Dado que los usuarios de tu aplicación no podrán utilizar el comando client
, Passport proporciona una API JSON que puedes usar para crear clientes. Esto te ahorra el problema de tener que codificar manualmente controladores para crear, actualizar y eliminar clientes.
Sin embargo, necesitarás emparejar la API JSON de Passport con tu propio frontend para proporcionar un panel de control para que tus usuarios gestionen sus clientes. A continuación, revisaremos todos los endpoints de la API para gestionar clientes. Para mayor comodidad, utilizaremos Axios para demostrar cómo realizar solicitudes HTTP a los endpoints.
GET /oauth/clients
Esta ruta devuelve todos los clientes para el usuario autenticado. Esto es útil principalmente para listar todos los clientes del usuario, de modo que pueda editarlos o eliminarlos:
axios.get('/oauth/clients') .then(response => { console.log(response.data); });
POST /oauth/clients
Esta ruta se utiliza para crear nuevos clientes. Requiere dos datos: el nombre
del cliente y una URL de redirección
. La URL de redirección
es a donde el usuario será redirigido después de aprobar o denegar una solicitud de autorización.
Cuando se crea un cliente, se le asignará un ID de cliente y un secreto de cliente. Estos valores se utilizarán al solicitar tokens de acceso desde su aplicación. La ruta de creación del cliente devolverá la nueva instancia del cliente:
const data = { name: 'Client Name', redirect: 'http://example.com/callback' }; axios.post('/oauth/clients', data) .then(response => { console.log(response.data); }) .catch (response => { // List errors on response... });
PUT /oauth/clients/{client-id}
Esta ruta se utiliza para actualizar clientes. Requiere dos datos: el nombre
del cliente y una URL de redirección
. La URL de redirección
es a donde el usuario será redirigido después de aprobar o denegar una solicitud de autorización. La ruta devolverá la instancia del cliente actualizada:
const data = { name: 'New Client Name', redirect: 'http://example.com/callback' }; axios.put('/oauth/clients/' + clientId, data) .then(response => { console.log(response.data); }) .catch (response => { // List errors on response... });
DELETE /oauth/clients/{client-id}
Esta ruta se utiliza para eliminar clientes:
axios.delete('/oauth/clients/' + clientId) .then(response => { // ... });
Una vez que se ha creado un cliente, los desarrolladores pueden usar su ID de cliente y secreto para solicitar un código de autorización y un token de acceso desde su aplicación. Primero, la aplicación consumidora debe hacer una solicitud de redirección a la ruta /oauth/authorize
de su aplicación de la siguiente manera:
use Illuminate\Http\Request; use Illuminate\Support\Str; Route::get('/redirect', function (Request $request) { $request->session()->put('state', $state = Str::random(40)); $query = http_build_query([ 'client_id' => 'client-id', 'redirect_uri' => 'http://third-party-app.com/callback', 'response_type' => 'code', 'scope' => '', 'state' => $state, // 'prompt' => '', // "none", "consent", or "login" ]); return redirect('http://passport-app.test/oauth/authorize?'.$query); });
El parámetro prompt
se puede usar para especificar el comportamiento de autenticación de la aplicación Passport.
Si el valor de prompt
es none
, Passport siempre lanzará un error de autenticación si el usuario no está ya autenticado con la aplicación Passport. Si el valor es consent
, Passport siempre mostrará la pantalla de aprobación de autorización, incluso si todos los alcances fueron concedidos previamente a la aplicación consumidora. Cuando el valor es login
, la aplicación Passport siempre solicitará al usuario que inicie sesión nuevamente en la aplicación, incluso si ya tiene una sesión existente.
Si no se proporciona ningún valor prompt
, se pedirá autorización al usuario solo si no ha autorizado previamente el acceso a la aplicación consumidora para los ámbitos solicitados.
Aprobando la Solicitud
Al recibir solicitudes de autorización, Passport responderá automáticamente según el valor del parámetro prompt
(si está presente) y puede mostrar una plantilla al usuario que le permita aprobar o denegar la solicitud de autorización. Si aprueban la solicitud, serán redirigidos de vuelta a la redirect_uri
que fue especificada por la aplicación consumidora. La redirect_uri
debe coincidir con la URL redirect
que fue especificada cuando se creó el cliente.
Si deseas personalizar la pantalla de aprobación de autorización, puedes publicar las vistas de Passport utilizando el comando Artisan vendor:publish
. Las vistas publicadas se colocarán en el directorio resources/views/vendor/passport
:
php artisan vendor:publish --tag=passport-views
A veces es posible que desees omitir el aviso de autorización, como cuando autorizas un cliente de primera parte. Puedes lograr esto extendiendo el modelo Client
y definiendo un método skipsAuthorization
. Si skipsAuthorization
devuelve true
, el cliente será aprobado y el usuario será redirigido de inmediato a la redirect_uri
, a menos que la aplicación consumidora haya configurado explícitamente el parámetro prompt
al redirigir para autorización:
<?php namespace App\Models\Passport; use Laravel\Passport\Client as BaseClient; class Client extends BaseClient { /** * Determine if the client should skip the authorization prompt. */ public function skipsAuthorization(): bool { return $this->firstParty(); } }
Si el usuario aprueba la solicitud de autorización, será redirigido de vuelta a la aplicación consumidora. El consumidor debe verificar primero el parámetro state
contra el valor que se almacenó antes de la redirección. Si el parámetro de estado coincide, entonces el consumidor debe emitir una solicitud POST
a su aplicación para solicitar un token de acceso. La solicitud debe incluir el código de autorización que fue emitido por su aplicación cuando el usuario aprobó la solicitud de autorización:
use Illuminate\Http\Request; use Illuminate\Support\Facades\Http; Route::get('/callback', function (Request $request) { $state = $request->session()->pull('state'); throw_unless( strlen($state) > 0 && $state === $request->state, InvalidArgumentException::class, 'Invalid state value.' ); $response = Http::asForm()->post('http://passport-app.test/oauth/token', [ 'grant_type' => 'authorization_code', 'client_id' => 'client-id', 'client_secret' => 'client-secret', 'redirect_uri' => 'http://third-party-app.com/callback', 'code' => $request->code, ]); return $response->json(); });
Al igual que la ruta /oauth/authorize
, la ruta /oauth/token
es definida para ti por Passport. No es necesario definir manualmente esta ruta.
Passport también incluye una API JSON para gestionar tokens de acceso autorizados. Puedes combinar esto con tu propio frontend para ofrecer a tus usuarios un panel de control para gestionar los tokens de acceso. Para mayor comodidad, utilizaremos Axios para demostrar cómo realizar solicitudes HTTP a los endpoints. La API JSON está protegida por el middleware web
y auth
; por lo tanto, solo puede ser llamada desde tu propia aplicación.
GET /oauth/tokens
Esta ruta devuelve todos los tokens de acceso autorizados que el usuario autenticado ha creado. Esto es útil principalmente para listar todos los tokens del usuario para que puedan ser revocados:
axios.get('/oauth/tokens') .then(response => { console.log(response.data); });
DELETE /oauth/tokens/{token-id}
Esta ruta puede utilizarse para revocar los tokens de acceso autorizados y sus tokens de actualización relacionados:
axios.delete('/oauth/tokens/' + tokenId);
Actualizando Tokens
Si tu aplicación emite tokens de acceso de corta duración, los usuarios necesitarán actualizar sus tokens de acceso a través del token de actualización que se les proporcionó cuando se emitió el token de acceso:
use Illuminate\Support\Facades\Http; $response = Http::asForm()->post('http://passport-app.test/oauth/token', [ 'grant_type' => 'refresh_token', 'refresh_token' => 'the-refresh-token', 'client_id' => 'client-id', 'client_secret' => 'client-secret', 'scope' => '', ]); return $response->json();
Esta ruta /oauth/token
devolverá una respuesta JSON que contiene los atributos access_token
, refresh_token
y expires_in
. El atributo expires_in
contiene el número de segundos hasta que expire el token de acceso.
Revocando Tokens
Puedes revocar un token utilizando el método revokeAccessToken
en el Laravel\Passport\TokenRepository
. Puedes revocar los tokens de actualización de un token utilizando el método revokeRefreshTokensByAccessTokenId
en el Laravel\Passport\RefreshTokenRepository
. Estas clases se pueden resolver utilizando el contenedor de servicios de Laravel:
use Laravel\Passport\TokenRepository; use Laravel\Passport\RefreshTokenRepository; $tokenRepository = app(TokenRepository::class); $refreshTokenRepository = app(RefreshTokenRepository::class); // Revoke an access token... $tokenRepository->revokeAccessToken($tokenId); // Revoke all of the token's refresh tokens... $refreshTokenRepository->revokeRefreshTokensByAccessTokenId($tokenId);
Purging de Tokens
Cuando los tokens han sido revocados o han expirado, es posible que desees purgarlos de la base de datos. El comando Artisan passport:purge
incluido en Passport puede hacer esto por ti:
# Purge revoked and expired tokens and auth codes... php artisan passport:purge # Only purge tokens expired for more than 6 hours... php artisan passport:purge --hours=6 # Only purge revoked tokens and auth codes... php artisan passport:purge --revoked # Only purge expired tokens and auth codes... php artisan passport:purge --expired
También puedes configurar un trabajo programado en el archivo routes/console.php
de tu aplicación para eliminar automáticamente tus tokens en un horario:
use Illuminate\Support\Facades\Schedule; Schedule::command('passport:purge')->hourly();
Grant de Código de Autorización Con PKCE
El flujo de concesión de código de autorización con "Proof Key for Code Exchange" (PKCE) es una forma segura de autenticar aplicaciones de una sola página o aplicaciones nativas para acceder a tu API. Este flujo debe usarse cuando no puedes garantizar que el secreto del cliente se almacenará de manera confidencial o para mitigar la amenaza de que el código de autorización sea interceptado por un atacante. Una combinación de un "verificador de código" y un "reto de código" reemplaza el secreto del cliente al intercambiar el código de autorización por un token de acceso.
Creando el Cliente
Antes de que tu aplicación pueda emitir tokens a través del flujo de autorización con código y PKCE, necesitarás crear un cliente habilitado para PKCE. Puedes hacer esto utilizando el comando Artisan passport:client
con la opción --public
:
php artisan passport:client --public
Verificador de Código y Desafío de Código
Como esta concesión de autorización no proporciona un secreto de cliente, los desarrolladores necesitarán generar una combinación de un verificador de código y un desafío de código para poder solicitar un token.
El verificador de código debe ser una cadena aleatoria de entre 43 y 128 caracteres que contenga letras, números y los caracteres "-"
, "."
, "_"
, "~"
, como se define en la especificación RFC 7636.
El desafío de código debe ser una cadena codificada en Base64 con caracteres seguros para URL y nombres de archivo. Los caracteres '='
finales deben ser eliminados y no deben existir saltos de línea, espacios en blanco u otros caracteres adicionales.
$encoded = base64_encode(hash('sha256', $code_verifier, true)); $codeChallenge = strtr(rtrim($encoded, '='), '+/', '-_');
Redirigiendo para Autorización
Una vez que se ha creado un cliente, puedes usar el ID del cliente y el verificador de código generado y el desafío de código para solicitar un código de autorización y un token de acceso desde tu aplicación. Primero, la aplicación consumidora debe hacer una solicitud de redirección a la ruta /oauth/authorize
de tu aplicación:
use Illuminate\Http\Request; use Illuminate\Support\Str; Route::get('/redirect', function (Request $request) { $request->session()->put('state', $state = Str::random(40)); $request->session()->put( 'code_verifier', $code_verifier = Str::random(128) ); $codeChallenge = strtr(rtrim( base64_encode(hash('sha256', $code_verifier, true)) , '='), '+/', '-_'); $query = http_build_query([ 'client_id' => 'client-id', 'redirect_uri' => 'http://third-party-app.com/callback', 'response_type' => 'code', 'scope' => '', 'state' => $state, 'code_challenge' => $codeChallenge, 'code_challenge_method' => 'S256', // 'prompt' => '', // "none", "consent", or "login" ]); return redirect('http://passport-app.test/oauth/authorize?'.$query); });
Convirtiendo Códigos de Autorización a Tokens de Acceso
Si el usuario aprueba la solicitud de autorización, será redirigido de vuelta a la aplicación consumidora. El consumidor debe verificar el parámetro state
con el valor que se almacenó antes de la redirección, como en el estándar Authorization Code Grant.
Si el parámetro de estado coincide, el consumidor debe emitir una solicitud POST
a tu aplicación para solicitar un token de acceso. La solicitud debe incluir el código de autorización que fue emitido por tu aplicación cuando el usuario aprobó la solicitud de autorización junto con el verificadores de código que se generó originalmente:
use Illuminate\Http\Request; use Illuminate\Support\Facades\Http; Route::get('/callback', function (Request $request) { $state = $request->session()->pull('state'); $codeVerifier = $request->session()->pull('code_verifier'); throw_unless( strlen($state) > 0 && $state === $request->state, InvalidArgumentException::class ); $response = Http::asForm()->post('http://passport-app.test/oauth/token', [ 'grant_type' => 'authorization_code', 'client_id' => 'client-id', 'redirect_uri' => 'http://third-party-app.com/callback', 'code_verifier' => $codeVerifier, 'code' => $request->code, ]); return $response->json(); });
Tokens de Grant por Contraseña
Ya no recomendamos usar tokens de concesión de contraseña. En su lugar, debes elegir un tipo de concesión que sea actualmente recomendado por OAuth2 Server.
El flujo de concesión de contraseña de OAuth2 permite que tus otros clientes de primera parte, como una aplicación móvil, obtengan un token de acceso utilizando una dirección de correo electrónico / nombre de usuario y una contraseña. Esto te permite emitir tokens de acceso de manera segura a tus clientes de primera parte sin requerir que tus usuarios pasen por todo el flujo de redireccionamiento de código de autorización OAuth2.
Para habilitar el grant de contraseña, llama al método enablePasswordGrant
en el método boot
de la clase App\Providers\AppServiceProvider
de tu aplicación:
/** * Bootstrap any application services. */ public function boot(): void { Passport::enablePasswordGrant(); }
Creando un Cliente de Grant por Contraseña
Antes de que tu aplicación pueda emitir tokens a través del grant de contraseña, necesitarás crear un cliente de grant de contraseña. Puedes hacer esto utilizando el comando Artisan passport:client
con la opción --password
. Si ya has ejecutado el comando passport:install
, no necesitas ejecutar este comando:
php artisan passport:client --password
Solicitando Tokens
Una vez que hayas creado un cliente de concesión de contraseña, puedes solicitar un token de acceso emitiendo una solicitud POST
a la ruta /oauth/token
con la dirección de correo electrónico y la contraseña del usuario. Recuerda que esta ruta ya está registrada por Passport, por lo que no es necesario definirla manualmente. Si la solicitud tiene éxito, recibirás un access_token
y un refresh_token
en la respuesta JSON del servidor:
use Illuminate\Support\Facades\Http; $response = Http::asForm()->post('http://passport-app.test/oauth/token', [ 'grant_type' => 'password', 'client_id' => 'client-id', 'client_secret' => 'client-secret', 'username' => 'taylor@laravel.com', 'password' => 'my-password', 'scope' => '', ]); return $response->json();
Recuerda que los tokens de acceso son de larga duración por defecto. Sin embargo, puedes configurar tu duración máxima de token de acceso si es necesario.
Solicitando Todos los Alcances
Al usar el grant de contraseña o el grant de credenciales del cliente, es posible que desees autorizar el token para todos los scopes soportados por tu aplicación. Puedes hacer esto solicitando el scope *
. Si solicitas el scope *
, el método can
en la instancia del token siempre devolverá true
. Este scope solo puede ser asignado a un token que sea emitido utilizando el grant password
o client_credentials
:
use Illuminate\Support\Facades\Http; $response = Http::asForm()->post('http://passport-app.test/oauth/token', [ 'grant_type' => 'password', 'client_id' => 'client-id', 'client_secret' => 'client-secret', 'username' => 'taylor@laravel.com', 'password' => 'my-password', 'scope' => '*', ]);
Personalizando el Proveedor de Usuario
Si tu aplicación utiliza más de un proveedor de usuario de autenticación, puedes especificar qué proveedor de usuario usa el cliente de concesión de contraseña proporcionando una opción --provider
al crear el cliente a través del comando artisan passport:client --password
. El nombre del proveedor dado debe coincidir con un proveedor válido definido en el archivo de configuración config/auth.php
de tu aplicación. Luego puedes proteger tu ruta utilizando middleware para asegurarte de que solo los usuarios del proveedor especificado por el guard estén autorizados.
Personalizando el Campo de Nombre de Usuario
Al autenticar utilizando el grant de contraseña, Passport utilizará el atributo email
de tu modelo autenticable como el "nombre de usuario". Sin embargo, puedes personalizar este comportamiento definiendo un método findForPassport
en tu modelo:
<?php namespace App\Models; use Illuminate\Foundation\Auth\User as Authenticatable; use Illuminate\Notifications\Notifiable; use Laravel\Passport\HasApiTokens; class User extends Authenticatable { use HasApiTokens, Notifiable; /** * Find the user instance for the given username. */ public function findForPassport(string $username): User { return $this->where('username', $username)->first(); } }
Personalizando la Validación de Contraseña
Al autenticar utilizando el grant de contraseña, Passport utilizará el atributo password
de tu modelo para validar la contraseña dada. Si tu modelo no tiene un atributo password
o deseas personalizar la lógica de validación de contraseña, puedes definir un método validateForPassportPasswordGrant
en tu modelo:
<?php namespace App\Models; use Illuminate\Foundation\Auth\User as Authenticatable; use Illuminate\Notifications\Notifiable; use Illuminate\Support\Facades\Hash; use Laravel\Passport\HasApiTokens; class User extends Authenticatable { use HasApiTokens, Notifiable; /** * Validate the password of the user for the Passport password grant. */ public function validateForPassportPasswordGrant(string $password): bool { return Hash::check($password, $this->password); } }
Tokens de Grant Implícito
Ya no recomendamos usar tokens de concesión implícita. En su lugar, deberías elegir un tipo de concesión que sea actualmente recomendado por el servidor OAuth2.
El otorgamiento implícito es similar al otorgamiento de código de autorización; sin embargo, el token se devuelve al cliente sin intercambiar un código de autorización. Este otorgamiento se utiliza comúnmente para aplicaciones JavaScript o móviles donde las credenciales del cliente no se pueden almacenar de manera segura. Para habilitar el otorgamiento, llama al método enableImplicitGrant
en el método boot
de la clase App\Providers\AppServiceProvider
de tu aplicación:
/** * Bootstrap any application services. */ public function boot(): void { Passport::enableImplicitGrant(); }
Una vez que se ha habilitado el otorgamiento, los desarrolladores pueden usar su ID de cliente para solicitar un token de acceso desde tu aplicación. La aplicación consumidora debe hacer una solicitud de redireccionamiento a la ruta /oauth/authorize
de tu aplicación de la siguiente manera:
use Illuminate\Http\Request; Route::get('/redirect', function (Request $request) { $request->session()->put('state', $state = Str::random(40)); $query = http_build_query([ 'client_id' => 'client-id', 'redirect_uri' => 'http://third-party-app.com/callback', 'response_type' => 'token', 'scope' => '', 'state' => $state, // 'prompt' => '', // "none", "consent", or "login" ]); return redirect('http://passport-app.test/oauth/authorize?'.$query); });
Recuerda que la ruta /oauth/authorize
ya está definida por Passport. No necesitas definir esta ruta manualmente.
Tokens de Grant de Credenciales de Cliente
El grant de credenciales del cliente es adecuado para la autenticación de máquina a máquina. Por ejemplo, podrías usar este grant en un trabajo programado que esté realizando tareas de mantenimiento a través de una API.
Antes de que tu aplicación pueda emitir tokens a través del grant de credenciales del cliente, necesitarás crear un cliente de grant de credenciales del cliente. Puedes hacer esto utilizando la opción --client
del comando Artisan passport:client
:
php artisan passport:client --client
A continuación, para usar este tipo de concesión, registra un alias de middleware para el middleware CheckClientCredentials
. Puedes definir alias de middleware en el archivo bootstrap/app.php
de tu aplicación:
use Laravel\Passport\Http\Middleware\CheckClientCredentials; ->withMiddleware(function (Middleware $middleware) { $middleware->alias([ 'client' => CheckClientCredentials::class ]); })
Entonces, adjunta el middleware a una ruta:
Route::get('/orders', function (Request $request) { ... })->middleware('client');
Para restringir el acceso a la ruta a ámbitos específicos, puedes proporcionar una lista delimitada por comas de los ámbitos requeridos al adjuntar el middleware client
a la ruta:
Route::get('/orders', function (Request $request) { ... })->middleware('client:check-status,your-scope');
Recuperando Tokens
Para recuperar un token utilizando este tipo de concesión, realiza una solicitud al endpoint oauth/token
:
use Illuminate\Support\Facades\Http; $response = Http::asForm()->post('http://passport-app.test/oauth/token', [ 'grant_type' => 'client_credentials', 'client_id' => 'client-id', 'client_secret' => 'client-secret', 'scope' => 'your-scope', ]); return $response->json()['access_token'];
Tokens de Acceso Personal
A veces, tus usuarios pueden querer emitir tokens de acceso para sí mismos sin pasar por el flujo típico de redirección de código de autorización. Permitir que los usuarios emitan tokens para sí mismos a través de la interfaz de usuario de tu aplicación puede ser útil para permitir que los usuarios experimenten con tu API o puede servir como un enfoque más simple para emitir tokens de acceso en general.
Si tu aplicación está utilizando principalmente Passport para emitir tokens de acceso personales, considera usar Laravel Sanctum, la biblioteca ligera de Laravel para emitir tokens de acceso a la API.
Creando un Cliente de Acceso Personal
Antes de que tu aplicación pueda emitir tokens de acceso personal, necesitarás crear un cliente de acceso personal. Puedes hacer esto ejecutando el comando Artisan passport:client
con la opción --personal
. Si ya has ejecutado el comando passport:install
, no necesitas ejecutar este comando:
php artisan passport:client --personal
Después de crear tu cliente de acceso personal, coloca el ID del cliente y el valor secreto en texto plano en el archivo .env
de tu aplicación:
PASSPORT_PERSONAL_ACCESS_CLIENT_ID="client-id-value" PASSPORT_PERSONAL_ACCESS_CLIENT_SECRET="unhashed-client-secret-value"
Gestionando Tokens de Acceso Personal
Una vez que hayas creado un cliente de acceso personal, puedes emitir tokens para un usuario dado utilizando el método createToken
en la instancia del modelo App\Models\User
. El método createToken
acepta el nombre del token como su primer argumento y un array opcional de alcances como su segundo argumento:
use App\Models\User; $user = User::find(1); // Creating a token without scopes... $token = $user->createToken('Token Name')->accessToken; // Creating a token with scopes... $token = $user->createToken('My Token', ['place-orders'])->accessToken;
API JSON
Passport también incluye una API JSON para gestionar tokens de acceso personal. Puedes combinar esto con tu propio frontend para ofrecer a tus usuarios un panel de control para gestionar sus tokens de acceso personal. A continuación, revisaremos todos los puntos finales de la API para gestionar tokens de acceso personal. Para mayor comodidad, utilizaremos Axios para demostrar cómo realizar solicitudes HTTP a los puntos finales.
La API JSON está protegida por los middleware web
y auth
; por lo tanto, solo puede ser llamada desde tu propia aplicación. No puede ser llamada desde una fuente externa.
GET /oauth/scopes
Esta ruta devuelve todos los alcances definidos para tu aplicación. Puedes usar esta ruta para listar los alcances que un usuario puede asignar a un token de acceso personal:
axios.get('/oauth/scopes') .then(response => { console.log(response.data); });
GET /oauth/personal-access-tokens
Esta ruta devuelve todos los tokens de acceso personal que ha creado el usuario autenticado. Esto es útil principalmente para listar todos los tokens del usuario para que pueda editarlos o revocarlos:
axios.get('/oauth/personal-access-tokens') .then(response => { console.log(response.data); });
POST /oauth/personal-access-tokens
Esta ruta crea nuevos tokens de acceso personal. Requiere dos piezas de información: el nombre
del token y los alcances
que deben asignarse al token:
const data = { name: 'Token Name', scopes: [] }; axios.post('/oauth/personal-access-tokens', data) .then(response => { console.log(response.data.accessToken); }) .catch (response => { // List errors on response... });
DELETE /oauth/personal-access-tokens/{token-id}
Esta ruta se puede usar para revocar tokens de acceso personal:
axios.delete('/oauth/personal-access-tokens/' + tokenId);
Protegiendo Rutas
A través de Middleware
Passport incluye un guardia de autenticación que validará los tokens de acceso en las solicitudes entrantes. Una vez que hayas configurado el guardia api
para usar el driver passport
, solo necesitas especificar el middleware auth:api
en las rutas que deben requerir un token de acceso válido:
Route::get('/user', function () { // ... })->middleware('auth:api');
Si estás utilizando el grant de credenciales de cliente, deberías usar el middleware client
para proteger tus rutas en lugar del middleware auth:api
.
Múltiples Guardias de Autenticación
Si tu aplicación autentica diferentes tipos de usuarios que pueden usar modelos Eloquent completamente diferentes, es probable que necesites definir una configuración de guardia para cada tipo de proveedor de usuarios en tu aplicación. Esto te permite proteger las solicitudes destinadas a proveedores de usuarios específicos. Por ejemplo, dada la siguiente configuración de guardia en el archivo de configuración config/auth.php
:
'api' => [ 'driver' => 'passport', 'provider' => 'users', ], 'api-customers' => [ 'driver' => 'passport', 'provider' => 'customers', ],
La siguiente ruta utilizará el guardia api-customers
, que usa el proveedor de usuarios customers
, para autenticar las solicitudes entrantes:
Route::get('/customer', function () { // ... })->middleware('auth:api-customers');
Para obtener más información sobre el uso de múltiples proveedores de usuarios con Passport, consulta la documentación de concesión de contraseña.
Pasando el Token de Acceso
Al llamar a rutas que están protegidas por Passport, los consumidores de la API de tu aplicación deben especificar su token de acceso como un token Bearer
en el encabezado Authorization
de su solicitud. Por ejemplo, al usar la biblioteca HTTP Guzzle:
use Illuminate\Support\Facades\Http; $response = Http::withHeaders([ 'Accept' => 'application/json', 'Authorization' => 'Bearer '.$accessToken, ])->get('https://passport-app.test/api/user'); return $response->json();
Alcances de Token
Los scopes permiten a tus clientes de API solicitar un conjunto específico de permisos al solicitar autorización para acceder a una cuenta. Por ejemplo, si estás construyendo una aplicación de comercio electrónico, no todos los consumidores de API necesitarán la capacidad de realizar pedidos. En su lugar, puedes permitir que los consumidores solo soliciten autorización para acceder a los estados de envío de pedidos. En otras palabras, los scopes permiten a los usuarios de tu aplicación limitar las acciones que una aplicación de terceros puede realizar en su nombre.
Definiendo Alcances
Puedes definir los ámbitos de tu API utilizando el método Passport::tokensCan
en el método boot
de la clase App\Providers\AppServiceProvider
de tu aplicación. El método tokensCan
acepta un array de nombres de ámbito y descripciones de ámbito. La descripción del ámbito puede ser lo que desees y se mostrará a los usuarios en la pantalla de aprobación de autorización:
/** * Bootstrap any application services. */ public function boot(): void { Passport::tokensCan([ 'place-orders' => 'Place orders', 'check-status' => 'Check order status', ]); }
Alcance Predeterminado
Si un cliente no solicita ningún alcance específico, puedes configurar tu servidor Passport para adjuntar un(los) alcance(s) predeterminado(s) al token utilizando el método setDefaultScope
. Típicamente, debes llamar a este método desde el método boot
de la clase App\Providers\AppServiceProvider
de tu aplicación:
use Laravel\Passport\Passport; Passport::tokensCan([ 'place-orders' => 'Place orders', 'check-status' => 'Check order status', ]); Passport::setDefaultScope([ 'check-status', 'place-orders', ]);
Los scopes predeterminados de Passport no se aplican a los tokens de acceso personal que son generados por el usuario.
Asignando Alcances a Tokens
Al Solicitar Códigos de Autorización
Al solicitar un token de acceso utilizando el flujo de autorización con código, los consumidores deben especificar sus ámbitos deseados como el parámetro de cadena de consulta scope
. El parámetro scope
debe ser una lista de ámbitos delimitada por espacios:
Route::get('/redirect', function () { $query = http_build_query([ 'client_id' => 'client-id', 'redirect_uri' => 'http://example.com/callback', 'response_type' => 'code', 'scope' => 'place-orders check-status', ]); return redirect('http://passport-app.test/oauth/authorize?'.$query); });
Al Emitir Tokens de Acceso Personales
Si estás emitiendo tokens de acceso personal utilizando el método createToken
del modelo App\Models\User
, puedes pasar el array de ámbitos deseados como segundo argumento al método:
$token = $user->createToken('My Token', ['place-orders'])->accessToken;
Verificando Alcances
Passport incluye dos middleware que se pueden usar para verificar que una solicitud entrante esté autenticada con un token que se ha otorgado un alcance dado. Para comenzar, define los siguientes alias de middleware en el archivo bootstrap/app.php
de tu aplicación:
use Laravel\Passport\Http\Middleware\CheckForAnyScope; use Laravel\Passport\Http\Middleware\CheckScopes; ->withMiddleware(function (Middleware $middleware) { $middleware->alias([ 'scopes' => CheckScopes::class, 'scope' => CheckForAnyScope::class, ]); })
Verificar Todos los Ámbitos
El middleware scopes
puede asignarse a una ruta para verificar que el token de acceso de la solicitud entrante tenga todos los scopes listados:
Route::get('/orders', function () { // Access token has both "check-status" and "place-orders" scopes... })->middleware(['auth:api', 'scopes:check-status,place-orders']);
Verificar si hay algún alcance
El middleware scope
puede ser asignado a una ruta para verificar que el token de acceso de la solicitud entrante tenga al menos uno de los alcances listados:
Route::get('/orders', function () { // Access token has either "check-status" or "place-orders" scope... })->middleware(['auth:api', 'scope:check-status,place-orders']);
Comprobando Alcances en una Instancia de Token
Una vez que una solicitud autenticada con un token de acceso ha ingresado a tu aplicación, aún puedes verificar si el token tiene un alcance dado utilizando el método tokenCan
en la instancia autenticada de App\Models\User
:
use Illuminate\Http\Request; Route::get('/orders', function (Request $request) { if ($request->user()->tokenCan('place-orders')) { // ... } });
Métodos de Alcance Adicionales
El método scopeIds
devolverá un array de todos los IDs / nombres definidos:
use Laravel\Passport\Passport; Passport::scopeIds();
El método scopes
devolverá un array con todos los scopes definidos como instancias de Laravel\Passport\Scope
:
Passport::scopes();
El método scopesFor
devolverá un array de instancias de Laravel\Passport\Scope
que coinciden con los ID / nombres dados:
Passport::scopesFor(['place-orders', 'check-status']);
Puedes determinar si un alcance dado ha sido definido utilizando el método hasScope
:
Passport::hasScope('place-orders');
Consumiendo Tu API Con JavaScript
Al construir una API, puede ser extremadamente útil poder consumir tu propia API desde tu aplicación JavaScript. Este enfoque de desarrollo de API permite que tu propia aplicación consuma la misma API que estás compartiendo con el mundo. La misma API puede ser consumida por tu aplicación web, aplicaciones móviles, aplicaciones de terceros y cualquier SDK que puedas publicar en varios gestores de paquetes.
Típicamente, si deseas consumir tu API desde tu aplicación JavaScript, necesitarías enviar manualmente un token de acceso a la aplicación y pasarlo con cada solicitud a tu aplicación. Sin embargo, Passport incluye un middleware que puede manejar esto por ti. Todo lo que necesitas hacer es añadir el middleware CreateFreshApiToken
al grupo de middleware web
en el archivo bootstrap/app.php
de tu aplicación:
use Laravel\Passport\Http\Middleware\CreateFreshApiToken; ->withMiddleware(function (Middleware $middleware) { $middleware->web(append: [ CreateFreshApiToken::class, ]); })
Debes asegurarte de que el middleware CreateFreshApiToken
sea el último middleware listado en tu pila de middleware.
Este middleware adjuntará una cookie laravel_token
a tus respuestas salientes. Esta cookie contiene un JWT encriptado que Passport utilizará para autenticar las solicitudes API desde tu aplicación JavaScript. El JWT tiene una duración igual al valor de configuración session.lifetime
. Ahora, dado que el navegador enviará automáticamente la cookie con todas las solicitudes posteriores, puedes hacer solicitudes a la API de tu aplicación sin pasar explícitamente un token de acceso:
axios.get('/api/user') .then(response => { console.log(response.data); });
Personalizando el Nombre de la Cookie
Si es necesario, puedes personalizar el nombre de la cookie laravel_token
utilizando el método Passport::cookie
. Típicamente, este método debe ser llamado desde el método boot
de la clase App\Providers\AppServiceProvider
de tu aplicación:
/** * Bootstrap any application services. */ public function boot(): void { Passport::cookie('custom_name'); }
Protección CSRF
Al utilizar este método de autenticación, deberás asegurarte de que se incluya un encabezado de token CSRF válido en tus solicitudes. El andamiaje JavaScript predeterminado de Laravel incluye una instancia de Axios, que utilizará automáticamente el valor de la cookie XSRF-TOKEN
encriptada para enviar un encabezado X-XSRF-TOKEN
en solicitudes de origen mismo.
Si eliges enviar el encabezado X-CSRF-TOKEN
en lugar de X-XSRF-TOKEN
, necesitarás usar el token no encriptado proporcionado por csrf_token()
.
Eventos
Passport genera eventos al emitir tokens de acceso y tokens de actualización. Puedes escuchar estos eventos para eliminar o revocar otros tokens de acceso en tu base de datos:
Nombre del Evento |
---|
Laravel\Passport\Events\AccessTokenCreated |
Laravel\Passport\Events\RefreshTokenCreated |
Pruebas
El método actingAs
de Passport se puede utilizar para especificar el usuario autenticado actualmente, así como sus scopes. El primer argumento dado al método actingAs
es la instancia del usuario y el segundo es un array de scopes que deben ser otorgados al token del usuario:
use App\Models\User; use Laravel\Passport\Passport; test('servers can be created', function () { Passport::actingAs( User::factory()->create(), ['create-servers'] ); $response = $this->post('/api/create-server'); $response->assertStatus(201); });
use App\Models\User; use Laravel\Passport\Passport; public function test_servers_can_be_created(): void { Passport::actingAs( User::factory()->create(), ['create-servers'] ); $response = $this->post('/api/create-server'); $response->assertStatus(201); }
El método actingAsClient
de Passport se puede utilizar para especificar el cliente autenticado actualmente, así como sus scopes. El primer argumento dado al método actingAsClient
es la instancia del cliente y el segundo es un array de scopes que se deben otorgar al token del cliente:
use Laravel\Passport\Client; use Laravel\Passport\Passport; test('orders can be retrieved', function () { Passport::actingAsClient( Client::factory()->create(), ['check-status'] ); $response = $this->get('/api/orders'); $response->assertStatus(200); });
use Laravel\Passport\Client; use Laravel\Passport\Passport; public function test_orders_can_be_retrieved(): void { Passport::actingAsClient( Client::factory()->create(), ['check-status'] ); $response = $this->get('/api/orders'); $response->assertStatus(200); }