All Downloads are FREE. Search and download functionalities are using the official Maven repository.

io.helidon.metrics.api.MetricsConfigBlueprint Maven / Gradle / Ivy

There is a newer version: 4.1.4
Show newest version
/*
 * Copyright (c) 2023, 2024 Oracle and/or its affiliates.
 *
 * 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 io.helidon.metrics.api;

import java.util.ArrayList;
import java.util.List;
import java.util.Map;
import java.util.Optional;
import java.util.TreeMap;
import java.util.regex.Matcher;

import io.helidon.builder.api.Option;
import io.helidon.builder.api.Prototype;
import io.helidon.common.config.Config;

/**
 * Configuration settings for metrics.
 *
 * 

Scope handling configuration

* Helidon allows developers to associate a scope with each meter. The {@value SCOPE_CONFIG_KEY} subsection of the * {@value METRICS_CONFIG_KEY} configuration controls *
    *
  • the default scope value to use if a meter is registered without an explicit scope setting, and
  • *
  • whether and how Helidon records each meter's scope as a tag in the underlying implementation meter registry. *

    * Specifically, users can specify whether scope tags are used at all and, if so, what tag name to use. *

  • *
*/ @Prototype.Configured(MetricsConfigBlueprint.METRICS_CONFIG_KEY) @Prototype.Blueprint(decorator = MetricsConfigSupport.BuilderDecorator.class) @Prototype.CustomMethods(MetricsConfigSupport.class) interface MetricsConfigBlueprint { /** * The config key containing settings for all of metrics. */ String METRICS_CONFIG_KEY = "metrics"; /** * Config key for scope-related settings. */ String SCOPE_CONFIG_KEY = "scoping"; /** * Config key for KPI metrics settings. */ String KEY_PERFORMANCE_INDICATORS_CONFIG_KEY = "key-performance-indicators"; @Prototype.FactoryMethod static List createTags(Config globalTagExpression) { return createTags(globalTagExpression.asString().get()); } static List createTags(String pairs) { // Use a TreeMap to order by tag name. Map result = new TreeMap<>(); List allErrors = new ArrayList<>(); String[] assignments = pairs.split("(? errorsForThisAssignment = new ArrayList<>(); if (assignment.isBlank()) { errorsForThisAssignment.add("empty assignment at position " + position + ": " + assignment); } else { // Pattern should yield group 1 = tag name and group 2 = tag value. Matcher matcher = MetricsConfigSupport.TAG_ASSIGNMENT_PATTERN.matcher(assignment); if (!matcher.matches()) { errorsForThisAssignment.add("expected tag=value but found '" + assignment + "'"); } else { String name = matcher.group(1); String value = matcher.group(2); if (name.isBlank()) { errorsForThisAssignment.add("missing tag name"); } if (value.isBlank()) { errorsForThisAssignment.add("missing tag value"); } if (!name.matches("[A-Za-z_][A-Za-z_0-9]*")) { errorsForThisAssignment.add( "tag name must start with a letter and include only letters, digits, and underscores"); } if (errorsForThisAssignment.isEmpty()) { result.put(name, // Do not use Tag.create in the next line. That would delegate to the MetricsFactoryManager // which, ultimately, might try to load config to set up the MetricFactory. But we are // already trying to load config and that would set up an infinite recursive loop. NoOpTag.create(name, value.replace("\\,", ",") .replace("\\=", "="))); } } } if (!errorsForThisAssignment.isEmpty()) { allErrors.add(String.format("Position %d with expression %s: %s", position, assignment, errorsForThisAssignment)); } position++; } if (!allErrors.isEmpty()) { throw new IllegalArgumentException("Error(s) in tag expression: " + allErrors); } return result.values() .stream() .toList(); } /** * Whether metrics functionality is enabled. * * @return if metrics are configured to be enabled */ @Option.Configured @Option.DefaultBoolean(true) boolean enabled(); /** * Whether to allow anybody to access the endpoint. * * @return whether to permit access to metrics endpoint to anybody, defaults to {@code true} * @see #roles() */ @Option.Configured @Option.DefaultBoolean(true) boolean permitAll(); /** * Hints for role names the user is expected to be in. * * @return list of hints */ @Option.Configured @Option.Default("observe") List roles(); /** * Key performance indicator metrics settings. * * @return key performance indicator metrics settings */ @Option.Configured(KEY_PERFORMANCE_INDICATORS_CONFIG_KEY) KeyPerformanceIndicatorMetricsConfig keyPerformanceIndicatorMetricsConfig(); /** * Global tags. * * @return name/value pairs for global tags */ @Option.Configured // for compatibility with MP metrics and earlier Helidon releases List tags(); /** * Value for the application tag to be added to each meter ID. * * @return application tag value */ @Option.Configured Optional appName(); /** * Name for the application tag to be added to each meter ID. * * @return application tag name */ @Option.Configured Optional appTagName(); /** * Settings related to scoping management. * * @return scoping settings */ @Option.Configured ScopingConfig scoping(); /** * Whether automatic REST request metrics should be measured. * * @return true/false */ @Option.Configured @Option.DefaultBoolean(false) boolean restRequestEnabled(); /** * Metrics configuration node. * * @return metrics configuration */ @Option.Redundant Config config(); /** * Whether the {@code gc.time} meter should be registered as a gauge (vs. a counter). * The {@code gc.time} meter is inspired by the MicroProfile Metrics spec, in which the meter was originally checked to * be a counter but starting in 5.1 was checked be a gauge. For the duration of Helidon 4.x users can choose which * type of meter Helidon registers for {@code gc.time}. * @return the type of meter to use for registering {@code gc.time} * @deprecated Provided for backward compatibility only; no replacement */ @Deprecated(since = "4.1", forRemoval = true) @Option.Configured @Option.Default("COUNTER") GcTimeType gcTimeType(); /** * Output format for built-in meter names. *

* {@link BuiltInMeterNameFormat#SNAKE} selects "snake_case" which does not conform to the MicroProfile * Metrics specification. * * @return the output format for built-in meter names */ @Option.Configured @Option.Default(BuiltInMeterNameFormat.DEFAULT) BuiltInMeterNameFormat builtInMeterNameFormat(); /** * Reports whether the specified scope is enabled, according to any scope configuration that * is part of this metrics configuration. * * @param scope scope name * @return true if the scope as a whole is enabled; false otherwise */ boolean isScopeEnabled(String scope); /** * Determines whether the meter with the specified name and within the indicated scope is enabled. * * @param name meter name * @param scope scope name * @return whether the meter is enabled */ boolean isMeterEnabled(String name, String scope); }





© 2015 - 2024 Weber Informatics LLC | Privacy Policy