2020-10-14 18:48:48 +08:00
|
|
|
package api
|
|
|
|
|
|
|
|
import (
|
2020-10-16 15:22:24 +08:00
|
|
|
"fmt"
|
2021-11-29 17:18:01 +08:00
|
|
|
"net/http"
|
2025-08-13 03:01:41 +08:00
|
|
|
"time"
|
2020-10-14 18:48:48 +08:00
|
|
|
|
2025-08-13 03:01:41 +08:00
|
|
|
"github.com/teris-io/shortid"
|
|
|
|
|
|
|
|
"k8s.io/apimachinery/pkg/api/errors"
|
|
|
|
v1 "k8s.io/apimachinery/pkg/apis/meta/v1"
|
|
|
|
"k8s.io/apimachinery/pkg/runtime/schema"
|
|
|
|
"k8s.io/client-go/dynamic"
|
|
|
|
|
|
|
|
"github.com/grafana/grafana/apps/shorturl/pkg/apis/shorturl/v1alpha1"
|
2020-10-14 18:48:48 +08:00
|
|
|
"github.com/grafana/grafana/pkg/api/dtos"
|
2021-01-15 21:43:20 +08:00
|
|
|
"github.com/grafana/grafana/pkg/api/response"
|
2025-07-30 21:31:11 +08:00
|
|
|
"github.com/grafana/grafana/pkg/api/routing"
|
|
|
|
"github.com/grafana/grafana/pkg/middleware"
|
2025-08-13 03:01:41 +08:00
|
|
|
"github.com/grafana/grafana/pkg/registry/apps/shorturl"
|
|
|
|
grafanaapiserver "github.com/grafana/grafana/pkg/services/apiserver"
|
|
|
|
"github.com/grafana/grafana/pkg/services/apiserver/endpoints/request"
|
2023-01-27 15:50:36 +08:00
|
|
|
contextmodel "github.com/grafana/grafana/pkg/services/contexthandler/model"
|
2025-08-13 03:01:41 +08:00
|
|
|
"github.com/grafana/grafana/pkg/services/featuremgmt"
|
2023-01-12 17:13:47 +08:00
|
|
|
"github.com/grafana/grafana/pkg/services/shorturls"
|
2020-10-14 18:48:48 +08:00
|
|
|
"github.com/grafana/grafana/pkg/setting"
|
|
|
|
"github.com/grafana/grafana/pkg/util"
|
2025-08-13 03:01:41 +08:00
|
|
|
"github.com/grafana/grafana/pkg/util/errhttp"
|
2021-10-11 20:30:59 +08:00
|
|
|
"github.com/grafana/grafana/pkg/web"
|
2020-10-14 18:48:48 +08:00
|
|
|
)
|
|
|
|
|
2025-07-30 21:31:11 +08:00
|
|
|
func (hs *HTTPServer) registerShortURLAPI(apiRoute routing.RouteRegister) {
|
|
|
|
reqSignedIn := middleware.ReqSignedIn
|
2025-08-13 03:01:41 +08:00
|
|
|
if hs.Features.IsEnabledGlobally(featuremgmt.FlagKubernetesShortURLs) {
|
|
|
|
handler := newShortURLK8sHandler(hs)
|
|
|
|
apiRoute.Post("/api/short-urls", reqSignedIn, handler.createKubernetesShortURLsHandler)
|
|
|
|
apiRoute.Get("/api/short-urls/:uid", reqSignedIn, handler.getKubernetesShortURLsHandler)
|
|
|
|
apiRoute.Get("/goto/:uid", reqSignedIn, handler.getKubernetesRedirectFromShortURL, hs.Index)
|
|
|
|
} else {
|
|
|
|
apiRoute.Post("/api/short-urls", reqSignedIn, hs.createShortURL)
|
|
|
|
apiRoute.Get("/api/short-urls/:uid", reqSignedIn, hs.getShortURL)
|
|
|
|
apiRoute.Get("/goto/:uid", reqSignedIn, hs.redirectFromShortURL, hs.Index)
|
|
|
|
}
|
2025-07-30 21:31:11 +08:00
|
|
|
}
|
|
|
|
|
2020-10-14 18:48:48 +08:00
|
|
|
// createShortURL handles requests to create short URLs.
|
2023-01-27 15:50:36 +08:00
|
|
|
func (hs *HTTPServer) createShortURL(c *contextmodel.ReqContext) response.Response {
|
2025-08-13 03:01:41 +08:00
|
|
|
cmd := &dtos.CreateShortURLCmd{}
|
2021-11-29 17:18:01 +08:00
|
|
|
if err := web.Bind(c.Req, &cmd); err != nil {
|
2023-01-12 17:13:47 +08:00
|
|
|
return response.Err(shorturls.ErrShortURLBadRequest.Errorf("bad request data: %w", err))
|
2021-11-29 17:18:01 +08:00
|
|
|
}
|
2020-10-14 18:48:48 +08:00
|
|
|
hs.log.Debug("Received request to create short URL", "path", cmd.Path)
|
2025-08-13 03:01:41 +08:00
|
|
|
shortURL, err := hs.ShortURLService.CreateShortURL(c.Req.Context(), c.SignedInUser, cmd)
|
2020-10-14 18:48:48 +08:00
|
|
|
if err != nil {
|
2022-06-15 21:11:36 +08:00
|
|
|
return response.Err(err)
|
2020-10-14 18:48:48 +08:00
|
|
|
}
|
|
|
|
|
2025-08-13 03:01:41 +08:00
|
|
|
shortURLDTO := hs.ShortURLService.ConvertShortURLToDTO(shortURL, hs.Cfg.AppURL)
|
|
|
|
c.Logger.Debug("Created short URL", "url", shortURLDTO.URL)
|
2020-10-14 18:48:48 +08:00
|
|
|
|
2025-08-13 03:01:41 +08:00
|
|
|
return response.JSON(http.StatusOK, shortURLDTO)
|
2020-10-14 18:48:48 +08:00
|
|
|
}
|
|
|
|
|
2023-01-27 15:50:36 +08:00
|
|
|
func (hs *HTTPServer) redirectFromShortURL(c *contextmodel.ReqContext) {
|
2021-10-11 20:30:59 +08:00
|
|
|
shortURLUID := web.Params(c.Req)[":uid"]
|
2020-10-14 18:48:48 +08:00
|
|
|
|
|
|
|
if !util.IsValidShortUID(shortURLUID) {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
shortURL, err := hs.ShortURLService.GetShortURLByUID(c.Req.Context(), c.SignedInUser, shortURLUID)
|
|
|
|
if err != nil {
|
2024-12-03 23:32:53 +08:00
|
|
|
// If we didn't get the URL for whatever reason, we redirect to the
|
|
|
|
// main page, otherwise we get into an endless loops of redirects, as
|
|
|
|
// we would try to redirect again.
|
2023-01-12 17:13:47 +08:00
|
|
|
if shorturls.ErrShortURLNotFound.Is(err) {
|
2024-12-03 23:32:53 +08:00
|
|
|
hs.log.Debug("Not redirecting short URL since not found", "uid", shortURLUID)
|
2025-08-13 03:01:41 +08:00
|
|
|
c.Redirect(hs.Cfg.AppURL, http.StatusPermanentRedirect)
|
2020-10-14 18:48:48 +08:00
|
|
|
return
|
|
|
|
}
|
|
|
|
hs.log.Error("Short URL redirection error", "err", err)
|
2025-08-13 03:01:41 +08:00
|
|
|
c.Redirect(hs.Cfg.AppURL, http.StatusTemporaryRedirect)
|
2020-10-14 18:48:48 +08:00
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2020-10-27 23:16:06 +08:00
|
|
|
// Failure to update LastSeenAt should still allow to redirect
|
|
|
|
if err := hs.ShortURLService.UpdateLastSeenAt(c.Req.Context(), shortURL); err != nil {
|
|
|
|
hs.log.Error("Failed to update short URL last seen at", "error", err)
|
|
|
|
}
|
|
|
|
|
2020-10-14 18:48:48 +08:00
|
|
|
hs.log.Debug("Redirecting short URL", "path", shortURL.Path)
|
2025-08-13 03:01:41 +08:00
|
|
|
c.Redirect(setting.ToAbsUrl(shortURL.Path), http.StatusFound)
|
|
|
|
}
|
|
|
|
|
|
|
|
// getShortURL handles requests to get short URLs.
|
|
|
|
func (hs *HTTPServer) getShortURL(c *contextmodel.ReqContext) response.Response {
|
|
|
|
shortURLUID := web.Params(c.Req)[":uid"]
|
|
|
|
|
|
|
|
if !util.IsValidShortUID(shortURLUID) {
|
|
|
|
return response.Err(shorturls.ErrShortURLBadRequest.Errorf("invalid uid"))
|
|
|
|
}
|
|
|
|
|
|
|
|
shortURL, err := hs.ShortURLService.GetShortURLByUID(c.Req.Context(), c.SignedInUser, shortURLUID)
|
|
|
|
if err != nil {
|
|
|
|
if shorturls.ErrShortURLNotFound.Is(err) {
|
|
|
|
return response.Err(shorturls.ErrShortURLNotFound.Errorf("shorturl not found: %w", err))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return response.JSON(http.StatusOK, shortURL)
|
|
|
|
}
|
|
|
|
|
|
|
|
type shortURLK8sHandler struct {
|
|
|
|
namespacer request.NamespaceMapper
|
|
|
|
gvr schema.GroupVersionResource
|
|
|
|
clientConfigProvider grafanaapiserver.DirectRestConfigProvider
|
|
|
|
cfg *setting.Cfg
|
|
|
|
}
|
|
|
|
|
|
|
|
func newShortURLK8sHandler(hs *HTTPServer) *shortURLK8sHandler {
|
|
|
|
gvr := schema.GroupVersionResource{
|
|
|
|
Group: v1alpha1.ShortURLKind().Group(),
|
|
|
|
Version: v1alpha1.ShortURLKind().Version(),
|
|
|
|
Resource: v1alpha1.ShortURLKind().Plural(),
|
|
|
|
}
|
|
|
|
return &shortURLK8sHandler{
|
|
|
|
gvr: gvr,
|
|
|
|
namespacer: request.GetNamespaceMapper(hs.Cfg),
|
|
|
|
clientConfigProvider: hs.clientConfigProvider,
|
|
|
|
cfg: hs.Cfg,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func (sk8s *shortURLK8sHandler) getKubernetesShortURLsHandler(c *contextmodel.ReqContext) {
|
|
|
|
client, ok := sk8s.getClient(c)
|
|
|
|
if !ok {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
shortURLUID := web.Params(c.Req)[":uid"]
|
|
|
|
if !util.IsValidShortUID(shortURLUID) {
|
|
|
|
c.JsonApiErr(http.StatusBadRequest, "Invalid short URL UID format", fmt.Errorf("invalid short URL UID: %s", shortURLUID))
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
c.Logger.Debug("Fetching short URL", "uid", shortURLUID)
|
|
|
|
out, err := client.Get(c.Req.Context(), shortURLUID, v1.GetOptions{})
|
|
|
|
if err != nil {
|
|
|
|
sk8s.writeError(c, err)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
c.JSON(http.StatusOK, shorturl.UnstructuredToLegacyShortURL(*out))
|
|
|
|
}
|
|
|
|
|
|
|
|
func (sk8s *shortURLK8sHandler) getKubernetesRedirectFromShortURL(c *contextmodel.ReqContext) {
|
|
|
|
client, ok := sk8s.getClient(c)
|
|
|
|
if !ok {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
shortURLUID := web.Params(c.Req)[":uid"]
|
|
|
|
if !util.IsValidShortUID(shortURLUID) {
|
|
|
|
c.Logger.Warn("Invalid short URL UID format", "uid", shortURLUID)
|
|
|
|
c.Redirect(sk8s.cfg.AppURL, http.StatusFound)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
// Get Object
|
|
|
|
obj, err := client.Get(c.Req.Context(), shortURLUID, v1.GetOptions{})
|
|
|
|
if err != nil {
|
|
|
|
sk8s.writeError(c, err)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
// Modify status
|
|
|
|
status := obj.Object["status"].(map[string]interface{})
|
|
|
|
newTimestamp := time.Now().Unix()
|
|
|
|
status["lastSeenAt"] = newTimestamp
|
|
|
|
|
|
|
|
// Try status subresource first (works in Mode 5), fallback to main resource (works in Mode 0)
|
|
|
|
out, err := client.Update(c.Req.Context(), obj, v1.UpdateOptions{}, "status")
|
|
|
|
if err != nil {
|
|
|
|
c.Logger.Debug("Status subresource update failed, trying main resource", "error", err)
|
|
|
|
// Fallback to main resource update (for Mode 0)
|
|
|
|
out, err = client.Update(c.Req.Context(), obj, v1.UpdateOptions{})
|
|
|
|
if err != nil {
|
|
|
|
c.Logger.Error("Both status and main resource updates failed", "error", err)
|
|
|
|
sk8s.writeError(c, err)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
spec := out.Object["spec"].(map[string]any)
|
|
|
|
path := spec["path"].(string)
|
|
|
|
c.Logger.Debug("Redirecting short URL", "uid", shortURLUID, "path", path)
|
|
|
|
c.Redirect(setting.ToAbsUrl(path), http.StatusFound)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (sk8s *shortURLK8sHandler) createKubernetesShortURLsHandler(c *contextmodel.ReqContext) {
|
|
|
|
client, ok := sk8s.getClient(c)
|
|
|
|
if !ok {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
cmd := dtos.CreateShortURLCmd{}
|
|
|
|
if err := web.Bind(c.Req, &cmd); err != nil {
|
|
|
|
c.Logger.Error("Failed to bind request data", "error", err)
|
|
|
|
c.JsonApiErr(http.StatusBadRequest, "bad request data", err)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
c.Logger.Debug("Creating short URL", "path", cmd.Path)
|
|
|
|
obj := shorturl.LegacyCreateCommandToUnstructured(cmd)
|
|
|
|
|
|
|
|
uid, err := shortid.Generate()
|
|
|
|
if err != nil {
|
|
|
|
c.JsonApiErr(http.StatusInternalServerError, "failed to generate uid", err)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
obj.SetGenerateName(uid)
|
|
|
|
|
|
|
|
out, err := client.Create(c.Req.Context(), &obj, v1.CreateOptions{})
|
|
|
|
if err != nil {
|
|
|
|
c.Logger.Error("Failed to create short URL in Kubernetes", "path", cmd.Path, "error", err)
|
|
|
|
sk8s.writeError(c, err)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
c.Logger.Info("Successfully created short URL", "path", cmd.Path, "uid", out.GetName())
|
|
|
|
c.JSON(http.StatusOK, shorturl.UnstructuredToLegacyShortURLDTO(*out, sk8s.cfg.AppURL))
|
|
|
|
}
|
|
|
|
|
|
|
|
//-----------------------------------------------------------------------------------------
|
|
|
|
// Utility functions
|
|
|
|
//-----------------------------------------------------------------------------------------
|
|
|
|
|
|
|
|
func (sk8s *shortURLK8sHandler) getClient(c *contextmodel.ReqContext) (dynamic.ResourceInterface, bool) {
|
|
|
|
dyn, err := dynamic.NewForConfig(sk8s.clientConfigProvider.GetDirectRestConfig(c))
|
|
|
|
if err != nil {
|
|
|
|
c.JsonApiErr(500, "client", err)
|
|
|
|
return nil, false
|
|
|
|
}
|
|
|
|
return dyn.Resource(sk8s.gvr).Namespace(sk8s.namespacer(c.OrgID)), true
|
|
|
|
}
|
|
|
|
|
|
|
|
func (sk8s *shortURLK8sHandler) writeError(c *contextmodel.ReqContext, err error) {
|
|
|
|
//nolint:errorlint
|
|
|
|
statusError, ok := err.(*errors.StatusError)
|
|
|
|
if ok {
|
|
|
|
c.JsonApiErr(int(statusError.Status().Code), statusError.Status().Message, err)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
errhttp.Write(c.Req.Context(), err, c.Resp)
|
2020-10-14 18:48:48 +08:00
|
|
|
}
|