2023-02-15 01:34:22 +05:30
|
|
|
package api
|
|
|
|
|
|
|
|
|
|
import (
|
|
|
|
|
"context"
|
|
|
|
|
"encoding/json"
|
2023-02-15 23:49:03 +05:30
|
|
|
"fmt"
|
2023-02-15 01:34:22 +05:30
|
|
|
"net/http"
|
|
|
|
|
"time"
|
|
|
|
|
|
2025-03-20 21:01:41 +05:30
|
|
|
"github.com/SigNoz/signoz/ee/query-service/model"
|
2025-04-01 23:49:37 +05:30
|
|
|
"github.com/SigNoz/signoz/ee/types"
|
|
|
|
|
eeTypes "github.com/SigNoz/signoz/ee/types"
|
2025-04-04 01:46:28 +05:30
|
|
|
"github.com/SigNoz/signoz/pkg/errors"
|
|
|
|
|
"github.com/SigNoz/signoz/pkg/http/render"
|
2025-03-20 21:01:41 +05:30
|
|
|
"github.com/SigNoz/signoz/pkg/query-service/auth"
|
|
|
|
|
baseconstants "github.com/SigNoz/signoz/pkg/query-service/constants"
|
|
|
|
|
basemodel "github.com/SigNoz/signoz/pkg/query-service/model"
|
2025-04-04 01:46:28 +05:30
|
|
|
"github.com/SigNoz/signoz/pkg/valuer"
|
2023-02-15 01:34:22 +05:30
|
|
|
"github.com/gorilla/mux"
|
|
|
|
|
"go.uber.org/zap"
|
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
func (ah *APIHandler) createPAT(w http.ResponseWriter, r *http.Request) {
|
|
|
|
|
ctx := context.Background()
|
|
|
|
|
|
2024-02-16 12:46:33 +05:30
|
|
|
req := model.CreatePATRequestBody{}
|
2023-02-15 01:34:22 +05:30
|
|
|
if err := json.NewDecoder(r.Body).Decode(&req); err != nil {
|
|
|
|
|
RespondError(w, model.BadRequest(err), nil)
|
|
|
|
|
return
|
|
|
|
|
}
|
2025-02-17 18:16:41 +05:30
|
|
|
user, err := auth.GetUserFromReqContext(r.Context())
|
2023-02-15 01:34:22 +05:30
|
|
|
if err != nil {
|
|
|
|
|
RespondError(w, &model.ApiError{
|
|
|
|
|
Typ: model.ErrorUnauthorized,
|
|
|
|
|
Err: err,
|
|
|
|
|
}, nil)
|
|
|
|
|
return
|
|
|
|
|
}
|
2025-04-01 23:49:37 +05:30
|
|
|
pat := eeTypes.NewGettablePAT(
|
|
|
|
|
req.Name,
|
|
|
|
|
req.Role,
|
|
|
|
|
user.ID,
|
|
|
|
|
req.ExpiresInDays,
|
|
|
|
|
)
|
2024-02-16 12:46:33 +05:30
|
|
|
err = validatePATRequest(pat)
|
|
|
|
|
if err != nil {
|
|
|
|
|
RespondError(w, model.BadRequest(err), nil)
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
2024-03-27 00:07:29 +05:30
|
|
|
zap.L().Info("Got Create PAT request", zap.Any("pat", pat))
|
2024-02-16 12:46:33 +05:30
|
|
|
var apierr basemodel.BaseApiError
|
2025-03-20 13:59:52 +05:30
|
|
|
if pat, apierr = ah.AppDao().CreatePAT(ctx, user.OrgID, pat); apierr != nil {
|
2024-02-16 12:46:33 +05:30
|
|
|
RespondError(w, apierr, nil)
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
ah.Respond(w, &pat)
|
|
|
|
|
}
|
|
|
|
|
|
2025-04-01 23:49:37 +05:30
|
|
|
func validatePATRequest(req types.GettablePAT) error {
|
2024-02-16 12:46:33 +05:30
|
|
|
if req.Role == "" || (req.Role != baseconstants.ViewerGroup && req.Role != baseconstants.EditorGroup && req.Role != baseconstants.AdminGroup) {
|
|
|
|
|
return fmt.Errorf("valid role is required")
|
|
|
|
|
}
|
|
|
|
|
if req.ExpiresAt < 0 {
|
|
|
|
|
return fmt.Errorf("valid expiresAt is required")
|
|
|
|
|
}
|
|
|
|
|
if req.Name == "" {
|
|
|
|
|
return fmt.Errorf("valid name is required")
|
|
|
|
|
}
|
|
|
|
|
return nil
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func (ah *APIHandler) updatePAT(w http.ResponseWriter, r *http.Request) {
|
|
|
|
|
ctx := context.Background()
|
2023-02-15 01:34:22 +05:30
|
|
|
|
2025-04-01 23:49:37 +05:30
|
|
|
req := types.GettablePAT{}
|
2024-02-16 12:46:33 +05:30
|
|
|
if err := json.NewDecoder(r.Body).Decode(&req); err != nil {
|
|
|
|
|
RespondError(w, model.BadRequest(err), nil)
|
|
|
|
|
return
|
|
|
|
|
}
|
2023-02-15 01:34:22 +05:30
|
|
|
|
2025-02-17 18:16:41 +05:30
|
|
|
user, err := auth.GetUserFromReqContext(r.Context())
|
2024-02-16 12:46:33 +05:30
|
|
|
if err != nil {
|
|
|
|
|
RespondError(w, &model.ApiError{
|
|
|
|
|
Typ: model.ErrorUnauthorized,
|
|
|
|
|
Err: err,
|
|
|
|
|
}, nil)
|
|
|
|
|
return
|
2023-12-13 17:05:59 +05:30
|
|
|
}
|
2024-02-16 12:46:33 +05:30
|
|
|
|
|
|
|
|
err = validatePATRequest(req)
|
|
|
|
|
if err != nil {
|
|
|
|
|
RespondError(w, model.BadRequest(err), nil)
|
|
|
|
|
return
|
2023-12-13 17:05:59 +05:30
|
|
|
}
|
|
|
|
|
|
2025-03-06 15:39:45 +05:30
|
|
|
req.UpdatedByUserID = user.ID
|
2025-04-04 01:46:28 +05:30
|
|
|
idStr := mux.Vars(r)["id"]
|
|
|
|
|
id, err := valuer.NewUUID(idStr)
|
|
|
|
|
if err != nil {
|
|
|
|
|
render.Error(w, errors.Newf(errors.TypeInvalidInput, errors.CodeInvalidInput, "id is not a valid uuid-v7"))
|
|
|
|
|
return
|
|
|
|
|
}
|
2025-03-20 13:59:52 +05:30
|
|
|
req.UpdatedAt = time.Now()
|
2024-03-27 00:07:29 +05:30
|
|
|
zap.L().Info("Got Update PAT request", zap.Any("pat", req))
|
2023-12-13 17:05:59 +05:30
|
|
|
var apierr basemodel.BaseApiError
|
2025-03-20 13:59:52 +05:30
|
|
|
if apierr = ah.AppDao().UpdatePAT(ctx, user.OrgID, req, id); apierr != nil {
|
2023-02-15 01:34:22 +05:30
|
|
|
RespondError(w, apierr, nil)
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
2024-02-16 12:46:33 +05:30
|
|
|
ah.Respond(w, map[string]string{"data": "pat updated successfully"})
|
2023-02-15 01:34:22 +05:30
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func (ah *APIHandler) getPATs(w http.ResponseWriter, r *http.Request) {
|
|
|
|
|
ctx := context.Background()
|
2025-02-17 18:16:41 +05:30
|
|
|
user, err := auth.GetUserFromReqContext(r.Context())
|
2023-02-15 23:49:03 +05:30
|
|
|
if err != nil {
|
|
|
|
|
RespondError(w, &model.ApiError{
|
|
|
|
|
Typ: model.ErrorUnauthorized,
|
|
|
|
|
Err: err,
|
|
|
|
|
}, nil)
|
|
|
|
|
return
|
|
|
|
|
}
|
2025-03-06 15:39:45 +05:30
|
|
|
zap.L().Info("Get PATs for user", zap.String("user_id", user.ID))
|
2025-03-20 13:59:52 +05:30
|
|
|
pats, apierr := ah.AppDao().ListPATs(ctx, user.OrgID)
|
2023-02-15 01:34:22 +05:30
|
|
|
if apierr != nil {
|
|
|
|
|
RespondError(w, apierr, nil)
|
|
|
|
|
return
|
|
|
|
|
}
|
2023-02-24 01:19:01 +05:30
|
|
|
ah.Respond(w, pats)
|
2023-02-15 01:34:22 +05:30
|
|
|
}
|
|
|
|
|
|
2024-02-16 12:46:33 +05:30
|
|
|
func (ah *APIHandler) revokePAT(w http.ResponseWriter, r *http.Request) {
|
2023-02-15 01:34:22 +05:30
|
|
|
ctx := context.Background()
|
2025-04-04 01:46:28 +05:30
|
|
|
idStr := mux.Vars(r)["id"]
|
|
|
|
|
id, err := valuer.NewUUID(idStr)
|
|
|
|
|
if err != nil {
|
|
|
|
|
render.Error(w, errors.Newf(errors.TypeInvalidInput, errors.CodeInvalidInput, "id is not a valid uuid-v7"))
|
|
|
|
|
return
|
|
|
|
|
}
|
2025-02-17 18:16:41 +05:30
|
|
|
user, err := auth.GetUserFromReqContext(r.Context())
|
2023-02-15 23:49:03 +05:30
|
|
|
if err != nil {
|
|
|
|
|
RespondError(w, &model.ApiError{
|
|
|
|
|
Typ: model.ErrorUnauthorized,
|
|
|
|
|
Err: err,
|
|
|
|
|
}, nil)
|
|
|
|
|
return
|
|
|
|
|
}
|
2024-02-28 17:37:30 +05:30
|
|
|
|
2025-04-04 01:46:28 +05:30
|
|
|
zap.L().Info("Revoke PAT with id", zap.String("id", id.StringValue()))
|
2025-03-20 13:59:52 +05:30
|
|
|
if apierr := ah.AppDao().RevokePAT(ctx, user.OrgID, id, user.ID); apierr != nil {
|
2023-02-15 01:34:22 +05:30
|
|
|
RespondError(w, apierr, nil)
|
|
|
|
|
return
|
|
|
|
|
}
|
2024-02-16 12:46:33 +05:30
|
|
|
ah.Respond(w, map[string]string{"data": "pat revoked successfully"})
|
2023-02-15 01:34:22 +05:30
|
|
|
}
|