Skip to content

Commit

Permalink
Keep track of optional extended plugins in separate set
Browse files Browse the repository at this point in the history
Signed-off-by: Craig Perkins <[email protected]>
  • Loading branch information
cwperks committed Jan 3, 2025
1 parent 0032cc3 commit 4a6d065
Show file tree
Hide file tree
Showing 4 changed files with 53 additions and 25 deletions.
28 changes: 21 additions & 7 deletions server/src/main/java/org/opensearch/plugins/PluginInfo.java
Original file line number Diff line number Diff line change
Expand Up @@ -56,10 +56,12 @@
import java.util.ArrayList;
import java.util.Arrays;
import java.util.Collections;
import java.util.HashSet;
import java.util.List;
import java.util.Locale;
import java.util.Map;
import java.util.Properties;
import java.util.Set;
import java.util.function.Function;
import java.util.stream.Collectors;

Expand All @@ -86,6 +88,8 @@ public class PluginInfo implements Writeable, ToXContentObject {
private final String classname;
private final String customFolderName;
private final List<String> extendedPlugins;
// Optional extended plugins are a subset of extendedPlugins that only contains the optional extended plugins
private final Set<String> optionalExtendedPlugins;
private final boolean hasNativeController;

/**
Expand Down Expand Up @@ -149,7 +153,11 @@ public PluginInfo(
this.javaVersion = javaVersion;
this.classname = classname;
this.customFolderName = customFolderName;
this.extendedPlugins = Collections.unmodifiableList(extendedPlugins);
this.extendedPlugins = extendedPlugins.stream().map(s -> s.split(";")[0]).collect(Collectors.toUnmodifiableList());
this.optionalExtendedPlugins = extendedPlugins.stream()
.filter(PluginInfo::isOptionalExtension)
.map(s -> s.split(";")[0])
.collect(Collectors.toUnmodifiableSet());
this.hasNativeController = hasNativeController;
}

Expand Down Expand Up @@ -209,6 +217,12 @@ public PluginInfo(final StreamInput in) throws IOException {
this.customFolderName = in.readString();
this.extendedPlugins = in.readStringList();
this.hasNativeController = in.readBoolean();
this.optionalExtendedPlugins = new HashSet<>();
}

static boolean isOptionalExtension(String extendedPlugin) {
String[] dependency = extendedPlugin.split(";");
return dependency.length > 1 && "optional=true".equals(dependency[1]);
}

@Override
Expand All @@ -232,7 +246,7 @@ This works for currently supported range notations (=,~)
} else {
out.writeString(name);
}
out.writeStringCollection(getExtendedPluginNames());
out.writeStringCollection(extendedPlugins);
out.writeBoolean(hasNativeController);
}

Expand Down Expand Up @@ -417,17 +431,17 @@ public String getFolderName() {
*
* @return the names of the plugins extended
*/
public List<String> getExtendedPluginNames() {
return extendedPlugins.stream().map(s -> s.split(";")[0]).collect(Collectors.toUnmodifiableList());
public boolean isExtendedPluginOptional(String extendedPlugin) {
return optionalExtendedPlugins.contains(extendedPlugin);
}

/**
* Other plugins this plugin extends through SPI including information about optionality.
* Other plugins this plugin extends through SPI
*
* @return the names of the plugins extended including optionality. i.e. opensearch-job-scheduler;optional=true
* @return the names of the plugins extended
*/
public List<String> getExtendedPlugins() {
return extendedPlugins;
return extendedPlugins.stream().map(s -> s.split(";")[0]).collect(Collectors.toUnmodifiableList());
}

/**
Expand Down
16 changes: 5 additions & 11 deletions server/src/main/java/org/opensearch/plugins/PluginsService.java
Original file line number Diff line number Diff line change
Expand Up @@ -497,11 +497,6 @@ static List<Bundle> sortBundles(Set<Bundle> bundles) {
return new ArrayList<>(sortedBundles);
}

static boolean isExtendedPluginOptional(String extendedPlugin) {
String[] dependency = extendedPlugin.split(";");
return dependency.length > 1 && "optional=true".equals(dependency[1]);
}

// add the given bundle to the sorted bundles, first adding dependencies
private static void addSortedBundle(
Bundle bundle,
Expand All @@ -527,15 +522,14 @@ private static void addSortedBundle(

dependencyStack.add(name);
for (String dependency : bundle.plugin.getExtendedPlugins()) {
String dependencyName = dependency.split(";")[0];
Bundle depBundle = bundles.get(dependencyName);
Bundle depBundle = bundles.get(dependency);
if (depBundle == null) {
if (isExtendedPluginOptional(dependency)) {
logger.warn("Missing plugin [" + dependencyName + "], dependency of [" + name + "]");
if (bundle.plugin.isExtendedPluginOptional(dependency)) {
logger.warn("Missing plugin [" + dependency + "], dependency of [" + name + "]");
logger.warn("Some features of this plugin may not function without the dependencies being installed.\n");
continue;
} else {
throw new IllegalArgumentException("Missing plugin [" + dependencyName + "], dependency of [" + name + "]");
throw new IllegalArgumentException("Missing plugin [" + dependency + "], dependency of [" + name + "]");
}
}
addSortedBundle(depBundle, bundles, sortedBundles, dependencyStack);
Expand Down Expand Up @@ -665,7 +659,7 @@ static void checkBundleJarHell(Set<URL> classpath, Bundle bundle, Map<String, Se
Set<URL> urls = new HashSet<>();
for (String extendedPlugin : exts) {
Set<URL> pluginUrls = transitiveUrls.get(extendedPlugin);
if (pluginUrls == null && isExtendedPluginOptional(extendedPlugin)) {
if (pluginUrls == null && bundle.plugin.isExtendedPluginOptional(extendedPlugin)) {
continue;
}
assert pluginUrls != null : "transitive urls should have already been set for " + extendedPlugin;
Expand Down
27 changes: 27 additions & 0 deletions server/src/test/java/org/opensearch/plugins/PluginInfoTests.java
Original file line number Diff line number Diff line change
Expand Up @@ -44,6 +44,7 @@
import org.opensearch.semver.SemverRange;
import org.opensearch.test.OpenSearchTestCase;

import java.io.IOException;
import java.nio.ByteBuffer;
import java.nio.file.Path;
import java.util.ArrayList;
Expand All @@ -55,6 +56,7 @@
import static org.hamcrest.Matchers.containsString;
import static org.hamcrest.Matchers.empty;
import static org.hamcrest.Matchers.equalTo;
import static org.hamcrest.Matchers.is;

public class PluginInfoTests extends OpenSearchTestCase {

Expand Down Expand Up @@ -281,6 +283,30 @@ public void testReadFromPropertiesJvmMissingClassname() throws Exception {
assertThat(e.getMessage(), containsString("property [classname] is missing"));
}

public void testExtendedPluginsSingleOptionalExtension() throws IOException {
Path pluginDir = createTempDir().resolve("fake-plugin");
PluginTestUtil.writePluginProperties(
pluginDir,
"description",
"fake desc",
"name",
"my_plugin",
"version",
"1.0",
"opensearch.version",
Version.CURRENT.toString(),
"java.version",
System.getProperty("java.specification.version"),
"classname",
"FakePlugin",
"extended.plugins",
"foo;optional=true"
);
PluginInfo info = PluginInfo.readFromProperties(pluginDir);
assertThat(info.getExtendedPlugins(), contains("foo"));
assertThat(info.isExtendedPluginOptional("foo"), is(true));
}

public void testExtendedPluginsSingleExtension() throws Exception {
Path pluginDir = createTempDir().resolve("fake-plugin");
PluginTestUtil.writePluginProperties(
Expand All @@ -302,6 +328,7 @@ public void testExtendedPluginsSingleExtension() throws Exception {
);
PluginInfo info = PluginInfo.readFromProperties(pluginDir);
assertThat(info.getExtendedPlugins(), contains("foo"));
assertThat(info.isExtendedPluginOptional("foo"), is(false));
}

public void testExtendedPluginsMultipleExtensions() throws Exception {
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -75,7 +75,6 @@
import static org.hamcrest.Matchers.hasSize;
import static org.hamcrest.Matchers.hasToString;
import static org.hamcrest.Matchers.instanceOf;
import static org.hamcrest.Matchers.is;
import static org.hamcrest.Matchers.notNullValue;
import static org.hamcrest.Matchers.sameInstance;

Expand Down Expand Up @@ -362,12 +361,6 @@ public void testSortBundlesNoDeps() throws Exception {
assertThat(sortedBundles, Matchers.contains(bundle1, bundle2, bundle3));
}

public void testIsOptionalExtendedPlugin() {
assertThat(PluginsService.isExtendedPluginOptional("plugin-dep"), is(false));
assertThat(PluginsService.isExtendedPluginOptional("plugin-dep;optional=true"), is(true));
assertThat(PluginsService.isExtendedPluginOptional("plugin-dep;optional=false"), is(false));
}

public void testSortBundlesMissingRequiredDep() throws Exception {
Path pluginDir = createTempDir();
PluginInfo info = new PluginInfo("foo", "desc", "1.0", Version.CURRENT, "1.8", "MyPlugin", Collections.singletonList("dne"), false);
Expand Down

0 comments on commit 4a6d065

Please sign in to comment.