The Best Fluffy Pancakes recipe you will fall in love with. Full of tips and tricks to help you make the best pancakes.
記憶點
- 安裝好套件後,會產生對應資料表,但都沒資料,需要執行
php artisan passport:client --password
來在oauth_clients建立一筆可以使用的secret- oauth_clients.passord_client 欄位需要是1,grant type=password 的才能使用該secret
- oauth_clients.provider 的值對應了使用者資料表,基本上應該會寫users
- clinet_id、
client_secret
為 oauth_clients.id、oauth_clients.secret,
username、password 為 users表中的 email、password- 不同使用者可以共用同一個secret
- 使用 後端代理請求 的做法,前端(使用者)只需要輸入帳號密碼,後端以此帳密加上已經設定好的secret取得token後回傳給使用者,如此secret便不會暴露出來,使用者也能順利取得token
Laravel Passport 的 OAuth 資料表
Laravel Passport 在安裝並運行 php artisan migrate
後,會生成以下主要的 OAuth 資料表:
- oauth_clients
- oauth_access_tokens
- oauth_refresh_tokens
- oauth_auth_codes
- oauth_personal_access_clients
- oauth_scopes
1. oauth_clients
用途:存儲所有的 OAuth 客戶端應用程式資訊。
主要欄位:
id
:主鍵。user_id
:關聯的使用者 ID(如果有)。name
:客戶端名稱。secret
:客戶端密鑰,用於驗證客戶端身份。redirect
:重定向 URI,用於授權碼授權流程。personal_access_client
:是否為個人訪問客戶端。password_client
:是否為密碼授權客戶端。revoked
:是否被撤銷。
2. oauth_access_tokens
用途:存儲所有發放的訪問令牌(Access Tokens)。
主要欄位:
id
:主鍵。user_id
:擁有此令牌的使用者 ID。client_id
:發行此令牌的客戶端 ID。name
:令牌名稱。scopes
:授權範圍。revoked
:是否被撤銷。expires_at
:令牌過期時間。
3. oauth_refresh_tokens
用途:存儲所有發放的刷新令牌(Refresh Tokens)。
主要欄位:
id
:主鍵。access_token_id
:關聯的訪問令牌 ID。revoked
:是否被撤銷。expires_at
:刷新令牌過期時間。
4. oauth_auth_codes
用途:存儲授權碼,用於授權碼授權流程。
主要欄位:
id
:主鍵。user_id
:擁有此授權碼的使用者 ID。client_id
:關聯的客戶端 ID。scopes
:授權範圍。revoked
:是否被撤銷。expires_at
:授權碼過期時間。redirect_uri
:重定向 URI。
5. oauth_personal_access_clients
用途:存儲個人訪問客戶端,用於生成個人訪問令牌。
主要欄位:
id
:主鍵。client_id
:關聯的客戶端 ID。created_at
、updated_at
:時間戳記。
6. oauth_scopes
用途:定義可用的授權範圍(Scopes)。
主要欄位:
id
:主鍵。description
:範圍描述。
資料表之間的關係
- oauth_clients 與 oauth_access_tokens:一對多關係,一個客戶端可以發行多個訪問令牌。
- oauth_access_tokens 與 oauth_refresh_tokens:一對一關係,一個訪問令牌對應一個刷新令牌。
- oauth_clients 與 oauth_auth_codes:一對多關係,一個客戶端可以生成多個授權碼。
- oauth_clients 與 oauth_personal_access_clients:一對一關係,一個客戶端可以是個人訪問客戶端。
- oauth_scopes 與 oauth_access_tokens/oauth_auth_codes:多對多關係,一個範圍可以被多個令牌或授權碼使用。
建立 OAuth 客戶端
Laravel Passport 提供了 Artisan 命令來創建不同類型的 OAuth 客戶端。根據你的應用需求,可以創建以下幾種類型的客戶端:
- 授權碼客戶端(Authorization Code Client)
- 密碼授權客戶端(Password Grant Client)
- 個人訪問客戶端(Personal Access Client)
1. 創建授權碼客戶端
這種客戶端適用於需要使用授權碼授權流程的應用,例如第三方應用。
bash複製程式碼php artisan passport:client
執行後,會提示輸入客戶端名稱和重定向 URI。例如:
less複製程式碼Enter a name for the client: My Authorization Code Client
Enter the redirect URL: https://myapp.com/callback
2. 創建密碼授權客戶端
適用於需要用戶直接通過應用提供的憑證(如用戶名和密碼)獲取令牌的情況,常用於移動應用或單頁應用(SPA)。
php artisan passport:client --password
獲取訪問令牌(Access Token)
取得訪問令牌的方式取決於使用的授權類型。以下介紹幾種常見的授權類型及其獲取令牌的方法。
1. 密碼授權流程(Password Grant)
適用於信任的應用,如官方的移動應用或單頁應用。
步驟:
- 創建密碼授權客戶端如上所述,使用
php artisan passport:client --password
創建。 - 發送請求獲取令牌使用 HTTP POST 請求向
/oauth/token
端點發送請求,包含以下參數:grant_type
: 固定為password
client_id
: 密碼授權客戶端的 IDclient_secret
: 密碼授權客戶端的密鑰username
: 用戶名(通常是 email)password
: 用戶密碼scope
: 可選,指定授權範圍
示例請求:
http複製程式碼POST /oauth/token HTTP/1.1
Host: your-app.com
Content-Type: application/json
{
"grant_type": "password",
"client_id": "client-id-here",
"client_secret": "client-secret-here",
"username": "user@example.com",
"password": "user-password",
"scope": ""
}
示例響應:
json複製程式碼{
"token_type": "Bearer",
"expires_in": 31536000,
"access_token": "access-token-here",
"refresh_token": "refresh-token-here"
}
2. 授權碼授權流程(Authorization Code Grant)
適用於第三方應用,通過重定向用戶到授權服務器以獲取授權碼,再交換令牌。
步驟:
- 引導用戶授權將用戶重定向到以下 URL,讓用戶授權應用:perl複製程式碼
https://your-app.com/oauth/authorize?response_type=code&client_id=client-id-here&redirect_uri=https://your-app.com/callback&scope=&state=optional-state
- 用戶授權後,獲取授權碼用戶同意後,會重定向到
redirect_uri
並帶上code
參數。 - 交換授權碼獲取令牌使用 HTTP POST 請求向
/oauth/token
端點發送請求,包含以下參數:grant_type
: 固定為authorization_code
client_id
: 授權碼客戶端的 IDclient_secret
: 授權碼客戶端的密鑰redirect_uri
: 重定向 URI,需與創建客戶端時相同code
: 授權碼
示例請求:
http複製程式碼POST /oauth/token HTTP/1.1
Host: your-app.com
Content-Type: application/json
{
"grant_type": "authorization_code",
"client_id": "client-id-here",
"client_secret": "client-secret-here",
"redirect_uri": "https://your-app.com/callback",
"code": "authorization-code-here"
}
示例響應:
json複製程式碼{
"token_type": "Bearer",
"expires_in": 31536000,
"access_token": "access-token-here",
"refresh_token": "refresh-token-here"
}
3. 客戶端憑證授權流程(Client Credentials Grant)
適用於機器對機器的應用,不涉及用戶。
步驟:
- 創建客戶端憑證客戶端使用
php artisan passport:client --client
創建。 - 發送請求獲取令牌使用 HTTP POST 請求向
/oauth/token
端點發送請求,包含以下參數:grant_type
: 固定為client_credentials
client_id
: 客戶端 IDclient_secret
: 客戶端密鑰scope
: 可選,指定授權範圍
示例請求:
http複製程式碼POST /oauth/token HTTP/1.1
Host: your-app.com
Content-Type: application/json
{
"grant_type": "client_credentials",
"client_id": "client-id-here",
"client_secret": "client-secret-here",
"scope": ""
}
示例響應:
json複製程式碼{
"token_type": "Bearer",
"expires_in": 31536000,
"access_token": "access-token-here"
}
4. 個人訪問授權流程(Personal Access Grant)
適用於為特定用戶生成長期有效的個人訪問令牌,通常用於個人 API 調試或工具。
步驟:
- 創建個人訪問客戶端如上所述,使用
php artisan passport:personal-client
創建。 - 發送請求獲取個人訪問令牌使用 HTTP POST 請求向
/oauth/token
端點發送請求,包含以下參數:grant_type
: 固定為personal_access
client_id
: 個人訪問客戶端的 IDclient_secret
: 個人訪問客戶端的密鑰username
: 用戶名(通常是 email)password
: 用戶密碼scope
: 可選,指定授權範圍
示例請求:
http複製程式碼POST /oauth/token HTTP/1.1
Host: your-app.com
Content-Type: application/json
{
"grant_type": "password",
"client_id": "personal-client-id",
"client_secret": "personal-client-secret",
"username": "user@example.com",
"password": "user-password",
"scope": ""
}
示例響應:
json複製程式碼{
"token_type": "Bearer",
"expires_in": 31536000,
"access_token": "personal-access-token-here",
"refresh_token": "refresh-token-here"
}
使用取得的訪問令牌
獲取訪問令牌後,需在後續 API 請求中將其包含在 Authorization
標頭中,以便通過 Passport 的身份驗證和授權機制。
示例:
http複製程式碼GET /api/posts HTTP/1.1
Host: your-app.com
Authorization: Bearer access-token-here
在前端應用中使用訪問令牌
假設你使用 JavaScript 的 fetch
API 來發送請求,可以這樣做:
javascript複製程式碼fetch('https://your-app.com/api/posts', {
method: 'GET',
headers: {
'Authorization': `Bearer ${accessToken}`,
'Content-Type': 'application/json'
}
})
.then(response => response.json())
.then(data => console.log(data))
.catch(error => console.error('Error:', error));
刷新訪問令牌(Refresh Token)
訪問令牌通常有有效期限,當訪問令牌過期時,可以使用刷新令牌獲取新的訪問令牌,而無需用戶重新登入。
步驟:
- 發送請求使用刷新令牌獲取新令牌使用 HTTP POST 請求向
/oauth/token
端點發送請求,包含以下參數:grant_type
: 固定為refresh_token
refresh_token
: 刷新令牌client_id
: 客戶端 IDclient_secret
: 客戶端密鑰scope
: 可選,指定授權範圍
示例請求:
http複製程式碼POST /oauth/token HTTP/1.1
Host: your-app.com
Content-Type: application/json
{
"grant_type": "refresh_token",
"refresh_token": "refresh-token-here",
"client_id": "client-id-here",
"client_secret": "client-secret-here",
"scope": ""
}
示例響應:
json複製程式碼{
"token_type": "Bearer",
"expires_in": 31536000,
"access_token": "new-access-token-here",
"refresh_token": "new-refresh-token-here"
}
在應用中實現刷新令牌
當你收到 401 未授權錯誤時,可以自動觸發刷新令牌流程。例如,使用 Axios 的攔截器:
javascript複製程式碼import axios from 'axios';
const api = axios.create({
baseURL: 'https://your-app.com/api',
headers: {
'Authorization': `Bearer ${accessToken}`,
'Content-Type': 'application/json'
}
});
// 攔截器處理 401 錯誤並嘗試刷新令牌
api.interceptors.response.use(
response => response,
async error => {
if (error.response.status === 401) {
// 發送刷新令牌請求
const refreshToken = localStorage.getItem('refresh_token');
const response = await axios.post('https://your-app.com/oauth/token', {
grant_type: 'refresh_token',
refresh_token: refreshToken,
client_id: 'client-id-here',
client_secret: 'client-secret-here',
scope: ''
});
const newAccessToken = response.data.access_token;
const newRefreshToken = response.data.refresh_token;
// 更新存儲的令牌
localStorage.setItem('access_token', newAccessToken);
localStorage.setItem('refresh_token', newRefreshToken);
// 更新請求頭並重試原始請求
error.config.headers['Authorization'] = `Bearer ${newAccessToken}`;
return axios(error.config);
}
return Promise.reject(error);
}
);
export default api;
總結
Laravel Passport 的 OAuth 資料表關係總結:
- oauth_clients:管理所有 OAuth 客戶端應用。
- oauth_access_tokens:存儲訪問令牌,與客戶端和用戶相關聯。
- oauth_refresh_tokens:存儲刷新令牌,與訪問令牌相關聯。
- oauth_auth_codes:存儲授權碼,與客戶端和用戶相關聯。
- oauth_personal_access_clients:管理個人訪問客戶端。
- oauth_scopes:定義授權範圍,與令牌和授權碼相關聯。
建立客戶端和獲取令牌的通用流程:
- 創建合適的 OAuth 客戶端(授權碼、密碼授權、客戶端憑證或個人訪問)。
- 根據授權類型,通過對應的流程(密碼授權、授權碼授權等)獲取訪問令牌。
- 在 API 請求中包含訪問令牌,通過
Authorization: Bearer {access_token}
標頭進行身份驗證。 - 當訪問令牌過期時,使用刷新令牌獲取新的訪問令牌,保持用戶的會話有效。
後端代理請求
- 流程概述: 前端應用不需要直接處理
client_secret
。相反,前端將使用者的認證資料(例如:使用者名稱和密碼)發送到你的後端伺服器,後端伺服器再代表前端處理與 OAuth 伺服器之間的所有安全交換,包括傳遞client_secret
。 - 步驟:
- 前端應用(如單頁應用或手機應用)向你的後端發送認證資料,例如使用者的 email 和密碼。
- 後端應用接收這些認證資料,然後將
client_id
和client_secret
與這些資料一起發送到/oauth/token
端點以交換access_token
。 - 後端將獲取的
access_token
返回給前端。