blob: 86b55bc7887b89d1e177436e18fe2201b9a11a4e [file] [log] [blame]
// Copyright (C) 2019 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.api.change;
import static com.google.common.collect.ImmutableList.toImmutableList;
import static com.google.common.truth.Truth.assertThat;
import static java.util.Objects.requireNonNull;
import com.google.common.base.MoreObjects;
import com.google.common.io.CharStreams;
import com.google.common.reflect.TypeToken;
import com.google.gerrit.acceptance.AbstractDaemonTest;
import com.google.gerrit.acceptance.RestResponse;
import com.google.gerrit.acceptance.UseSsh;
import com.google.gerrit.common.Nullable;
import com.google.gerrit.extensions.annotations.Exports;
import com.google.gerrit.extensions.common.ChangeInfo;
import com.google.gerrit.extensions.common.PluginDefinedInfo;
import com.google.gerrit.json.OutputFormat;
import com.google.gerrit.reviewdb.client.Change;
import com.google.gerrit.server.query.change.ChangeQueryProcessor.ChangeAttributeFactory;
import com.google.gerrit.server.query.change.OutputStreamQuery;
import com.google.gson.Gson;
import com.google.inject.AbstractModule;
import java.io.StringReader;
import java.util.ArrayList;
import java.util.List;
import java.util.Map;
import java.util.Objects;
import org.junit.Test;
@UseSsh
public class PluginFieldsIT extends AbstractDaemonTest {
static class MyInfo extends PluginDefinedInfo {
@Nullable String theAttribute;
MyInfo(@Nullable String theAttribute) {
this.theAttribute = theAttribute;
}
MyInfo(String name, @Nullable String theAttribute) {
this.name = requireNonNull(name);
this.theAttribute = theAttribute;
}
@Override
public boolean equals(Object o) {
if (!(o instanceof MyInfo)) {
return false;
}
MyInfo i = (MyInfo) o;
return Objects.equals(name, i.name) && Objects.equals(theAttribute, i.theAttribute);
}
@Override
public int hashCode() {
return Objects.hash(name, theAttribute);
}
@Override
public String toString() {
return MoreObjects.toStringHelper(this)
.add("name", name)
.add("theAttribute", theAttribute)
.toString();
}
}
static class NullAttributeModule extends AbstractModule {
@Override
public void configure() {
bind(ChangeAttributeFactory.class)
.annotatedWith(Exports.named("always-null"))
.toInstance((cd, qp, p) -> null);
}
}
static class SimpleAttributeModule extends AbstractModule {
@Override
public void configure() {
bind(ChangeAttributeFactory.class)
.annotatedWith(Exports.named("simple"))
.toInstance((cd, qp, p) -> new MyInfo("change " + cd.getId()));
}
}
@Test
public void queryChangeApiWithNullAttribute() throws Exception {
queryChangeWithNullAttribute(
id -> pluginInfoFromSingletonList(gApi.changes().query(id.toString()).get()));
}
@Test
public void queryChangeRestWithNullAttribute() throws Exception {
queryChangeWithNullAttribute(
id -> pluginInfoFromSingletonListRest(adminRestSession.get(changeQueryUrl(id))));
}
@Test
public void queryChangeSshWithNullAttribute() throws Exception {
queryChangeWithNullAttribute(
id -> pluginInfoFromSingletonListSsh(adminSshSession.exec(changeQueryCmd(id))));
}
private void queryChangeWithNullAttribute(PluginInfoGetter getter) throws Exception {
Change.Id id = createChange().getChange().getId();
assertThat(getter.call(id)).isNull();
try (AutoCloseable ignored = installPlugin("my-plugin", NullAttributeModule.class)) {
assertThat(getter.call(id)).isNull();
}
assertThat(getter.call(id)).isNull();
}
@Test
public void queryChangeApiWithSimpleAttribute() throws Exception {
queryChangeWithSimpleAttribute(
id -> pluginInfoFromSingletonList(gApi.changes().query(id.toString()).get()));
}
@Test
public void queryChangeRestWithSimpleAttribute() throws Exception {
queryChangeWithSimpleAttribute(
id -> pluginInfoFromSingletonListRest(adminRestSession.get(changeQueryUrl(id))));
}
@Test
public void queryChangeSshWithSimpleAttribute() throws Exception {
queryChangeWithSimpleAttribute(
id -> pluginInfoFromSingletonListSsh(adminSshSession.exec(changeQueryCmd(id))));
}
private void queryChangeWithSimpleAttribute(PluginInfoGetter getter) throws Exception {
Change.Id id = createChange().getChange().getId();
assertThat(getter.call(id)).isNull();
try (AutoCloseable ignored = installPlugin("my-plugin", SimpleAttributeModule.class)) {
List<MyInfo> infos = getter.call(id);
assertThat(infos).containsExactly(new MyInfo("my-plugin", "change " + id));
}
assertThat(getter.call(id)).isNull();
}
private String changeQueryUrl(Change.Id id) {
return "/changes/?q=" + id;
}
private String changeQueryCmd(Change.Id id) {
return "gerrit query --format json " + id;
}
private static List<MyInfo> pluginInfoFromSingletonList(List<ChangeInfo> changeInfos) {
assertThat(changeInfos).hasSize(1);
List<PluginDefinedInfo> pluginInfo = changeInfos.get(0).plugins;
if (pluginInfo == null) {
return null;
}
return pluginInfo.stream().map(MyInfo.class::cast).collect(toImmutableList());
}
@Nullable
private static List<MyInfo> pluginInfoFromSingletonListRest(RestResponse res) throws Exception {
res.assertOK();
// Don't deserialize to ChangeInfo directly, since that would treat the plugins field as
// List<PluginDefinedInfo> and ignore the unknown keys found in MyInfo.
Gson gson = OutputFormat.JSON.newGson();
List<Map<String, Object>> changeInfos =
gson.fromJson(res.getReader(), new TypeToken<List<Map<String, Object>>>() {}.getType());
assertThat(changeInfos).hasSize(1);
Object plugins = changeInfos.get(0).get("plugins");
if (plugins == null) {
return null;
}
return gson.fromJson(gson.toJson(plugins), new TypeToken<List<MyInfo>>() {}.getType());
}
@Nullable
private static List<MyInfo> pluginInfoFromSingletonListSsh(String sshOutput) throws Exception {
Gson gson = OutputStreamQuery.GSON;
List<Map<String, Object>> changeAttrs = new ArrayList<>();
for (String line : CharStreams.readLines(new StringReader(sshOutput))) {
// Don't deserialize to ChangeAttribute directly, since that would treat the plugins field as
// List<PluginDefinedInfo> and ignore the unknown keys found in MyInfo.
Map<String, Object> changeAttr =
gson.fromJson(line, new TypeToken<Map<String, Object>>() {}.getType());
if (!"stats".equals(changeAttr.get("type"))) {
changeAttrs.add(changeAttr);
}
}
assertThat(changeAttrs).hasSize(1);
Object plugins = changeAttrs.get(0).get("plugins");
if (plugins == null) {
return null;
}
return gson.fromJson(gson.toJson(plugins), new TypeToken<List<MyInfo>>() {}.getType());
}
@FunctionalInterface
private interface PluginInfoGetter {
List<MyInfo> call(Change.Id id) throws Exception;
}
}