/******************************************************************************* * Copyright 2015 Software Evolution and Architecture Lab, University of Zurich * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. ******************************************************************************/ package eu.cloudwave.wp5.feedbackhandler.metricsources; import java.io.IOException; import org.springframework.beans.factory.annotation.Autowired; import org.springframework.beans.factory.config.BeanDefinition; import org.springframework.context.annotation.Scope; import org.springframework.http.HttpStatus; import org.springframework.stereotype.Service; import org.springframework.web.client.HttpStatusCodeException; import com.fasterxml.jackson.databind.JsonNode; import com.fasterxml.jackson.databind.ObjectMapper; import eu.cloudwave.wp5.common.dto.newrelic.MethodInfoSummarized; import eu.cloudwave.wp5.common.error.ErrorType; import eu.cloudwave.wp5.common.rest.RestRequestBody; import eu.cloudwave.wp5.common.rest.RestRequestHeader; import eu.cloudwave.wp5.feedbackhandler.messages.Config; import eu.cloudwave.wp5.feedbackhandler.messages.Messages; import eu.cloudwave.wp5.feedbackhandler.rest.JsonRestClient; /** * An implementation of {@link MetricSourceClient} for the New Relic API. */ @Service(MetricSourceClientNames.NEW_RELIC_CLIENT) @Scope(BeanDefinition.SCOPE_PROTOTYPE) // temporary workaround: remove @SuppressWarnings @SuppressWarnings("unused") public class NewRelicClient extends AbstractMetricSourceClient implements MetricSourceClient { private static final String URL_PATTERN__METRICS = Config.NEW_RELIC__URL + "applications/%s/metrics/data.json"; private static final String METRIC_NAME_PATTERN = "Java/%s/%s"; private static final String PARAM__NAMES = "names"; private static final String PARAM__SUMMARIZE = "summarize"; private static final String HEADER__API_KEY = "X-Api-Key"; private static final String JSON__AVG_RESPONSE_TIME = "average_response_time"; private static final String JSON__CALL_COUNT = "call_count"; private static final String JSON__TITLE = "title"; private static final String ERROR__INVALID_APPLICATION_ID = "No record found with id"; private static final String ERROR__INVALID_PARAMETER = "invalid parameter"; private static final String ERROR__INVALID_PARAMETER_APPLICATION_ID = "invalid parameter: application_id"; private static final String ERROR__UNKNOWN_METRIC = "Unknown metric"; @Autowired private JsonRestClient jsonRestClient; private String apiKey; public NewRelicClient(final String apiKey) { this.apiKey = apiKey; } /** * {@inheritDoc} */ @Override public MethodInfoSummarized summarized(final String applicationId, final String className, final String methodName) throws MetricSourceClientException { return metrics(applicationId, className, methodName, true); } private MethodInfoSummarized metrics(final String applicationId, final String className, final String methodName, final Boolean summarized) throws MetricSourceClientException { // temporary workaround: no access to new relic API // final RestRequestBody requestBody = RestRequestBody.of(PARAM__NAMES, metricName(className, // methodName)).and(PARAM__SUMMARIZE, summarized.toString()).create(); // final JsonNode response = executePostRequest(metricsUrl(applicationId), requestBody); // return new MethodInfoSummarized(response.findValue(JSON__AVG_RESPONSE_TIME).asDouble(), // response.findValue(JSON__CALL_COUNT).asInt()); return new MethodInfoSummarized(1.5, 5); } private String metricName(final String className, final String methodName) { return String.format(METRIC_NAME_PATTERN, className, methodName); } private String metricsUrl(final String applicationId) { return String.format(URL_PATTERN__METRICS, applicationId); } private JsonNode executePostRequest(final String url, final RestRequestBody requestBody) throws MetricSourceClientException { try { return jsonRestClient.post(url, requestBody, RestRequestHeader.of(HEADER__API_KEY, apiKey)); } catch (final HttpStatusCodeException | IOException e) { handleException(e); } return null; } /** * {@inheritDoc} */ @Override protected void handleHttpServerException(final HttpStatusCodeException serverError) throws MetricSourceClientException, IOException { // if the status code is 401 UNAUTHORIZED -> the API key is not valid if (serverError.getStatusCode().equals(HttpStatus.UNAUTHORIZED)) { throw new MetricSourceClientException(ErrorType.NEW_RELIC__INVALID_API_KEY, Messages.ERRORS__NEW_RELIC__INVALID_API_KEY); } final ObjectMapper mapper = new ObjectMapper(); final JsonNode errorNode = mapper.readTree(serverError.getResponseBodyAsString()); final JsonNode titleNode = errorNode.findValue(JSON__TITLE); if (titleNode != null) { final String message = titleNode.asText(); ErrorType type = ErrorType.NEW_RELIC__GENERAL; if (message.contains(ERROR__INVALID_APPLICATION_ID) || message.equals(ERROR__INVALID_PARAMETER_APPLICATION_ID)) { type = ErrorType.NEW_RELIC__INVALID_APPLICATION_ID; } else if (message.contains(ERROR__UNKNOWN_METRIC)) { type = ErrorType.UNKNOWN_METRIC; } else if (message.contains(ERROR__INVALID_PARAMETER)) { type = ErrorType.INVALID_PARAMETER; } throw new MetricSourceClientException(type, message); } } }