da48a3652d
後端: - Migration:diving_offers 新增 provider_id(nullable FK) - Migration:users.role ENUM 加入 provider 值 - Migration:diving_offers.spot 改為 nullable - AuthController:registerProvider business_name 改為選填 - AuthController:updateProviderProfile 補上 certifications / dive_sites / services / facilities / website / social_media - ProviderOfferController:教練課程 CRUD(index/show/store/update/destroy),實作 provider_id 所有權不變式(404 → 403 兩步驟) 前端(frontend/): - coachAuth store、coachAxios(獨立於 member auth) - /coach/* 路由群組 + beforeEach guard - CoachNavBar、CoachLayout(教練頁隱藏會員 NavBar) - LoginView、RegisterView、DashboardView(表格 + 刪除確認) - OfferFormView(新增/編輯共用)、ProfileView OpenSpec: - openspec/config.yaml 補入專案 context 與 rules - 新增 specs:coach-offers-api / coach-portal-ui / provider-auth - 更新 spec:diving-offers-api 加入 provider_id - 歸檔:openspec/changes/archive/2026-05-10-coach-portal Co-Authored-By: Claude Sonnet 4.6 (1M context) <noreply@anthropic.com>
57 lines
2.2 KiB
Markdown
57 lines
2.2 KiB
Markdown
## ADDED Requirements
|
||
|
||
### Requirement: 教練帳號登入
|
||
後端 SHALL 提供 `POST /api/provider/login`,驗證 email/password 並回傳 Sanctum Bearer token,僅限 role=provider 帳號。
|
||
|
||
#### Scenario: 正確帳密登入成功
|
||
- **WHEN** 教練送出正確的 email 與 password
|
||
- **THEN** 回傳 HTTP 200,包含 `{ status: true, data: { user, token, token_type: "Bearer" } }`
|
||
|
||
#### Scenario: 錯誤帳密登入失敗
|
||
- **WHEN** 教練送出錯誤的 email 或 password
|
||
- **THEN** 回傳 HTTP 401,`{ status: false, message: "帳號或密碼錯誤" }`
|
||
|
||
#### Scenario: 會員帳號無法用教練登入
|
||
- **WHEN** role=member 的帳號嘗試呼叫 `/api/provider/login`
|
||
- **THEN** 回傳 HTTP 403,`{ status: false, message: "此帳號非教練角色" }`
|
||
|
||
---
|
||
|
||
### Requirement: 教練帳號註冊
|
||
後端 SHALL 提供 `POST /api/provider/register`,建立 role=provider 的 User 與對應 ProviderProfile。
|
||
|
||
#### Scenario: 新帳號註冊成功
|
||
- **WHEN** 送出有效的 name / email / password / password_confirmation
|
||
- **THEN** 回傳 HTTP 201,`{ status: true, data: { user } }`
|
||
|
||
#### Scenario: Email 重複
|
||
- **WHEN** 送出已存在的 email
|
||
- **THEN** 回傳 HTTP 422,`{ status: false, message: "此 Email 已被使用" }`
|
||
|
||
---
|
||
|
||
### Requirement: 教練登出
|
||
後端 SHALL 提供 `POST /api/provider/logout`(需 Bearer token),撤銷當前 token。
|
||
|
||
#### Scenario: 登出成功
|
||
- **WHEN** 已登入教練送出登出請求
|
||
- **THEN** 回傳 HTTP 200,`{ status: true, message: "已登出" }`,token 失效
|
||
|
||
---
|
||
|
||
### Requirement: 教練個人資料讀取
|
||
後端 SHALL 提供 `GET /api/provider/profile`(需 Bearer token),回傳教練基本資訊與 ProviderProfile。
|
||
|
||
#### Scenario: 取得個人資料
|
||
- **WHEN** 已登入教練送出 GET 請求
|
||
- **THEN** 回傳 HTTP 200,包含 `{ id, name, email, role, profile: { bio, expertise, certification, avatar } }`
|
||
|
||
---
|
||
|
||
### Requirement: 教練個人資料更新
|
||
後端 SHALL 提供 `PUT /api/provider/profile`(需 Bearer token),更新教練基本資訊與 ProviderProfile。
|
||
|
||
#### Scenario: 更新成功
|
||
- **WHEN** 教練送出合法的更新資料
|
||
- **THEN** 回傳 HTTP 200,`{ status: true, message: "資料已更新", data: { ...profile } }`
|