Partager via


Remplacements d’échantillonnage - Azure Monitor Application Insights pour Java

Remarque

La fonctionnalité de remplacements d’échantillonnage est en disponibilité générale, à partir de la version 3.5.0.

Les remplacements d’échantillonnage vous permettent de remplacer le pourcentage d’échantillonnage par défaut, par exemple :

  • Définissez le pourcentage d’échantillonnage sur 0 (ou une valeur faible) pour les contrôles d’intégrité bruyants.
  • Définissez le pourcentage d’échantillonnage sur 0 (ou une valeur faible) pour les appels de dépendance bruyants.
  • Définissez le pourcentage d’échantillonnage sur 100 pour un type de requête important (par exemple, /login) même si vous avez configuré l’échantillonnage par défaut sur une valeur inférieure.

Terminologie

Avant d’en apprendre davantage sur les remplacements d’échantillonnage, vous devez comprendre le terme étendue. Une étendue est un terme général désignant l’un de ces trois éléments :

  • Une requête entrante.
  • Une dépendance sortante (par exemple, un appel distant à un autre service).
  • Une dépendance « in-process » (par exemple, une tâche effectuée par les sous-composants du service).

Pour les remplacements d’échantillonnage, les composantes d’étendue suivants sont importants :

  • Attributs

Les attributs d’étendue représentent à la fois les propriétés standard et personnalisées d’une requête ou d’une dépendance donnée.

Prise en main

Pour commencer, créez un fichier de configuration nommé applicationinsights.json. Enregistrez-le dans le même répertoire que le fichier applicationinsights-agent-*.jar. Utilisez le modèle suivant.

{
  "connectionString": "...",
  "sampling": {
    "percentage": 10,
    "overrides": [
      {
        "telemetryType": "request",
        "attributes": [
          ...
        ],
        "percentage": 0
      },
      {
        "telemetryType": "request",
        "attributes": [
          ...
        ],
        "percentage": 100
      }
    ]
  }
}

Fonctionnement

telemetryType(telemetryKind dans Application Insights 3.4.0) doit être l’un de request, dependency, trace (journal) ou exception.

Au démarrage d’une étendue, le type d’étendue et les attributs présents dessus sont utilisés pour vérifier si l’un des remplacements d’échantillonnage correspond.

Les correspondances peuvent être strict ou regexp. Les correspondances des expressions régulières sont effectuées par rapport à la valeur entière de l’attribut. Par conséquent, si vous souhaitez faire correspondre une valeur contenant abc n’importe où dans celle-ci, vous devez utiliser .*abc.*. Un remplacement d’échantillonnage peut spécifier plusieurs critères d’attributs, auquel cas tous doivent être satisfaits pour que le remplacement d’échantillonnage soit en correspondance.

Si un des remplacements d’échantillonnage est en correspondance, son pourcentage d’échantillonnage est utilisé pour décider d’échantillonner ou non l’étendue.

Seul le premier remplacement d’échantillonnage correspondant est utilisé.

Si aucun remplacement d’échantillonnage ne correspond :

Attributs d’étendue disponibles pour l’échantillonnage

Les attributs d’étendue OpenTelemetry sont collectés automatiquement et basés sur les conventions sémantiques OpenTelemetry.

Vous pouvez ajouter par programme des attributs d’étendue et les utiliser pour l’échantillonnage.

Remarque

Pour voir l’ensemble exact d’attributs capturés par Application Insights Java pour votre application, définissez le niveau auto-diagnostics sur débogage, et recherchez les messages de débogage commençant par le texte « exporting span ».

Remarque

Seuls les attributs définis au début de l’étendue sont disponibles pour l’échantillonnage. Par conséquent, les attributs, tels que http.response.status_code ou la durée de la requête, qui sont capturés ultérieurement, peuvent être filtrés via les extensions OpenTelemetry Java. Voici un exemple d’extension qui filtre les étendues en fonction de la durée de la requête.

Remarque

Les attributs ajoutés avec un processeur de télémétrie ne sont pas disponibles pour l’échantillonnage.

Cas d’utilisation

Supprimer la collecte de télémétrie pour les contrôles d’intégrité

Cet exemple supprime la collecte de télémétrie pour toutes les requêtes de /health-checks.

Cet exemple supprimera également la collecte des étendues en aval (dépendances) qui seraient normalement collectées sous /health-checks.

{
  "connectionString": "...",
  "sampling": {
    "overrides": [
      {
        "telemetryType": "request",
        "attributes": [
          {
            "key": "url.path",
            "value": "/health-check",
            "matchType": "strict"
          }
        ],
        "percentage": 0
      }
    ]
  }
}

Supprimer la collecte de télémétrie pour les appels de dépendance bruyants

Cet exemple supprime la collecte de télémétrie pour tous les appels redis de GET my-noisy-key.

{
  "connectionString": "...",
  "sampling": {
    "overrides": [
      {
        "telemetryType": "dependency",
        "attributes": [
          {
            "key": "db.system",
            "value": "redis",
            "matchType": "strict"
          },
          {
            "key": "db.statement",
            "value": "GET my-noisy-key",
            "matchType": "strict"
          }
        ],
        "percentage": 0
      }
    ]
  }
}

Collecter 100 % de la télémétrie pour un type de requête important

Cet exemple recueille 100 % des données télémétriques pour /login.

Étant donné que les étendues en aval (dépendances) respectent la décision d’échantillonnage du parent (sans remplacement d’échantillonnage pour cette étendue en aval), elles sont également collectées pour toutes les demandes « /login ».

{
  "connectionString": "...",
  "sampling": {
    "percentage": 10
  },
  "sampling": {
    "overrides": [
      {
        "telemetryType": "request",
        "attributes": [
          {
            "key": "url.path",
            "value": "/login",
            "matchType": "strict"
          }
        ],
        "percentage": 100
      }
    ]
  }
}

Exposition d’attributs d’étendue pour supprimer des appels de dépendance SQL

Cet exemple décrit l’expérience de recherche d’attributs disponibles pour supprimer les appels SQL bruyants. La requête ci-dessous illustre les différents appels SQL et le nombre d’enregistrements associés au cours des 30 derniers jours :

dependencies
| where timestamp > ago(30d)
| where name == 'SQL: DB Query'
| summarize count() by name, operation_Name, data
| sort by count_ desc
SQL: DB Query    POST /Order             DECLARE @MyVar varbinary(20); SET @MyVar = CONVERT(VARBINARY(20), 'Hello World');SET CONTEXT_INFO @MyVar;    36712549    
SQL: DB Query    POST /Receipt           DECLARE @MyVar varbinary(20); SET @MyVar = CONVERT(VARBINARY(20), 'Hello World');SET CONTEXT_INFO @MyVar;    2220248    
SQL: DB Query    POST /CheckOutForm      DECLARE @MyVar varbinary(20); SET @MyVar = CONVERT(VARBINARY(20), 'Hello World');SET CONTEXT_INFO @MyVar;    554074    
SQL: DB Query    GET /ClientInfo         DECLARE @MyVar varbinary(20); SET @MyVar = CONVERT(VARBINARY(20), 'Hello World');SET CONTEXT_INFO @MyVar;    37064

Dans les résultats ci-dessus, il peut être observé que toutes les opérations partagent la même valeur dans le champ data : DECLARE @MyVar varbinary(20); SET @MyVar = CONVERT(VARBINARY(20), 'Hello World');SET CONTEXT_INFO @MyVar;. Le point commun entre tous ces enregistrements en fait un bon candidat pour un remplacement d’échantillonnage.

En définissant l’autodiagnostic pour déboguer, les entrées de journal suivantes sont visibles dans la sortie :

2023-10-26 15:48:25.407-04:00 DEBUG c.m.a.a.i.exporter.AgentSpanExporter - exporting span: SpanData{spanContext=ImmutableSpanContext...

La zone d’intérêt de ces journaux est la section « attributs » :

{
  "attributes": {
    "data": {
      "thread.name": "DefaultDatabaseBroadcastTransport: MessageReader thread",
      "thread.id": 96,
      "db.connection_string": "apache:",
      "db.statement": "DECLARE @MyVar varbinary(20); SET @MyVar = CONVERT(VARBINARY(20), 'Hello World');SET CONTEXT_INFO @MyVar;",
      "db.system": "other_sql",
      "applicationinsights.internal.item_count": 1
    }
  }
}

À l’aide de cette sortie, vous pouvez configurer un remplacement d’échantillonnage similaire à celui ci-dessous qui filtrera nos appels SQL bruyants :

{
  "connectionString": "...",
  "preview": {
    "sampling": {
      "overrides": [
        {
          "telemetryType": "dependency",
          "attributes": [
            {
              "key": "db.statement",
              "value": "DECLARE @MyVar varbinary(20); SET @MyVar = CONVERT(VARBINARY(20), 'Hello World');SET CONTEXT_INFO @MyVar;",
              "matchType": "strict"
            }
          ],
          "percentage": 0
        }
      ]
    }
  }
}

Une fois les modifications appliquées, la requête suivante nous permet de déterminer la dernière fois que ces dépendances ont été ingérées dans Application Insights :

dependencies
| where timestamp > ago(30d)
| where data contains 'DECLARE @MyVar'
| summarize max(timestamp) by data
| sort by max_timestamp desc
DECLARE @MyVar varbinary(20); SET @MyVar = CONVERT(VARBINARY(20), 'Hello World');SET CONTEXT_INFO @MyVar;    11/13/2023 8:52:41 PM 

Supprimer la collecte de télémétrie pour les journaux

Avec SL4J, vous pouvez ajouter des attributs de journal :

import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.slf4j.MDC;

public class MdcClass {

  private static final Logger logger = LoggerFactory.getLogger(MdcClass.class);

  void method {
	
    MDC.put("key", "value");
    try {
       logger.info(...); // Application log to remove
    finally {
       MDC.remove("key"); // In a finally block in case an exception happens with logger.info
    }
	
  }
  
}

Vous pouvez ensuite supprimer le journal ayant l’attribut ajouté :

{
  "sampling": {
    "overrides": [
      {
        "telemetryType": "trace",
        "percentage": 0,
        "attributes": [
          {
            "key": "key",
            "value": "value",
            "matchType": "strict"
          }
        ]
      }
    ]
  }
}

Supprimer la collecte de télémétrie pour une méthode Java

Nous allons ajouter une étendue à une méthode Java et supprimer cette étendue avec un remplacement d’échantillonnage.

Ajoutons d’abord la dépendance opentelemetry-instrumentation-annotations :

    <dependency>
      <groupId>io.opentelemetry.instrumentation</groupId>
      <artifactId>opentelemetry-instrumentation-annotations</artifactId>
    </dependency>

Nous pouvons maintenant ajouter l’annotation WithSpan à une méthode Java exécutant des requêtes SQL :

package org.springframework.samples.petclinic.vet;

@Controller
class VetController {

	private final VetRepository vetRepository;

	public VetController(VetRepository vetRepository) {
		this.vetRepository = vetRepository;
	}

	@GetMapping("/vets.html")
	public String showVetList(@RequestParam(defaultValue = "1") int page, Model model) {
		Vets vets = new Vets();
		Page<Vet> paginated = findPaginated(page);
		vets.getVetList().addAll(paginated.toList());
		return addPaginationModel(page, paginated, model);
	}

	@WithSpan
	private Page<Vet> findPaginated(int page) {
		int pageSize = 5;
		Pageable pageable = PageRequest.of(page - 1, pageSize);
		return vetRepository.findAll(pageable);  // Execution of SQL requests
	}

La configuration de remplacement d’échantillonnage suivante vous permet de supprimer l’étendue ajoutée par l’annotation WithSpan :

  "sampling": {
    "overrides": [
      {
        "telemetryType": "dependency",
        "attributes": [
          {
            "key": "code.function",
            "value": "findPaginated",
            "matchType": "strict"
          }
        ],
        "percentage": 0
      }
    ]
  }

La valeur d’attribut est le nom de la méthode Java.

Cette configuration supprime toutes les données de télémétrie créées à partir de la méthode findPaginated. Les dépendances SQL ne sont pas créées pour les exécutions SQL provenant de la méthode findPaginated.

La configuration suivante supprime toutes les données de télémétrie émises à partir de méthodes de la classe VetController ayant l’annotation WithSpan :

 "sampling": {
    "overrides": [
      {
        "telemetryType": "dependency",
        "attributes": [
          {
            "key": "code.namespace",
            "value": "org.springframework.samples.petclinic.vet.VetController",
            "matchType": "strict"
          }
        ],
        "percentage": 0
      }
    ]
  }

Dépannage

Si vous utilisez regexp et que le remplacement d’échantillonnage ne fonctionne pas, essayez avec .* regex. Si l’échantillonnage fonctionne maintenant, cela signifie que vous rencontrez un problème avec le premier regex et lisez cette documentation regex.

S’il ne fonctionne pas avec .*, il se peut que vous ayez un problème de syntaxe dans votre application-insights.json file. Consultez les journaux Application Insights pour voir si vous remarquez des messages d’avertissement.