From 24554c53120257b03fbd6f9dfa37eb0e433e525c Mon Sep 17 00:00:00 2001 From: Keith Massey Date: Tue, 1 Mar 2022 15:20:26 -0600 Subject: [PATCH] Getting deprecation.skip_deprecated_settings to work with dynamic settings (#81836) (#84523) (#84527) Previously, deprecation.skip_deprecated_settings did not work with dynamic settings. The reason is that when the deprecation check was done, only the current settings were available. When the setting is a node setting that is fine because deprecation.skip_deprecated_settings is also a node setting. However when the setting is dynamic, deprecation.skip_deprecated_settings is not in the Settings object. --- .../common/logging/DeprecationLogger.java | 33 +++++++++++++++---- .../common/settings/Setting.java | 13 +++----- .../java/org/elasticsearch/node/Node.java | 2 ++ .../common/settings/SettingTests.java | 4 ++- 4 files changed, 36 insertions(+), 16 deletions(-) diff --git a/server/src/main/java/org/elasticsearch/common/logging/DeprecationLogger.java b/server/src/main/java/org/elasticsearch/common/logging/DeprecationLogger.java index 38eeffad1cc52..6772a452abc9f 100644 --- a/server/src/main/java/org/elasticsearch/common/logging/DeprecationLogger.java +++ b/server/src/main/java/org/elasticsearch/common/logging/DeprecationLogger.java @@ -11,9 +11,13 @@ import org.apache.logging.log4j.Level; import org.apache.logging.log4j.LogManager; import org.apache.logging.log4j.Logger; +import org.elasticsearch.common.regex.Regex; +import org.elasticsearch.common.settings.Settings; import java.security.AccessController; import java.security.PrivilegedAction; +import java.util.Collections; +import java.util.List; /** * A logger that logs deprecation notices. Logger should be initialized with a class or name which will be used @@ -35,7 +39,7 @@ public class DeprecationLogger { * More serious that WARN by 1, but less serious than ERROR */ public static Level CRITICAL = Level.forName("CRITICAL", Level.WARN.intLevel() - 1); - + private static volatile List skipTheseDeprecations = Collections.emptyList(); private final Logger logger; /** @@ -56,6 +60,19 @@ public static DeprecationLogger getLogger(String name) { return new DeprecationLogger(name); } + /** + * The DeprecationLogger uses the "deprecation.skip_deprecated_settings" setting to decide whether to log a deprecation for a setting. + * This is a node setting. This method initializes the DeprecationLogger class with the node settings for the node in order to read the + * "deprecation.skip_deprecated_settings" setting. This only needs to be called once per JVM. If it is not called, the default behavior + * is to assume that the "deprecation.skip_deprecated_settings" setting is not set. + * @param nodeSettings The settings for this node + */ + public static void initialize(Settings nodeSettings) { + skipTheseDeprecations = nodeSettings == null + ? Collections.emptyList() + : nodeSettings.getAsList("deprecation.skip_deprecated_settings"); + } + private DeprecationLogger(String parentLoggerName) { this.logger = LogManager.getLogger(getLoggerName(parentLoggerName)); } @@ -95,12 +112,14 @@ public DeprecationLogger warn(final DeprecationCategory category, final String k } private DeprecationLogger logDeprecation(Level level, DeprecationCategory category, String key, String msg, Object[] params) { - assert category != DeprecationCategory.COMPATIBLE_API - : "DeprecationCategory.COMPATIBLE_API should be logged with compatibleApiWarning method"; - String opaqueId = HeaderWarning.getXOpaqueId(); - String productOrigin = HeaderWarning.getProductOrigin(); - ESLogMessage deprecationMessage = DeprecatedMessage.of(category, key, opaqueId, productOrigin, msg, params); - doPrivilegedLog(level, deprecationMessage); + if (Regex.simpleMatch(skipTheseDeprecations, key) == false) { + assert category != DeprecationCategory.COMPATIBLE_API + : "DeprecationCategory.COMPATIBLE_API should be logged with compatibleApiWarning method"; + String opaqueId = HeaderWarning.getXOpaqueId(); + String productOrigin = HeaderWarning.getProductOrigin(); + ESLogMessage deprecationMessage = DeprecatedMessage.of(category, key, opaqueId, productOrigin, msg, params); + doPrivilegedLog(level, deprecationMessage); + } return this; } diff --git a/server/src/main/java/org/elasticsearch/common/settings/Setting.java b/server/src/main/java/org/elasticsearch/common/settings/Setting.java index 563696ce84302..f0a17f8845427 100644 --- a/server/src/main/java/org/elasticsearch/common/settings/Setting.java +++ b/server/src/main/java/org/elasticsearch/common/settings/Setting.java @@ -577,14 +577,11 @@ void checkDeprecation(Settings settings) { if (this.isDeprecated() && this.exists(settings)) { // It would be convenient to show its replacement key, but replacement is often not so simple final String key = getKey(); - List skipTheseDeprecations = settings.getAsList("deprecation.skip_deprecated_settings"); - if (Regex.simpleMatch(skipTheseDeprecations, key) == false) { - String message = "[{}] setting was deprecated in Elasticsearch and will be removed in a future release."; - if (this.isDeprecatedWarningOnly()) { - Settings.DeprecationLoggerHolder.deprecationLogger.warn(DeprecationCategory.SETTINGS, key, message, key); - } else { - Settings.DeprecationLoggerHolder.deprecationLogger.critical(DeprecationCategory.SETTINGS, key, message, key); - } + String message = "[{}] setting was deprecated in Elasticsearch and will be removed in a future release."; + if (this.isDeprecatedWarningOnly()) { + Settings.DeprecationLoggerHolder.deprecationLogger.warn(DeprecationCategory.SETTINGS, key, message, key); + } else { + Settings.DeprecationLoggerHolder.deprecationLogger.critical(DeprecationCategory.SETTINGS, key, message, key); } } } diff --git a/server/src/main/java/org/elasticsearch/node/Node.java b/server/src/main/java/org/elasticsearch/node/Node.java index b2de391e19bb8..28481a9895b28 100644 --- a/server/src/main/java/org/elasticsearch/node/Node.java +++ b/server/src/main/java/org/elasticsearch/node/Node.java @@ -293,6 +293,8 @@ protected Node( final List resourcesToClose = new ArrayList<>(); // register everything we need to release in the case of an error boolean success = false; try { + // Pass the node settings to the DeprecationLogger class so that it can have the deprecation.skip_deprecated_settings setting: + DeprecationLogger.initialize(initialEnvironment.settings()); Settings tmpSettings = Settings.builder() .put(initialEnvironment.settings()) .put(Client.CLIENT_TYPE_SETTING_S.getKey(), CLIENT_TYPE) diff --git a/server/src/test/java/org/elasticsearch/common/settings/SettingTests.java b/server/src/test/java/org/elasticsearch/common/settings/SettingTests.java index 9819f8b1918ac..9824bbf6f36dc 100644 --- a/server/src/test/java/org/elasticsearch/common/settings/SettingTests.java +++ b/server/src/test/java/org/elasticsearch/common/settings/SettingTests.java @@ -12,6 +12,7 @@ import org.apache.logging.log4j.Logger; import org.apache.logging.log4j.core.LogEvent; import org.elasticsearch.cluster.metadata.IndexMetadata; +import org.elasticsearch.common.logging.DeprecationLogger; import org.elasticsearch.common.logging.Loggers; import org.elasticsearch.common.settings.AbstractScopedSettings.SettingUpdater; import org.elasticsearch.common.settings.Setting.Property; @@ -1396,7 +1397,7 @@ public void testCheckForDeprecation() { } public void testCheckForDeprecationWithSkipSetting() { - final String settingName = "foo.bar"; + final String settingName = "foo.bar.hide.this"; final String settingValue = "blat"; final Setting setting = Setting.simpleString(settingName, settingValue); final Settings settings = Settings.builder().put(settingName, settingValue).build(); @@ -1409,6 +1410,7 @@ public void testCheckForDeprecationWithSkipSetting() { .put(settingName, settingValue) .putList("deprecation.skip_deprecated_settings", settingName) .build(); + DeprecationLogger.initialize(settingsWithSkipDeprecationSetting); deprecatedSetting.checkDeprecation(settingsWithSkipDeprecationSetting); ensureNoWarnings(); }