diff --git a/.github/workflows/BWC.yml b/.github/workflows/BWC.yml new file mode 100644 index 0000000..70cdcd3 --- /dev/null +++ b/.github/workflows/BWC.yml @@ -0,0 +1,32 @@ +name: Build and Run BWC Tests + +on: + push: + branches: + - "*" + pull_request: + branches: + - "*" + +jobs: + build: + strategy: + matrix: + os: [ubuntu-latest] + java: [21] + name: Build and Run BWC Tests + runs-on: ${{ matrix.os }} + permissions: + contents: read + + steps: + - uses: actions/checkout@v4 + - name: Set up JDK ${{ matrix.java }} + uses: actions/setup-java@v4 + with: + distribution: "temurin" + java-version: ${{ matrix.java }} + + - name: Build and Run Tests + run: | + ./gradlew --info bwcTestSuite -Dtests.security.manager=false diff --git a/.gitignore b/.gitignore index 4b74090..ceb19b0 100644 --- a/.gitignore +++ b/.gitignore @@ -2,6 +2,9 @@ # https://discuss.elastic.co/t/leftovers-after-integtestrunner-in-root/152610 \.local*-integTestRunner-execution-times.log +# This location is used by BWC test clusters to host older and actual versions of plugin ZIP files. +src/test/resources/org/opensearch/prometheus-exporter/bwc/ + # intellij files .idea/ *.iml diff --git a/README.md b/README.md index 71c9d59..015a7a8 100644 --- a/README.md +++ b/README.md @@ -12,6 +12,7 @@ The [Prometheus® exporter](https://prometheus.io/docs/instrumenting/writing_exp - [Usage](#usage) - [Build from Source](#build-from-source) - [Testing](#testing) + - [BWC Testing](#bwc-testing) - [License](#license) - [Trademarks & Attributions](#trademarks--attributions) @@ -279,6 +280,20 @@ To run individual integration rest test file use: -Dtests.method="test {yaml=/20_11_index_level_metrics_disabled/Dynamically disable index level metrics}" ``` +### BWC Testing + +Backward Compatibility (BWC) Testing is run manually using provided shell script: + +``` +./bwctest.sh +``` + +It is not part of `./gradlew [build|check]` task(s), but it is included in the CI workflow. + +OpenSearch versions used during BWC tests use determined by properties located in `gradle.properties` file. Specifically `project.version` and `project.BWCversion`. Version of plugin deployed into `project.BWCversion` cluster is specified by `project.BWCPluginVersion` property. + +In the beginning of BWC tests the actual version of plugin (`project.version`) is build using `bundlePlugin` gradle task and the `project.BWCPluginVersion` plugin is downloaded from GitHub releases. Both ZIP files are placed into `src/test/resource/org/opensearch/prometheus-exrpoter/bwc/prometheus-exporter` folder (this folder is ignored by git). + ## License Licensed under the Apache License, Version 2.0 (the "License"); diff --git a/build.gradle b/build.gradle index 7c214f3..b01731f 100644 --- a/build.gradle +++ b/build.gradle @@ -1,5 +1,6 @@ import org.opensearch.gradle.test.RestIntegTestTask +import java.util.concurrent.Callable import java.util.regex.Matcher import java.util.regex.Pattern @@ -36,6 +37,12 @@ buildscript { "opensearch": opensearch_version, "prometheus": "0.16.0" ] + + bwcPluginDownloadLink = 'https://github.com/Aiven-Open/prometheus-exporter-plugin-for-opensearch/releases/download/' + + project.BWCPluginVersion + '/prometheus-exporter-' + project.BWCPluginVersion + '.zip' + baseName = "bwcCluster" + bwcFilePath = "src/test/resources/org/opensearch/prometheus-exporter/bwc/" + bwcPrometheusExporterPath = bwcFilePath + "prometheus-exporter/" } repositories { @@ -46,7 +53,12 @@ buildscript { maven { url "https://plugins.gradle.org/m2/" } } + configurations { + zipArchive + } + dependencies { + zipArchive group: 'org.opensearch.plugin.prometheus', name:'prometheus-exporter', version: "${versions.opensearch}" classpath "org.opensearch.gradle:build-tools:${versions.opensearch}" } } @@ -118,6 +130,9 @@ task integTest(type: RestIntegTestTask) { description = "Run tests against a cluster" testClassesDirs = sourceSets.test.output.classesDirs classpath = sourceSets.test.runtimeClasspath + filter { + excludeTestsMatching "org.opensearch.plugin.bwc.*IT" + } } tasks.named("check").configure { dependsOn(integTest) } @@ -146,6 +161,178 @@ testClusters.integTest { plugin(project.tasks.bundlePlugin.archiveFile) } +task copyZIPBundle { + dependsOn(bundlePlugin) + + doLast { + // By using ant.copy we can "hack" around gradle check for other task's resources modification. + // It seems like a dirty hack but some official plugins seem to use this practice; + // for instance see https://github.com/opensearch-project/anomaly-detection/ + ant.copy(todir: bwcPrometheusExporterPath + project.version) { + ant.fileset(dir: 'build/distributions/', includes: 'prometheus-exporter-' + project.version + '.zip') + } + } +} + +// Clusters for BWC tests +2.times { i -> + testClusters { + "${baseName}$i" { + versions = [project.BWCversion, "2.17.1"] + numberOfNodes = 3 + plugin(provider(new Callable() { + @Override + RegularFile call() throws Exception { + return new RegularFile() { + @Override + File getAsFile() { + if (new File("$project.rootDir/$bwcFilePath/prometheus-exporter/$project.BWCPluginVersion").exists()) { + project.delete(files("$project.rootDir/$bwcFilePath/prometheus-exporter/$project.BWCPluginVersion")) + } + project.mkdir bwcPrometheusExporterPath + project.BWCPluginVersion + ant.get(src: bwcPluginDownloadLink, + dest: bwcPrometheusExporterPath + project.BWCPluginVersion, + httpusecaches: false) + return fileTree(bwcPrometheusExporterPath + project.BWCPluginVersion).getSingleFile() + } + } + } + })) + setting 'path.repo', "${buildDir}/cluster/shared/repo/${baseName}" + setting 'http.content_type.required', 'true' + } + } +} + +List> plugins = [ + provider(new Callable(){ + @Override + RegularFile call() throws Exception { + return new RegularFile() { + @Override + File getAsFile() { + return fileTree(bwcPrometheusExporterPath + project.version).getSingleFile() + } + } + } + }) +] + +// Creates 2 test clusters with 3 nodes of the old version. +2.times {i -> + task "${baseName}#oldVersionClusterTask$i"(type: RestIntegTestTask) { + useCluster testClusters."${baseName}$i" + filter { + includeTestsMatching "org.opensearch.plugin.bwc.*IT" + } + systemProperty 'tests.rest.bwcsuite', 'old_cluster' + systemProperty 'tests.rest.bwcsuite_round', 'old' + systemProperty 'tests.plugin_bwc_version', project.BWCPluginVersion + nonInputProperties.systemProperty('tests.rest.cluster', "${-> testClusters."${baseName}$i".allHttpSocketURI.join(",")}") + nonInputProperties.systemProperty('tests.clustername', "${-> testClusters."${baseName}$i".getName()}") + } +} + +// Upgrades one node of the old cluster to new OpenSearch version with upgraded plugin version +// This results in a mixed cluster with 2 nodes on the old version and 1 upgraded node. +// This is also used as a one third upgraded cluster for a rolling upgrade. +task "${baseName}#mixedClusterTask"(type: RestIntegTestTask) { + dependsOn tasks.named("copyZIPBundle") + useCluster testClusters."${baseName}0" + dependsOn "${baseName}#oldVersionClusterTask0" + doFirst { + testClusters."${baseName}0".upgradeNodeAndPluginToNextVersion(plugins) + } + filter { + includeTestsMatching "org.opensearch.plugin.bwc.*IT" + } + systemProperty 'tests.rest.bwcsuite', 'mixed_cluster' + systemProperty 'tests.rest.bwcsuite_round', 'first' + systemProperty 'tests.plugin_bwc_version', project.BWCPluginVersion + nonInputProperties.systemProperty('tests.rest.cluster', "${-> testClusters."${baseName}0".allHttpSocketURI.join(",")}") + nonInputProperties.systemProperty('tests.clustername', "${-> testClusters."${baseName}0".getName()}") +} + +// Upgrades the second node to new OpenSearch version with upgraded plugin version after the first node is upgraded. +// This results in a mixed cluster with 1 node on the old version and 2 upgraded nodes. +// This is used for rolling upgrade. +task "${baseName}#twoThirdsUpgradedClusterTask"(type: RestIntegTestTask) { + dependsOn tasks.named("copyZIPBundle") + dependsOn "${baseName}#mixedClusterTask" + useCluster testClusters."${baseName}0" + doFirst { + testClusters."${baseName}0".upgradeNodeAndPluginToNextVersion(plugins) + } + filter { + includeTestsMatching "org.opensearch.plugin.bwc.*IT" + } + systemProperty 'tests.rest.bwcsuite', 'mixed_cluster' + systemProperty 'tests.rest.bwcsuite_round', 'second' + systemProperty 'tests.plugin_bwc_version', project.BWCPluginVersion + nonInputProperties.systemProperty('tests.rest.cluster', "${-> testClusters."${baseName}0".allHttpSocketURI.join(",")}") + nonInputProperties.systemProperty('tests.clustername', "${-> testClusters."${baseName}0".getName()}") +} + +// Upgrades the third node to new OpenSearch version with upgraded plugin version after the second node is upgraded. +// This results in a fully upgraded cluster. +// This is used for rolling upgrade. +task "${baseName}#rollingUpgradeClusterTask"(type: RestIntegTestTask) { + dependsOn tasks.named("copyZIPBundle") + dependsOn "${baseName}#twoThirdsUpgradedClusterTask" + useCluster testClusters."${baseName}0" + doFirst { + testClusters."${baseName}0".upgradeNodeAndPluginToNextVersion(plugins) + } + filter { + includeTestsMatching "org.opensearch.plugin.bwc.*IT" + } + mustRunAfter "${baseName}#mixedClusterTask" + systemProperty 'tests.rest.bwcsuite', 'mixed_cluster' + systemProperty 'tests.rest.bwcsuite_round', 'third' + systemProperty 'tests.plugin_bwc_version', project.BWCPluginVersion + nonInputProperties.systemProperty('tests.rest.cluster', "${-> testClusters."${baseName}0".allHttpSocketURI.join(",")}") + nonInputProperties.systemProperty('tests.clustername', "${-> testClusters."${baseName}0".getName()}") +} + +// Upgrades all the nodes of the old cluster to new OpenSearch version with upgraded plugin version +// at the same time resulting in a fully upgraded cluster. +task "${baseName}#fullRestartClusterTask"(type: RestIntegTestTask) { + dependsOn tasks.named("copyZIPBundle") + dependsOn "${baseName}#oldVersionClusterTask1" + useCluster testClusters."${baseName}1" + doFirst { + testClusters."${baseName}1".upgradeAllNodesAndPluginsToNextVersion(plugins) + } + filter { + includeTestsMatching "org.opensearch.plugin.bwc.*IT" + } + systemProperty 'tests.rest.bwcsuite', 'upgraded_cluster' + systemProperty 'tests.plugin_bwc_version', project.BWCPluginVersion + nonInputProperties.systemProperty('tests.rest.cluster', "${-> testClusters."${baseName}1".allHttpSocketURI.join(",")}") + nonInputProperties.systemProperty('tests.clustername', "${-> testClusters."${baseName}1".getName()}") +} + +// A BWC test suite which runs all the bwc tasks combined. +task bwcTestSuite(type: RestIntegTestTask) { + + // Delete all downloaded and built plugin ZIP files. + // Again – we are using ant task to workaround gradle resources modification alert. + doFirst { + ant.delete(includeEmptyDirs: true, verbose: true, removeNotFollowedSymlinks: true) { + ant.fileset( + dir: 'src/test/resources/org/opensearch/prometheus-exporter/bwc/prometheus-exporter' + ) + } + } + + exclude '**/*Test*' + exclude '**/*IT*' + + dependsOn tasks.named("${baseName}#mixedClusterTask") + dependsOn tasks.named("${baseName}#rollingUpgradeClusterTask") + dependsOn tasks.named("${baseName}#fullRestartClusterTask") +} + run { useCluster testClusters.integTest } \ No newline at end of file diff --git a/bwctest.sh b/bwctest.sh new file mode 100755 index 0000000..3f92417 --- /dev/null +++ b/bwctest.sh @@ -0,0 +1,36 @@ +#!/bin/bash + +set -e + +function usage() { + echo "" + echo "This script is used to run Backwards Compatibility tests" + echo "--------------------------------------------------------------------------" + echo "Usage: $0 [args]" + echo "" + echo "Required arguments:" + echo "None" + echo "" + echo -e "-h\tPrint this message." + echo "--------------------------------------------------------------------------" +} + +while getopts ":h" arg; do + case $arg in + h) + usage + exit 1 + ;; + ?) + echo "Invalid option: -${OPTARG}" + exit 1 + ;; + esac +done + +# Warning! +# This should be done from gradle, see bwcTestSuite task, but that task is skipped. +# TODO: Skipping task ':bwcTestSuite' as it has no source files and no previous output files. +rm -rf src/test/resources/org/opensearch/prometheus-exporter/bwc/prometheus-exporter + +./gradlew bwcTestSuite -Dtests.security.manager=false \ No newline at end of file diff --git a/gradle.properties b/gradle.properties index 8b3d132..ac4f89f 100644 --- a/gradle.properties +++ b/gradle.properties @@ -1,7 +1,14 @@ #group = org.opensearch.plugin.prometheus +# An actual version of plugin version = 2.17.1.0 +# A version of OpenSearch cluster to run BWC tests against +BWCversion = 2.17.0 + +# A version of plugin to deploy to BWC clusters +BWCPluginVersion = 2.17.0.0 + pluginName = prometheus-exporter pluginClassname = org.opensearch.plugin.prometheus.PrometheusExporterPlugin pluginDescription = Prometheus exporter plugin for OpenSearch diff --git a/src/test/java/org/opensearch/plugin/bwc/PluginBackwardsCompatibilityIT.java b/src/test/java/org/opensearch/plugin/bwc/PluginBackwardsCompatibilityIT.java new file mode 100644 index 0000000..386f5e0 --- /dev/null +++ b/src/test/java/org/opensearch/plugin/bwc/PluginBackwardsCompatibilityIT.java @@ -0,0 +1,140 @@ +/* + * Copyright [2024] [Lukáš VLČEK] + * + * 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.opensearch.plugin.bwc; + +import org.junit.Assert; +import org.opensearch.Version; +import org.opensearch.common.settings.Settings; +import org.opensearch.test.rest.OpenSearchRestTestCase; + +import java.util.List; +import java.util.Map; +import java.util.Set; +import java.util.stream.Collectors; + +/** + * IMPORTANT: When upgrading to a new version of OpenSearch this test has to be updated + * to reflect appropriate versions, see {@link #BWCVersion} and {@link #NewVersion}. + * These two variables must match 'testClusters' versions used in 'build.gradle' file. + */ +public class PluginBackwardsCompatibilityIT extends OpenSearchRestTestCase { + + public static final Version BWCVersion = Version.V_2_17_0; + public static final Version NewVersion = Version.V_2_17_1; + + private static final ClusterType CLUSTER_TYPE = ClusterType.parse(System.getProperty("tests.rest.bwcsuite")); + private static final String CLUSTER_NAME = System.getProperty("tests.clustername"); +// private static final String MIXED_CLUSTER_TEST_ROUND = System.getProperty("tests.rest.bwcsuite_round"); + + @Override + protected final Settings restClientSettings() { + return Settings + .builder() + .put(super.restClientSettings()) + // increase the timeout here to 90 seconds to handle long waits for a green + // cluster health. the waits for green need to be longer than a minute to + // account for delayed shards + .put(OpenSearchRestTestCase.CLIENT_SOCKET_TIMEOUT, "90s") + .build(); + } + + private enum ClusterType { + OLD, + MIXED, + UPGRADED; + + public static ClusterType parse(String value) { + switch (value) { + case "old_cluster": + return OLD; + case "mixed_cluster": + return MIXED; + case "upgraded_cluster": + return UPGRADED; + default: + throw new AssertionError("unknown cluster type: " + value); + } + } + } + + public void testBackwardsCompatibility() throws Exception { + int testedNodes = 0; + String uri = getNodesPluginsUri(); + Map> responseMap = (Map>) getAsMap(uri).get("nodes"); + for (Map response : responseMap.values()) { + + String version = (String) response.get("version"); + Assert.assertFalse(version.isEmpty()); + + List> plugins = (List>) response.get("plugins"); + Set pluginNames = plugins.stream().map(map -> map.get("name")).collect(Collectors.toSet()); + + switch (CLUSTER_TYPE) { + case OLD: + testedNodes++; + Assert.assertTrue(pluginNames.contains("prometheus-exporter")); + Assert.assertEquals(BWCVersion, Version.fromString(version)); + break; + case MIXED: + testedNodes++; + Assert.assertTrue(pluginNames.contains("prometheus-exporter")); + break; + case UPGRADED: + testedNodes++; + Assert.assertTrue(pluginNames.contains("prometheus-exporter")); + Assert.assertEquals(NewVersion, Version.fromString(version)); + break; + } + } + + assertTrue(testedNodes > 0); + + switch (CLUSTER_TYPE) { + case OLD: + assertEquals(1, testedNodes); + break; + case MIXED: + assertEquals(1, testedNodes); + break; + case UPGRADED: + assertEquals(3, testedNodes); + break; + } + + } + + private String getNodesPluginsUri() { + switch (CLUSTER_TYPE) { + case OLD: + return "_nodes/" + CLUSTER_NAME + "-0/plugins"; + case MIXED: + String round = System.getProperty("tests.rest.bwcsuite_round"); + if (round.equals("second")) { + return "_nodes/" + CLUSTER_NAME + "-1/plugins"; + } else if (round.equals("third")) { + return "_nodes/" + CLUSTER_NAME + "-2/plugins"; + } else { + return "_nodes/" + CLUSTER_NAME + "-0/plugins"; + } + case UPGRADED: + return "_nodes/plugins"; + default: + throw new AssertionError("unknown cluster type: " + CLUSTER_TYPE); + } + } +}