Merge "Fix DynamicOptions to support a custom bean"
This commit is contained in:
@@ -0,0 +1,117 @@
|
|||||||
|
// Copyright (C) 2020 The Android Open Source Project
|
||||||
|
//
|
||||||
|
// 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 com.google.gerrit.acceptance;
|
||||||
|
|
||||||
|
import static com.google.gerrit.sshd.CommandMetaData.Mode.MASTER_OR_SLAVE;
|
||||||
|
|
||||||
|
import com.google.common.collect.Lists;
|
||||||
|
import com.google.gerrit.extensions.annotations.Exports;
|
||||||
|
import com.google.gerrit.json.OutputFormat;
|
||||||
|
import com.google.gerrit.server.DynamicOptions;
|
||||||
|
import com.google.gerrit.sshd.CommandMetaData;
|
||||||
|
import com.google.gerrit.sshd.CommandModule;
|
||||||
|
import com.google.gerrit.sshd.SshCommand;
|
||||||
|
import com.google.gson.reflect.TypeToken;
|
||||||
|
import com.google.inject.AbstractModule;
|
||||||
|
import com.google.inject.Inject;
|
||||||
|
import java.io.BufferedWriter;
|
||||||
|
import java.io.OutputStream;
|
||||||
|
import java.io.OutputStreamWriter;
|
||||||
|
import java.io.PrintWriter;
|
||||||
|
import java.util.Collections;
|
||||||
|
import java.util.List;
|
||||||
|
|
||||||
|
public class AbstractDynamicOptionsTest extends AbstractDaemonTest {
|
||||||
|
protected static final String LS_SAMPLES = "ls-samples";
|
||||||
|
|
||||||
|
protected interface Bean {
|
||||||
|
void setSamples(List<String> samples);
|
||||||
|
}
|
||||||
|
|
||||||
|
protected static class ListSamples implements Bean, DynamicOptions.BeanReceiver {
|
||||||
|
protected List<String> samples = Collections.emptyList();
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void setSamples(List<String> samples) {
|
||||||
|
this.samples = samples;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void display(OutputStream displayOutputStream) throws Exception {
|
||||||
|
PrintWriter stdout =
|
||||||
|
new PrintWriter(new BufferedWriter(new OutputStreamWriter(displayOutputStream, "UTF-8")));
|
||||||
|
try {
|
||||||
|
OutputFormat.JSON
|
||||||
|
.newGson()
|
||||||
|
.toJson(samples, new TypeToken<List<String>>() {}.getType(), stdout);
|
||||||
|
stdout.print('\n');
|
||||||
|
} finally {
|
||||||
|
stdout.flush();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void setDynamicBean(String plugin, DynamicOptions.DynamicBean dynamicBean) {}
|
||||||
|
}
|
||||||
|
|
||||||
|
@CommandMetaData(name = LS_SAMPLES, runsAt = MASTER_OR_SLAVE)
|
||||||
|
protected static class ListSamplesCommand extends SshCommand {
|
||||||
|
@Inject private ListSamples impl;
|
||||||
|
|
||||||
|
@Override
|
||||||
|
protected void run() throws Exception {
|
||||||
|
impl.display(out);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
protected void parseCommandLine(DynamicOptions pluginOptions) throws UnloggedFailure {
|
||||||
|
parseCommandLine(impl, pluginOptions);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public static class PluginOneSshModule extends CommandModule {
|
||||||
|
@Override
|
||||||
|
public void configure() {
|
||||||
|
command(LS_SAMPLES).to(ListSamplesCommand.class);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
protected static class ListSamplesOptions implements DynamicOptions.BeanParseListener {
|
||||||
|
@Override
|
||||||
|
public void onBeanParseStart(String plugin, Object bean) {
|
||||||
|
((Bean) bean).setSamples(Lists.newArrayList("sample1", "sample2"));
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void onBeanParseEnd(String plugin, Object bean) {}
|
||||||
|
}
|
||||||
|
|
||||||
|
protected static class PluginTwoModule extends AbstractModule {
|
||||||
|
@Override
|
||||||
|
public void configure() {
|
||||||
|
bind(DynamicOptions.DynamicBean.class)
|
||||||
|
.annotatedWith(
|
||||||
|
Exports.named("com.google.gerrit.acceptance.AbstractDynamicOptionsTest.ListSamples"))
|
||||||
|
.to(ListSamplesOptionsClassNameProvider.class);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
protected static class ListSamplesOptionsClassNameProvider
|
||||||
|
implements DynamicOptions.ClassNameProvider {
|
||||||
|
@Override
|
||||||
|
public String getClassName() {
|
||||||
|
return "com.google.gerrit.acceptance.AbstractDynamicOptionsTest$ListSamplesOptions";
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -161,6 +161,8 @@ public class ParameterParser {
|
|||||||
HttpServletResponse res)
|
HttpServletResponse res)
|
||||||
throws IOException {
|
throws IOException {
|
||||||
CmdLineParser clp = parserFactory.create(param);
|
CmdLineParser clp = parserFactory.create(param);
|
||||||
|
pluginOptions.setBean(param);
|
||||||
|
pluginOptions.startLifecycleListeners();
|
||||||
pluginOptions.parseDynamicBeans(clp);
|
pluginOptions.parseDynamicBeans(clp);
|
||||||
pluginOptions.setDynamicBeans();
|
pluginOptions.setDynamicBeans();
|
||||||
pluginOptions.onBeanParseStart();
|
pluginOptions.onBeanParseStart();
|
||||||
|
|||||||
@@ -507,7 +507,7 @@ public class RestApiServlet extends HttpServlet {
|
|||||||
}
|
}
|
||||||
|
|
||||||
try (DynamicOptions pluginOptions =
|
try (DynamicOptions pluginOptions =
|
||||||
new DynamicOptions(viewData.view, globals.injector, globals.dynamicBeans)) {
|
new DynamicOptions(globals.injector, globals.dynamicBeans)) {
|
||||||
if (!globals
|
if (!globals
|
||||||
.paramParser
|
.paramParser
|
||||||
.get()
|
.get()
|
||||||
|
|||||||
@@ -193,6 +193,7 @@ public class DynamicOptions implements AutoCloseable {
|
|||||||
protected Object bean;
|
protected Object bean;
|
||||||
protected Map<String, DynamicBean> beansByPlugin;
|
protected Map<String, DynamicBean> beansByPlugin;
|
||||||
protected Injector injector;
|
protected Injector injector;
|
||||||
|
protected DynamicMap<DynamicBean> dynamicBeans;
|
||||||
protected LifecycleManager lifecycleManager;
|
protected LifecycleManager lifecycleManager;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -200,7 +201,9 @@ public class DynamicOptions implements AutoCloseable {
|
|||||||
* this class so the following methods can be called if desired:
|
* this class so the following methods can be called if desired:
|
||||||
*
|
*
|
||||||
* <pre>
|
* <pre>
|
||||||
* DynamicOptions pluginOptions = new DynamicOptions(bean, injector, dynamicBeans);
|
* DynamicOptions pluginOptions = new DynamicOptions(injector, dynamicBeans);
|
||||||
|
* pluginOptions.setBean(bean);
|
||||||
|
* pluginOptions.startLifecycleListeners();
|
||||||
* pluginOptions.parseDynamicBeans(clp);
|
* pluginOptions.parseDynamicBeans(clp);
|
||||||
* pluginOptions.setDynamicBeans();
|
* pluginOptions.setDynamicBeans();
|
||||||
* pluginOptions.onBeanParseStart();
|
* pluginOptions.onBeanParseStart();
|
||||||
@@ -210,11 +213,15 @@ public class DynamicOptions implements AutoCloseable {
|
|||||||
* pluginOptions.onBeanParseEnd();
|
* pluginOptions.onBeanParseEnd();
|
||||||
* </pre>
|
* </pre>
|
||||||
*/
|
*/
|
||||||
public DynamicOptions(Object bean, Injector injector, DynamicMap<DynamicBean> dynamicBeans) {
|
public DynamicOptions(Injector injector, DynamicMap<DynamicBean> dynamicBeans) {
|
||||||
this.bean = bean;
|
|
||||||
this.injector = injector;
|
this.injector = injector;
|
||||||
|
this.dynamicBeans = dynamicBeans;
|
||||||
lifecycleManager = new LifecycleManager();
|
lifecycleManager = new LifecycleManager();
|
||||||
beansByPlugin = new HashMap<>();
|
beansByPlugin = new HashMap<>();
|
||||||
|
}
|
||||||
|
|
||||||
|
public void setBean(Object bean) {
|
||||||
|
this.bean = bean;
|
||||||
Class<?> beanClass =
|
Class<?> beanClass =
|
||||||
(bean instanceof BeanReceiver)
|
(bean instanceof BeanReceiver)
|
||||||
? ((BeanReceiver) bean).getExportedBeanReceiver()
|
? ((BeanReceiver) bean).getExportedBeanReceiver()
|
||||||
@@ -226,7 +233,6 @@ public class DynamicOptions implements AutoCloseable {
|
|||||||
beansByPlugin.put(plugin, getDynamicBean(bean, provider.get()));
|
beansByPlugin.put(plugin, getDynamicBean(bean, provider.get()));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
startLifecycleListeners();
|
|
||||||
}
|
}
|
||||||
|
|
||||||
@SuppressWarnings("unchecked")
|
@SuppressWarnings("unchecked")
|
||||||
|
|||||||
@@ -177,6 +177,8 @@ class ChangeApiImpl implements ChangeApi {
|
|||||||
private final Provider<GetPureRevert> getPureRevertProvider;
|
private final Provider<GetPureRevert> getPureRevertProvider;
|
||||||
private final StarredChangesUtil stars;
|
private final StarredChangesUtil stars;
|
||||||
private final DynamicOptionParser dynamicOptionParser;
|
private final DynamicOptionParser dynamicOptionParser;
|
||||||
|
private final Injector injector;
|
||||||
|
private final DynamicMap<DynamicOptions.DynamicBean> dynamicBeans;
|
||||||
|
|
||||||
@Inject
|
@Inject
|
||||||
ChangeApiImpl(
|
ChangeApiImpl(
|
||||||
@@ -230,7 +232,9 @@ class ChangeApiImpl implements ChangeApi {
|
|||||||
Provider<GetPureRevert> getPureRevertProvider,
|
Provider<GetPureRevert> getPureRevertProvider,
|
||||||
StarredChangesUtil stars,
|
StarredChangesUtil stars,
|
||||||
DynamicOptionParser dynamicOptionParser,
|
DynamicOptionParser dynamicOptionParser,
|
||||||
@Assisted ChangeResource change) {
|
@Assisted ChangeResource change,
|
||||||
|
Injector injector,
|
||||||
|
DynamicMap<DynamicOptions.DynamicBean> dynamicBeans) {
|
||||||
this.changeApi = changeApi;
|
this.changeApi = changeApi;
|
||||||
this.revert = revert;
|
this.revert = revert;
|
||||||
this.revertSubmission = revertSubmission;
|
this.revertSubmission = revertSubmission;
|
||||||
@@ -282,6 +286,8 @@ class ChangeApiImpl implements ChangeApi {
|
|||||||
this.stars = stars;
|
this.stars = stars;
|
||||||
this.dynamicOptionParser = dynamicOptionParser;
|
this.dynamicOptionParser = dynamicOptionParser;
|
||||||
this.change = change;
|
this.change = change;
|
||||||
|
this.injector = injector;
|
||||||
|
this.dynamicBeans = dynamicBeans;
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
@@ -500,10 +506,10 @@ class ChangeApiImpl implements ChangeApi {
|
|||||||
public ChangeInfo get(
|
public ChangeInfo get(
|
||||||
EnumSet<ListChangesOption> options, ImmutableListMultimap<String, String> pluginOptions)
|
EnumSet<ListChangesOption> options, ImmutableListMultimap<String, String> pluginOptions)
|
||||||
throws RestApiException {
|
throws RestApiException {
|
||||||
try {
|
try (DynamicOptions dynamicOptions = new DynamicOptions(injector, dynamicBeans)) {
|
||||||
GetChange getChange = getChangeProvider.get();
|
GetChange getChange = getChangeProvider.get();
|
||||||
options.forEach(getChange::addOption);
|
options.forEach(getChange::addOption);
|
||||||
dynamicOptionParser.parseDynamicOptions(getChange, pluginOptions);
|
dynamicOptionParser.parseDynamicOptions(getChange, pluginOptions, dynamicOptions);
|
||||||
return getChange.apply(change).value();
|
return getChange.apply(change).value();
|
||||||
} catch (Exception e) {
|
} catch (Exception e) {
|
||||||
throw asRestApiException("Cannot retrieve change", e);
|
throw asRestApiException("Cannot retrieve change", e);
|
||||||
@@ -759,8 +765,6 @@ class ChangeApiImpl implements ChangeApi {
|
|||||||
@Singleton
|
@Singleton
|
||||||
static class DynamicOptionParser {
|
static class DynamicOptionParser {
|
||||||
private final CmdLineParser.Factory cmdLineParserFactory;
|
private final CmdLineParser.Factory cmdLineParserFactory;
|
||||||
private final Injector injector;
|
|
||||||
private final DynamicMap<DynamicOptions.DynamicBean> dynamicBeans;
|
|
||||||
|
|
||||||
@Inject
|
@Inject
|
||||||
DynamicOptionParser(
|
DynamicOptionParser(
|
||||||
@@ -768,14 +772,14 @@ class ChangeApiImpl implements ChangeApi {
|
|||||||
Injector injector,
|
Injector injector,
|
||||||
DynamicMap<DynamicOptions.DynamicBean> dynamicBeans) {
|
DynamicMap<DynamicOptions.DynamicBean> dynamicBeans) {
|
||||||
this.cmdLineParserFactory = cmdLineParserFactory;
|
this.cmdLineParserFactory = cmdLineParserFactory;
|
||||||
this.injector = injector;
|
|
||||||
this.dynamicBeans = dynamicBeans;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void parseDynamicOptions(Object bean, ListMultimap<String, String> pluginOptions)
|
void parseDynamicOptions(
|
||||||
|
Object bean, ListMultimap<String, String> pluginOptions, DynamicOptions dynamicOptions)
|
||||||
throws BadRequestException {
|
throws BadRequestException {
|
||||||
CmdLineParser clp = cmdLineParserFactory.create(bean);
|
CmdLineParser clp = cmdLineParserFactory.create(bean);
|
||||||
DynamicOptions dynamicOptions = new DynamicOptions(bean, injector, dynamicBeans);
|
dynamicOptions.setBean(bean);
|
||||||
|
dynamicOptions.startLifecycleListeners();
|
||||||
dynamicOptions.parseDynamicBeans(clp);
|
dynamicOptions.parseDynamicBeans(clp);
|
||||||
dynamicOptions.setDynamicBeans();
|
dynamicOptions.setDynamicBeans();
|
||||||
dynamicOptions.onBeanParseStart();
|
dynamicOptions.onBeanParseStart();
|
||||||
|
|||||||
@@ -26,15 +26,18 @@ import com.google.gerrit.extensions.api.changes.Changes;
|
|||||||
import com.google.gerrit.extensions.client.ListChangesOption;
|
import com.google.gerrit.extensions.client.ListChangesOption;
|
||||||
import com.google.gerrit.extensions.common.ChangeInfo;
|
import com.google.gerrit.extensions.common.ChangeInfo;
|
||||||
import com.google.gerrit.extensions.common.ChangeInput;
|
import com.google.gerrit.extensions.common.ChangeInput;
|
||||||
|
import com.google.gerrit.extensions.registration.DynamicMap;
|
||||||
import com.google.gerrit.extensions.restapi.IdString;
|
import com.google.gerrit.extensions.restapi.IdString;
|
||||||
import com.google.gerrit.extensions.restapi.RestApiException;
|
import com.google.gerrit.extensions.restapi.RestApiException;
|
||||||
import com.google.gerrit.extensions.restapi.TopLevelResource;
|
import com.google.gerrit.extensions.restapi.TopLevelResource;
|
||||||
import com.google.gerrit.extensions.restapi.Url;
|
import com.google.gerrit.extensions.restapi.Url;
|
||||||
|
import com.google.gerrit.server.DynamicOptions;
|
||||||
import com.google.gerrit.server.api.changes.ChangeApiImpl.DynamicOptionParser;
|
import com.google.gerrit.server.api.changes.ChangeApiImpl.DynamicOptionParser;
|
||||||
import com.google.gerrit.server.restapi.change.ChangesCollection;
|
import com.google.gerrit.server.restapi.change.ChangesCollection;
|
||||||
import com.google.gerrit.server.restapi.change.CreateChange;
|
import com.google.gerrit.server.restapi.change.CreateChange;
|
||||||
import com.google.gerrit.server.restapi.change.QueryChanges;
|
import com.google.gerrit.server.restapi.change.QueryChanges;
|
||||||
import com.google.inject.Inject;
|
import com.google.inject.Inject;
|
||||||
|
import com.google.inject.Injector;
|
||||||
import com.google.inject.Provider;
|
import com.google.inject.Provider;
|
||||||
import com.google.inject.Singleton;
|
import com.google.inject.Singleton;
|
||||||
import java.util.List;
|
import java.util.List;
|
||||||
@@ -46,6 +49,8 @@ class ChangesImpl implements Changes {
|
|||||||
private final CreateChange createChange;
|
private final CreateChange createChange;
|
||||||
private final DynamicOptionParser dynamicOptionParser;
|
private final DynamicOptionParser dynamicOptionParser;
|
||||||
private final Provider<QueryChanges> queryProvider;
|
private final Provider<QueryChanges> queryProvider;
|
||||||
|
private final Injector injector;
|
||||||
|
private final DynamicMap<DynamicOptions.DynamicBean> dynamicBeans;
|
||||||
|
|
||||||
@Inject
|
@Inject
|
||||||
ChangesImpl(
|
ChangesImpl(
|
||||||
@@ -53,12 +58,16 @@ class ChangesImpl implements Changes {
|
|||||||
ChangeApiImpl.Factory api,
|
ChangeApiImpl.Factory api,
|
||||||
CreateChange createChange,
|
CreateChange createChange,
|
||||||
DynamicOptionParser dynamicOptionParser,
|
DynamicOptionParser dynamicOptionParser,
|
||||||
Provider<QueryChanges> queryProvider) {
|
Provider<QueryChanges> queryProvider,
|
||||||
|
Injector injector,
|
||||||
|
DynamicMap<DynamicOptions.DynamicBean> dynamicBeans) {
|
||||||
this.changes = changes;
|
this.changes = changes;
|
||||||
this.api = api;
|
this.api = api;
|
||||||
this.createChange = createChange;
|
this.createChange = createChange;
|
||||||
this.dynamicOptionParser = dynamicOptionParser;
|
this.dynamicOptionParser = dynamicOptionParser;
|
||||||
this.queryProvider = queryProvider;
|
this.queryProvider = queryProvider;
|
||||||
|
this.injector = injector;
|
||||||
|
this.dynamicBeans = dynamicBeans;
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
@@ -123,34 +132,36 @@ class ChangesImpl implements Changes {
|
|||||||
}
|
}
|
||||||
|
|
||||||
private List<ChangeInfo> get(QueryRequest q) throws RestApiException {
|
private List<ChangeInfo> get(QueryRequest q) throws RestApiException {
|
||||||
QueryChanges qc = queryProvider.get();
|
try (DynamicOptions dynamicOptions = new DynamicOptions(injector, dynamicBeans)) {
|
||||||
if (q.getQuery() != null) {
|
QueryChanges qc = queryProvider.get();
|
||||||
qc.addQuery(q.getQuery());
|
if (q.getQuery() != null) {
|
||||||
}
|
qc.addQuery(q.getQuery());
|
||||||
qc.setLimit(q.getLimit());
|
|
||||||
qc.setStart(q.getStart());
|
|
||||||
qc.setNoLimit(q.getNoLimit());
|
|
||||||
for (ListChangesOption option : q.getOptions()) {
|
|
||||||
qc.addOption(option);
|
|
||||||
}
|
|
||||||
dynamicOptionParser.parseDynamicOptions(qc, q.getPluginOptions());
|
|
||||||
|
|
||||||
try {
|
|
||||||
List<?> result = qc.apply(TopLevelResource.INSTANCE).value();
|
|
||||||
if (result.isEmpty()) {
|
|
||||||
return ImmutableList.of();
|
|
||||||
}
|
}
|
||||||
|
qc.setLimit(q.getLimit());
|
||||||
|
qc.setStart(q.getStart());
|
||||||
|
qc.setNoLimit(q.getNoLimit());
|
||||||
|
for (ListChangesOption option : q.getOptions()) {
|
||||||
|
qc.addOption(option);
|
||||||
|
}
|
||||||
|
dynamicOptionParser.parseDynamicOptions(qc, q.getPluginOptions(), dynamicOptions);
|
||||||
|
|
||||||
// Check type safety of result; the extension API should be safer than the
|
try {
|
||||||
// REST API in this case, since it's intended to be used in Java.
|
List<?> result = qc.apply(TopLevelResource.INSTANCE).value();
|
||||||
Object first = requireNonNull(result.iterator().next());
|
if (result.isEmpty()) {
|
||||||
checkState(first instanceof ChangeInfo);
|
return ImmutableList.of();
|
||||||
@SuppressWarnings("unchecked")
|
}
|
||||||
List<ChangeInfo> infos = (List<ChangeInfo>) result;
|
|
||||||
|
|
||||||
return ImmutableList.copyOf(infos);
|
// Check type safety of result; the extension API should be safer than the
|
||||||
} catch (Exception e) {
|
// REST API in this case, since it's intended to be used in Java.
|
||||||
throw asRestApiException("Cannot query changes", e);
|
Object first = requireNonNull(result.iterator().next());
|
||||||
|
checkState(first instanceof ChangeInfo);
|
||||||
|
@SuppressWarnings("unchecked")
|
||||||
|
List<ChangeInfo> infos = (List<ChangeInfo>) result;
|
||||||
|
|
||||||
|
return ImmutableList.copyOf(infos);
|
||||||
|
} catch (Exception e) {
|
||||||
|
throw asRestApiException("Cannot query changes", e);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -235,6 +235,8 @@ public abstract class BaseCommand implements Command {
|
|||||||
protected void parseCommandLine(Object options, DynamicOptions pluginOptions)
|
protected void parseCommandLine(Object options, DynamicOptions pluginOptions)
|
||||||
throws UnloggedFailure {
|
throws UnloggedFailure {
|
||||||
final CmdLineParser clp = newCmdLineParser(options);
|
final CmdLineParser clp = newCmdLineParser(options);
|
||||||
|
pluginOptions.setBean(options);
|
||||||
|
pluginOptions.startLifecycleListeners();
|
||||||
pluginOptions.parseDynamicBeans(clp);
|
pluginOptions.parseDynamicBeans(clp);
|
||||||
pluginOptions.setDynamicBeans();
|
pluginOptions.setDynamicBeans();
|
||||||
pluginOptions.onBeanParseStart();
|
pluginOptions.onBeanParseStart();
|
||||||
@@ -468,8 +470,7 @@ public abstract class BaseCommand implements Command {
|
|||||||
|
|
||||||
try {
|
try {
|
||||||
if (thunk instanceof ProjectCommandRunnable) {
|
if (thunk instanceof ProjectCommandRunnable) {
|
||||||
try (DynamicOptions pluginOptions =
|
try (DynamicOptions pluginOptions = new DynamicOptions(injector, dynamicBeans)) {
|
||||||
new DynamicOptions(BaseCommand.this, injector, dynamicBeans)) {
|
|
||||||
((ProjectCommandRunnable) thunk).executeParseCommand(pluginOptions);
|
((ProjectCommandRunnable) thunk).executeParseCommand(pluginOptions);
|
||||||
projectName = ((ProjectCommandRunnable) thunk).getProjectName();
|
projectName = ((ProjectCommandRunnable) thunk).getProjectName();
|
||||||
thunk.run();
|
thunk.run();
|
||||||
|
|||||||
@@ -72,8 +72,7 @@ final class DispatchCommand extends BaseCommand {
|
|||||||
|
|
||||||
@Override
|
@Override
|
||||||
public void start(ChannelSession channel, Environment env) throws IOException {
|
public void start(ChannelSession channel, Environment env) throws IOException {
|
||||||
try (DynamicOptions pluginOptions =
|
try (DynamicOptions pluginOptions = new DynamicOptions(injector, dynamicBeans)) {
|
||||||
new DynamicOptions(DispatchCommand.this, injector, dynamicBeans)) {
|
|
||||||
parseCommandLine(pluginOptions);
|
parseCommandLine(pluginOptions);
|
||||||
if (Strings.isNullOrEmpty(commandName)) {
|
if (Strings.isNullOrEmpty(commandName)) {
|
||||||
StringWriter msg = new StringWriter();
|
StringWriter msg = new StringWriter();
|
||||||
|
|||||||
@@ -50,8 +50,7 @@ public abstract class SshCommand extends BaseCommand {
|
|||||||
public void start(ChannelSession channel, Environment env) throws IOException {
|
public void start(ChannelSession channel, Environment env) throws IOException {
|
||||||
startThread(
|
startThread(
|
||||||
() -> {
|
() -> {
|
||||||
try (DynamicOptions pluginOptions =
|
try (DynamicOptions pluginOptions = new DynamicOptions(injector, dynamicBeans)) {
|
||||||
new DynamicOptions(SshCommand.this, injector, dynamicBeans)) {
|
|
||||||
parseCommandLine(pluginOptions);
|
parseCommandLine(pluginOptions);
|
||||||
stdout = toPrintWriter(out);
|
stdout = toPrintWriter(out);
|
||||||
stderr = toPrintWriter(err);
|
stderr = toPrintWriter(err);
|
||||||
|
|||||||
@@ -93,7 +93,7 @@ public final class SuExec extends BaseCommand {
|
|||||||
|
|
||||||
@Override
|
@Override
|
||||||
public void start(ChannelSession channel, Environment env) throws IOException {
|
public void start(ChannelSession channel, Environment env) throws IOException {
|
||||||
try (DynamicOptions pluginOptions = new DynamicOptions(SuExec.this, injector, dynamicBeans)) {
|
try (DynamicOptions pluginOptions = new DynamicOptions(injector, dynamicBeans)) {
|
||||||
checkCanRunAs();
|
checkCanRunAs();
|
||||||
parseCommandLine(pluginOptions);
|
parseCommandLine(pluginOptions);
|
||||||
|
|
||||||
|
|||||||
@@ -108,8 +108,7 @@ public final class StreamEvents extends BaseCommand {
|
|||||||
|
|
||||||
@Override
|
@Override
|
||||||
public void start(ChannelSession channel, Environment env) throws IOException {
|
public void start(ChannelSession channel, Environment env) throws IOException {
|
||||||
try (DynamicOptions pluginOptions =
|
try (DynamicOptions pluginOptions = new DynamicOptions(injector, dynamicBeans)) {
|
||||||
new DynamicOptions(StreamEvents.this, injector, dynamicBeans)) {
|
|
||||||
try {
|
try {
|
||||||
parseCommandLine(pluginOptions);
|
parseCommandLine(pluginOptions);
|
||||||
} catch (UnloggedFailure e) {
|
} catch (UnloggedFailure e) {
|
||||||
|
|||||||
@@ -0,0 +1,52 @@
|
|||||||
|
// Copyright (C) 2020 The Android Open Source Project
|
||||||
|
//
|
||||||
|
// 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 com.google.gerrit.acceptance.ssh;
|
||||||
|
|
||||||
|
import static com.google.gerrit.server.query.change.OutputStreamQuery.GSON;
|
||||||
|
import static junit.framework.TestCase.assertEquals;
|
||||||
|
|
||||||
|
import com.google.common.collect.Lists;
|
||||||
|
import com.google.gerrit.acceptance.AbstractDynamicOptionsTest;
|
||||||
|
import com.google.gerrit.acceptance.NoHttpd;
|
||||||
|
import com.google.gerrit.acceptance.UseSsh;
|
||||||
|
import com.google.gson.reflect.TypeToken;
|
||||||
|
import com.google.inject.Module;
|
||||||
|
import java.io.IOException;
|
||||||
|
import java.util.List;
|
||||||
|
import org.junit.Test;
|
||||||
|
|
||||||
|
@NoHttpd
|
||||||
|
@UseSsh
|
||||||
|
public class DynamicOptionsIT extends AbstractDynamicOptionsTest {
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public Module createSshModule() {
|
||||||
|
return new AbstractDynamicOptionsTest.PluginOneSshModule();
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testDynamicPluginOptions() throws Exception {
|
||||||
|
try (AutoCloseable ignored =
|
||||||
|
installPlugin("my-plugin", AbstractDynamicOptionsTest.PluginTwoModule.class)) {
|
||||||
|
List<String> samples = getSamplesList(adminSshSession.exec("ls-samples"));
|
||||||
|
adminSshSession.assertSuccess();
|
||||||
|
assertEquals(Lists.newArrayList("sample1", "sample2"), samples);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
protected List<String> getSamplesList(String sshOutput) throws IOException {
|
||||||
|
return GSON.fromJson(sshOutput, new TypeToken<List<String>>() {}.getType());
|
||||||
|
}
|
||||||
|
}
|
||||||
Reference in New Issue
Block a user