package authz import ( "context" "net/http" "github.com/porter-dev/porter/api/server/authz/policy" "github.com/porter-dev/porter/api/server/shared" "github.com/porter-dev/porter/api/server/shared/apierrors" "github.com/porter-dev/porter/api/types" "github.com/porter-dev/porter/internal/models" "helm.sh/helm/v3/pkg/release" ) type ReleaseScopedFactory struct { config *shared.Config } func NewReleaseScopedFactory( config *shared.Config, ) *ReleaseScopedFactory { return &ReleaseScopedFactory{config} } func (p *ReleaseScopedFactory) Middleware(next http.Handler) http.Handler { return &ReleaseScopedMiddleware{next, p.config, NewOutOfClusterAgentGetter(p.config)} } type ReleaseScopedMiddleware struct { next http.Handler config *shared.Config agentGetter KubernetesAgentGetter } func (p *ReleaseScopedMiddleware) ServeHTTP(w http.ResponseWriter, r *http.Request) { cluster, _ := r.Context().Value(types.ClusterScope).(*models.Cluster) helmAgent, err := p.agentGetter.GetHelmAgent(r, cluster) if err != nil { apierrors.HandleAPIError(w, p.config.Logger, apierrors.NewErrInternal(err)) return } // get the name of the application reqScopes, _ := r.Context().Value(RequestScopeCtxKey).(map[types.PermissionScope]*policy.RequestAction) name := reqScopes[types.ReleaseScope].Resource.Name // get the version for the application version, _ := GetURLParamUint(r, string(types.URLParamReleaseVersion)) release, err := helmAgent.GetRelease(name, int(version)) if err != nil { apierrors.HandleAPIError(w, p.config.Logger, apierrors.NewErrInternal(err)) return } ctx := NewReleaseContext(r.Context(), release) r = r.WithContext(ctx) p.next.ServeHTTP(w, r) } func NewReleaseContext(ctx context.Context, helmRelease *release.Release) context.Context { return context.WithValue(ctx, types.ReleaseScope, helmRelease) }