diff --git a/src/autoscaler/api/publicapiserver/public_api_handler.go b/src/autoscaler/api/publicapiserver/public_api_handler.go index 1b35cdb2da..d1b3db0d19 100644 --- a/src/autoscaler/api/publicapiserver/public_api_handler.go +++ b/src/autoscaler/api/publicapiserver/public_api_handler.go @@ -17,12 +17,11 @@ import ( "code.cloudfoundry.org/app-autoscaler/src/autoscaler/cred_helper" "code.cloudfoundry.org/app-autoscaler/src/autoscaler/db" "code.cloudfoundry.org/app-autoscaler/src/autoscaler/helpers" + "code.cloudfoundry.org/app-autoscaler/src/autoscaler/helpers/handlers" "code.cloudfoundry.org/app-autoscaler/src/autoscaler/models" "code.cloudfoundry.org/app-autoscaler/src/autoscaler/routes" - "github.com/google/uuid" - - "code.cloudfoundry.org/app-autoscaler/src/autoscaler/helpers/handlers" "code.cloudfoundry.org/lager/v3" + "github.com/google/uuid" ) type PublicApiHandler struct { @@ -55,26 +54,31 @@ func NewPublicApiHandler(logger lager.Logger, conf *config.Config, policydb db.P policydb: policydb, bindingdb: bindingdb, eventGeneratorClient: egClient, - policyValidator: policyvalidator.NewPolicyValidator( - conf.PolicySchemaPath, - conf.ScalingRules.CPU.LowerThreshold, - conf.ScalingRules.CPU.UpperThreshold, - conf.ScalingRules.CPUUtil.LowerThreshold, - conf.ScalingRules.CPUUtil.UpperThreshold, - conf.ScalingRules.DiskUtil.LowerThreshold, - conf.ScalingRules.DiskUtil.UpperThreshold, - conf.ScalingRules.Disk.LowerThreshold, - conf.ScalingRules.Disk.UpperThreshold, - ), - schedulerUtil: schedulerclient.New(conf, logger), - credentials: credentials, + policyValidator: createPolicyValidator(conf), + schedulerUtil: schedulerclient.New(conf, logger), + credentials: credentials, } } +func createPolicyValidator(conf *config.Config) *policyvalidator.PolicyValidator { + return policyvalidator.NewPolicyValidator( + conf.PolicySchemaPath, + conf.ScalingRules.CPU.LowerThreshold, + conf.ScalingRules.CPU.UpperThreshold, + conf.ScalingRules.CPUUtil.LowerThreshold, + conf.ScalingRules.CPUUtil.UpperThreshold, + conf.ScalingRules.DiskUtil.LowerThreshold, + conf.ScalingRules.DiskUtil.UpperThreshold, + conf.ScalingRules.Disk.LowerThreshold, + conf.ScalingRules.Disk.UpperThreshold, + ) +} + func writeErrorResponse(w http.ResponseWriter, statusCode int, message string) { handlers.WriteJSONResponse(w, statusCode, models.ErrorResponse{ Code: http.StatusText(statusCode), - Message: message}) + Message: message, + }) } func (h *PublicApiHandler) GetScalingPolicy(w http.ResponseWriter, r *http.Request, vars map[string]string) { @@ -84,6 +88,7 @@ func (h *PublicApiHandler) GetScalingPolicy(w http.ResponseWriter, r *http.Reque writeErrorResponse(w, http.StatusBadRequest, ErrorMessageAppidIsRequired) return } + logger := h.logger.Session("GetScalingPolicy", lager.Data{"appId": appId}) logger.Info("Get Scaling Policy") @@ -129,15 +134,14 @@ func (h *PublicApiHandler) AttachScalingPolicy(w http.ResponseWriter, r *http.Re } policyGuid := uuid.NewString() - err = h.policydb.SaveAppPolicy(r.Context(), appId, policy, policyGuid) - if err != nil { + if err := h.policydb.SaveAppPolicy(r.Context(), appId, policy, policyGuid); err != nil { logger.Error("Failed to save policy", err) writeErrorResponse(w, http.StatusInternalServerError, "Error saving policy") return } + h.logger.Info("creating/updating schedules", lager.Data{"policy": policy}) - err = h.schedulerUtil.CreateOrUpdateSchedule(r.Context(), appId, policy, policyGuid) - if err != nil { + if err := h.schedulerUtil.CreateOrUpdateSchedule(r.Context(), appId, policy, policyGuid); err != nil { logger.Error("Failed to create/update schedule", err) writeErrorResponse(w, http.StatusInternalServerError, err.Error()) return @@ -151,7 +155,7 @@ func (h *PublicApiHandler) AttachScalingPolicy(w http.ResponseWriter, r *http.Re } _, err = w.Write(response) if err != nil { - logger.Error("Failed to write body", err) + h.logger.Error("Failed to write body", err) } } @@ -162,17 +166,18 @@ func (h *PublicApiHandler) DetachScalingPolicy(w http.ResponseWriter, r *http.Re writeErrorResponse(w, http.StatusBadRequest, ErrorMessageAppidIsRequired) return } + logger := h.logger.Session("DetachScalingPolicy", lager.Data{"appId": appId}) logger.Info("Deleting policy json", lager.Data{"appId": appId}) - err := h.policydb.DeletePolicy(r.Context(), appId) - if err != nil { + + if err := h.policydb.DeletePolicy(r.Context(), appId); err != nil { logger.Error("Failed to delete policy from database", err) writeErrorResponse(w, http.StatusInternalServerError, "Error deleting policy") return } + logger.Info("Deleting schedules") - err = h.schedulerUtil.DeleteSchedule(r.Context(), appId) - if err != nil { + if err := h.schedulerUtil.DeleteSchedule(r.Context(), appId); err != nil { logger.Error("Failed to delete schedule", err) writeErrorResponse(w, http.StatusInternalServerError, "Error deleting schedules") return @@ -181,51 +186,54 @@ func (h *PublicApiHandler) DetachScalingPolicy(w http.ResponseWriter, r *http.Re if h.bindingdb != nil && !reflect.ValueOf(h.bindingdb).IsNil() { //TODO this is a copy of part of the attach ... this should use a common function. // brokered offering: check if there's a default policy that could apply - serviceInstance, err := h.bindingdb.GetServiceInstanceByAppId(appId) - if err != nil { - logger.Error("Failed to find service instance for app", err) - writeErrorResponse(w, http.StatusInternalServerError, "Error retrieving service instance") - return - } - if serviceInstance.DefaultPolicy != "" { - policyStr := serviceInstance.DefaultPolicy - policyGuidStr := serviceInstance.DefaultPolicyGuid - logger.Info("saving default policy json for app", lager.Data{"policy": policyStr}) - var policy *models.ScalingPolicy - err := json.Unmarshal([]byte(policyStr), &policy) - if err != nil { - h.logger.Error("default policy invalid", err, lager.Data{"appId": appId, "policy": policyStr}) - writeErrorResponse(w, http.StatusInternalServerError, "Default policy not valid") - return - } - - err = h.policydb.SaveAppPolicy(r.Context(), appId, policy, policyGuidStr) - if err != nil { - logger.Error("failed to save policy", err, lager.Data{"policy": policyStr}) - writeErrorResponse(w, http.StatusInternalServerError, "Error attaching the default policy") - return - } - - logger.Info("creating/updating schedules", lager.Data{"policy": policyStr}) - err = h.schedulerUtil.CreateOrUpdateSchedule(r.Context(), appId, policy, policyGuidStr) - //while there is synchronization between policy and schedule, so creating schedule error does not break - //the whole creating binding process - if err != nil { - logger.Error("failed to create/update schedules", err, lager.Data{"policy": policyStr}) - writeErrorResponse(w, http.StatusInternalServerError, fmt.Sprintf("Failed to update schedule:%s", err)) - } - } - } - // find via the app id the binding -> service instance - // default policy? then apply that + h.handleDefaultPolicy(w, r, logger, appId) + } w.WriteHeader(http.StatusOK) - _, err = w.Write([]byte("{}")) + _, err := w.Write([]byte("{}")) if err != nil { logger.Error(ActionWriteBody, err) } } +func (h *PublicApiHandler) handleDefaultPolicy(w http.ResponseWriter, r *http.Request, logger lager.Logger, appId string) { + serviceInstance, err := h.bindingdb.GetServiceInstanceByAppId(appId) + if err != nil { + logger.Error("Failed to find service instance for app", err) + writeErrorResponse(w, http.StatusInternalServerError, "Error retrieving service instance") + return + } + + if serviceInstance.DefaultPolicy != "" { + h.saveDefaultPolicy(w, r, logger, appId, serviceInstance) + } +} + +func (h *PublicApiHandler) saveDefaultPolicy(w http.ResponseWriter, r *http.Request, logger lager.Logger, appId string, serviceInstance *models.ServiceInstance) { + policyStr := serviceInstance.DefaultPolicy + policyGuidStr := serviceInstance.DefaultPolicyGuid + logger.Info("saving default policy json for app", lager.Data{"policy": policyStr}) + + var policy *models.ScalingPolicy + if err := json.Unmarshal([]byte(policyStr), &policy); err != nil { + h.logger.Error("default policy invalid", err, lager.Data{"appId": appId, "policy": policyStr}) + writeErrorResponse(w, http.StatusInternalServerError, "Default policy not valid") + return + } + + if err := h.policydb.SaveAppPolicy(r.Context(), appId, policy, policyGuidStr); err != nil { + logger.Error("failed to save policy", err, lager.Data{"policy": policyStr}) + writeErrorResponse(w, http.StatusInternalServerError, "Error attaching the default policy") + return + } + + logger.Info("creating/updating schedules", lager.Data{"policy": policyStr}) + if err := h.schedulerUtil.CreateOrUpdateSchedule(r.Context(), appId, policy, policyGuidStr); err != nil { + logger.Error("failed to create/update schedules", err, lager.Data{"policy": policyStr}) + writeErrorResponse(w, http.StatusInternalServerError, fmt.Sprintf("Failed to update schedule:%s", err)) + } +} + func (h *PublicApiHandler) proxyRequest(logger lager.Logger, appId string, metricType string, w http.ResponseWriter, req *http.Request, parameters *url.Values, requestDescription string) { reqUrl := req.URL r := routes.NewRouter() @@ -242,26 +250,13 @@ func (h *PublicApiHandler) proxyRequest(logger lager.Logger, appId string, metri } aUrl := h.conf.EventGenerator.EventGeneratorUrl + path.RequestURI() + "?" + parameters.Encode() - req, err = http.NewRequest("GET", aUrl, nil) if h.conf.CfInstanceCert != "" { - certPEM := []byte(h.conf.CfInstanceCert) - - // Calculate SHA-256 hash of the certificate - hash := sha256.Sum256(certPEM) - - // URL encode the PEM certificate - encodedCert := url.QueryEscape(string(certPEM)) - - // Construct the XFCC header value - xfccHeader := fmt.Sprintf("Hash=%x;Cert=\"%s\"", hash, encodedCert) - - req.Header.Set("X-Forwarded-Client-Cert", xfccHeader) + h.setXForwardedClientCertHeader(req) } resp, err := h.eventGeneratorClient.Do(req) - if err != nil { logger.Error("Failed to retrieve "+requestDescription, err, lager.Data{"url": aUrl}) writeErrorResponse(w, http.StatusInternalServerError, "Error retrieving "+requestDescription) @@ -281,6 +276,7 @@ func (h *PublicApiHandler) proxyRequest(logger lager.Logger, appId string, metri writeErrorResponse(w, resp.StatusCode, string(responseData)) return } + paginatedResponse, err := paginateResource(responseData, parameters, reqUrl) if err != nil { handlers.WriteJSONResponse(w, http.StatusInternalServerError, err.Error()) @@ -290,6 +286,14 @@ func (h *PublicApiHandler) proxyRequest(logger lager.Logger, appId string, metri handlers.WriteJSONResponse(w, resp.StatusCode, paginatedResponse) } +func (h *PublicApiHandler) setXForwardedClientCertHeader(req *http.Request) { + certPEM := []byte(h.conf.CfInstanceCert) + hash := sha256.Sum256(certPEM) + encodedCert := url.QueryEscape(string(certPEM)) + xfccHeader := fmt.Sprintf("Hash=%x;Cert=\"%s\"", hash, encodedCert) + req.Header.Set("X-Forwarded-Client-Cert", xfccHeader) +} + func (h *PublicApiHandler) GetAggregatedMetricsHistories(w http.ResponseWriter, req *http.Request, vars map[string]string) { appId := vars["appId"] metricType := vars["metricType"] @@ -309,7 +313,6 @@ func (h *PublicApiHandler) GetAggregatedMetricsHistories(w http.ResponseWriter, } h.proxyRequest(logger, appId, metricType, w, req, parameters, "metrics history from eventgenerator") - //proxyRequest(pathFn, h.eventGeneratorClient, w, req.URL, parameters, "metrics history from eventgenerator", logger) } func (h *PublicApiHandler) GetApiInfo(w http.ResponseWriter, _ *http.Request, _ map[string]string) { diff --git a/src/autoscaler/api/publicapiserver/public_api_handler_test.go b/src/autoscaler/api/publicapiserver/public_api_handler_test.go index 62a7bc957c..0f383b2f6e 100644 --- a/src/autoscaler/api/publicapiserver/public_api_handler_test.go +++ b/src/autoscaler/api/publicapiserver/public_api_handler_test.go @@ -378,8 +378,9 @@ var _ = Describe("PublicApiHandler", func() { BeforeEach(func() { policydb.SaveAppPolicyReturns(fmt.Errorf("failed to save new (default) policy")) }) - It("should error", func() { + FIt("should error", func() { Expect(resp.Code).To(Equal(http.StatusInternalServerError)) + fmt.Println("BANANA", resp.Body.String()) Expect(resp.Body.String()).To(Equal(`{"code":"Internal Server Error","message":"Error attaching the default policy"}`)) }) })