2024-12-10 15:59:25 +01:00
|
|
|
package users
|
2023-03-21 16:02:19 +01:00
|
|
|
|
|
|
|
import (
|
|
|
|
"encoding/json"
|
|
|
|
"net/http"
|
|
|
|
|
|
|
|
"github.com/gorilla/mux"
|
|
|
|
|
|
|
|
"github.com/netbirdio/netbird/management/server"
|
|
|
|
"github.com/netbirdio/netbird/management/server/http/api"
|
2024-12-10 15:59:25 +01:00
|
|
|
"github.com/netbirdio/netbird/management/server/http/configs"
|
2023-03-21 16:02:19 +01:00
|
|
|
"github.com/netbirdio/netbird/management/server/http/util"
|
|
|
|
"github.com/netbirdio/netbird/management/server/jwtclaims"
|
|
|
|
"github.com/netbirdio/netbird/management/server/status"
|
2024-12-20 11:30:28 +01:00
|
|
|
"github.com/netbirdio/netbird/management/server/types"
|
2023-03-21 16:02:19 +01:00
|
|
|
)
|
|
|
|
|
2024-12-10 15:59:25 +01:00
|
|
|
// patHandler is the nameserver group handler of the account
|
|
|
|
type patHandler struct {
|
2023-03-21 16:02:19 +01:00
|
|
|
accountManager server.AccountManager
|
|
|
|
claimsExtractor *jwtclaims.ClaimsExtractor
|
|
|
|
}
|
|
|
|
|
2024-12-10 15:59:25 +01:00
|
|
|
func addUsersTokensEndpoint(accountManager server.AccountManager, authCfg configs.AuthCfg, router *mux.Router) {
|
|
|
|
tokenHandler := newPATsHandler(accountManager, authCfg)
|
|
|
|
router.HandleFunc("/users/{userId}/tokens", tokenHandler.getAllTokens).Methods("GET", "OPTIONS")
|
|
|
|
router.HandleFunc("/users/{userId}/tokens", tokenHandler.createToken).Methods("POST", "OPTIONS")
|
|
|
|
router.HandleFunc("/users/{userId}/tokens/{tokenId}", tokenHandler.getToken).Methods("GET", "OPTIONS")
|
|
|
|
router.HandleFunc("/users/{userId}/tokens/{tokenId}", tokenHandler.deleteToken).Methods("DELETE", "OPTIONS")
|
|
|
|
}
|
|
|
|
|
|
|
|
// newPATsHandler creates a new patHandler HTTP handler
|
|
|
|
func newPATsHandler(accountManager server.AccountManager, authCfg configs.AuthCfg) *patHandler {
|
|
|
|
return &patHandler{
|
2023-03-21 16:02:19 +01:00
|
|
|
accountManager: accountManager,
|
|
|
|
claimsExtractor: jwtclaims.NewClaimsExtractor(
|
|
|
|
jwtclaims.WithAudience(authCfg.Audience),
|
|
|
|
jwtclaims.WithUserIDClaim(authCfg.UserIDClaim),
|
|
|
|
),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-12-10 15:59:25 +01:00
|
|
|
// getAllTokens is HTTP GET handler that returns a list of all personal access tokens for the given user
|
|
|
|
func (h *patHandler) getAllTokens(w http.ResponseWriter, r *http.Request) {
|
2023-03-21 16:02:19 +01:00
|
|
|
claims := h.claimsExtractor.FromRequestContext(r)
|
2024-09-27 16:10:50 +02:00
|
|
|
accountID, userID, err := h.accountManager.GetAccountIDFromToken(r.Context(), claims)
|
2023-03-21 16:02:19 +01:00
|
|
|
if err != nil {
|
2024-07-03 11:33:02 +02:00
|
|
|
util.WriteError(r.Context(), err, w)
|
2023-03-21 16:02:19 +01:00
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
vars := mux.Vars(r)
|
2024-09-27 16:10:50 +02:00
|
|
|
targetUserID := vars["userId"]
|
2023-03-21 16:02:19 +01:00
|
|
|
if len(userID) == 0 {
|
2024-07-03 11:33:02 +02:00
|
|
|
util.WriteError(r.Context(), status.Errorf(status.InvalidArgument, "invalid user ID"), w)
|
2023-03-21 16:02:19 +01:00
|
|
|
return
|
|
|
|
}
|
2023-03-30 13:58:44 +02:00
|
|
|
|
2024-09-27 16:10:50 +02:00
|
|
|
pats, err := h.accountManager.GetAllPATs(r.Context(), accountID, userID, targetUserID)
|
2023-03-30 13:58:44 +02:00
|
|
|
if err != nil {
|
2024-07-03 11:33:02 +02:00
|
|
|
util.WriteError(r.Context(), err, w)
|
2023-03-21 16:02:19 +01:00
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2023-03-30 13:58:44 +02:00
|
|
|
var patResponse []*api.PersonalAccessToken
|
|
|
|
for _, pat := range pats {
|
|
|
|
patResponse = append(patResponse, toPATResponse(pat))
|
2023-03-21 16:02:19 +01:00
|
|
|
}
|
|
|
|
|
2024-07-03 11:33:02 +02:00
|
|
|
util.WriteJSONObject(r.Context(), w, patResponse)
|
2023-03-21 16:02:19 +01:00
|
|
|
}
|
|
|
|
|
2024-12-10 15:59:25 +01:00
|
|
|
// getToken is HTTP GET handler that returns a personal access token for the given user
|
|
|
|
func (h *patHandler) getToken(w http.ResponseWriter, r *http.Request) {
|
2023-03-21 16:02:19 +01:00
|
|
|
claims := h.claimsExtractor.FromRequestContext(r)
|
2024-09-27 16:10:50 +02:00
|
|
|
accountID, userID, err := h.accountManager.GetAccountIDFromToken(r.Context(), claims)
|
2023-03-21 16:02:19 +01:00
|
|
|
if err != nil {
|
2024-07-03 11:33:02 +02:00
|
|
|
util.WriteError(r.Context(), err, w)
|
2023-03-21 16:02:19 +01:00
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
vars := mux.Vars(r)
|
2023-03-30 13:58:44 +02:00
|
|
|
targetUserID := vars["userId"]
|
|
|
|
if len(targetUserID) == 0 {
|
2024-07-03 11:33:02 +02:00
|
|
|
util.WriteError(r.Context(), status.Errorf(status.InvalidArgument, "invalid user ID"), w)
|
2023-03-21 16:02:19 +01:00
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
tokenID := vars["tokenId"]
|
|
|
|
if len(tokenID) == 0 {
|
2024-07-03 11:33:02 +02:00
|
|
|
util.WriteError(r.Context(), status.Errorf(status.InvalidArgument, "invalid token ID"), w)
|
2023-03-21 16:02:19 +01:00
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2024-09-27 16:10:50 +02:00
|
|
|
pat, err := h.accountManager.GetPAT(r.Context(), accountID, userID, targetUserID, tokenID)
|
2023-03-30 13:58:44 +02:00
|
|
|
if err != nil {
|
2024-07-03 11:33:02 +02:00
|
|
|
util.WriteError(r.Context(), err, w)
|
2023-03-28 14:47:15 +02:00
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2024-07-03 11:33:02 +02:00
|
|
|
util.WriteJSONObject(r.Context(), w, toPATResponse(pat))
|
2023-03-21 16:02:19 +01:00
|
|
|
}
|
|
|
|
|
2024-12-10 15:59:25 +01:00
|
|
|
// createToken is HTTP POST handler that creates a personal access token for the given user
|
|
|
|
func (h *patHandler) createToken(w http.ResponseWriter, r *http.Request) {
|
2023-03-21 16:02:19 +01:00
|
|
|
claims := h.claimsExtractor.FromRequestContext(r)
|
2024-09-27 16:10:50 +02:00
|
|
|
accountID, userID, err := h.accountManager.GetAccountIDFromToken(r.Context(), claims)
|
2023-03-21 16:02:19 +01:00
|
|
|
if err != nil {
|
2024-07-03 11:33:02 +02:00
|
|
|
util.WriteError(r.Context(), err, w)
|
2023-03-21 16:02:19 +01:00
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
vars := mux.Vars(r)
|
2023-03-30 13:58:44 +02:00
|
|
|
targetUserID := vars["userId"]
|
|
|
|
if len(targetUserID) == 0 {
|
2024-07-03 11:33:02 +02:00
|
|
|
util.WriteError(r.Context(), status.Errorf(status.InvalidArgument, "invalid user ID"), w)
|
2023-03-21 16:02:19 +01:00
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
var req api.PostApiUsersUserIdTokensJSONRequestBody
|
|
|
|
err = json.NewDecoder(r.Body).Decode(&req)
|
|
|
|
if err != nil {
|
|
|
|
util.WriteErrorResponse("couldn't parse JSON request", http.StatusBadRequest, w)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2024-09-27 16:10:50 +02:00
|
|
|
pat, err := h.accountManager.CreatePAT(r.Context(), accountID, userID, targetUserID, req.Name, req.ExpiresIn)
|
2023-03-21 16:02:19 +01:00
|
|
|
if err != nil {
|
2024-07-03 11:33:02 +02:00
|
|
|
util.WriteError(r.Context(), err, w)
|
2023-03-21 16:02:19 +01:00
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2024-07-03 11:33:02 +02:00
|
|
|
util.WriteJSONObject(r.Context(), w, toPATGeneratedResponse(pat))
|
2023-03-21 16:02:19 +01:00
|
|
|
}
|
|
|
|
|
2024-12-10 15:59:25 +01:00
|
|
|
// deleteToken is HTTP DELETE handler that deletes a personal access token for the given user
|
|
|
|
func (h *patHandler) deleteToken(w http.ResponseWriter, r *http.Request) {
|
2023-03-21 16:02:19 +01:00
|
|
|
claims := h.claimsExtractor.FromRequestContext(r)
|
2024-09-27 16:10:50 +02:00
|
|
|
accountID, userID, err := h.accountManager.GetAccountIDFromToken(r.Context(), claims)
|
2023-03-21 16:02:19 +01:00
|
|
|
if err != nil {
|
2024-07-03 11:33:02 +02:00
|
|
|
util.WriteError(r.Context(), err, w)
|
2023-03-21 16:02:19 +01:00
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
vars := mux.Vars(r)
|
2023-03-30 13:58:44 +02:00
|
|
|
targetUserID := vars["userId"]
|
|
|
|
if len(targetUserID) == 0 {
|
2024-07-03 11:33:02 +02:00
|
|
|
util.WriteError(r.Context(), status.Errorf(status.InvalidArgument, "invalid user ID"), w)
|
2023-03-21 16:02:19 +01:00
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
tokenID := vars["tokenId"]
|
|
|
|
if len(tokenID) == 0 {
|
2024-07-03 11:33:02 +02:00
|
|
|
util.WriteError(r.Context(), status.Errorf(status.InvalidArgument, "invalid token ID"), w)
|
2023-03-21 16:02:19 +01:00
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2024-09-27 16:10:50 +02:00
|
|
|
err = h.accountManager.DeletePAT(r.Context(), accountID, userID, targetUserID, tokenID)
|
2023-03-21 16:02:19 +01:00
|
|
|
if err != nil {
|
2024-07-03 11:33:02 +02:00
|
|
|
util.WriteError(r.Context(), err, w)
|
2023-03-21 16:02:19 +01:00
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2024-12-10 15:59:25 +01:00
|
|
|
util.WriteJSONObject(r.Context(), w, util.EmptyObject{})
|
2023-03-21 16:02:19 +01:00
|
|
|
}
|
|
|
|
|
2024-12-20 11:30:28 +01:00
|
|
|
func toPATResponse(pat *types.PersonalAccessToken) *api.PersonalAccessToken {
|
2023-03-21 16:02:19 +01:00
|
|
|
return &api.PersonalAccessToken{
|
|
|
|
CreatedAt: pat.CreatedAt,
|
|
|
|
CreatedBy: pat.CreatedBy,
|
2023-03-27 16:28:49 +02:00
|
|
|
Name: pat.Name,
|
2025-01-06 13:38:30 +01:00
|
|
|
ExpirationDate: pat.GetExpirationDate(),
|
2023-03-21 16:02:19 +01:00
|
|
|
Id: pat.ID,
|
2025-01-06 13:38:30 +01:00
|
|
|
LastUsed: pat.LastUsed,
|
2023-03-21 16:02:19 +01:00
|
|
|
}
|
|
|
|
}
|
2023-03-28 14:47:15 +02:00
|
|
|
|
2024-12-20 11:30:28 +01:00
|
|
|
func toPATGeneratedResponse(pat *types.PersonalAccessTokenGenerated) *api.PersonalAccessTokenGenerated {
|
2023-03-28 14:47:15 +02:00
|
|
|
return &api.PersonalAccessTokenGenerated{
|
|
|
|
PlainToken: pat.PlainToken,
|
|
|
|
PersonalAccessToken: *toPATResponse(&pat.PersonalAccessToken),
|
|
|
|
}
|
|
|
|
}
|