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

org.glowroot.agent.config.ConfigFile Maven / Gradle / Ivy

There is a newer version: 0.14.0-beta.3
Show newest version
/*
 * Copyright 2013-2019 the original author or authors.
 *
 * 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 org.glowroot.agent.config;

import java.io.File;
import java.io.IOException;
import java.util.List;
import java.util.Map;

import org.glowroot.agent.shaded.com.fasterxml.jackson.core.type.TypeReference;
import org.glowroot.agent.shaded.com.fasterxml.jackson.databind.JsonNode;
import org.glowroot.agent.shaded.com.fasterxml.jackson.databind.ObjectMapper;
import org.glowroot.agent.shaded.com.fasterxml.jackson.databind.node.ObjectNode;
import org.glowroot.agent.shaded.org.glowroot.agent.it.harness.shaded.com.google.common.collect.ImmutableList;
import org.glowroot.agent.shaded.org.checkerframework.checker.nullness.qual.Nullable;
import org.glowroot.agent.shaded.org.glowroot.agent.shaded.org.slf4j.Logger;
import org.glowroot.agent.shaded.org.glowroot.agent.shaded.org.slf4j.LoggerFactory;

import org.glowroot.agent.shaded.org.glowroot.common.util.ObjectMappers;
import org.glowroot.agent.shaded.org.glowroot.common.util.OnlyUsedByTests;

// TODO if config.json file has unrecognized top-level node (something other than "transactions",
// "uiDefaults", "advanced", etc) then log warning and remove that node
class ConfigFile {

    private static final Logger logger = LoggerFactory.getLogger(ConfigFile.class);
    private static final ObjectMapper mapper = ObjectMappers.create();

    private static final List keyOrder =
            ImmutableList.of("transactions", "jvm", "uiDefaults", "advanced", "gauges",
                    "syntheticMonitors", "alerts", "plugins", "instrumentation");

    private final File file;
    private final ObjectNode rootObjectNode;
    private final boolean readOnly;

    ConfigFile(List confDirs, boolean readOnly) {
        file = new File(confDirs.get(0), "config.json");
        if (file.exists()) {
            rootObjectNode = readRootObjectNode(file);
        } else {
            File defaultFile = getConfigDefaultFile(confDirs);
            if (defaultFile == null) {
                rootObjectNode = mapper.createObjectNode();
            } else {
                rootObjectNode = readRootObjectNode(defaultFile);
            }
        }
        this.readOnly = readOnly;
    }

     /*@Nullable*/ T getConfig(String key, Class clazz) {
        return ConfigFileUtil.getConfig(rootObjectNode, key, clazz);
    }

     /*@Nullable*/ T getConfig(String key, TypeReference typeReference) {
        return ConfigFileUtil.getConfig(rootObjectNode, key, typeReference);
    }

    void writeConfig(String key, Object config) throws IOException {
        if (readOnly) {
            throw new IllegalStateException("Running with config.readOnly=true so config updates"
                    + " are not allowed");
        }
        rootObjectNode.replace(key, mapper.valueToTree(config));
        ConfigFileUtil.writeToFileIfNeeded(file, rootObjectNode, keyOrder, false);
    }

    void writeAllConfigs(Map configs) throws IOException {
        if (readOnly) {
            throw new IllegalStateException("Running with config.readOnly=true so config updates"
                    + " are not allowed");
        }
        for (Map.Entry entry : configs.entrySet()) {
            rootObjectNode.replace(entry.getKey(), mapper.valueToTree(entry.getValue()));
        }
        ConfigFileUtil.writeToFileIfNeeded(file, rootObjectNode, keyOrder, false);
    }

    void writeAllConfigsOnStartup(Map configs) throws IOException {
        for (Map.Entry entry : configs.entrySet()) {
            rootObjectNode.replace(entry.getKey(), mapper.valueToTree(entry.getValue()));
        }
        ConfigFileUtil.writeToFileIfNeeded(file, rootObjectNode, keyOrder, readOnly);
    }

    @OnlyUsedByTests
    void delete() throws IOException {
        if (!file.delete()) {
            throw new IOException("Could not delete file: " + file.getCanonicalPath());
        }
    }

    private static ObjectNode readRootObjectNode(File file) {
        ObjectNode rootObjectNode = ConfigFileUtil.getRootObjectNode(file);
        upgradeAlertsIfNeeded(rootObjectNode);
        upgradeUiIfNeeded(rootObjectNode);
        upgradeAdvancedIfNeeded(rootObjectNode);
        upgradePluginPropertiesIfNeeded(rootObjectNode);
        upgradeSlowThresholdOverrideIfNeeded(rootObjectNode);
        removeUserRecordingIfNeeded(rootObjectNode);
        return rootObjectNode;
    }

    private static @Nullable File getConfigDefaultFile(List confDirs) {
        for (File confDir : confDirs) {
            File defaultFile = new File(confDir, "config-default.json");
            if (defaultFile.exists()) {
                return defaultFile;
            }
        }
        return null;
    }

    private static void upgradeAlertsIfNeeded(ObjectNode rootObjectNode) {
        JsonNode alertsNode = rootObjectNode.get("alerts");
        if (alertsNode == null || !alertsNode.isArray()) {
            return;
        }
        for (JsonNode alertNode : alertsNode) {
            if (!(alertNode instanceof ObjectNode)) {
                continue;
            }
            ObjectNode alertObjectNode = (ObjectNode) alertNode;
            if (alertObjectNode.has("transactionThresholdMillis")) {
                // upgrade from 0.9.9 to 0.9.10
                alertObjectNode.set("thresholdMillis",
                        alertObjectNode.remove("transactionThresholdMillis"));
            }
            if (alertObjectNode.has("kind")) {
                // upgrade from 0.9.17 to 0.9.18
                String alertKind = alertObjectNode.remove("kind").asText();
                if (alertKind.equals("transaction")) {
                    ObjectNode conditionObjectNode = mapper.createObjectNode();
                    conditionObjectNode.put("conditionType", "metric");
                    conditionObjectNode.put("metric", "transaction:x-percentile");
                    conditionObjectNode.set("transactionType",
                            alertObjectNode.remove("transactionType"));
                    conditionObjectNode.set("percentile",
                            alertObjectNode.remove("transactionPercentile"));
                    conditionObjectNode.set("threshold", alertObjectNode.remove("thresholdMillis"));
                    conditionObjectNode.set("timePeriodSeconds",
                            alertObjectNode.remove("timePeriodSeconds"));
                    conditionObjectNode.set("minTransactionCount",
                            alertObjectNode.remove("minTransactionCount"));
                    alertObjectNode.set("condition", conditionObjectNode);
                } else if (alertKind.equals("gauge")) {
                    ObjectNode conditionObjectNode = mapper.createObjectNode();
                    conditionObjectNode.put("conditionType", "metric");
                    conditionObjectNode.put("metric",
                            "gauge:" + alertObjectNode.remove("gaugeName").asText());
                    conditionObjectNode.set("threshold", alertObjectNode.remove("gaugeThreshold"));
                    conditionObjectNode.set("timePeriodSeconds",
                            alertObjectNode.remove("timePeriodSeconds"));
                    alertObjectNode.set("condition", conditionObjectNode);
                } else {
                    logger.error("unexpected alert kind: {}", alertKind);
                }
                ObjectNode emailNotificationObjectNode = mapper.createObjectNode();
                emailNotificationObjectNode.set("emailAddresses",
                        alertObjectNode.remove("emailAddresses"));
                alertObjectNode.set("emailNotification", emailNotificationObjectNode);
            }
            if (!alertObjectNode.has("severity")) {
                // upgrade from 0.9.21 to 0.9.22
                alertObjectNode.put("severity", "critical");
            }
            if (alertObjectNode.has("condition")) {
                JsonNode conditionNode = alertObjectNode.get("condition");
                if (conditionNode instanceof ObjectNode) {
                    ObjectNode conditionObjectNode = (ObjectNode) conditionNode;
                    if (conditionObjectNode.path("conditionType").asText().equals(
                            "synthetic-monitor") && !conditionObjectNode.has("consecutiveCount")) {
                        // upgrade from 0.12.3 to 0.13.0
                        conditionObjectNode.put("consecutiveCount", 1);
                    }
                }
            }
        }
    }

    private static void upgradeUiIfNeeded(ObjectNode configRootObjectNode) {
        JsonNode uiNode = configRootObjectNode.get("ui");
        if (uiNode == null || !uiNode.isObject()) {
            return;
        }
        ObjectNode uiObjectNode = (ObjectNode) uiNode;
        if (uiObjectNode.has("defaultDisplayedTransactionType")) {
            // upgrade from 0.9.28 to 0.10.0
            uiObjectNode.set("defaultTransactionType",
                    uiObjectNode.remove("defaultDisplayedTransactionType"));
        }
        if (uiObjectNode.has("defaultDisplayedPercentiles")) {
            // upgrade from 0.9.28 to 0.10.0
            uiObjectNode.set("defaultPercentiles",
                    uiObjectNode.remove("defaultDisplayedPercentiles"));
        }
        // upgrade from 0.10.12 to 0.11.0
        configRootObjectNode.set("uiDefaults", configRootObjectNode.remove("ui"));
    }

    private static void upgradeAdvancedIfNeeded(ObjectNode rootObjectNode) {
        JsonNode advancedNode = rootObjectNode.get("advanced");
        if (advancedNode == null || !advancedNode.isObject()) {
            return;
        }
        ObjectNode advancedObjectNode = (ObjectNode) advancedNode;
        if (advancedObjectNode.has("maxAggregateTransactionsPerType")) {
            // upgrade from 0.10.5 to 0.10.6
            advancedObjectNode.set("maxTransactionAggregates",
                    advancedObjectNode.remove("maxAggregateTransactionsPerType"));
        }
        if (advancedObjectNode.has("maxAggregateQueriesPerType")) {
            // upgrade from 0.10.5 to 0.10.6
            advancedObjectNode.set("maxQueryAggregates",
                    advancedObjectNode.remove("maxAggregateQueriesPerType"));
        }
        if (advancedObjectNode.has("maxAggregateServiceCallsPerType")) {
            // upgrade from 0.10.5 to 0.10.6
            advancedObjectNode.set("maxServiceCallAggregates",
                    advancedObjectNode.remove("maxAggregateServiceCallsPerType"));
        }
        if (advancedObjectNode.has("maxStackTraceSamplesPerTransaction")) {
            // upgrade from 0.10.5 to 0.10.6
            advancedObjectNode.set("maxProfileSamplesPerTransaction",
                    advancedObjectNode.remove("maxStackTraceSamplesPerTransaction"));
        }
    }

    private static void upgradePluginPropertiesIfNeeded(ObjectNode rootObjectNode) {
        JsonNode pluginsNode = rootObjectNode.get("plugins");
        if (pluginsNode == null || !pluginsNode.isArray()) {
            return;
        }
        for (JsonNode pluginNode : pluginsNode) {
            if (!(pluginNode instanceof ObjectNode)) {
                continue;
            }
            ObjectNode pluginObjectNode = (ObjectNode) pluginNode;
            if (pluginObjectNode.path("id").asText().equals("jdbc")) {
                JsonNode propertiesNode = pluginObjectNode.get("properties");
                if (propertiesNode != null && propertiesNode.isObject()) {
                    upgradeJdbcPluginPropertiesIfNeeded((ObjectNode) propertiesNode);
                }
                // since no other plugin property upgrades at this point, can return now
                return;
            }
        }
    }

    private static void upgradeSlowThresholdOverrideIfNeeded(ObjectNode rootObjectNode) {
        JsonNode transactionsNode = rootObjectNode.get("transactions");
        if (transactionsNode == null || !transactionsNode.isObject()) {
            return;
        }
        ObjectNode transactionsObjectNode = (ObjectNode) transactionsNode;
        if (transactionsObjectNode.has("slowThresholds")) {
            // upgrade from 0.11.1 to 0.12.0
            transactionsObjectNode.set("slowThresholdOverrides",
                    transactionsObjectNode.remove("slowThresholds"));
        }
    }

    private static void removeUserRecordingIfNeeded(ObjectNode rootObjectNode) {
        rootObjectNode.remove("userRecording");
    }

    private static void upgradeJdbcPluginPropertiesIfNeeded(ObjectNode propertiesObjectNode) {
        if (propertiesObjectNode.path("captureBindParameters").asBoolean()
                && !propertiesObjectNode.has("captureBindParametersIncludes")) {
            // upgrade from 0.11.1 to 0.11.2
            propertiesObjectNode.set("captureBindParametersIncludes",
                    mapper.createArrayNode().add(".*"));
            propertiesObjectNode.remove("captureBindParameters");
        }
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy